mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-24 07:13:33 +03:00 
			
		
		
		
	2617.31.12, 2617.31.15, 2617.31.15, 2617.31.16, 2617.43.1 - initial changeset that introduced the fix for Bug#989 and follow up fixes for all test suite failures introduced in the initial changeset. ------------------------------------------------------------ revno: 2617.31.1 committer: Davi Arnaut <Davi.Arnaut@Sun.COM> branch nick: 4284-6.0 timestamp: Fri 2009-03-06 19:17:00 -0300 message: Bug#989: If DROP TABLE while there's an active transaction, wrong binlog order WL#4284: Transactional DDL locking Currently the MySQL server does not keep metadata locks on schema objects for the duration of a transaction, thus failing to guarantee the integrity of the schema objects being used during the transaction and to protect then from concurrent DDL operations. This also poses a problem for replication as a DDL operation might be replicated even thought there are active transactions using the object being modified. The solution is to defer the release of metadata locks until a active transaction is either committed or rolled back. This prevents other statements from modifying the table for the entire duration of the transaction. This provides commitment ordering for guaranteeing serializability across multiple transactions. - Incompatible change: If MySQL's metadata locking system encounters a lock conflict, the usual schema is to use the try and back-off technique to avoid deadlocks -- this schema consists in releasing all locks and trying to acquire them all in one go. But in a transactional context this algorithm can't be utilized as its not possible to release locks acquired during the course of the transaction without breaking the transaction commitments. To avoid deadlocks in this case, the ER_LOCK_DEADLOCK will be returned if a lock conflict is encountered during a transaction. Let's consider an example: A transaction has two statements that modify table t1, then table t2, and then commits. The first statement of the transaction will acquire a shared metadata lock on table t1, and it will be kept utill COMMIT to ensure serializability. At the moment when the second statement attempts to acquire a shared metadata lock on t2, a concurrent ALTER or DROP statement might have locked t2 exclusively. The prescription of the current locking protocol is that the acquirer of the shared lock backs off -- gives up all his current locks and retries. This implies that the entire multi-statement transaction has to be rolled back. - Incompatible change: FLUSH commands such as FLUSH PRIVILEGES and FLUSH TABLES WITH READ LOCK won't cause locked tables to be implicitly unlocked anymore. mysql-test/extra/binlog_tests/drop_table.test: Add test case for Bug#989. mysql-test/extra/binlog_tests/mix_innodb_myisam_binlog.test: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction. mysql-test/include/mix1.inc: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction. mysql-test/include/mix2.inc: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction. mysql-test/r/flush_block_commit.result: Update test case result (WL#4284). mysql-test/r/flush_block_commit_notembedded.result: Update test case result (WL#4284). mysql-test/r/innodb.result: Update test case result (WL#4284). mysql-test/r/innodb_mysql.result: Update test case result (WL#4284). mysql-test/r/lock.result: Add test case result for an effect of WL#4284/Bug#989 (all locks should be released when a connection terminates). mysql-test/r/mix2_myisam.result: Update test case result (effects of WL#4284/Bug#989). mysql-test/r/not_embedded_server.result: Update test case result (effects of WL#4284/Bug#989). Add a test case for interaction of WL#4284 and FLUSH PRIVILEGES. mysql-test/r/partition_innodb_semi_consistent.result: Update test case result (effects of WL#4284/Bug#989). mysql-test/r/partition_sync.result: Temporarily disable the test case for Bug#43867, which will be fixed by a subsequent backport. mysql-test/r/ps.result: Add a test case for effect of PREPARE on transactional locks: we take a savepoint at beginning of PREAPRE and release it at the end. Thus PREPARE does not accumulate metadata locks (Bug#989/WL#4284). mysql-test/r/read_only_innodb.result: Update test case result (effects of WL#4284/Bug#989). mysql-test/suite/binlog/r/binlog_row_drop_tbl.result: Add a test case result (WL#4284/Bug#989). mysql-test/suite/binlog/r/binlog_row_mix_innodb_myisam.result: Update test case result (effects of WL#4284/Bug#989). mysql-test/suite/binlog/r/binlog_stm_drop_tbl.result: Add a test case result (WL#4284/Bug#989). mysql-test/suite/binlog/r/binlog_stm_mix_innodb_myisam.result: Update test case result (effects of WL#4284/Bug#989). mysql-test/suite/binlog/r/binlog_unsafe.result: A side effect of Bug#989 -- slightly different table map ids. mysql-test/suite/binlog/t/binlog_row_drop_tbl.test: Add a test case for WL#4284/Bug#989. mysql-test/suite/binlog/t/binlog_stm_drop_tbl.test: Add a test case for WL#4284/Bug#989. mysql-test/suite/binlog/t/binlog_stm_row.test: Update to the new state name. This is actually a follow up to another patch for WL#4284, that changes Locked thread state to Table lock. mysql-test/suite/ndb/r/ndb_index_ordered.result: Remove result for disabled part of the test case. mysql-test/suite/ndb/t/disabled.def: Temporarily disable a test case (Bug#45621). mysql-test/suite/ndb/t/ndb_index_ordered.test: Disable a part of a test case (needs update to reflect semantics of Bug#989). mysql-test/suite/rpl/t/disabled.def: Disable tests made meaningless by transactional metadata locking. mysql-test/suite/sys_vars/r/autocommit_func.result: Add a commit (Bug#989). mysql-test/suite/sys_vars/t/autocommit_func.test: Add a commit (Bug#989). mysql-test/t/flush_block_commit.test: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction. mysql-test/t/flush_block_commit_notembedded.test: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction. Add a test case for transaction-scope locks and the global read lock (Bug#989/WL#4284). mysql-test/t/innodb.test: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction (effects of Bug#989/WL#4284). mysql-test/t/lock.test: Add a test case for Bug#989/WL#4284. mysql-test/t/not_embedded_server.test: Add a test case for Bug#989/WL#4284. mysql-test/t/partition_innodb_semi_consistent.test: Replace TRUNCATE with DELETE, to not issue an implicit commit of a transaction, and not depend on metadata locks. mysql-test/t/partition_sync.test: Temporarily disable the test case for Bug#43867, which needs a fix to be backported from 6.0. mysql-test/t/ps.test: Add a test case for semantics of PREPARE and transaction-scope locks: metadata locks on tables used in PREPARE are enclosed into a temporary savepoint, taken at the beginning of PREPARE, and released at the end. Thus PREPARE does not effect what locks a transaction owns. mysql-test/t/read_only_innodb.test: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction (Bug#989/WL#4284). Wait for the read_only statement to actually flush tables before sending other concurrent statements that depend on its state. mysql-test/t/xa.test: Fix test case to reflect the fact that transactions now hold metadata locks for the duration of a transaction (Bug#989/WL#4284). sql/ha_ndbcluster_binlog.cc: Backport bits of changes of ha_ndbcluster_binlog.cc from 6.0, to fix the failing binlog test suite with WL#4284. WL#4284 implementation does not work with 5.1 implementation of ndbcluster binlog index. sql/log_event.cc: Release metadata locks after issuing a commit. sql/mdl.cc: Style changes (WL#4284). sql/mysql_priv.h: Rename parameter to match the name used in the definition (WL#4284). sql/rpl_injector.cc: Release metadata locks on commit (WL#4284). sql/rpl_rli.cc: Remove assert made meaningless, metadata locks are released at the end of the transaction. sql/set_var.cc: Close tables and release locks if autocommit mode is set. sql/slave.cc: Release metadata locks after a rollback. sql/sql_acl.cc: Don't implicitly unlock locked tables. Issue a implicit commit at the end and unlock tables. sql/sql_base.cc: Defer the release of metadata locks when closing tables if not required to. Issue a deadlock error if the locking protocol requires that a transaction re-acquire its locks. Release metadata locks when closing tables for reopen. sql/sql_class.cc: Release metadata locks if the thread is killed. sql/sql_parse.cc: Release metadata locks after implicitly committing a active transaction, or after explicit commits or rollbacks. sql/sql_plugin.cc: Allocate MDL request on the stack as the use of the table is contained within the function. It will be removed from the context once close_thread_tables is called at the end of the function. sql/sql_prepare.cc: The problem is that the prepare phase of the CREATE TABLE statement takes a exclusive metadata lock lock and this can cause a self-deadlock the thread already holds a shared lock on the table being that should be created. The solution is to make the prepare phase take a shared metadata lock when preparing a CREATE TABLE statement. The execution of the statement will still acquire a exclusive lock, but won't cause any problem as it issues a implicit commit. After some discussions with stakeholders it has been decided that metadata locks acquired during a PREPARE statement must be released once the statement is prepared even if it is prepared within a multi statement transaction. sql/sql_servers.cc: Don't implicitly unlock locked tables. Issue a implicit commit at the end and unlock tables. sql/sql_table.cc: Close table and release metadata locks after a admin operation. sql/table.h: The problem is that the prepare phase of the CREATE TABLE statement takes a exclusive metadata lock lock and this can cause a self-deadlock the thread already holds a shared lock on the table being that should be created. The solution is to make the prepare phase take a shared metadata lock when preparing a CREATE TABLE statement. The execution of the statement will still acquire a exclusive lock, but won't cause any problem as it issues a implicit commit. sql/transaction.cc: Release metadata locks after the implicitly committed due to a new transaction being started. Also, release metadata locks acquired after a savepoint if the transaction is rolled back to the save point. The problem is that in some cases transaction-long metadata locks could be released before the transaction was committed. This could happen when a active transaction was ended by a "START TRANSACTION" or "BEGIN" statement, in which case the metadata locks would be released before the actual commit of the active transaction. The solution is to defer the release of metadata locks to after the transaction has been implicitly committed. No test case is provided as the effort to provide one is too disproportional to the size of the fix.
		
			
				
	
	
		
			198 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| -- source include/have_partition.inc
 | |
