Commit 5dd028f8 authored by Marko Mäkelä's avatar Marko Mäkelä

MDEV-24700 Assertion "lock not found"==0 in lock_table_x_unlock()

After an ignored INSERT IGNORE statement into an empty table, we would
wrongly use the MDEV-515 table-level undo logging for a subsequent
REPLACE statement.

ha_innobase::reset_template(): Clear m_prebuilt->ins_node->bulk_insert
on every statement boundary.

ha_innobase::start_stmt(): Invoke end_bulk_insert().

ha_innobase::extra(): Avoid accessing m_prebuilt->trx. Do not call
thd_to_trx(). Invoke end_bulk_insert() and try to reset bulk_insert
when changing the REPLACE or IGNORE settings.

trx_mod_table_time_t::WAS_BULK: Use a distinct value from BULK.

trx_undo_report_row_operation(): Add debug assertions.

Note: Some calls to end_bulk_insert() may be redundant, but statement
boundaries are not always clear in the API (especially in the
presence of LOCK TABLES or stored procedures).
parent 121d0f7f
...@@ -22,3 +22,18 @@ SELECT * FROM t2; ...@@ -22,3 +22,18 @@ SELECT * FROM t2;
a a
3 3
DROP TABLE t1, t2; DROP TABLE t1, t2;
#
# MDEV-24700 Assertion "lock not found"==0 in lock_table_x_unlock()
#
SET FOREIGN_KEY_CHECKS=OFF;
CREATE TABLE t1 (id INT PRIMARY KEY, f INT REFERENCES nonexistent(x))
ENGINE=InnoDB;
SET FOREIGN_KEY_CHECKS=ON;
BEGIN;
INSERT IGNORE INTO t1 VALUES (1,11);
Warnings:
Warning 1452 Cannot add or update a child row: a foreign key constraint fails (`test`.`t1`, CONSTRAINT `t1_ibfk_1` FOREIGN KEY (`f`) REFERENCES `nonexistent` (`x`))
REPLACE INTO t1 VALUES (1,12);
ERROR 23000: Cannot add or update a child row: a foreign key constraint fails (`test`.`t1`, CONSTRAINT `t1_ibfk_1` FOREIGN KEY (`f`) REFERENCES `nonexistent` (`x`))
COMMIT;
DROP TABLE t1;
...@@ -135,7 +135,6 @@ drop table t1; ...@@ -135,7 +135,6 @@ drop table t1;
connection default; connection default;
CREATE TABLE t1 (a INT PRIMARY KEY, b INT NOT NULL) ENGINE=InnoDB; CREATE TABLE t1 (a INT PRIMARY KEY, b INT NOT NULL) ENGINE=InnoDB;
INSERT INTO t1 VALUES(3,1); INSERT INTO t1 VALUES(3,1);
BEGIN; BEGIN;
......
...@@ -24,3 +24,18 @@ COMMIT; ...@@ -24,3 +24,18 @@ COMMIT;
SELECT * FROM t1; SELECT * FROM t1;
SELECT * FROM t2; SELECT * FROM t2;
DROP TABLE t1, t2; DROP TABLE t1, t2;
--echo #
--echo # MDEV-24700 Assertion "lock not found"==0 in lock_table_x_unlock()
--echo #
SET FOREIGN_KEY_CHECKS=OFF;
CREATE TABLE t1 (id INT PRIMARY KEY, f INT REFERENCES nonexistent(x))
ENGINE=InnoDB;
SET FOREIGN_KEY_CHECKS=ON;
BEGIN;
INSERT IGNORE INTO t1 VALUES (1,11);
--error ER_NO_REFERENCED_ROW_2
REPLACE INTO t1 VALUES (1,12);
COMMIT;
DROP TABLE t1;
...@@ -3039,6 +3039,9 @@ ha_innobase::reset_template(void) ...@@ -3039,6 +3039,9 @@ ha_innobase::reset_template(void)
m_prebuilt->pk_filter = NULL; m_prebuilt->pk_filter = NULL;
m_prebuilt->template_type = ROW_MYSQL_NO_TEMPLATE; m_prebuilt->template_type = ROW_MYSQL_NO_TEMPLATE;
} }
if (ins_node_t* node = m_prebuilt->ins_node) {
node->bulk_insert = false;
}
} }
/*****************************************************************//** /*****************************************************************//**
...@@ -15062,11 +15065,9 @@ ha_innobase::extra( ...@@ -15062,11 +15065,9 @@ ha_innobase::extra(
enum ha_extra_function operation) enum ha_extra_function operation)
/*!< in: HA_EXTRA_FLUSH or some other flag */ /*!< in: HA_EXTRA_FLUSH or some other flag */
{ {
check_trx_exists(ha_thd()); /* Warning: since it is not sure that MariaDB calls external_lock()
before calling this function, m_prebuilt->trx can be obsolete! */
/* Warning: since it is not sure that MySQL calls external_lock trx_t* trx = check_trx_exists(ha_thd());
before calling this function, the trx field in m_prebuilt can be
obsolete! */
switch (operation) { switch (operation) {
case HA_EXTRA_FLUSH: case HA_EXTRA_FLUSH:
...@@ -15076,7 +15077,19 @@ ha_innobase::extra( ...@@ -15076,7 +15077,19 @@ ha_innobase::extra(
break; break;
case HA_EXTRA_RESET_STATE: case HA_EXTRA_RESET_STATE:
reset_template(); reset_template();
thd_to_trx(ha_thd())->duplicates = 0; trx->duplicates = 0;
/* fall through */
case HA_EXTRA_IGNORE_INSERT:
/* HA_EXTRA_IGNORE_INSERT is very similar to
HA_EXTRA_IGNORE_DUP_KEY, but with one crucial difference:
we want !trx->duplicates for INSERT IGNORE so that
row_ins_duplicate_error_in_clust() will acquire a
shared lock instead of an exclusive lock. */
stmt_boundary:
trx->end_bulk_insert(*m_prebuilt->table);
if (ins_node_t* node = m_prebuilt->ins_node) {
node->bulk_insert = false;
}
break; break;
case HA_EXTRA_NO_KEYREAD: case HA_EXTRA_NO_KEYREAD:
m_prebuilt->read_just_key = 0; m_prebuilt->read_just_key = 0;
...@@ -15087,33 +15100,27 @@ ha_innobase::extra( ...@@ -15087,33 +15100,27 @@ ha_innobase::extra(
case HA_EXTRA_KEYREAD_PRESERVE_FIELDS: case HA_EXTRA_KEYREAD_PRESERVE_FIELDS:
m_prebuilt->keep_other_fields_on_keyread = 1; m_prebuilt->keep_other_fields_on_keyread = 1;
break; break;
/* IMPORTANT: m_prebuilt->trx can be obsolete in
this method, because it is not sure that MySQL
calls external_lock before this method with the
parameters below. We must not invoke update_thd()
either, because the calling threads may change.
CAREFUL HERE, OR MEMORY CORRUPTION MAY OCCUR! */
case HA_EXTRA_INSERT_WITH_UPDATE: case HA_EXTRA_INSERT_WITH_UPDATE:
thd_to_trx(ha_thd())->duplicates |= TRX_DUP_IGNORE; trx->duplicates |= TRX_DUP_IGNORE;
break; goto stmt_boundary;
case HA_EXTRA_NO_IGNORE_DUP_KEY: case HA_EXTRA_NO_IGNORE_DUP_KEY:
thd_to_trx(ha_thd())->duplicates &= ~TRX_DUP_IGNORE; trx->duplicates &= ~TRX_DUP_IGNORE;
break; goto stmt_boundary;
case HA_EXTRA_WRITE_CAN_REPLACE: case HA_EXTRA_WRITE_CAN_REPLACE:
thd_to_trx(ha_thd())->duplicates |= TRX_DUP_REPLACE; trx->duplicates |= TRX_DUP_REPLACE;
break; goto stmt_boundary;
case HA_EXTRA_WRITE_CANNOT_REPLACE: case HA_EXTRA_WRITE_CANNOT_REPLACE:
thd_to_trx(ha_thd())->duplicates &= ~TRX_DUP_REPLACE; trx->duplicates &= ~TRX_DUP_REPLACE;
break; goto stmt_boundary;
case HA_EXTRA_BEGIN_ALTER_COPY: case HA_EXTRA_BEGIN_ALTER_COPY:
m_prebuilt->table->skip_alter_undo = 1; m_prebuilt->table->skip_alter_undo = 1;
if (m_prebuilt->table->is_temporary() if (m_prebuilt->table->is_temporary()
|| !m_prebuilt->table->versioned_by_id()) { || !m_prebuilt->table->versioned_by_id()) {
break; break;
} }
trx_start_if_not_started(m_prebuilt->trx, true); ut_ad(trx == m_prebuilt->trx);
m_prebuilt->trx->mod_tables.emplace( trx_start_if_not_started(trx, true);
trx->mod_tables.emplace(
const_cast<dict_table_t*>(m_prebuilt->table), 0) const_cast<dict_table_t*>(m_prebuilt->table), 0)
.first->second.set_versioned(0); .first->second.set_versioned(0);
break; break;
...@@ -15123,14 +15130,7 @@ ha_innobase::extra( ...@@ -15123,14 +15130,7 @@ ha_innobase::extra(
case HA_EXTRA_FAKE_START_STMT: case HA_EXTRA_FAKE_START_STMT:
trx_register_for_2pc(m_prebuilt->trx); trx_register_for_2pc(m_prebuilt->trx);
m_prebuilt->sql_stat_start = true; m_prebuilt->sql_stat_start = true;
break; goto stmt_boundary;
case HA_EXTRA_IGNORE_INSERT:
if (ins_node_t* node = m_prebuilt->ins_node) {
if (node->bulk_insert) {
m_prebuilt->trx->end_bulk_insert(*node->table);
}
}
break;
default:/* Do nothing */ default:/* Do nothing */
; ;
} }
...@@ -15195,6 +15195,7 @@ ha_innobase::start_stmt( ...@@ -15195,6 +15195,7 @@ ha_innobase::start_stmt(
m_prebuilt->sql_stat_start = TRUE; m_prebuilt->sql_stat_start = TRUE;
m_prebuilt->hint_need_to_fetch_extra_cols = 0; m_prebuilt->hint_need_to_fetch_extra_cols = 0;
reset_template(); reset_template();
trx->end_bulk_insert(*m_prebuilt->table);
if (m_prebuilt->table->is_temporary() if (m_prebuilt->table->is_temporary()
&& m_mysql_has_locked && m_mysql_has_locked
...@@ -15367,7 +15368,7 @@ ha_innobase::external_lock( ...@@ -15367,7 +15368,7 @@ ha_innobase::external_lock(
m_prebuilt->hint_need_to_fetch_extra_cols = 0; m_prebuilt->hint_need_to_fetch_extra_cols = 0;
reset_template(); reset_template();
trx->end_bulk_insert(); trx->end_bulk_insert(*m_prebuilt->table);
switch (m_prebuilt->table->quiesce) { switch (m_prebuilt->table->quiesce) {
case QUIESCE_START: case QUIESCE_START:
......
...@@ -512,9 +512,9 @@ class trx_mod_table_time_t ...@@ -512,9 +512,9 @@ class trx_mod_table_time_t
/** Flag in 'first' to indicate that some operations were /** Flag in 'first' to indicate that some operations were
covered by a TRX_UNDO_EMPTY record (for the first statement to covered by a TRX_UNDO_EMPTY record (for the first statement to
insert into an empty table) */ insert into an empty table) */
static constexpr undo_no_t WAS_BULK= 1ULL << 63; static constexpr undo_no_t WAS_BULK= 1ULL << 62;
/** First modification of the table, possibly ORed with BULK */ /** First modification of the table, possibly ORed with BULK or WAS_BULK */
undo_no_t first; undo_no_t first;
/** First modification of a system versioned column (or NONE) */ /** First modification of a system versioned column (or NONE) */
undo_no_t first_versioned= NONE; undo_no_t first_versioned= NONE;
......
...@@ -2010,6 +2010,9 @@ trx_undo_report_row_operation( ...@@ -2010,6 +2010,9 @@ trx_undo_report_row_operation(
(below) can set the flag. */ (below) can set the flag. */
ut_ad(!m.second); ut_ad(!m.second);
/* We already wrote a TRX_UNDO_EMPTY record. */ /* We already wrote a TRX_UNDO_EMPTY record. */
ut_ad(thr->run_node);
ut_ad(que_node_get_type(thr->run_node) == QUE_NODE_INSERT);
ut_ad(static_cast<ins_node_t*>(thr->run_node)->bulk_insert);
return DB_SUCCESS; return DB_SUCCESS;
} else if (m.second } else if (m.second
&& thr->run_node && thr->run_node
......
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