diff options
author | Sergei Golubchik <sergii@pisem.net> | 2011-10-19 21:45:18 +0200 |
---|---|---|
committer | Sergei Golubchik <sergii@pisem.net> | 2011-10-19 21:45:18 +0200 |
commit | 76f0b94bb0b2994d639353530c5b251d0f1a204b (patch) | |
tree | 9ed50628aac34f89a37637bab2fc4915b86b5eb4 /sql/sql_cache.cc | |
parent | 4e46d8e5bff140f2549841167dc4b65a3c0a645d (diff) | |
parent | 5dc1a2231f55bacc9aaf0e24816f3d9c2ee1f21d (diff) | |
download | mariadb-git-76f0b94bb0b2994d639353530c5b251d0f1a204b.tar.gz |
merge with 5.3
sql/sql_insert.cc:
CREATE ... IF NOT EXISTS may do nothing, but
it is still not a failure. don't forget to my_ok it.
******
CREATE ... IF NOT EXISTS may do nothing, but
it is still not a failure. don't forget to my_ok it.
sql/sql_table.cc:
small cleanup
******
small cleanup
Diffstat (limited to 'sql/sql_cache.cc')
-rw-r--r-- | sql/sql_cache.cc | 534 |
1 files changed, 411 insertions, 123 deletions
diff --git a/sql/sql_cache.cc b/sql/sql_cache.cc index feb9614efe7..256b2a874cb 100644 --- a/sql/sql_cache.cc +++ b/sql/sql_cache.cc @@ -345,6 +345,8 @@ TODO list: #include "log_slow.h" #include "transaction.h" +const uchar *query_state_map; + #ifdef EMBEDDED_LIBRARY #include "emb_qcache.h" #endif @@ -430,6 +432,136 @@ struct Query_cache_wait_state }; +/* + Check if character is a white space. +*/ + +inline bool is_white_space(char c) +{ + return (query_state_map[(uint) ((uchar) c)] == MY_LEX_SKIP); +} + + +/** + Generate a query_string without query comments or duplicated space + + @param new_query New query without 'fluff' is stored here + @param query Original query + @param query_length Length of original query + @param additional_length Extra space for query cache we need to allocate + in new_query buffer. + + Note: + If there is no space to allocate new_query, we will put original query + into new_query. +*/ + +static void make_base_query(String *new_query, + const char *query, size_t query_length, + size_t additional_length) +{ + char *buffer; + const char *query_end, *last_space; + + /* The following is guaranteed by the query_cache interface */ + DBUG_ASSERT(query[query_length] == 0); + DBUG_ASSERT(!is_white_space(query[0])); + + if (new_query->realloc(query_length + additional_length)) + { + /* + We could not allocate the query. Use original query for + the query cache; Better than nothing.... + */ + new_query->set(query, query_length, system_charset_info); + return; + } + + buffer= (char*) new_query->ptr(); // Store base query here + query_end= query + query_length; + last_space= 0; // No space found yet + + while (query < query_end) + { + char current = *(query++); + switch (current) { + case '\'': + case '`': + case '"': + *(buffer++)= current; // copy first quote + while (query < query_end) + { + *(buffer++)= *query; + if (*(query++) == current) // found pair quote + break; + } + continue; // Continue with next symbol + case '/': // Start of comment ? + /* + Comment of format /#!number #/, must be skipped. + These may include '"' and other comments, but it should + be safe to parse the content as a normal string. + */ + if (query[0] != '*' || query[1] == '!') + break; + + query++; // skip "/" + while (++query < query_end) + { + if (query[0] == '*' && query[1] == '/') + { + query+= 2; + goto insert_space; + } + } + continue; // Will end outer loop + case '-': + if (*query != '-' || !is_white_space(query[1])) // Not a comment + break; + query++; // skip second "-", and go to search of "\n" + /* fall through */ + case '#': + while (query < query_end) + { + if (*(query++) == '\n') + goto insert_space; + } + continue; // Will end outer loop + default: + if (is_white_space(current)) + goto insert_space; + break; + } + *(buffer++)= current; + continue; + +insert_space: + if (buffer != last_space) + { + *(buffer++)= ' '; + last_space= buffer; + } + } + if (buffer == last_space) + buffer--; // Remove the last space + *buffer= 0; + new_query->length((size_t) (buffer - new_query->ptr())); +} + + +/** + Check and change local variable if global one is switched + + @param thd thread handle +*/ + +void inline fix_local_query_cache_mode(THD *thd) +{ + if (global_system_variables.query_cache_type == 0) + thd->variables.query_cache_type= 0; +} + + /** Serialize access to the query cache. If the lock cannot be granted the thread hangs in a conditional wait which @@ -439,32 +571,42 @@ struct Query_cache_wait_state effect by another thread. This enables a quick path in execution to skip waits when the outcome is known. - @param use_timeout TRUE if the lock can abort because of a timeout. + @param mode TIMEOUT the lock can abort because of a timeout + TRY the lock can abort because it is locked now + WAIT wait for lock (default) - @note use_timeout is optional and default value is FALSE. + @note mode is optional and default value is WAIT. @return @retval FALSE An exclusive lock was taken @retval TRUE The locking attempt failed */ -bool Query_cache::try_lock(bool use_timeout) +bool Query_cache::try_lock(THD *thd, Cache_try_lock_mode mode) { - bool interrupt= FALSE; - THD *thd= current_thd; + bool interrupt= TRUE; Query_cache_wait_state wait_state(thd, __func__, __FILE__, __LINE__); DBUG_ENTER("Query_cache::try_lock"); mysql_mutex_lock(&structure_guard_mutex); + DBUG_EXECUTE_IF("status_wait_query_cache_mutex_sleep", { sleep(5); }); + if (m_cache_status == DISABLED) + { + mysql_mutex_unlock(&structure_guard_mutex); + DBUG_RETURN(TRUE); + } + m_requests_in_progress++; + fix_local_query_cache_mode(thd); + while (1) { if (m_cache_lock_status == Query_cache::UNLOCKED) { m_cache_lock_status= Query_cache::LOCKED; #ifndef DBUG_OFF - if (thd) - m_cache_lock_thread_id= thd->thread_id; + m_cache_lock_thread_id= thd->thread_id; #endif + interrupt= FALSE; break; } else if (m_cache_lock_status == Query_cache::LOCKED_NO_WAIT) @@ -473,7 +615,6 @@ bool Query_cache::try_lock(bool use_timeout) If query cache is protected by a LOCKED_NO_WAIT lock this thread should avoid using the query cache as it is being evicted. */ - interrupt= TRUE; break; } else @@ -483,24 +624,34 @@ bool Query_cache::try_lock(bool use_timeout) To prevent send_result_to_client() and query_cache_insert() from blocking execution for too long a timeout is put on the lock. */ - if (use_timeout) + if (mode == WAIT) + { + mysql_cond_wait(&COND_cache_status_changed, &structure_guard_mutex); + } + else if (mode == TIMEOUT) { struct timespec waittime; set_timespec_nsec(waittime,(ulong)(50000000L)); /* Wait for 50 msec */ int res= mysql_cond_timedwait(&COND_cache_status_changed, &structure_guard_mutex, &waittime); if (res == ETIMEDOUT) - { - interrupt= TRUE; break; - } } else { - mysql_cond_wait(&COND_cache_status_changed, &structure_guard_mutex); + /** + If we are here, then mode is == TRY and there was someone else using + the query cache. (m_cache_lock_status != Query_cache::UNLOCKED). + Signal that we didn't get a lock. + */ + DBUG_ASSERT(m_requests_in_progress > 1); + DBUG_ASSERT(mode == TRY); + break; } } } + if (interrupt) + m_requests_in_progress--; mysql_mutex_unlock(&structure_guard_mutex); DBUG_RETURN(interrupt); @@ -525,10 +676,12 @@ void Query_cache::lock_and_suspend(void) DBUG_ENTER("Query_cache::lock_and_suspend"); mysql_mutex_lock(&structure_guard_mutex); + m_requests_in_progress++; while (m_cache_lock_status != Query_cache::UNLOCKED) mysql_cond_wait(&COND_cache_status_changed, &structure_guard_mutex); m_cache_lock_status= Query_cache::LOCKED_NO_WAIT; #ifndef DBUG_OFF + /* Here thd may not be set during shutdown */ if (thd) m_cache_lock_thread_id= thd->thread_id; #endif @@ -547,19 +700,19 @@ void Query_cache::lock_and_suspend(void) It is used by all methods which invalidates one or more tables. */ -void Query_cache::lock(void) +void Query_cache::lock(THD *thd) { - THD *thd= current_thd; Query_cache_wait_state wait_state(thd, __func__, __FILE__, __LINE__); DBUG_ENTER("Query_cache::lock"); mysql_mutex_lock(&structure_guard_mutex); + m_requests_in_progress++; + fix_local_query_cache_mode(thd); while (m_cache_lock_status != Query_cache::UNLOCKED) mysql_cond_wait(&COND_cache_status_changed, &structure_guard_mutex); m_cache_lock_status= Query_cache::LOCKED; #ifndef DBUG_OFF - if (thd) - m_cache_lock_thread_id= thd->thread_id; + m_cache_lock_thread_id= thd->thread_id; #endif mysql_mutex_unlock(&structure_guard_mutex); @@ -576,6 +729,7 @@ void Query_cache::unlock(void) DBUG_ENTER("Query_cache::unlock"); mysql_mutex_lock(&structure_guard_mutex); #ifndef DBUG_OFF + /* Thd may not be set in resize() at mysqld start */ THD *thd= current_thd; if (thd) DBUG_ASSERT(m_cache_lock_thread_id == thd->thread_id); @@ -585,6 +739,14 @@ void Query_cache::unlock(void) m_cache_lock_status= Query_cache::UNLOCKED; DBUG_PRINT("Query_cache",("Sending signal")); mysql_cond_signal(&COND_cache_status_changed); + DBUG_ASSERT(m_requests_in_progress > 0); + m_requests_in_progress--; + if (m_requests_in_progress == 0 && m_cache_status == DISABLE_REQUEST) + { + /* No clients => just free query cache */ + free_cache(); + m_cache_status= DISABLED; + } mysql_mutex_unlock(&structure_guard_mutex); DBUG_VOID_RETURN; } @@ -601,25 +763,24 @@ void Query_cache::unlock(void) @retval FALSE No directive found. */ -static bool has_no_cache_directive(char *sql) +static bool has_no_cache_directive(const char *sql) { - int i=0; - while (sql[i] == ' ') - ++i; + while (is_white_space(*sql)) + sql++; - if (my_toupper(system_charset_info, sql[i]) == 'S' && - my_toupper(system_charset_info, sql[i+1]) == 'Q' && - my_toupper(system_charset_info, sql[i+2]) == 'L' && - my_toupper(system_charset_info, sql[i+3]) == '_' && - my_toupper(system_charset_info, sql[i+4]) == 'N' && - my_toupper(system_charset_info, sql[i+5]) == 'O' && - my_toupper(system_charset_info, sql[i+6]) == '_' && - my_toupper(system_charset_info, sql[i+7]) == 'C' && - my_toupper(system_charset_info, sql[i+8]) == 'A' && - my_toupper(system_charset_info, sql[i+9]) == 'C' && - my_toupper(system_charset_info, sql[i+10]) == 'H' && - my_toupper(system_charset_info, sql[i+11]) == 'E' && - my_toupper(system_charset_info, sql[i+12]) == ' ') + if (my_toupper(system_charset_info, sql[0]) == 'S' && + my_toupper(system_charset_info, sql[1]) == 'Q' && + my_toupper(system_charset_info, sql[2]) == 'L' && + my_toupper(system_charset_info, sql[3]) == '_' && + my_toupper(system_charset_info, sql[4]) == 'N' && + my_toupper(system_charset_info, sql[5]) == 'O' && + my_toupper(system_charset_info, sql[6]) == '_' && + my_toupper(system_charset_info, sql[7]) == 'C' && + my_toupper(system_charset_info, sql[8]) == 'A' && + my_toupper(system_charset_info, sql[9]) == 'C' && + my_toupper(system_charset_info, sql[10]) == 'H' && + my_toupper(system_charset_info, sql[11]) == 'E' && + my_isspace(system_charset_info, sql[12])) return TRUE; return FALSE; @@ -893,13 +1054,19 @@ Query_cache::insert(Query_cache_tls *query_cache_tls, { DBUG_ENTER("Query_cache::insert"); - /* See the comment on double-check locking usage above. */ + /* First we check if query cache is disable without doing a mutex lock */ if (is_disabled() || query_cache_tls->first_query_block == NULL) DBUG_VOID_RETURN; + DBUG_ASSERT(current_thd); + QC_DEBUG_SYNC("wait_in_query_cache_insert"); - if (try_lock()) + /* + Lock the cache with try_lock(). try_lock() will fail if + cache was disabled between the above test and lock. + */ + if (try_lock(current_thd, Query_cache::WAIT)) DBUG_VOID_RETURN; Query_cache_block *query_block = query_cache_tls->first_query_block; @@ -957,7 +1124,7 @@ Query_cache::abort(Query_cache_tls *query_cache_tls) if (is_disabled() || query_cache_tls->first_query_block == NULL) DBUG_VOID_RETURN; - if (try_lock()) + if (try_lock(current_thd, Query_cache::WAIT)) DBUG_VOID_RETURN; /* @@ -1008,7 +1175,7 @@ void Query_cache::end_of_result(THD *thd) emb_count_querycache_size(thd), 0); #endif - if (try_lock()) + if (try_lock(thd, Query_cache::WAIT)) DBUG_VOID_RETURN; query_block= query_cache_tls->first_query_block; @@ -1092,7 +1259,8 @@ Query_cache::Query_cache(ulong query_cache_limit_arg, :query_cache_size(0), query_cache_limit(query_cache_limit_arg), queries_in_cache(0), hits(0), inserts(0), refused(0), - total_blocks(0), lowmem_prunes(0), m_query_cache_is_disabled(FALSE), + total_blocks(0), lowmem_prunes(0), + m_cache_status(OK), min_allocation_unit(ALIGN_SIZE(min_allocation_unit_arg)), min_result_data_size(ALIGN_SIZE(min_result_data_size_arg)), def_query_hash_size(ALIGN_SIZE(def_query_hash_size_arg)), @@ -1116,6 +1284,13 @@ ulong Query_cache::resize(ulong query_cache_size_arg) query_cache_size_arg)); DBUG_ASSERT(initialized); + if (global_system_variables.query_cache_type == 0) + { + if (query_cache_size_arg != 0) + my_error(ER_QUERY_CACHE_IS_DISABLED, MYF(0)); + DBUG_RETURN(0); + } + lock_and_suspend(); /* @@ -1148,8 +1323,17 @@ ulong Query_cache::resize(ulong query_cache_size_arg) query_cache_size= query_cache_size_arg; new_query_cache_size= init_cache(); + /* + m_cache_status is internal query cache switch so switching it on/off + will not be reflected on global_system_variables.query_cache_type + */ if (new_query_cache_size) + { DBUG_EXECUTE("check_querycache",check_integrity(1);); + m_cache_status= OK; // size > 0 => enable cache + } + else + m_cache_status= DISABLED; // size 0 means the cache disabled unlock(); DBUG_RETURN(new_query_cache_size); @@ -1168,6 +1352,9 @@ void Query_cache::store_query(THD *thd, TABLE_LIST *tables_used) { TABLE_COUNTER_TYPE local_tables; ulong tot_length; + const char *query; + size_t query_length; + uint8 tables_type; DBUG_ENTER("Query_cache::store_query"); /* Testing 'query_cache_size' without a lock here is safe: the thing @@ -1177,12 +1364,23 @@ void Query_cache::store_query(THD *thd, TABLE_LIST *tables_used) See also a note on double-check locking usage above. */ - if (thd->locked_tables_mode || query_cache_size == 0) + if (!thd->query_cache_is_applicable || query_cache_size == 0) + { + DBUG_PRINT("qcache", ("Query cache not ready")); DBUG_VOID_RETURN; - uint8 tables_type= 0; + } + if (thd->lex->sql_command != SQLCOM_SELECT) + { + DBUG_PRINT("qcache", ("Ignoring not SELECT command")); + DBUG_VOID_RETURN; + } - if ((local_tables= is_cacheable(thd, thd->query_length(), - thd->query(), thd->lex, tables_used, + /* The following assert fails if we haven't called send_result_to_client */ + DBUG_ASSERT(thd->base_query.is_alloced() || + thd->base_query.ptr() == thd->query()); + + tables_type= 0; + if ((local_tables= is_cacheable(thd, thd->lex, tables_used, &tables_type))) { NET *net= &thd->net; @@ -1261,7 +1459,7 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", The 'TRUE' parameter indicate that the lock is allowed to timeout */ - if (try_lock(TRUE)) + if (try_lock(thd, Query_cache::WAIT)) DBUG_VOID_RETURN; if (query_cache_size == 0) { @@ -1277,11 +1475,13 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", DBUG_VOID_RETURN; } + query= thd->base_query.ptr(); + query_length= thd->base_query.length(); + /* Key is query + database + flag */ if (thd->db_length) { - memcpy(thd->query() + thd->query_length() + 1, thd->db, - thd->db_length); + memcpy((char*) (query + query_length + 1), thd->db, thd->db_length); DBUG_PRINT("qcache", ("database: %s length: %u", thd->db, (unsigned) thd->db_length)); } @@ -1289,24 +1489,24 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", { DBUG_PRINT("qcache", ("No active database")); } - tot_length= thd->query_length() + thd->db_length + 1 + + tot_length= query_length + thd->db_length + 1 + QUERY_CACHE_FLAGS_SIZE; /* We should only copy structure (don't use it location directly) because of alignment issue */ - memcpy((void*) (thd->query() + (tot_length - QUERY_CACHE_FLAGS_SIZE)), + memcpy((void*) (query + (tot_length - QUERY_CACHE_FLAGS_SIZE)), &flags, QUERY_CACHE_FLAGS_SIZE); /* Check if another thread is processing the same query? */ Query_cache_block *competitor = (Query_cache_block *) - my_hash_search(&queries, (uchar*) thd->query(), tot_length); + my_hash_search(&queries, (uchar*) query, tot_length); DBUG_PRINT("qcache", ("competitor 0x%lx", (ulong) competitor)); if (competitor == 0) { /* Query is not in cache and no one is working with it; Store it */ Query_cache_block *query_block; - query_block= write_block_data(tot_length, (uchar*) thd->query(), + query_block= write_block_data(tot_length, (uchar*) query, ALIGN_SIZE(sizeof(Query_cache_query)), Query_cache_block::QUERY, local_tables); if (query_block != 0) @@ -1363,7 +1563,7 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", DBUG_PRINT("qcache", ("Another thread process same query")); } } - else if (thd->lex->sql_command == SQLCOM_SELECT) + else statistic_increment(refused, &structure_guard_mutex); end: @@ -1427,7 +1627,7 @@ send_data_in_chunks(NET *net, const uchar *packet, ulong len) to the user. @param thd Pointer to the thread handler - @param sql A pointer to the sql statement * + @param org_sql A pointer to the sql statement * @param query_length Length of the statement in characters @return status code @@ -1442,7 +1642,7 @@ send_data_in_chunks(NET *net, const uchar *packet, ulong len) */ int -Query_cache::send_result_to_client(THD *thd, char *sql, uint query_length) +Query_cache::send_result_to_client(THD *thd, char *org_sql, uint query_length) { ulonglong engine_data; Query_cache_query *query; @@ -1453,6 +1653,7 @@ Query_cache::send_result_to_client(THD *thd, char *sql, uint query_length) Query_cache_block_table *block_table, *block_table_end; ulong tot_length; Query_cache_query_flags flags; + const char *sql, *sql_end; DBUG_ENTER("Query_cache::send_result_to_client"); /* @@ -1463,50 +1664,95 @@ Query_cache::send_result_to_client(THD *thd, char *sql, uint query_length) See also a note on double-check locking usage above. */ if (is_disabled() || thd->locked_tables_mode || - thd->variables.query_cache_type == 0 || query_cache_size == 0) + thd->variables.query_cache_type == 0) goto err; + DBUG_ASSERT(query_cache_size != 0); // otherwise cache would be disabled - if (!thd->lex->safe_to_cache_query) + thd->query_cache_is_applicable= 1; + sql= org_sql; sql_end= sql + query_length; + + /* + Skip all comments at start of query. The following tests is false for + all normal queries. + */ + if (!my_isalpha(system_charset_info, *sql)) + { + while (sql < sql_end) + { + char current= *sql; + switch (current) { + case '/': + if (sql[1] != '*') + break; + sql+= 2; // Skip '/*' + if (*sql == '!') + { + /* + Found / *!number comment; Skip number to see if sql + starts with 'select' + */ + sql++; + while (my_isdigit(system_charset_info, *sql)) + sql++; + } + else + { + while (sql++ < sql_end) + { + if (sql[-1] == '*' && *sql == '/') + { + sql++; + break; + } + } + } + continue; + case '-': + if (sql[1] != '-' || !is_white_space(sql[2])) // Not a comment + break; + sql++; // Skip first '-' + /* Fall through */ + case '#': + while (++sql < sql_end) + { + if (*sql == '\n') + { + sql++; // Skip '\n' + break; + } + } + /* Continue with analyzing current symbol */ + continue; + case '\r': + case '\n': + case '\t': + case ' ': + case '(': // To handle (select a from t1) union (select a from t1); + sql++; + continue; + default: + break; + } + /* We only come here when we found the first word of the sql */ + break; + } + } + if ((my_toupper(system_charset_info, sql[0]) != 'S' || + my_toupper(system_charset_info, sql[1]) != 'E' || + my_toupper(system_charset_info, sql[2]) != 'L')) { - DBUG_PRINT("qcache", ("SELECT is non-cacheable")); + DBUG_PRINT("qcache", ("The statement is not a SELECT; Not cached")); goto err; } + if ((sql_end - sql) > 20 && has_no_cache_directive(sql+6)) { - uint i= 0; - /* - Skip '(' characters in queries like following: - (select a from t1) union (select a from t1); - */ - while (sql[i]=='(') - i++; - /* - Test if the query is a SELECT - (pre-space is removed in dispatch_command). - - First '/' looks like comment before command it is not - frequently appeared in real life, consequently we can - check all such queries, too. + We do not increase 'refused' statistics here since it will be done + later when the query is parsed. */ - if ((my_toupper(system_charset_info, sql[i]) != 'S' || - my_toupper(system_charset_info, sql[i + 1]) != 'E' || - my_toupper(system_charset_info, sql[i + 2]) != 'L') && - sql[i] != '/') - { - DBUG_PRINT("qcache", ("The statement is not a SELECT; Not cached")); - goto err; - } - - if (query_length > 20 && has_no_cache_directive(&sql[i+6])) - { - /* - We do not increase 'refused' statistics here since it will be done - later when the query is parsed. - */ - DBUG_PRINT("qcache", ("The statement has a SQL_NO_CACHE directive")); - goto err; - } + DBUG_PRINT("qcache", ("The statement has a SQL_NO_CACHE directive")); + goto err; } /* @@ -1514,20 +1760,32 @@ Query_cache::send_result_to_client(THD *thd, char *sql, uint query_length) disabled or if a full cache flush is in progress, the attempt to get the lock is aborted. - The 'TRUE' parameter indicate that the lock is allowed to timeout + The WAIT parameter indicate that the lock is allowed to timeout. */ - if (try_lock(TRUE)) + if (try_lock(thd, Query_cache::WAIT)) goto err; if (query_cache_size == 0) goto err_unlock; Query_cache_block *query_block; + if (opt_query_cache_strip_comments) + { + make_base_query(&thd->base_query, sql, (size_t) (sql_end - sql), + thd->db_length + 1 + QUERY_CACHE_FLAGS_SIZE); + sql= thd->base_query.ptr(); + query_length= thd->base_query.length(); + } + else + { + sql= org_sql; + thd->base_query.set(sql, query_length, system_charset_info); + } tot_length= query_length + thd->db_length + 1 + QUERY_CACHE_FLAGS_SIZE; if (thd->db_length) { - memcpy(sql+query_length+1, thd->db, thd->db_length); + memcpy((char*) (sql+query_length+1), thd->db, thd->db_length); DBUG_PRINT("qcache", ("database: '%s' length: %u", thd->db, (unsigned)thd->db_length)); } @@ -1658,7 +1916,7 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", temporary tables => assign following variable to make check faster. */ - thd->lex->safe_to_cache_query=0; + thd->query_cache_is_applicable= 0; // Query can't be cached BLOCK_UNLOCK_RD(query_block); DBUG_RETURN(-1); } @@ -1674,7 +1932,7 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", ("probably no SELECT access to %s.%s => return to normal processing", table_list.db, table_list.alias)); unlock(); - thd->lex->safe_to_cache_query=0; // Don't try to cache this + thd->query_cache_is_applicable= 0; // Query can't be cached BLOCK_UNLOCK_RD(query_block); DBUG_RETURN(-1); // Privilege error } @@ -1683,7 +1941,7 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", DBUG_PRINT("qcache", ("Need to check column privileges for %s.%s", table_list.db, table_list.alias)); BLOCK_UNLOCK_RD(query_block); - thd->lex->safe_to_cache_query= 0; // Don't try to cache this + thd->query_cache_is_applicable= 0; // Query can't be cached goto err_unlock; // Parse query } #endif /*!NO_EMBEDDED_ACCESS_CHECKS*/ @@ -1707,7 +1965,7 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", table->key_length()); } else - thd->lex->safe_to_cache_query= 0; // Don't try to cache this + thd->query_cache_is_applicable= 0; // Query can't be cached /* End the statement transaction potentially started by engine. */ trans_rollback_stmt(thd); goto err_unlock; // Parse query @@ -1769,13 +2027,16 @@ def_week_frmt: %lu, in_trans: %d, autocommit: %d", err_unlock: unlock(); -err: MYSQL_QUERY_CACHE_MISS(thd->query()); /* query_plan_flags doesn't have to be changed here as it contains QPLAN_QC_NO by default */ DBUG_RETURN(0); // Query was not cached + +err: + thd->query_cache_is_applicable= 0; // Query can't be cached + DBUG_RETURN(0); // Query was not cached } @@ -1814,13 +2075,12 @@ void Query_cache::invalidate(THD *thd, TABLE_LIST *tables_used, DBUG_VOID_RETURN; } -void Query_cache::invalidate(CHANGED_TABLE_LIST *tables_used) +void Query_cache::invalidate(THD *thd, CHANGED_TABLE_LIST *tables_used) { DBUG_ENTER("Query_cache::invalidate (changed table list)"); if (is_disabled()) DBUG_VOID_RETURN; - THD *thd= current_thd; for (; tables_used; tables_used= tables_used->next) { thd_proc_info(thd, "invalidating query cache entries (table list)"); @@ -1843,13 +2103,13 @@ void Query_cache::invalidate(CHANGED_TABLE_LIST *tables_used) NOTE can be used only for opened tables */ -void Query_cache::invalidate_locked_for_write(TABLE_LIST *tables_used) +void Query_cache::invalidate_locked_for_write(THD *thd, + TABLE_LIST *tables_used) { DBUG_ENTER("Query_cache::invalidate_locked_for_write"); if (is_disabled()) DBUG_VOID_RETURN; - THD *thd= current_thd; for (; tables_used; tables_used= tables_used->next_local) { thd_proc_info(thd, "invalidating query cache entries (table)"); @@ -1905,9 +2165,8 @@ void Query_cache::invalidate(THD *thd, const char *key, uint32 key_length, Remove all cached queries that uses the given database. */ -void Query_cache::invalidate(char *db) +void Query_cache::invalidate(THD *thd, char *db) { - DBUG_ENTER("Query_cache::invalidate (db)"); if (is_disabled()) DBUG_VOID_RETURN; @@ -1917,9 +2176,7 @@ void Query_cache::invalidate(char *db) Lock the query cache and queue all invalidation attempts to avoid the risk of a race between invalidation, cache inserts and flushes. */ - lock(); - - THD *thd= current_thd; + lock(thd); if (query_cache_size > 0) { @@ -2026,7 +2283,7 @@ void Query_cache::flush() */ -void Query_cache::pack(ulong join_limit, uint iteration_limit) +void Query_cache::pack(THD *thd, ulong join_limit, uint iteration_limit) { DBUG_ENTER("Query_cache::pack"); @@ -2037,7 +2294,7 @@ void Query_cache::pack(ulong join_limit, uint iteration_limit) If the entire qc is being invalidated we can bail out early instead of waiting for the lock. */ - if (try_lock()) + if (try_lock(thd, Query_cache::WAIT)) DBUG_VOID_RETURN; if (query_cache_size == 0) @@ -2074,11 +2331,25 @@ void Query_cache::destroy() mysql_cond_destroy(&COND_cache_status_changed); mysql_mutex_destroy(&structure_guard_mutex); initialized = 0; + DBUG_ASSERT(m_requests_in_progress == 0); } DBUG_VOID_RETURN; } +void Query_cache::disable_query_cache(THD *thd) +{ + m_cache_status= DISABLE_REQUEST; + /* + If there is no requests in progress try to free buffer. + try_lock(TRY) will exit immediately if there is lock. + unlock() should free block. + */ + if (m_requests_in_progress == 0 && !try_lock(thd, TRY)) + unlock(); +} + + /***************************************************************************** init/destroy *****************************************************************************/ @@ -2091,16 +2362,21 @@ void Query_cache::init() mysql_cond_init(key_COND_cache_status_changed, &COND_cache_status_changed, NULL); m_cache_lock_status= Query_cache::UNLOCKED; + m_cache_status= Query_cache::OK; + m_requests_in_progress= 0; initialized = 1; + query_state_map= default_charset_info->state_map; /* - If we explicitly turn off query cache from the command line query cache will - be disabled for the reminder of the server life time. This is because we - want to avoid locking the QC specific mutex if query cache isn't going to - be used. + If we explicitly turn off query cache from the command line query + cache will be disabled for the reminder of the server life + time. This is because we want to avoid locking the QC specific + mutex if query cache isn't going to be used. */ if (global_system_variables.query_cache_type == 0) - query_cache.disable_query_cache(); - + { + free_cache(); + m_cache_status= DISABLED; + } DBUG_VOID_RETURN; } @@ -2313,6 +2589,18 @@ void Query_cache::free_cache() { DBUG_ENTER("Query_cache::free_cache"); + /* Destroy locks */ + Query_cache_block *block= queries_blocks; + if (block) + { + do + { + Query_cache_query *query= block->query(); + mysql_rwlock_destroy(&query->lock); + block= block->next; + } while (block != queries_blocks); + } + my_free(cache); make_disabled(); my_hash_free(&queries); @@ -2792,7 +3080,7 @@ void Query_cache::invalidate_table(THD *thd, uchar * key, uint32 key_length) Lock the query cache and queue all invalidation attempts to avoid the risk of a race between invalidation, cache inserts and flushes. */ - lock(); + lock(thd); DEBUG_SYNC(thd, "wait_in_query_cache_invalidate2"); @@ -3570,7 +3858,7 @@ Query_cache::process_and_count_tables(THD *thd, TABLE_LIST *tables_used, { DBUG_PRINT("qcache", ("Don't cache statement as it refers to " "tables with column privileges.")); - thd->lex->safe_to_cache_query= 0; + thd->query_cache_is_applicable= 0; // Query can't be cached DBUG_RETURN(0); } #endif @@ -3583,16 +3871,17 @@ Query_cache::process_and_count_tables(THD *thd, TABLE_LIST *tables_used, } else { - DBUG_PRINT("qcache", ("table: %s db: %s type: %u", - tables_used->table->s->table_name.str, - tables_used->table->s->db.str, - tables_used->table->s->db_type()->db_type)); if (tables_used->derived) { + DBUG_PRINT("qcache", ("table: %s", tables_used->alias)); table_count--; DBUG_PRINT("qcache", ("derived table skipped")); continue; } + DBUG_PRINT("qcache", ("table: %s db: %s type: %u", + tables_used->table->s->table_name.str, + tables_used->table->s->db.str, + tables_used->table->s->db_type()->db_type)); *tables_type|= tables_used->table->file->table_cache_type(); /* @@ -3635,14 +3924,13 @@ Query_cache::process_and_count_tables(THD *thd, TABLE_LIST *tables_used, */ TABLE_COUNTER_TYPE -Query_cache::is_cacheable(THD *thd, size_t query_len, const char *query, - LEX *lex, +Query_cache::is_cacheable(THD *thd, LEX *lex, TABLE_LIST *tables_used, uint8 *tables_type) { TABLE_COUNTER_TYPE table_count; DBUG_ENTER("Query_cache::is_cacheable"); - if (query_cache_is_cacheable_query(lex) && + if (thd->lex->safe_to_cache_query && (thd->variables.query_cache_type == 1 || (thd->variables.query_cache_type == 2 && (lex->select_lex.options & OPTION_TO_QUERY_CACHE)))) @@ -3707,7 +3995,7 @@ my_bool Query_cache::ask_handler_allowance(THD *thd, { DBUG_PRINT("qcache", ("Handler does not allow caching for %s.%s", tables_used->db, tables_used->alias)); - thd->lex->safe_to_cache_query= 0; // Don't try to cache this + thd->query_cache_is_applicable= 0; // Query can't be cached DBUG_RETURN(1); } } |