| -- source include/not_embedded.inc
 | |
| -- source include/have_innodb.inc
 | |
| 
 | |
| --disable_warnings
 | |
| drop table if exists t1;
 | |
| --enable_warnings
 | |
| 
 | |
| # basic tests of semi-consistent reads
 | |
| # for verifying Bug#40595: Non-matching rows not released with READ-COMMITTED
 | |
| #               on tables with partitions
 | |
| 
 | |
| connect (a,localhost,root,,);
 | |
| connect (b,localhost,root,,);
 | |
| connection a;
 | |
| set binlog_format=mixed;
 | |
| set session transaction isolation level repeatable read;
 | |
| create table t1(a int not null)
 | |
| engine=innodb
 | |
| DEFAULT CHARSET=latin1
 | |
| PARTITION BY RANGE(a)
 | |
| (PARTITION p0 VALUES LESS THAN (20),
 | |
|  PARTITION p1 VALUES LESS THAN MAXVALUE);
 | |
| insert into t1 values (1),(2),(3),(4),(5),(6),(7);
 | |
| set autocommit=0;
 | |
| # this should lock the entire table
 | |
| select * from t1 where a=3 lock in share mode;
 | |
| connection b;
 | |
| set binlog_format=mixed;
 | |
| set session transaction isolation level repeatable read;
 | |
