mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-24 07:13:33 +03:00 
			
		
		
		
	The test case problem stemmed from the fact that a debug sync signal is a global variable that persists until overwritten by a new signal. This means that if two different signals are raised in sequence, a thread waiting for the first signal might miss it if the second signal sets the global variable before the thread wakes up. The solution is to deliver a subsequent signal only after the waiting thread has received it. mysql-test/t/query_cache_debug.test: Wait for signal to be delivered.
		
			
				
	
	
		
			331 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			331 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| --source include/not_embedded.inc
 | |
| --source include/have_query_cache.inc
 | |
| --source include/have_debug_sync.inc
 | |
| 
 | |
| #
 | |
| # Bug #30887 Server crashes on SET GLOBAL query_cache_size=0
 | |
| #
 | |
| flush status;
 | |
| set query_cache_type=DEMAND;
 | |
| set global query_cache_size= 1024*768;
 | |
| --disable_warnings
 | |
| drop table if exists t1;
 | |
| --enable_warnings
 | |
| create table t1 (a varchar(100));
 | |
| insert into t1 values ('aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'),('bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb');
 | |
| connect (bug30887con1, localhost, root, ,test);
 | |
| connect (bug30887con2, localhost, root, ,test);
 | |
| 
 | |
| connection bug30887con1;
 | |
| --echo Activate debug hook and attempt to retrieve the statement from the cache.
 | |
| set debug_sync="wait_in_query_cache_insert SIGNAL parked WAIT_FOR go";
 | |
| --send select SQL_CACHE * from t1;
 | |
| 
 | |
| connection default;
 | |
| set debug_sync="now WAIT_FOR parked";
 | |
| 
 | |
| connection bug30887con2;
 | |
| --echo On a second connection; clear the query cache.
 | |
| show status like 'Qcache_queries_in_cache';
 | |
| set global query_cache_size= 0;
 | |
| 
 | |
| connection default;
 | |
| --echo Signal the debug hook to release the lock.
 | |
| set debug_sync="now SIGNAL go";
 | |
| 
 | |
| --echo Show query cache status.
 | |
| show status like 'Qcache_queries_in_cache';
 | |
| 
 | |
| connection bug30887con1;
 | |
| --reap
 | |
| 
 | |
| disconnect bug30887con1;
 | |
| disconnect bug30887con2;
 | |
| connection default;
 | |
| set debug_sync= 'RESET';
 | |
| set global query_cache_size= 0;
 | |
| use test;
 | |
| drop table t1;
 | |
| 
 | |
| #
 | |
| # Bug#41098: Query Cache returns wrong result with concurrent insert
 | |
| #
 | |
| 
 | |
| SET @old_concurrent_insert= @@GLOBAL.concurrent_insert;
 | |
| SET @old_query_cache_size= @@GLOBAL.query_cache_size;
 | |
| 
 | |
| --disable_warnings
 | |
| DROP TABLE IF EXISTS t1, t2;
 | |
| --enable_warnings
 | |
| CREATE TABLE t1 (a INT);
 | |
| CREATE TABLE t2 (a INT);
 | |
| INSERT INTO t1 VALUES (1),(2),(3);
 | |
| 
 | |
| SET GLOBAL concurrent_insert= 1;
 | |
| SET GLOBAL query_cache_size= 1024*512;
 | |
| SET GLOBAL query_cache_type= ON;
 | |
| 
 | |
| connect(con1,localhost,root,,test,,);
 | |
| connect(con2,localhost,root,,test,,);
 | |
| 
 | |
| connection con1;
 | |
| --echo # Switch to connection con1
 | |
| SET DEBUG_SYNC = "wait_after_query_cache_invalidate SIGNAL parked WAIT_FOR go";
 | |
| --echo # Send concurrent insert, will wait in the query cache table invalidate
 | |
| --send INSERT INTO t1 VALUES (4)
 | |
| 
 | |
| connection default;
 | |
| --echo # Switch to connection default
 | |
| --echo # Wait for concurrent insert to reach the debug point
 | |
| SET DEBUG_SYNC = "now WAIT_FOR parked";
 | |
| 
 | |
| connection con2;
 | |
| --echo # Switch to connection con2
 | |
