Commit ecda78f0 authored by marko's avatar marko

branches/zip: get_share(), free_share(): Make table locking case sensitive.

If lower_case_table_names=1, MySQL will pass the table names in lower case.
Thus, we can use a binary comparison (strcmp) in the hash table.

rb://87 approved by Heikki Tuuri, to address Bug #41676 and Issue #167.
parent da9083a0
2009-02-10 The InnoDB Team
* handler/ha_innodb.h, handler/ha_innodb.cc:
Fix Bug#41676 Table names are case insensitive in locking
2009-02-10 The InnoDB Team 2009-02-10 The InnoDB Team
* ut/ut0mem.c: * ut/ut0mem.c:
......
...@@ -8203,50 +8203,30 @@ bool innobase_show_status(handlerton *hton, THD* thd, ...@@ -8203,50 +8203,30 @@ bool innobase_show_status(handlerton *hton, THD* thd,
locking. locking.
****************************************************************************/ ****************************************************************************/
/****************************************************************************
Folds a string in system_charset_info. */
static
ulint
innobase_fold_name(
/*===============*/
/* out: fold value of the name */
const uchar* name, /* in: string to be folded */
size_t length) /* in: length of the name in bytes */
{
ulong n1 = 1, n2 = 4;
system_charset_info->coll->hash_sort(system_charset_info,
name, length, &n1, &n2);
return((ulint) n1);
}
static INNOBASE_SHARE* get_share(const char* table_name) static INNOBASE_SHARE* get_share(const char* table_name)
{ {
INNOBASE_SHARE *share; INNOBASE_SHARE *share;
pthread_mutex_lock(&innobase_share_mutex); pthread_mutex_lock(&innobase_share_mutex);
uint length=(uint) strlen(table_name);
ulint fold = innobase_fold_name((const uchar*) table_name, length); ulint fold = ut_fold_string(table_name);
HASH_SEARCH(table_name_hash, innobase_open_tables, fold, HASH_SEARCH(table_name_hash, innobase_open_tables, fold,
INNOBASE_SHARE*, share, INNOBASE_SHARE*, share,
ut_ad(share->use_count > 0), ut_ad(share->use_count > 0),
!my_strnncoll(system_charset_info, !strcmp(share->table_name, table_name));
share->table_name,
share->table_name_length,
(const uchar*) table_name, length));
if (!share) { if (!share) {
uint length = (uint) strlen(table_name);
/* TODO: invoke HASH_MIGRATE if innobase_open_tables /* TODO: invoke HASH_MIGRATE if innobase_open_tables
grows too big */ grows too big */
share = (INNOBASE_SHARE *) my_malloc(sizeof(*share)+length+1, share = (INNOBASE_SHARE *) my_malloc(sizeof(*share)+length+1,
MYF(MY_FAE | MY_ZEROFILL)); MYF(MY_FAE | MY_ZEROFILL));
share->table_name_length = length; share->table_name = (char*) memcpy(share + 1,
share->table_name = (uchar*) memcpy(share + 1, table_name, length + 1);
table_name, length + 1);
HASH_INSERT(INNOBASE_SHARE, table_name_hash, HASH_INSERT(INNOBASE_SHARE, table_name_hash,
innobase_open_tables, fold, share); innobase_open_tables, fold, share);
...@@ -8267,24 +8247,18 @@ static void free_share(INNOBASE_SHARE* share) ...@@ -8267,24 +8247,18 @@ static void free_share(INNOBASE_SHARE* share)
#ifdef UNIV_DEBUG #ifdef UNIV_DEBUG
INNOBASE_SHARE* share2; INNOBASE_SHARE* share2;
ulint fold = innobase_fold_name(share->table_name, ulint fold = ut_fold_string(share->table_name);
share->table_name_length);
HASH_SEARCH(table_name_hash, innobase_open_tables, fold, HASH_SEARCH(table_name_hash, innobase_open_tables, fold,
INNOBASE_SHARE*, share2, INNOBASE_SHARE*, share2,
ut_ad(share->use_count > 0), ut_ad(share->use_count > 0),
!my_strnncoll(system_charset_info, !strcmp(share->table_name, share2->table_name));
share->table_name,
share->table_name_length,
share2->table_name,
share2->table_name_length));
ut_a(share2 == share); ut_a(share2 == share);
#endif /* UNIV_DEBUG */ #endif /* UNIV_DEBUG */
if (!--share->use_count) { if (!--share->use_count) {
ulint fold = innobase_fold_name(share->table_name, ulint fold = ut_fold_string(share->table_name);
share->table_name_length);
HASH_DELETE(INNOBASE_SHARE, table_name_hash, HASH_DELETE(INNOBASE_SHARE, table_name_hash,
innobase_open_tables, fold, share); innobase_open_tables, fold, share);
......
...@@ -27,8 +27,8 @@ ...@@ -27,8 +27,8 @@
typedef struct st_innobase_share { typedef struct st_innobase_share {
THR_LOCK lock; THR_LOCK lock;
pthread_mutex_t mutex; pthread_mutex_t mutex;
const uchar *table_name; const char* table_name;
uint table_name_length,use_count; uint use_count;
void* table_name_hash; void* table_name_hash;
} INNOBASE_SHARE; } INNOBASE_SHARE;
......
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