| set autocommit=0;
 | |
| -- error ER_LOCK_WAIT_TIMEOUT
 | |
| update t1 set a=10 where a=5;
 | |
| connection a;
 | |
| #DELETE FROM t1 WHERE a=5;
 | |
| commit;
 | |
| connection b;
 | |
| # perform a semi-consisent read (and unlock non-matching rows)
 | |
| set session transaction isolation level read committed;
 | |
| update t1 set a=10 where a=5;
 | |
| connection a;
 | |
| -- error ER_LOCK_WAIT_TIMEOUT
 | |
| select * from t1 where a=2 for update;
 | |
| # this should lock the records (1),(2)
 | |
| select * from t1 where a=2 limit 1 for update;
 | |
| connection b;
 | |
| # semi-consistent read will skip non-matching locked rows a=1, a=2
 | |
| update t1 set a=11 where a=6;
 | |
| -- error ER_LOCK_WAIT_TIMEOUT
 | |
| update t1 set a=12 where a=2;
 | |
| -- error ER_LOCK_WAIT_TIMEOUT
 | |
| update t1 set a=13 where a=1;
 | |
| connection a;
 | |
| commit;
 | |
| connection b;
 | |
| update t1 set a=14 where a=1;
 | |
| commit;
 | |
| connection a;
 | |
| --sorted_result
 | |
| select * from t1;
 | |
| drop table t1;
 | |
| 
 | |
| connection default;
 | |
| disconnect a;
 | |
| disconnect b;
 | |
| 
 | |
| #
 | |
| # Bug #31310: Locked rows silently skipped in read-committed isolation level.
 | |
| # (This also tests the '*_semi_consistent*' functions in partitioning)
 | |