| --echo # Send SELECT that shouldn't be cached
 | |
| SELECT * FROM t1;
 | |
| 
 | |
| connection default;
 | |
| --echo # Switch to connection default
 | |
| --echo # Notify the concurrent insert to proceed
 | |
| SET DEBUG_SYNC = "now SIGNAL go";
 | |
| 
 | |
| connection con1;
 | |
| --echo # Switch to connection con1
 | |
| --echo # Gather insert result
 | |
| --reap
 | |
| SHOW STATUS LIKE "Qcache_queries_in_cache";
 | |
| --echo # Test that it's cacheable
 | |
| SELECT * FROM t1;
 | |
| SHOW STATUS LIKE "Qcache_queries_in_cache";
 | |
| 
 | |
| --echo # Disconnect
 | |
| disconnect con1;
 | |
| disconnect con2;
 | |
| 
 | |
| connection default;
 | |
| --echo # Restore defaults
 | |
| SET DEBUG_SYNC= 'RESET';
 | |
| RESET QUERY CACHE;
 | |
| DROP TABLE t1,t2;
 | |
| SET GLOBAL concurrent_insert= DEFAULT;
 | |
| SET GLOBAL query_cache_size= DEFAULT;
 | |
| SET GLOBAL query_cache_type= DEFAULT;
 | |
| 
 | |
| 
 | |
| --echo #
 | |
| --echo # Bug43758 Query cache can lock up threads in 'freeing items' state
 | |
| --echo #
 | |
| FLUSH STATUS;
 | |
| SET GLOBAL query_cache_type=DEMAND;
 | |
| SET GLOBAL query_cache_size= 1024*768;
 | |
| --disable_warnings
 | |
| DROP TABLE IF EXISTS t1,t2,t3,t4,t5;
 | |
| --enable_warnings
 | |
| CREATE TABLE t1 (a VARCHAR(100));
 | |
| CREATE TABLE t2 (a VARCHAR(100));
 | |
| CREATE TABLE t3 (a VARCHAR(100));
 | |
| CREATE TABLE t4 (a VARCHAR(100));
 | |
| CREATE TABLE t5 (a VARCHAR(100));
 | |
| 
 | |
| INSERT INTO t1 VALUES ('aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'),('bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb');
 | |
| INSERT INTO t2 VALUES ('aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'),('bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb');
 | |
| INSERT INTO t3 VALUES ('aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'),('bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb');
 | |
| INSERT INTO t4 VALUES ('aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'),('bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb');
 | |
| INSERT INTO t5 VALUES ('aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa'),('bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb');
 | |
| 
 | |
| connect (thd2, localhost, root, ,test);
 | |
| connect (thd3, localhost, root, ,test);
 | |
| connect (thd1, localhost, root, ,test);
 | |
| 
 | |
| connection thd1;
 | |
| --echo =================================== Connection thd1
 | |
| --echo **
 | |
| --echo ** Load Query Cache with a result set and one table.
 | |
| --echo **
 | |
| SELECT SQL_CACHE * FROM t1;
 | |
| --echo *************************************************************************
 | |
| --echo ** We want to accomplish the following state:
 | |
| --echo **  - Query cache status: TABLE_FLUSH_IN_PROGRESS
 | |
| --echo **  - THD1: invalidate_table_internal (iterating query blocks)
 | |
| --echo **  - THD2: query_cache_insert (cond_wait)
 | |
| --echo **  - THD3: query_cache_insert (cond_wait)
 | |
| --echo **  - No thread should be holding the structure_guard_mutex.
 | |
| --echo **
 | |
| --echo ** First step is to place a DELETE-statement on the debug hook just
 | |
| --echo ** before the mutex lock in invalidate_table_internal.
 | |
| --echo ** This will allow new result sets to be written into the QC.
 | |
| --echo ** 
 | |
| SET DEBUG_SYNC="wait_in_query_cache_invalidate1 SIGNAL parked1_1 WAIT_FOR go1_1";
 | |
| SET DEBUG_SYNC="wait_in_query_cache_invalidate2 SIGNAL parked1_2 WAIT_FOR go1_2";
 | |
| --send DELETE FROM t1 WHERE a like '%a%';
 | |
