Commit bb718f79 authored by Paul McCullagh's avatar Paul McCullagh

Fixed a bug that resulted in 2-phase commit not being used between PBXT and the binlog

parent 4b643332
PBXT Release Notes PBXT Release Notes
================== ==================
------- 1.0.09g RC3 - 2009-12-16
RN292: Fixed a bug that resulted in 2-phase commit not being used between PBXT and the binlog. This bug was a result of a hack which as added to solve a problem in an pre-release version of MySQL 5.1. The hack was removed.
------- 1.0.09f RC3 - 2009-11-30 ------- 1.0.09f RC3 - 2009-11-30
RN291: Fixed bug #489088: On shutdown MySQL reports: [Warning] Plugin 'PBXT' will be forced to shutdown. RN291: Fixed bug #489088: On shutdown MySQL reports: [Warning] Plugin 'PBXT' will be forced to shutdown.
......
...@@ -355,10 +355,10 @@ static int sort_keys(KEY *a, KEY *b) ...@@ -355,10 +355,10 @@ static int sort_keys(KEY *a, KEY *b)
{ {
if (!(b_flags & HA_NOSAME)) if (!(b_flags & HA_NOSAME))
return -1; return -1;
if ((a_flags ^ b_flags) & (HA_NULL_PART_KEY | HA_END_SPACE_KEY)) if ((a_flags ^ b_flags) & HA_NULL_PART_KEY)
{ {
/* Sort NOT NULL keys before other keys */ /* Sort NOT NULL keys before other keys */
return (a_flags & (HA_NULL_PART_KEY | HA_END_SPACE_KEY)) ? 1 : -1; return (a_flags & HA_NULL_PART_KEY) ? 1 : -1;
} }
if (a->name == primary_key_name) if (a->name == primary_key_name)
return -1; return -1;
......
...@@ -1442,7 +1442,7 @@ static int pbxt_commit(handlerton *hton, THD *thd, bool all) ...@@ -1442,7 +1442,7 @@ static int pbxt_commit(handlerton *hton, THD *thd, bool all)
XTThreadPtr self; XTThreadPtr self;
if ((self = (XTThreadPtr) *thd_ha_data(thd, hton))) { if ((self = (XTThreadPtr) *thd_ha_data(thd, hton))) {
XT_PRINT1(self, "pbxt_commit all=%d\n", all); XT_PRINT2(self, "%s pbxt_commit all=%d\n", all ? "END CONN XACT" : "END STAT", all);
if (self->st_xact_data) { if (self->st_xact_data) {
/* There are no table locks, commit immediately in all cases /* There are no table locks, commit immediately in all cases
...@@ -1474,7 +1474,7 @@ static int pbxt_rollback(handlerton *hton, THD *thd, bool all) ...@@ -1474,7 +1474,7 @@ static int pbxt_rollback(handlerton *hton, THD *thd, bool all)
XTThreadPtr self; XTThreadPtr self;
if ((self = (XTThreadPtr) *thd_ha_data(thd, hton))) { if ((self = (XTThreadPtr) *thd_ha_data(thd, hton))) {
XT_PRINT1(self, "pbxt_rollback all=%d in pbxt_commit\n", all); XT_PRINT2(self, "%s pbxt_rollback all=%d\n", all ? "CONN END XACT" : "STAT END", all);
if (self->st_xact_data) { if (self->st_xact_data) {
/* There are no table locks, rollback immediately in all cases /* There are no table locks, rollback immediately in all cases
...@@ -1538,7 +1538,7 @@ static int pbxt_prepare(handlerton *hton, THD *thd, bool all) ...@@ -1538,7 +1538,7 @@ static int pbxt_prepare(handlerton *hton, THD *thd, bool all)
* except when this is a statement commit with an explicit * except when this is a statement commit with an explicit
* transaction (!all && !self->st_auto_commit). * transaction (!all && !self->st_auto_commit).
*/ */
if (all) { if (all || self->st_auto_commit) {
XID xid; XID xid;
XT_PRINT0(self, "xt_xn_prepare in pbxt_prepare\n"); XT_PRINT0(self, "xt_xn_prepare in pbxt_prepare\n");
...@@ -2620,26 +2620,7 @@ int ha_pbxt::write_row(byte *buf) ...@@ -2620,26 +2620,7 @@ int ha_pbxt::write_row(byte *buf)
} }
#endif #endif
/* GOTCHA: I have a huge problem with the transaction statement. /* {START-STAT-HACK} previously position of start statement hack. */
* It is not ALWAYS committed (I mean ha_commit_trans() is
* not always called - for example in SELECT).
*
* If I call trans_register_ha() but ha_commit_trans() is not called
* then MySQL thinks a transaction is still running (while
* I have committed the auto-transaction in ha_pbxt::external_lock()).
*
* This causes all kinds of problems, like transactions
* are killed when they should not be.
*
* To prevent this, I only inform MySQL that a transaction
* has beens started when an update is performed. I have determined that
* ha_commit_trans() is only guarenteed to be called if an update is done.
*/
if (!pb_open_tab->ot_thread->st_stat_trans) {
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(pb_open_tab->ot_thread, "ha_pbxt::write_row trans_register_ha all=FALSE\n");
pb_open_tab->ot_thread->st_stat_trans = TRUE;
}
xt_xlog_check_long_writer(pb_open_tab->ot_thread); xt_xlog_check_long_writer(pb_open_tab->ot_thread);
...@@ -2730,11 +2711,7 @@ int ha_pbxt::update_row(const byte * old_data, byte * new_data) ...@@ -2730,11 +2711,7 @@ int ha_pbxt::update_row(const byte * old_data, byte * new_data)
XT_DISABLED_TRACE(("UPDATE tx=%d val=%d\n", (int) self->st_xact_data->xd_start_xn_id, (int) XT_GET_DISK_4(&new_data[1]))); XT_DISABLED_TRACE(("UPDATE tx=%d val=%d\n", (int) self->st_xact_data->xd_start_xn_id, (int) XT_GET_DISK_4(&new_data[1])));
//statistic_increment(ha_update_count,&LOCK_status); //statistic_increment(ha_update_count,&LOCK_status);
if (!self->st_stat_trans) { /* {START-STAT-HACK} previously position of start statement hack. */
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(self, "ha_pbxt::update_row trans_register_ha all=FALSE\n");
self->st_stat_trans = TRUE;
}
xt_xlog_check_long_writer(self); xt_xlog_check_long_writer(self);
...@@ -2821,11 +2798,7 @@ int ha_pbxt::delete_row(const byte * buf) ...@@ -2821,11 +2798,7 @@ int ha_pbxt::delete_row(const byte * buf)
} }
#endif #endif
if (!pb_open_tab->ot_thread->st_stat_trans) { /* {START-STAT-HACK} previously position of start statement hack. */
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(pb_open_tab->ot_thread, "ha_pbxt::delete_row trans_register_ha all=FALSE\n");
pb_open_tab->ot_thread->st_stat_trans = TRUE;
}
xt_xlog_check_long_writer(pb_open_tab->ot_thread); xt_xlog_check_long_writer(pb_open_tab->ot_thread);
...@@ -3155,15 +3128,12 @@ int ha_pbxt::index_init(uint idx, bool XT_UNUSED(sorted)) ...@@ -3155,15 +3128,12 @@ int ha_pbxt::index_init(uint idx, bool XT_UNUSED(sorted))
printf("index_init %s index %d cols req=%d/%d read_bits=%X write_bits=%X index_bits=%X\n", pb_open_tab->ot_table->tab_name->ps_path, (int) idx, pb_open_tab->ot_cols_req, pb_open_tab->ot_cols_req, (int) *table->read_set->bitmap, (int) *table->write_set->bitmap, (int) *ind->mi_col_map.bitmap); printf("index_init %s index %d cols req=%d/%d read_bits=%X write_bits=%X index_bits=%X\n", pb_open_tab->ot_table->tab_name->ps_path, (int) idx, pb_open_tab->ot_cols_req, pb_open_tab->ot_cols_req, (int) *table->read_set->bitmap, (int) *table->write_set->bitmap, (int) *ind->mi_col_map.bitmap);
#endif #endif
/* {START-STAT-HACK} previously position of start statement hack,
* previous comment to code below: */
/* Start a statement based transaction as soon /* Start a statement based transaction as soon
* as a read is done for a modify type statement! * as a read is done for a modify type statement!
* Previously, this was done too late! * Previously, this was done too late!
*/ */
if (!thread->st_stat_trans) {
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(thread, "ha_pbxt::update_row trans_register_ha all=FALSE\n");
thread->st_stat_trans = TRUE;
}
} }
else { else {
pb_open_tab->ot_cols_req = ha_get_max_bit(table->read_set); pb_open_tab->ot_cols_req = ha_get_max_bit(table->read_set);
...@@ -3612,15 +3582,12 @@ int ha_pbxt::rnd_init(bool scan) ...@@ -3612,15 +3582,12 @@ int ha_pbxt::rnd_init(bool scan)
/* The number of columns required: */ /* The number of columns required: */
if (pb_open_tab->ot_is_modify) { if (pb_open_tab->ot_is_modify) {
pb_open_tab->ot_cols_req = table->read_set->MX_BIT_SIZE(); pb_open_tab->ot_cols_req = table->read_set->MX_BIT_SIZE();
/* {START-STAT-HACK} previously position of start statement hack,
* previous comment to code below: */
/* Start a statement based transaction as soon /* Start a statement based transaction as soon
* as a read is done for a modify type statement! * as a read is done for a modify type statement!
* Previously, this was done too late! * Previously, this was done too late!
*/ */
if (!thread->st_stat_trans) {
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(thread, "ha_pbxt::update_row trans_register_ha all=FALSE\n");
thread->st_stat_trans = TRUE;
}
} }
else { else {
pb_open_tab->ot_cols_req = ha_get_max_bit(table->read_set); pb_open_tab->ot_cols_req = ha_get_max_bit(table->read_set);
...@@ -4631,7 +4598,7 @@ xtPublic int ha_pbxt::external_lock(THD *thd, int lock_type) ...@@ -4631,7 +4598,7 @@ xtPublic int ha_pbxt::external_lock(THD *thd, int lock_type)
cont_(b); cont_(b);
} }
/* See (***) */ /* See {IS-UPDATE-STAT} */
self->st_is_update = FALSE; self->st_is_update = FALSE;
/* Auto begin a transaction (if one is not already running): */ /* Auto begin a transaction (if one is not already running): */
...@@ -4660,7 +4627,7 @@ xtPublic int ha_pbxt::external_lock(THD *thd, int lock_type) ...@@ -4660,7 +4627,7 @@ xtPublic int ha_pbxt::external_lock(THD *thd, int lock_type)
} }
/* /*
* (**) GOTCHA: trans_register_ha() is not mentioned in the documentation. * {START-TRANS} GOTCHA: trans_register_ha() is not mentioned in the documentation.
* It must be called to inform MySQL that we have a transaction (see start_stmt). * It must be called to inform MySQL that we have a transaction (see start_stmt).
* *
* Here are some tests that confirm whether things are done correctly: * Here are some tests that confirm whether things are done correctly:
...@@ -4698,8 +4665,44 @@ xtPublic int ha_pbxt::external_lock(THD *thd, int lock_type) ...@@ -4698,8 +4665,44 @@ xtPublic int ha_pbxt::external_lock(THD *thd, int lock_type)
*/ */
if (!self->st_auto_commit) { if (!self->st_auto_commit) {
trans_register_ha(thd, TRUE, pbxt_hton); trans_register_ha(thd, TRUE, pbxt_hton);
XT_PRINT0(self, "ha_pbxt::external_lock trans_register_ha all=TRUE\n"); XT_PRINT0(self, "CONN START XACT - ha_pbxt::external_lock --> trans_register_ha\n");
}
} }
/* Start a statment transaction: */
/* {START-STAT-HACK} The problem that ha_commit_trans() is not
* called by MySQL seems to be fixed (tests confirm this).
* Here is the previous comment when this code was execute
* here {START-STAT-HACK}
*
* GOTCHA: I have a huge problem with the transaction statement.
* It is not ALWAYS committed (I mean ha_commit_trans() is
* not always called - for example in SELECT).
*
* If I call trans_register_ha() but ha_commit_trans() is not called
* then MySQL thinks a transaction is still running (while
* I have committed the auto-transaction in ha_pbxt::external_lock()).
*
* This causes all kinds of problems, like transactions
* are killed when they should not be.
*
* To prevent this, I only inform MySQL that a transaction
* has beens started when an update is performed. I have determined that
* ha_commit_trans() is only guarenteed to be called if an update is done.
* --------
*
* So, this is the correct place to start a statement transaction.
*
* Note: if trans_register_ha() is not called before ha_write_row(), then
* PBXT is not registered correctly as a modification transaction.
* (mark_trx_read_write call in ha_write_row).
* This leads to 2-phase commit not being called as it should when
* binary logging is enabled.
*/
if (!pb_open_tab->ot_thread->st_stat_trans) {
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(pb_open_tab->ot_thread, "STAT START - ha_pbxt::external_lock --> trans_register_ha\n");
pb_open_tab->ot_thread->st_stat_trans = TRUE;
} }
if (lock_type == F_WRLCK || self->st_xact_mode < XT_XACT_REPEATABLE_READ) if (lock_type == F_WRLCK || self->st_xact_mode < XT_XACT_REPEATABLE_READ)
...@@ -4826,7 +4829,7 @@ int ha_pbxt::start_stmt(THD *thd, thr_lock_type lock_type) ...@@ -4826,7 +4829,7 @@ int ha_pbxt::start_stmt(THD *thd, thr_lock_type lock_type)
} }
} }
/* (***) This is required at this level! /* {IS-UPDATE-STAT} This is required at this level!
* No matter how often it is called, it is still the start of a * No matter how often it is called, it is still the start of a
* statement. We need to make sure statements that are NOT mistaken * statement. We need to make sure statements that are NOT mistaken
* for different type of statement. * for different type of statement.
...@@ -4841,7 +4844,7 @@ int ha_pbxt::start_stmt(THD *thd, thr_lock_type lock_type) ...@@ -4841,7 +4844,7 @@ int ha_pbxt::start_stmt(THD *thd, thr_lock_type lock_type)
*/ */
self->st_is_update = FALSE; self->st_is_update = FALSE;
/* See comment (**) */ /* See comment {START-TRANS} */
if (!self->st_xact_data) { if (!self->st_xact_data) {
self->st_xact_mode = thd_tx_isolation(thd) <= ISO_READ_COMMITTED ? XT_XACT_COMMITTED_READ : XT_XACT_REPEATABLE_READ; self->st_xact_mode = thd_tx_isolation(thd) <= ISO_READ_COMMITTED ? XT_XACT_COMMITTED_READ : XT_XACT_REPEATABLE_READ;
self->st_ignore_fkeys = (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) != 0; self->st_ignore_fkeys = (thd_test_options(thd, OPTION_NO_FOREIGN_KEY_CHECKS)) != 0;
...@@ -4858,10 +4861,17 @@ int ha_pbxt::start_stmt(THD *thd, thr_lock_type lock_type) ...@@ -4858,10 +4861,17 @@ int ha_pbxt::start_stmt(THD *thd, thr_lock_type lock_type)
} }
if (!self->st_auto_commit) { if (!self->st_auto_commit) {
trans_register_ha(thd, TRUE, pbxt_hton); trans_register_ha(thd, TRUE, pbxt_hton);
XT_PRINT0(self, "ha_pbxt::start_stmt trans_register_ha all=TRUE\n"); XT_PRINT0(self, "START CONN XACT - ha_pbxt::start_stmt --> trans_register_ha\n");
} }
} }
/* Start a statment (see {START-STAT-HACK}): */
if (!pb_open_tab->ot_thread->st_stat_trans) {
trans_register_ha(pb_mysql_thd, FALSE, pbxt_hton);
XT_PRINT0(pb_open_tab->ot_thread, "START STAT - ha_pbxt::start_stmt --> trans_register_ha\n");
pb_open_tab->ot_thread->st_stat_trans = TRUE;
}
if (pb_open_tab->ot_for_update || self->st_xact_mode < XT_XACT_REPEATABLE_READ) if (pb_open_tab->ot_for_update || self->st_xact_mode < XT_XACT_REPEATABLE_READ)
self->st_visible_time = self->st_database->db_xn_end_time; self->st_visible_time = self->st_database->db_xn_end_time;
......
...@@ -380,7 +380,7 @@ xtPublic void xt_int8_to_byte_size(xtInt8 value, char *string) ...@@ -380,7 +380,7 @@ xtPublic void xt_int8_to_byte_size(xtInt8 value, char *string)
/* Version number must also be set in configure.in! */ /* Version number must also be set in configure.in! */
xtPublic c_char *xt_get_version(void) xtPublic c_char *xt_get_version(void)
{ {
return "1.0.09f RC"; return "1.0.09g RC";
} }
/* Copy and URL decode! */ /* Copy and URL decode! */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment