mirror of
https://github.com/MariaDB/server.git
synced 2025-12-13 20:03:16 +03:00
an atomic counter" Split the large LOCK_open section in open_table(). Do not call open_table_from_share() under LOCK_open. Remove thd->version. This fixes Bug#50589 "Server hang on a query evaluated using a temporary table" Bug#51557 "LOCK_open and kernel_mutex are not happy together" Bug#49463 "LOCK_table and innodb are not nice when handler instances are created". This patch has effect on storage engines that rely on ha_open() PSEA method being called under LOCK_open. In particular: 1) NDB is broken and left unfixed. NDB relies on LOCK_open being kept as part of ha_open(), since it uses auto-discovery. While previously the NDB open code was race-prone, now it simply fails on asserts. 2) HEAP engine had a race in ha_heap::open() when a share for the same table could be added twice to the list of shares, or a dangling reference to a share stored in HEAP handler. This patch aims to address this problem by 'pinning' the newly created share in the internal HEAP engine share list until at least one handler instance is created using that share.
157 lines
4.0 KiB
C
157 lines
4.0 KiB
C
/* Copyright (C) 2000-2004, 2006 MySQL AB, 2008-2009 Sun Microsystems, Inc
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
/* open a heap-database */
|
|
|
|
#include "heapdef.h"
|
|
#ifdef VMS
|
|
#include "hp_static.c" /* Stupid vms-linker */
|
|
#endif
|
|
|
|
#include "my_sys.h"
|
|
|
|
/*
|
|
Open heap table based on HP_SHARE structure
|
|
|
|
NOTE
|
|
This doesn't register the table in the open table list.
|
|
*/
|
|
|
|
HP_INFO *heap_open_from_share(HP_SHARE *share, int mode)
|
|
{
|
|
HP_INFO *info;
|
|
DBUG_ENTER("heap_open_from_share");
|
|
|
|
if (!(info= (HP_INFO*) my_malloc((uint) sizeof(HP_INFO) +
|
|
2 * share->max_key_length,
|
|
MYF(MY_ZEROFILL))))
|
|
{
|
|
DBUG_RETURN(0);
|
|
}
|
|
share->open_count++;
|
|
#ifdef THREAD
|
|
thr_lock_data_init(&share->lock,&info->lock,NULL);
|
|
#endif
|
|
info->s= share;
|
|
info->lastkey= (uchar*) (info + 1);
|
|
info->recbuf= (uchar*) (info->lastkey + share->max_key_length);
|
|
info->mode= mode;
|
|
info->current_record= (ulong) ~0L; /* No current record */
|
|
info->lastinx= info->errkey= -1;
|
|
#ifndef DBUG_OFF
|
|
info->opt_flag= READ_CHECK_USED; /* Check when changing */
|
|
#endif
|
|
DBUG_PRINT("exit",("heap: 0x%lx reclength: %d records_in_block: %d",
|
|
(long) info, share->reclength,
|
|
share->block.records_in_block));
|
|
DBUG_RETURN(info);
|
|
}
|
|
|
|
|
|
/*
|
|
Open heap table based on HP_SHARE structure and register it
|
|
*/
|
|
|
|
HP_INFO *heap_open_from_share_and_register(HP_SHARE *share, int mode)
|
|
{
|
|
HP_INFO *info;
|
|
DBUG_ENTER("heap_open_from_share_and_register");
|
|
|
|
mysql_mutex_lock(&THR_LOCK_heap);
|
|
if ((info= heap_open_from_share(share, mode)))
|
|
{
|
|
info->open_list.data= (void*) info;
|
|
heap_open_list= list_add(heap_open_list,&info->open_list);
|
|
/* Unpin the share, it is now pinned by the file. */
|
|
share->open_count--;
|
|
}
|
|
mysql_mutex_unlock(&THR_LOCK_heap);
|
|
DBUG_RETURN(info);
|
|
}
|
|
|
|
|
|
/**
|
|
Dereference a HEAP share and free it if it's not referenced.
|
|
We don't check open_count for internal tables since they
|
|
are always thread-local, i.e. referenced by a single thread.
|
|
*/
|
|
void heap_release_share(HP_SHARE *share, my_bool internal_table)
|
|
{
|
|
/* Couldn't open table; Remove the newly created table */
|
|
if (internal_table)
|
|
hp_free(share);
|
|
else
|
|
{
|
|
mysql_mutex_lock(&THR_LOCK_heap);
|
|
if (--share->open_count == 0)
|
|
hp_free(share);
|
|
mysql_mutex_unlock(&THR_LOCK_heap);
|
|
}
|
|
}
|
|
|
|
/*
|
|
Open heap table based on name
|
|
|
|
NOTE
|
|
This register the table in the open table list. so that it can be
|
|
found by future heap_open() calls.
|
|
*/
|
|
|
|
HP_INFO *heap_open(const char *name, int mode)
|
|
{
|
|
HP_INFO *info;
|
|
HP_SHARE *share;
|
|
DBUG_ENTER("heap_open");
|
|
|
|
mysql_mutex_lock(&THR_LOCK_heap);
|
|
if (!(share= hp_find_named_heap(name)))
|
|
{
|
|
my_errno= ENOENT;
|
|
mysql_mutex_unlock(&THR_LOCK_heap);
|
|
DBUG_RETURN(0);
|
|
}
|
|
if ((info= heap_open_from_share(share, mode)))
|
|
{
|
|
info->open_list.data= (void*) info;
|
|
heap_open_list= list_add(heap_open_list,&info->open_list);
|
|
}
|
|
mysql_mutex_unlock(&THR_LOCK_heap);
|
|
DBUG_RETURN(info);
|
|
}
|
|
|
|
|
|
/* map name to a heap-nr. If name isn't found return 0 */
|
|
|
|
HP_SHARE *hp_find_named_heap(const char *name)
|
|
{
|
|
LIST *pos;
|
|
HP_SHARE *info;
|
|
DBUG_ENTER("heap_find");
|
|
DBUG_PRINT("enter",("name: %s",name));
|
|
|
|
for (pos= heap_share_list; pos; pos= pos->next)
|
|
{
|
|
info= (HP_SHARE*) pos->data;
|
|
if (!strcmp(name, info->name))
|
|
{
|
|
DBUG_PRINT("exit", ("Old heap_database: 0x%lx", (long) info));
|
|
DBUG_RETURN(info);
|
|
}
|
|
}
|
|
DBUG_RETURN((HP_SHARE *) 0);
|
|
}
|
|
|
|
|