| 
 | |
| connection default;
 | |
| --echo =================================== Connection default
 | |
| --echo ** Assert that the expect process status is obtained.
 | |
| SET DEBUG_SYNC="now WAIT_FOR parked1_1";
 | |
| -- echo **
 | |
| 
 | |
| connection thd2;
 | |
| --echo =================================== Connection thd2
 | |
| --echo ** On THD2: Insert a result into the cache. This attempt will be blocked
 | |
| --echo ** because of a debug hook placed just before the mutex lock after which
 | |
| --echo ** the first part of the result set is written.
 | |
| SET DEBUG_SYNC="wait_in_query_cache_insert SIGNAL parked2 WAIT_FOR go2 EXECUTE 1";
 | |
| --send SELECT SQL_CACHE * FROM t2 UNION SELECT * FROM t3
 | |
| 
 | |
| connection default;
 | |
| --echo =================================== Connection default
 | |
| --echo ** Assert that the SELECT-stmt thread reaches the sync point.
 | |
| SET DEBUG_SYNC="now WAIT_FOR parked2";
 | |
| --echo **
 | |
| --echo **
 | |
| 
 | |
| connection thd3;
 | |
| --echo =================================== Connection thd3
 | |
| --echo ** On THD3: Insert another result into the cache and block on the same
 | |
| --echo ** debug hook.
 | |
| SET DEBUG_SYNC="wait_in_query_cache_insert SIGNAL parked3 WAIT_FOR go3 EXECUTE 1";
 | |
| --send SELECT SQL_CACHE * FROM t4 UNION SELECT * FROM t5
 | |
| 
 | |
| connection default;
 | |
| --echo =================================== Connection default
 | |
| --echo ** Assert that the SELECT-stmt thread reaches the sync point.
 | |
| SET DEBUG_SYNC="now WAIT_FOR parked3";
 | |
| --echo **
 | |
| --echo **
 | |
| 
 | |
| --echo ** Signal the DELETE thread, THD1, to continue. It will enter the mutex
 | |
| --echo ** lock and set query cache status to TABLE_FLUSH_IN_PROGRESS and then
 | |
| --echo ** unlock the mutex before stopping on the next debug hook.
 | |
| SET DEBUG_SYNC="now SIGNAL go1_1";
 | |
| --echo ** Assert that we reach the next debug hook.
 | |
| SET DEBUG_SYNC="now WAIT_FOR parked1_2";
 | |
| 
 | |
| --echo **
 | |
| --echo ** Signal the remaining debug hooks blocking THD2 and THD3.
 | |
| --echo ** The threads will grab the guard mutex enter the wait condition and
 | |
| --echo ** and finally release the mutex. The threads will continue to wait
 | |
| --echo ** until a broadcast signal reaches them causing both threads to 
 | |
| --echo ** come alive and check the condition.
 | |
| SET DEBUG_SYNC="now SIGNAL go2";
 | |
| --echo ** Wait for thd2 to receive the signal
 | |
| let $wait_condition=
 | |
|   SELECT COUNT(*) = 1 FROM information_schema.processlist
 | |
|   WHERE state = "Waiting for query cache lock";
 | |
| --source include/wait_condition.inc
 | |
| SET DEBUG_SYNC="now SIGNAL go3";
 | |
| --echo ** Wait for thd3 to receive the signal
 | |
| let $wait_condition=
 | |
|   SELECT COUNT(*) = 2 FROM information_schema.processlist
 | |
|   WHERE state = "Waiting for query cache lock";
 | |
| --source include/wait_condition.inc
 | |
| 
 | |
| --echo **
 | |
| --echo ** Finally signal the DELETE statement on THD1 one last time.
 | |
| --echo ** The stmt will complete the query cache invalidation and return 
 | |
| --echo ** cache status to NO_FLUSH_IN_PROGRESS. On the status change
 | |
| --echo ** One signal will be sent to the thread group waiting for executing
 | |
| --echo ** invalidations and a broadcast signal will be sent to the thread 
 | |
| --echo ** group holding result set writers.
 | |
| SET DEBUG_SYNC="now SIGNAL go1_2";
 | |
| 
 | |
| --echo **
 | |
| --echo *************************************************************************
 | |