| # Copied from include/mix1.inc
 | |
| 
 | |
| connect (con1,localhost,root,,);
 | |
| connect (con2,localhost,root,,);
 | |
| SET SESSION AUTOCOMMIT = 0;
 | |
| SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
 | |
| set binlog_format=mixed;
 | |
| --echo # Switch to connection con1
 | |
| connection con1;
 | |
| 
 | |
| eval
 | |
| CREATE TABLE t1 (a INT PRIMARY KEY, b VARCHAR(256))
 | |
| ENGINE = InnoDB
 | |
| PARTITION BY RANGE (a)
 | |
| (PARTITION p0 VALUES LESS THAN (300),
 | |
|  PARTITION p1 VALUES LESS THAN MAXVALUE);
 | |
| INSERT INTO t1 VALUES (1,2);
 | |
| 
 | |
| --echo # 1. test for locking:
 | |
| 
 | |
| BEGIN;
 | |
| --enable_info
 | |
| UPDATE t1 SET b = 12 WHERE a = 1;
 | |
| --disable_info
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| --echo # Switch to connection con2
 | |
| connection con2;
 | |
| 
 | |
| --enable_info
 | |
| --disable_abort_on_error
 | |
| --error ER_LOCK_WAIT_TIMEOUT
 | |
| UPDATE t1 SET b = 21 WHERE a = 1;
 | |
| --disable_info
 | |
| 
 | |
| --echo # Switch to connection con1
 | |
| connection con1;
 | |
| SELECT * FROM t1;
 | |
| ROLLBACK;
 | |
| 
 | |
| --echo # 2. test for serialized update:
 | |
| 
 | |
| CREATE TABLE t2 (a INT);
 | |
| 
 | |
| TRUNCATE t1;
 | |
| INSERT INTO t1 VALUES (1,'init');
 | |
| 
 | |
| DELIMITER |;
 | |
| CREATE PROCEDURE p1()
 | |
| BEGIN
 | |
|   UPDATE t1 SET b = CONCAT(b, '+con2')  WHERE a = 1;
 | |
|   INSERT INTO t2 VALUES ();
 | |
| END|
 | |
| DELIMITER ;|
 | |
| 
 | |
| BEGIN;
 | |
| --enable_info
 | |
| UPDATE t1 SET b = CONCAT(b, '+con1') WHERE a = 1;
 | |
| --disable_info
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| --echo # Switch to connection con2
 | |
| connection con2;
 | |
| 
 | |
| --send CALL p1;
 | |
| 
 | |
| --echo # Switch to connection con1
 | |
| connection con1;
 | |
| SELECT * FROM t1;
 | |
| COMMIT;
 | |
| 
 | |
| let $bug31310 = 1;
 | |
| while ($bug31310)
 | |
| {
 | |
|   let $bug31310= `SELECT 1 - COUNT(*) FROM t2`;
 | |
| }
 | |
| 
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| --echo # Switch to connection con2
 | |
| connection con2;
 | |
| --reap
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| --echo # Switch to connection con1
 | |
| connection con1;
 | |
| 
 | |
| --echo # 3. test for updated key column:
 | |
| 
 | |
| TRUNCATE t1;
 | |
| DELETE FROM t2;
 | |
| 
 | |
| INSERT INTO t1 VALUES (1,'init');
 | |
| 
 | |
| BEGIN;
 | |
| --enable_info
 | |
| UPDATE t1 SET a = 2, b = CONCAT(b, '+con1') WHERE a = 1;
 | |
| --disable_info
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| --echo # Switch to connection con2
 | |
| connection con2;
 | |
| 
 | |
| --send CALL p1;
 | |
| 
 | |
| --echo # Switch to connection con1
 | |
| connection con1;
 | |
| SELECT * FROM t1;
 | |
| COMMIT;
 | |
| 
 | |
| let $bug31310 = 1;
 | |
| while ($bug31310)
 | |
| {
 | |
|   let $bug31310= `SELECT 1 - COUNT(*) FROM t2`;
 | |
| }
 | |
| 
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| --echo # Switch to connection con2
 | |
| connection con2;
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| connection default;
 | |
| disconnect con1;
 | |
| disconnect con2;
 | |
| DROP PROCEDURE p1;
 | |
| DROP TABLE t1, t2;
 | |
| 
 |