| --echo ** No tables should be locked
 | |
| connection thd2;
 | |
| --echo =================================== Connection thd2
 | |
| reap;
 | |
| DELETE FROM t1;
 | |
| DELETE FROM t2;
 | |
| DELETE FROM t3;
 | |
| 
 | |
| connection thd3;
 | |
| --echo =================================== Connection thd3
 | |
| reap;
 | |
| DELETE FROM t4;
 | |
| DELETE FROM t5;
 | |
| 
 | |
| connection thd1;
 | |
| --echo =================================== Connection thd1
 | |
| reap;
 | |
| 
 | |
| --echo ** Done.
 | |
| 
 | |
| connection default;
 | |
| disconnect thd1;
 | |
| disconnect thd2;
 | |
| disconnect thd3;
 | |
| SET DEBUG_SYNC= 'RESET';
 | |
| SET GLOBAL query_cache_size= 0;
 | |
| 
 | |
| connection default;
 | |
| --echo # Restore defaults
 | |
| RESET QUERY CACHE;
 | |
| FLUSH STATUS;
 | |
| DROP TABLE t1,t2,t3,t4,t5;
 | |
| SET GLOBAL query_cache_size= DEFAULT;
 | |
| SET GLOBAL query_cache_type= DEFAULT;
 | |
| 
 | |
| --echo #
 | |
| --echo # Bug#56822: Add a thread state for sessions waiting on the query cache lock
 | |
| --echo #
 | |
| 
 | |
| SET @old_query_cache_size= @@GLOBAL.query_cache_size;
 | |
| 
 | |
| --disable_warnings
 | |
| DROP TABLE IF EXISTS t1;
 | |
| --enable_warnings
 | |
| CREATE TABLE t1 (a INT);
 | |
| INSERT INTO t1 VALUES (1),(2),(3);
 | |
| 
 | |
| SET GLOBAL concurrent_insert= 1;
 | |
| SET GLOBAL query_cache_size= 1024*512;
 | |
| SET GLOBAL query_cache_type= ON;
 | |
| 
 | |
| connect(con1,localhost,root,,test,,);
 | |
| connect(con2,localhost,root,,test,,);
 | |
| 
 | |
| connection con1;
 | |
| --echo # Switch to connection con1
 | |
| SET DEBUG_SYNC = "wait_in_query_cache_invalidate2 SIGNAL parked WAIT_FOR go";
 | |
| --echo # Send INSERT, will wait in the query cache table invalidation
 | |
| --send INSERT INTO t1 VALUES (4);
 | |
| 
 | |
| connection default;
 | |
| --echo # Switch to connection default
 | |
| --echo # Wait for insert to reach the debug point
 | |
| SET DEBUG_SYNC = "now WAIT_FOR parked";
 | |
| 
 | |
| connection con2;
 | |
| --echo # Switch to connection con2
 | |
| --echo # Send a query that should wait on the query cache lock
 | |
| --send RESET QUERY CACHE
 | |
| 
 | |
| connection default;
 | |
| --echo # Switch to connection default
 | |
| --echo # Wait for the state to be reflected in the processlist
 | |
| let $wait_condition=
 | |
|   SELECT COUNT(*) = 1 FROM information_schema.processlist
 | |
|   WHERE state = "Waiting for query cache lock" AND info = "RESET QUERY CACHE";
 | |
| --source include/wait_condition.inc
 | |
| --echo # Signal that the query cache can be unlocked
 | |
| SET DEBUG_SYNC="now SIGNAL go";
 | |
| 
 | |
| connection con1;
 | |
| --echo # Reap con1 and disconnect
 | |
| --reap
 | |
| disconnect con1;
 | |
| 
 | |
| connection con2;
 | |
| --echo # Reap con2 and disconnect
 | |
| --reap
 | |
| disconnect con2;
 | |
| 
 | |
| connection default;
 | |
| --echo # Restore defaults
 | |
| SET DEBUG_SYNC= 'RESET';
 | |
| RESET QUERY CACHE;
 | |
| DROP TABLE t1;
 | |
| SET GLOBAL query_cache_size= DEFAULT;
 | |
| SET GLOBAL query_cache_type= DEFAULT;
 |