sql_base.cc 81.5 KB
Newer Older
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
2

bk@work.mysql.com's avatar
bk@work.mysql.com committed
3 4 5 6
   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; either version 2 of the License, or
   (at your option) any later version.
7

bk@work.mysql.com's avatar
bk@work.mysql.com committed
8 9 10 11
   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.
12

bk@work.mysql.com's avatar
bk@work.mysql.com committed
13 14 15 16 17
   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 */


18
/* Basic functions needed by many modules */
bk@work.mysql.com's avatar
bk@work.mysql.com committed
19 20

#include "mysql_priv.h"
21
#include "sql_select.h"
bk@work.mysql.com's avatar
bk@work.mysql.com committed
22 23 24 25 26 27 28 29 30 31
#include <m_ctype.h>
#include <my_dir.h>
#include <hash.h>
#include <nisam.h>
#ifdef	__WIN__
#include <io.h>
#endif

TABLE *unused_tables;				/* Used by mysql_test */
HASH open_cache;				/* Used by mysql_test */
igor@rurik.mysql.com's avatar
igor@rurik.mysql.com committed
32
HASH assign_cache;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
33

34
static int open_unireg_entry(THD *thd,TABLE *entry,const char *db,
35
			     const char *name, const char *alias);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
36 37 38 39
static void free_cache_entry(TABLE *entry);
static void mysql_rm_tmp_tables(void);


40 41
extern "C" byte *table_cache_key(const byte *record,uint *length,
				 my_bool not_used __attribute__((unused)))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
42 43 44 45 46 47
{
  TABLE *entry=(TABLE*) record;
  *length=entry->key_length;
  return (byte*) entry->table_cache_key;
}

48
bool table_cache_init(void)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
49 50
{
  mysql_rm_tmp_tables();
monty@mysql.com's avatar
monty@mysql.com committed
51
  return hash_init(&open_cache, &my_charset_bin, table_cache_size+16,
monty@mysql.com's avatar
monty@mysql.com committed
52
		   0, 0,table_cache_key,
monty@mysql.com's avatar
monty@mysql.com committed
53
		   (hash_free_key) free_cache_entry, 0) != 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
54 55 56 57 58
}

void table_cache_free(void)
{
  DBUG_ENTER("table_cache_free");
59
  close_cached_tables((THD*) 0,0,(TABLE_LIST*) 0);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
  if (!open_cache.records)			// Safety first
    hash_free(&open_cache);
  DBUG_VOID_RETURN;
}

uint cached_tables(void)
{
  return open_cache.records;
}

#ifdef EXTRA_DEBUG
static void check_unused(void)
{
  uint count=0,idx=0;
  TABLE *cur_link,*start_link;

  if ((start_link=cur_link=unused_tables))
  {
    do
    {
      if (cur_link != cur_link->next->prev || cur_link != cur_link->prev->next)
      {
	DBUG_PRINT("error",("Unused_links aren't linked properly")); /* purecov: inspected */
	return; /* purecov: inspected */
      }
    } while (count++ < open_cache.records &&
	     (cur_link=cur_link->next) != start_link);
    if (cur_link != start_link)
    {
      DBUG_PRINT("error",("Unused_links aren't connected")); /* purecov: inspected */
    }
  }
  for (idx=0 ; idx < open_cache.records ; idx++)
  {
    TABLE *entry=(TABLE*) hash_element(&open_cache,idx);
    if (!entry->in_use)
      count--;
  }
  if (count != 0)
  {
100
    DBUG_PRINT("error",("Unused_links doesn't match open_cache: diff: %d", /* purecov: inspected */
bk@work.mysql.com's avatar
bk@work.mysql.com committed
101 102 103 104 105 106 107
			count)); /* purecov: inspected */
  }
}
#else
#define check_unused()
#endif

monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
/*
  Create a list for all open tables matching SQL expression

  SYNOPSIS
    list_open_tables()
    thd			Thread THD
    wild		SQL like expression

  NOTES
    One gets only a list of tables for which one has any kind of privilege.
    db and table names are allocated in result struct, so one doesn't need
    a lock on LOCK_open when traversing the return list.

  RETURN VALUES
    NULL	Error (Probably OOM)
    #		Pointer to list of names of open tables.
*/

126
OPEN_TABLE_LIST *list_open_tables(THD *thd, const char *wild)
127 128
{
  int result = 0;
129
  OPEN_TABLE_LIST **start_list, *open_list;
130
  TABLE_LIST table_list;
131
  char name[NAME_LEN*2];
132
  DBUG_ENTER("list_open_tables");
133

134 135
  VOID(pthread_mutex_lock(&LOCK_open));
  bzero((char*) &table_list,sizeof(table_list));
136 137
  start_list= &open_list;
  open_list=0;
138

139
  for (uint idx=0 ; result == 0 && idx < open_cache.records; idx++)
140
  {
141
    OPEN_TABLE_LIST *table;
142
    TABLE *entry=(TABLE*) hash_element(&open_cache,idx);
143

144 145
    DBUG_ASSERT(entry->real_name);
    if ((!entry->real_name))			// To be removed
146 147
      continue;					// Shouldn't happen
    if (wild)
148
    {
149
      strxmov(name,entry->table_cache_key,".",entry->real_name,NullS);
150
      if (wild_compare(name,wild,0))
151
	continue;
152 153
    }

154 155 156 157
    /* Check if user has SELECT privilege for any column in the table */
    table_list.db= (char*) entry->table_cache_key;
    table_list.real_name= entry->real_name;
    table_list.grant.privilege=0;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
158

159
    if (check_table_access(thd,SELECT_ACL | EXTRA_ACL,&table_list,1))
160 161 162
      continue;
    /* need to check if we haven't already listed it */
    for (table= open_list  ; table ; table=table->next)
163
    {
164 165 166 167 168 169 170 171 172
      if (!strcmp(table->table,entry->real_name) &&
	  !strcmp(table->db,entry->table_cache_key))
      {
	if (entry->in_use)
	  table->in_use++;
	if (entry->locked_by_name)
	  table->locked++;
	break;
      }
173
    }
174
    if (table)
175
      continue;
176
    if (!(*start_list = (OPEN_TABLE_LIST *)
177
	  sql_alloc(sizeof(**start_list)+entry->key_length)))
178
    {
179
      open_list=0;				// Out of memory
180
      break;
181
    }
182 183 184 185
    strmov((*start_list)->table=
	   strmov(((*start_list)->db= (char*) ((*start_list)+1)),
		  entry->table_cache_key)+1,
	   entry->real_name);
186 187 188
    (*start_list)->in_use= entry->in_use ? 1 : 0;
    (*start_list)->locked= entry->locked_by_name ? 1 : 0;
    start_list= &(*start_list)->next;
189
    *start_list=0;
190 191
  }
  VOID(pthread_mutex_unlock(&LOCK_open));
192
  DBUG_RETURN(open_list);
193
}
bk@work.mysql.com's avatar
bk@work.mysql.com committed
194 195 196 197 198 199 200 201 202 203 204 205 206

/*****************************************************************************
 *	 Functions to free open table cache
 ****************************************************************************/


void intern_close_table(TABLE *table)
{						// Free all structures
  free_io_cache(table);
  if (table->file)
    VOID(closefrm(table));			// close file
}

207 208 209 210 211 212 213 214 215 216
/*
  Remove table from the open table cache

  SYNOPSIS
    free_cache_entry()
    table		Table to remove

  NOTE
    We need to have a lock on LOCK_open when calling this
*/
bk@work.mysql.com's avatar
bk@work.mysql.com committed
217 218 219 220

static void free_cache_entry(TABLE *table)
{
  DBUG_ENTER("free_cache_entry");
221
  safe_mutex_assert_owner(&LOCK_open);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239

  intern_close_table(table);
  if (!table->in_use)
  {
    table->next->prev=table->prev;		/* remove from used chain */
    table->prev->next=table->next;
    if (table == unused_tables)
    {
      unused_tables=unused_tables->next;
      if (table == unused_tables)
	unused_tables=0;
    }
    check_unused();				// consisty check
  }
  my_free((gptr) table,MYF(0));
  DBUG_VOID_RETURN;
}

240
/* Free resources allocated by filesort() and read_record() */
bk@work.mysql.com's avatar
bk@work.mysql.com committed
241 242 243

void free_io_cache(TABLE *table)
{
244
  DBUG_ENTER("free_io_cache");
igor@hundin.mysql.fi's avatar
igor@hundin.mysql.fi committed
245
  if (table->sort.io_cache)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
246
  {
igor@hundin.mysql.fi's avatar
igor@hundin.mysql.fi committed
247 248 249
    close_cached_file(table->sort.io_cache);
    my_free((gptr) table->sort.io_cache,MYF(0));
    table->sort.io_cache=0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
250
  }
251
  DBUG_VOID_RETURN;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
252 253
}

254 255 256 257 258 259
/*
  Close all tables which aren't in use by any thread

  THD can be NULL, but then if_wait_for_refresh must be FALSE
  and tables must be NULL.
*/
bk@work.mysql.com's avatar
bk@work.mysql.com committed
260

261 262
bool close_cached_tables(THD *thd, bool if_wait_for_refresh,
			 TABLE_LIST *tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
263 264 265
{
  bool result=0;
  DBUG_ENTER("close_cached_tables");
266
  DBUG_ASSERT(thd || (!if_wait_for_refresh && !tables));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
267 268

  VOID(pthread_mutex_lock(&LOCK_open));
269
  if (!tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
270
  {
271 272
    while (unused_tables)
    {
bk@work.mysql.com's avatar
bk@work.mysql.com committed
273
#ifdef EXTRA_DEBUG
274 275
      if (hash_delete(&open_cache,(byte*) unused_tables))
	printf("Warning: Couldn't delete open table from hash\n");
bk@work.mysql.com's avatar
bk@work.mysql.com committed
276
#else
277
      VOID(hash_delete(&open_cache,(byte*) unused_tables));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
278
#endif
279 280
    }
    refresh_version++;				// Force close of open tables
bk@work.mysql.com's avatar
bk@work.mysql.com committed
281
  }
282
  else
bk@work.mysql.com's avatar
bk@work.mysql.com committed
283
  {
284 285 286
    bool found=0;
    for (TABLE_LIST *table=tables ; table ; table=table->next)
    {
287
      if (remove_table_from_cache(thd, table->db, table->real_name, 1))
288 289 290 291
	found=1;
    }
    if (!found)
      if_wait_for_refresh=0;			// Nothing to wait for
bk@work.mysql.com's avatar
bk@work.mysql.com committed
292
  }
293
#ifndef EMBEDDED_LIBRARY
294 295
  if (!tables)
    kill_delayed_threads();
296
#endif
bk@work.mysql.com's avatar
bk@work.mysql.com committed
297 298 299 300 301 302 303 304 305 306
  if (if_wait_for_refresh)
  {
    /*
      If there is any table that has a lower refresh_version, wait until
      this is closed (or this thread is killed) before returning
    */
    thd->mysys_var->current_mutex= &LOCK_open;
    thd->mysys_var->current_cond= &COND_refresh;
    thd->proc_info="Flushing tables";

307
    close_old_data_files(thd,thd->open_tables,1,1);
308
    mysql_ha_flush(thd, tables, MYSQL_HA_REOPEN_ON_USAGE | MYSQL_HA_FLUSH_ALL);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
309 310
    bool found=1;
    /* Wait until all threads has closed all the tables we had locked */
311 312
    DBUG_PRINT("info",
	       ("Waiting for others threads to close their open tables"));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334
    while (found && ! thd->killed)
    {
      found=0;
      for (uint idx=0 ; idx < open_cache.records ; idx++)
      {
	TABLE *table=(TABLE*) hash_element(&open_cache,idx);
	if ((table->version) < refresh_version && table->db_stat)
	{
	  found=1;
	  pthread_cond_wait(&COND_refresh,&LOCK_open);
	  break;
	}
      }
    }
    /*
      No other thread has the locked tables open; reopen them and get the
      old locks. This should always succeed (unless some external process
      has removed the tables)
    */
    thd->in_lock_tables=1;
    result=reopen_tables(thd,1,1);
    thd->in_lock_tables=0;
335 336 337
    /* Set version for table */
    for (TABLE *table=thd->open_tables; table ; table=table->next)
      table->version=refresh_version;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
338 339 340 341 342 343 344 345 346 347 348 349 350 351
  }
  VOID(pthread_mutex_unlock(&LOCK_open));
  if (if_wait_for_refresh)
  {
    pthread_mutex_lock(&thd->mysys_var->mutex);
    thd->mysys_var->current_mutex= 0;
    thd->mysys_var->current_cond= 0;
    thd->proc_info=0;
    pthread_mutex_unlock(&thd->mysys_var->mutex);
  }
  DBUG_RETURN(result);
}


352 353
/*
  Close all tables used by thread
bk@work.mysql.com's avatar
bk@work.mysql.com committed
354

355 356 357 358 359 360 361 362 363 364 365 366
  SYNOPSIS
    close_thread_tables()
    thd			Thread handler
    lock_in_use		Set to 1 (0 = default) if caller has a lock on
			LOCK_open
    skip_derived	Set to 1 (0 = default) if we should not free derived
			tables.

  IMPLEMENTATION
    Unlocks tables and frees derived tables.
    Put all normal tables used by thread in free list.
*/
bk@work.mysql.com's avatar
bk@work.mysql.com committed
367

368
void close_thread_tables(THD *thd, bool lock_in_use, bool skip_derived)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
369
{
370
  bool found_old_table;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
371 372
  DBUG_ENTER("close_thread_tables");

373 374 375 376 377 378 379 380 381 382 383 384 385 386
  if (thd->derived_tables && !skip_derived)
  {
    TABLE *table, *next;
    /*
      Close all derived tables generated from questions like
      SELECT * from (select * from t1))
    */
    for (table= thd->derived_tables ; table ; table= next)
    {
      next= table->next;
      free_tmp_table(thd, table);
    }
    thd->derived_tables= 0;
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
387
  if (thd->locked_tables)
388 389
  {
    ha_commit_stmt(thd);			// If select statement
bk@work.mysql.com's avatar
bk@work.mysql.com committed
390
    DBUG_VOID_RETURN;				// LOCK TABLES in use
391
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
392 393 394

  if (thd->lock)
  {
395 396
    mysql_unlock_tables(thd, thd->lock);
    thd->lock=0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
397 398
  }
  /* VOID(pthread_sigmask(SIG_SETMASK,&thd->block_signals,NULL)); */
399
  if (!lock_in_use)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
400
    VOID(pthread_mutex_lock(&LOCK_open));
401
  safe_mutex_assert_owner(&LOCK_open);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
402 403

  DBUG_PRINT("info", ("thd->open_tables=%p", thd->open_tables));
404

405
 found_old_table= 0;
406 407
  while (thd->open_tables)
    found_old_table|=close_thread_table(thd, &thd->open_tables);
408
  thd->some_tables_deleted=0;
409

bk@work.mysql.com's avatar
bk@work.mysql.com committed
410
  /* Free tables to hold down open files */
411
  while (open_cache.records > table_cache_size && unused_tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
412 413 414 415 416 417 418
    VOID(hash_delete(&open_cache,(byte*) unused_tables)); /* purecov: tested */
  check_unused();
  if (found_old_table)
  {
    /* Tell threads waiting for refresh that something has happened */
    VOID(pthread_cond_broadcast(&COND_refresh));
  }
419
  if (!lock_in_use)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
420 421 422 423 424
    VOID(pthread_mutex_unlock(&LOCK_open));
  /*  VOID(pthread_sigmask(SIG_SETMASK,&thd->signals,NULL)); */
  DBUG_VOID_RETURN;
}

425 426 427 428 429 430
/* move one table to free list */

bool close_thread_table(THD *thd, TABLE **table_ptr)
{
  DBUG_ENTER("close_thread_table");

431 432 433
  bool found_old_table= 0;
  TABLE *table= *table_ptr;
  DBUG_ASSERT(table->key_read == 0);
434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451

  *table_ptr=table->next;
  if (table->version != refresh_version ||
      thd->version != refresh_version || !table->db_stat)
  {
    VOID(hash_delete(&open_cache,(byte*) table));
    found_old_table=1;
  }
  else
  {
    if (table->flush_version != flush_version)
    {
      table->flush_version=flush_version;
      table->file->extra(HA_EXTRA_FLUSH);
    }
    else
    {
      // Free memory and reset for next loop
452
      table->file->reset();
453 454 455 456 457 458 459 460 461 462 463 464 465 466 467
    }
    table->in_use=0;
    if (unused_tables)
    {
      table->next=unused_tables;		/* Link in last */
      table->prev=unused_tables->prev;
      unused_tables->prev=table;
      table->prev->next=table;
    }
    else
      unused_tables=table->next=table->prev=table;
  }
  DBUG_RETURN(found_old_table);
}

bk@work.mysql.com's avatar
bk@work.mysql.com committed
468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487
	/* Close and delete temporary tables */

void close_temporary(TABLE *table,bool delete_table)
{
  DBUG_ENTER("close_temporary");
  char path[FN_REFLEN];
  db_type table_type=table->db_type;
  strmov(path,table->path);
  free_io_cache(table);
  closefrm(table);
  my_free((char*) table,MYF(0));
  if (delete_table)
    rm_temporary_table(table_type, path);
  DBUG_VOID_RETURN;
}


void close_temporary_tables(THD *thd)
{
  TABLE *table,*next;
488 489
  char *query, *name_in_query, *end;
  uint greatest_key_length= 0;
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
490

491 492 493
  if (!thd->temporary_tables)
    return;
  
494 495 496 497 498
  /*
    We write a DROP TEMPORARY TABLE for each temp table left, so that our
    replication slave can clean them up. Not one multi-table DROP TABLE binlog
    event: this would cause problems if slave uses --replicate-*-table.
  */
499
  LINT_INIT(end);
500

501
  /* We'll re-use always same buffer so make it big enough for longest name */
502
  for (table=thd->temporary_tables ; table ; table=table->next)
503
    greatest_key_length= max(greatest_key_length, table->key_length);
504

505
  if ((query = alloc_root(thd->mem_root, greatest_key_length+50)))
506
    // Better add "if exists", in case a RESET MASTER has been done
507
    name_in_query= strmov(query, "DROP /*!40005 TEMPORARY */ TABLE IF EXISTS `");
508

bk@work.mysql.com's avatar
bk@work.mysql.com committed
509 510
  for (table=thd->temporary_tables ; table ; table=next)
  {
511 512 513 514 515
    /*
      In we are OOM for 'query' this is not fatal. We skip temporary tables
      not created directly by the user.
    */
    if (query && mysql_bin_log.is_open() && (table->real_name[0] != '#'))
516
    {
517 518 519 520
      /*
        Here we assume table_cache_key always starts
        with \0 terminated db name
      */
521 522 523 524 525 526 527 528 529 530 531 532 533 534 535
      end = strxmov(name_in_query, table->table_cache_key, "`.`",
                    table->real_name, "`", NullS);
      Query_log_event qinfo(thd, query, (ulong)(end-query), 0, FALSE);
      /*
        Imagine the thread had created a temp table, then was doing a SELECT, and
        the SELECT was killed. Then it's not clever to mark the statement above as
        "killed", because it's not really a statement updating data, and there
        are 99.99% chances it will succeed on slave. And, if thread is
        killed now, it's not clever either.
        If a real update (one updating a persistent table) was killed on the
        master, then this real update will be logged with error_code=killed,
        rightfully causing the slave to stop.
      */
      qinfo.error_code= 0;
      mysql_bin_log.write(&qinfo);
536
    }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
537 538 539 540 541 542
    next=table->next;
    close_temporary(table);
  }
  thd->temporary_tables=0;
}

543
/*
544
  Find first suitable table by alias in given list.
545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561

  SYNOPSIS
    find_table_in_list()
    table - pointer to table list
    db_name - data base name or 0 for any
    table_name - table name or 0 for any

  RETURN VALUES
    NULL	Table not found
    #		Pointer to found table.
*/

TABLE_LIST * find_table_in_list(TABLE_LIST *table,
				const char *db_name, const char *table_name)
{
  for (; table; table= table->next)
    if ((!db_name || !strcmp(table->db, db_name)) &&
562 563
	(!table_name || !my_strcasecmp(table_alias_charset,
				       table->alias, table_name)))
564 565 566
      break;
  return table;
}
bk@work.mysql.com's avatar
bk@work.mysql.com committed
567

568 569 570 571
/*
  Find real table in given list.

  SYNOPSIS
572
    find_real_table_in_list()
573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592
    table - pointer to table list
    db_name - data base name
    table_name - table name

  RETURN VALUES
    NULL	Table not found
    #		Pointer to found table.
*/

TABLE_LIST * find_real_table_in_list(TABLE_LIST *table,
				     const char *db_name,
				     const char *table_name)
{
  for (; table; table= table->next)
    if (!strcmp(table->db, db_name) &&
	!strcmp(table->real_name, table_name))
      break;
  return table;
}

bk@work.mysql.com's avatar
bk@work.mysql.com committed
593 594 595 596 597 598
TABLE **find_temporary_table(THD *thd, const char *db, const char *table_name)
{
  char	key[MAX_DBKEY_LENGTH];
  uint	key_length= (uint) (strmov(strmov(key,db)+1,table_name)-key)+1;
  TABLE *table,**prev;

guilhem@mysql.com's avatar
guilhem@mysql.com committed
599 600
  int4store(key+key_length,thd->server_id);
  key_length += 4;
601
  int4store(key+key_length,thd->variables.pseudo_thread_id);
602
  key_length += 4;
603

bk@work.mysql.com's avatar
bk@work.mysql.com committed
604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623
  prev= &thd->temporary_tables;
  for (table=thd->temporary_tables ; table ; table=table->next)
  {
    if (table->key_length == key_length &&
	!memcmp(table->table_cache_key,key,key_length))
      return prev;
    prev= &table->next;
  }
  return 0;					// Not a temporary table
}

bool close_temporary_table(THD *thd, const char *db, const char *table_name)
{
  TABLE *table,**prev;

  if (!(prev=find_temporary_table(thd,db,table_name)))
    return 1;
  table= *prev;
  *prev= table->next;
  close_temporary(table);
624
  if (thd->slave_thread)
625
    --slave_open_temp_tables;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
626 627 628
  return 0;
}

guilhem@mysql.com's avatar
guilhem@mysql.com committed
629 630 631 632 633 634 635
/*
  Used by ALTER TABLE when the table is a temporary one. It changes something
  only if the ALTER contained a RENAME clause (otherwise, table_name is the old
  name).
  Prepares a table cache key, which is the concatenation of db, table_name and
  thd->slave_proxy_id, separated by '\0'.
*/
636
bool rename_temporary_table(THD* thd, TABLE *table, const char *db,
bk@work.mysql.com's avatar
bk@work.mysql.com committed
637 638 639 640
			    const char *table_name)
{
  char *key;
  if (!(key=(char*) alloc_root(&table->mem_root,
641
			       (uint) strlen(db)+
guilhem@mysql.com's avatar
guilhem@mysql.com committed
642
			       (uint) strlen(table_name)+6+4)))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
643 644 645 646 647
    return 1;				/* purecov: inspected */
  table->key_length=(uint)
    (strmov((table->real_name=strmov(table->table_cache_key=key,
				     db)+1),
	    table_name) - table->table_cache_key)+1;
guilhem@mysql.com's avatar
guilhem@mysql.com committed
648 649
  int4store(key+table->key_length,thd->server_id);
  table->key_length += 4;
650
  int4store(key+table->key_length,thd->variables.pseudo_thread_id);
651
  table->key_length += 4;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708
  return 0;
}


	/* move table first in unused links */

static void relink_unused(TABLE *table)
{
  if (table != unused_tables)
  {
    table->prev->next=table->next;		/* Remove from unused list */
    table->next->prev=table->prev;
    table->next=unused_tables;			/* Link in unused tables */
    table->prev=unused_tables->prev;
    unused_tables->prev->next=table;
    unused_tables->prev=table;
    unused_tables=table;
    check_unused();
  }
}


/*
  Remove all instances of table from the current open list
  Free all locks on tables that are done with LOCK TABLES
 */

TABLE *unlink_open_table(THD *thd, TABLE *list, TABLE *find)
{
  char key[MAX_DBKEY_LENGTH];
  uint key_length=find->key_length;
  TABLE *start=list,**prev,*next;
  prev= &start;
  memcpy(key,find->table_cache_key,key_length);
  for (; list ; list=next)
  {
    next=list->next;
    if (list->key_length == key_length &&
	!memcmp(list->table_cache_key,key,key_length))
    {
      if (thd->locked_tables)
	mysql_lock_remove(thd, thd->locked_tables,list);
      VOID(hash_delete(&open_cache,(byte*) list)); // Close table
    }
    else
    {
      *prev=list;				// put in use list
      prev= &list->next;
    }
  }
  *prev=0;
  // Notify any 'refresh' threads
  pthread_cond_broadcast(&COND_refresh);
  return start;
}


709
/*
bk@work.mysql.com's avatar
bk@work.mysql.com committed
710
   When we call the following function we must have a lock on
711
   LOCK_open ; This lock will be unlocked on return.
bk@work.mysql.com's avatar
bk@work.mysql.com committed
712 713 714 715
*/

void wait_for_refresh(THD *thd)
{
716 717
  safe_mutex_assert_owner(&LOCK_open);

bk@work.mysql.com's avatar
bk@work.mysql.com committed
718 719 720 721 722 723
  /* Wait until the current table is up to date */
  const char *proc_info;
  thd->mysys_var->current_mutex= &LOCK_open;
  thd->mysys_var->current_cond= &COND_refresh;
  proc_info=thd->proc_info;
  thd->proc_info="Waiting for table";
724 725
  if (!thd->killed)
    (void) pthread_cond_wait(&COND_refresh,&LOCK_open);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
726 727 728 729 730 731 732 733 734

  pthread_mutex_unlock(&LOCK_open);	// Must be unlocked first
  pthread_mutex_lock(&thd->mysys_var->mutex);
  thd->mysys_var->current_mutex= 0;
  thd->mysys_var->current_cond= 0;
  thd->proc_info= proc_info;
  pthread_mutex_unlock(&thd->mysys_var->mutex);
}

735

736 737 738 739 740 741
TABLE *reopen_name_locked_table(THD* thd, TABLE_LIST* table_list)
{
  DBUG_ENTER("reopen_name_locked_table");
  if (thd->killed)
    DBUG_RETURN(0);
  TABLE* table;
742
  if (!(table = table_list->table))
743 744 745
    DBUG_RETURN(0);

  char* db = thd->db ? thd->db : table_list->db;
746
  char* table_name = table_list->real_name;
747 748 749 750 751
  char	key[MAX_DBKEY_LENGTH];
  uint	key_length;
  key_length=(uint) (strmov(strmov(key,db)+1,table_name)-key)+1;

  pthread_mutex_lock(&LOCK_open);
752
  if (open_unireg_entry(thd, table, db, table_name, table_name) ||
753 754
      !(table->table_cache_key =memdup_root(&table->mem_root,(char*) key,
					    key_length)))
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
755 756 757 758 759
  {
    closefrm(table);
    pthread_mutex_unlock(&LOCK_open);
    DBUG_RETURN(0);
  }
760

761
  table->key_length=key_length;
sasha@mysql.sashanet.com's avatar
sasha@mysql.sashanet.com committed
762 763
  table->version=0;
  table->flush_version=0;
764 765 766
  table->in_use = thd;
  check_unused();
  pthread_mutex_unlock(&LOCK_open);
sasha@mysql.sashanet.com's avatar
sasha@mysql.sashanet.com committed
767 768
  table->next = thd->open_tables;
  thd->open_tables = table;
769 770 771
  table->tablenr=thd->current_tablenr++;
  table->used_fields=0;
  table->const_table=0;
772
  table->outer_join= table->null_row= table->maybe_null= table->force_index= 0;
773
  table->status=STATUS_NO_RECORD;
774 775
  table->keys_in_use_for_query= table->keys_in_use;
  table->used_keys= table->keys_for_keyread;
776
  DBUG_RETURN(table);
777 778
}

bk@work.mysql.com's avatar
bk@work.mysql.com committed
779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802

/******************************************************************************
** open a table
** Uses a cache of open tables to find a table not in use.
** If refresh is a NULL pointer, then the is no version number checking and
** the table is not put in the thread-open-list
** If the return value is NULL and refresh is set then one must close
** all tables and retry the open
******************************************************************************/


TABLE *open_table(THD *thd,const char *db,const char *table_name,
		  const char *alias,bool *refresh)
{
  reg1	TABLE *table;
  char	key[MAX_DBKEY_LENGTH];
  uint	key_length;
  DBUG_ENTER("open_table");

  /* find a unused table in the open table cache */
  if (refresh)
    *refresh=0;
  if (thd->killed)
    DBUG_RETURN(0);
803
  key_length= (uint) (strmov(strmov(key,db)+1,table_name)-key)+1;
guilhem@mysql.com's avatar
guilhem@mysql.com committed
804
  int4store(key + key_length, thd->server_id);
monty@narttu.mysql.fi's avatar
monty@narttu.mysql.fi committed
805
  int4store(key + key_length + 4, thd->variables.pseudo_thread_id);
806

bk@work.mysql.com's avatar
bk@work.mysql.com committed
807 808
  for (table=thd->temporary_tables; table ; table=table->next)
  {
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
809 810 811
    if (table->key_length == key_length + TMP_TABLE_KEY_EXTRA &&
	!memcmp(table->table_cache_key, key,
                key_length + TMP_TABLE_KEY_EXTRA))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
812 813 814 815 816 817 818 819
    {
      if (table->query_id == thd->query_id)
      {
	my_printf_error(ER_CANT_REOPEN_TABLE,
			ER(ER_CANT_REOPEN_TABLE),MYF(0),table->table_name);
	DBUG_RETURN(0);
      }
      table->query_id=thd->query_id;
820
      table->clear_query_id=1;
821
      thd->tmp_table_used= 1;
822
      DBUG_PRINT("info",("Using temporary table"));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
823 824 825 826 827 828 829 830 831 832
      goto reset;
    }
  }

  if (thd->locked_tables)
  {						// Using table locks
    for (table=thd->open_tables; table ; table=table->next)
    {
      if (table->key_length == key_length &&
	  !memcmp(table->table_cache_key,key,key_length) &&
monty@mysql.com's avatar
monty@mysql.com committed
833
	  !my_strcasecmp(system_charset_info, table->table_name, alias) &&
834 835 836
	  table->query_id != thd->query_id)
      {
	table->query_id=thd->query_id;
837
        DBUG_PRINT("info",("Using locked table"));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
838
	goto reset;
839
      }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
840 841 842 843
    }
    my_printf_error(ER_TABLE_NOT_LOCKED,ER(ER_TABLE_NOT_LOCKED),MYF(0),alias);
    DBUG_RETURN(0);
  }
igor@rurik.mysql.com's avatar
igor@rurik.mysql.com committed
844

bk@work.mysql.com's avatar
bk@work.mysql.com committed
845 846 847 848 849 850 851 852 853 854 855 856
  VOID(pthread_mutex_lock(&LOCK_open));

  if (!thd->open_tables)
    thd->version=refresh_version;
  else if (thd->version != refresh_version && refresh)
  {
    /* Someone did a refresh while thread was opening tables */
    *refresh=1;
    VOID(pthread_mutex_unlock(&LOCK_open));
    DBUG_RETURN(0);
  }

857
  /* close handler tables which are marked for flush */
858
  mysql_ha_flush(thd, (TABLE_LIST*) NULL, MYSQL_HA_REOPEN_ON_USAGE);
859

bk@work.mysql.com's avatar
bk@work.mysql.com committed
860 861 862 863 864 865 866 867 868 869
  for (table=(TABLE*) hash_search(&open_cache,(byte*) key,key_length) ;
       table && table->in_use ;
       table = (TABLE*) hash_next(&open_cache,(byte*) key,key_length))
  {
    if (table->version != refresh_version)
    {
      /*
      ** There is a refresh in progress for this table
      ** Wait until the table is freed or the thread is killed.
      */
870
      close_old_data_files(thd,thd->open_tables,0,0);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889
      if (table->in_use != thd)
	wait_for_refresh(thd);
      else
	VOID(pthread_mutex_unlock(&LOCK_open));
      if (refresh)
	*refresh=1;
      DBUG_RETURN(0);
    }
  }
  if (table)
  {
    if (table == unused_tables)
    {						// First unused
      unused_tables=unused_tables->next;	// Remove from link
      if (table == unused_tables)
	unused_tables=0;
    }
    table->prev->next=table->next;		/* Remove from unused list */
    table->next->prev=table->prev;
igor@rurik.mysql.com's avatar
igor@rurik.mysql.com committed
890

bk@work.mysql.com's avatar
bk@work.mysql.com committed
891 892 893 894
  }
  else
  {
    /* Free cache if too big */
895
    while (open_cache.records > table_cache_size && unused_tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
896 897 898 899
      VOID(hash_delete(&open_cache,(byte*) unused_tables)); /* purecov: tested */

    /* make a new table */
    if (!(table=(TABLE*) my_malloc(sizeof(*table),MYF(MY_WME))))
900 901
    {
      VOID(pthread_mutex_unlock(&LOCK_open));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
902
      DBUG_RETURN(NULL);
903
    }
904
    if (open_unireg_entry(thd, table,db,table_name,alias) ||
bk@work.mysql.com's avatar
bk@work.mysql.com committed
905 906 907 908 909 910 911 912 913 914 915 916
	!(table->table_cache_key=memdup_root(&table->mem_root,(char*) key,
					     key_length)))
    {
      table->next=table->prev=table;
      free_cache_entry(table);
      VOID(pthread_mutex_unlock(&LOCK_open));
      DBUG_RETURN(NULL);
    }
    table->key_length=key_length;
    table->version=refresh_version;
    table->flush_version=flush_version;
    DBUG_PRINT("info", ("inserting table %p into the cache", table));
hf@deer.(none)'s avatar
SCRUM  
hf@deer.(none) committed
917
    VOID(my_hash_insert(&open_cache,(byte*) table));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
918 919 920
  }

  table->in_use=thd;
921
  check_unused();				// Debugging call
igor@rurik.mysql.com's avatar
igor@rurik.mysql.com committed
922
       
bk@work.mysql.com's avatar
bk@work.mysql.com committed
923 924 925 926 927 928 929 930 931 932 933 934
  VOID(pthread_mutex_unlock(&LOCK_open));
  if (refresh)
  {
    table->next=thd->open_tables;		/* Link into simple list */
    thd->open_tables=table;
  }
  table->reginfo.lock_type=TL_READ;		/* Assume read */

 reset:
  /* Fix alias if table name changes */
  if (strcmp(table->table_name,alias))
  {
935
    uint length=(uint) strlen(alias)+1;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
936 937 938 939 940 941
    table->table_name= (char*) my_realloc(table->table_name,length,
					  MYF(MY_WME));
    memcpy(table->table_name,alias,length);
    for (uint i=0 ; i < table->fields ; i++)
      table->field[i]->table_name=table->table_name;
  }
942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966
#if MYSQL_VERSION_ID < 40100
  /*
    If per-connection "new" variable (represented by variables.new_mode)
    is set then we should pretend that the length of TIMESTAMP field is 19.
    The cheapest (from perfomance viewpoint) way to achieve that is to set
    field_length of all Field_timestamp objects in a table after opening
    it (to 19 if new_mode is true or to original field length otherwise).
    We save value of new_mode variable in TABLE::timestamp_mode to
    not perform this setup if new_mode value is the same between sequential
    table opens.
  */
  my_bool new_mode= thd->variables.new_mode;
  if (table->timestamp_mode != new_mode)
  {
    for (uint i=0 ; i < table->fields ; i++)
    {
      Field *field= table->field[i];

      if (field->type() == FIELD_TYPE_TIMESTAMP)
        field->field_length= new_mode ? 19 :
                             ((Field_timestamp *)(field))->orig_field_length;
    }
    table->timestamp_mode= new_mode;
  }
#endif
bk@work.mysql.com's avatar
bk@work.mysql.com committed
967 968 969 970
  /* These variables are also set in reopen_table() */
  table->tablenr=thd->current_tablenr++;
  table->used_fields=0;
  table->const_table=0;
971
  table->outer_join= table->null_row= table->maybe_null= table->force_index= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
972
  table->status=STATUS_NO_RECORD;
973 974
  table->keys_in_use_for_query= table->keys_in_use;
  table->used_keys= table->keys_for_keyread;
975
  if (table->timestamp_field)
976
    table->timestamp_field_type= table->timestamp_field->get_auto_set_type();
Sinisa@sinisa.nasamreza.org's avatar
Sinisa@sinisa.nasamreza.org committed
977
  DBUG_ASSERT(table->key_read == 0);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020
  DBUG_RETURN(table);
}


TABLE *find_locked_table(THD *thd, const char *db,const char *table_name)
{
  char	key[MAX_DBKEY_LENGTH];
  uint key_length=(uint) (strmov(strmov(key,db)+1,table_name)-key)+1;

  for (TABLE *table=thd->open_tables; table ; table=table->next)
  {
    if (table->key_length == key_length &&
	!memcmp(table->table_cache_key,key,key_length))
      return table;
  }
  return(0);
}


/****************************************************************************
** Reopen an table because the definition has changed. The date file for the
** table is already closed.
** Returns 0 if ok.
** If table can't be reopened, the entry is unchanged.
****************************************************************************/

bool reopen_table(TABLE *table,bool locked)
{
  TABLE tmp;
  char *db=table->table_cache_key;
  char *table_name=table->real_name;
  bool error=1;
  Field **field;
  uint key,part;
  DBUG_ENTER("reopen_table");

#ifdef EXTRA_DEBUG
  if (table->db_stat)
    sql_print_error("Table %s had a open data handler in reopen_table",
		    table->table_name);
#endif
  if (!locked)
    VOID(pthread_mutex_lock(&LOCK_open));
1021
  safe_mutex_assert_owner(&LOCK_open);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1022

1023
  if (open_unireg_entry(current_thd,&tmp,db,table_name,table->table_name))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1024 1025 1026 1027 1028 1029 1030 1031 1032 1033
    goto end;
  free_io_cache(table);

  if (!(tmp.table_cache_key= memdup_root(&tmp.mem_root,db,
					 table->key_length)))
  {
    closefrm(&tmp);				// End of memory
    goto end;
  }

1034
  /* This list copies variables set by open_table */
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1035 1036 1037 1038 1039
  tmp.tablenr=		table->tablenr;
  tmp.used_fields=	table->used_fields;
  tmp.const_table=	table->const_table;
  tmp.outer_join=	table->outer_join;
  tmp.null_row=		table->null_row;
1040
  tmp.maybe_null=	table->maybe_null;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1041
  tmp.status=		table->status;
1042 1043
  tmp.keys_in_use_for_query= tmp.keys_in_use;
  tmp.used_keys= 	tmp.keys_for_keyread;
1044
  tmp.force_index=	tmp.force_index;
1045 1046 1047 1048 1049 1050 1051

  /* Get state */
  tmp.key_length=	table->key_length;
  tmp.in_use=    	table->in_use;
  tmp.reginfo.lock_type=table->reginfo.lock_type;
  tmp.version=		refresh_version;
  tmp.tmp_table=	table->tmp_table;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1052 1053
  tmp.grant=		table->grant;

1054
  /* Replace table in open list */
1055 1056
  tmp.next=		table->next;
  tmp.prev=		table->prev;
1057

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1058 1059 1060 1061 1062 1063
  if (table->file)
    VOID(closefrm(table));		// close file, free everything

  *table=tmp;
  table->file->change_table_ptr(table);

1064
  DBUG_ASSERT(table->table_name);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1065 1066
  for (field=table->field ; *field ; field++)
  {
1067
    (*field)->table= (*field)->orig_table= table;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1068 1069 1070 1071
    (*field)->table_name=table->table_name;
  }
  for (key=0 ; key < table->keys ; key++)
    for (part=0 ; part < table->key_info[key].usable_key_parts ; part++)
1072
      table->key_info[key].key_part[part].field->table= table;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113
  VOID(pthread_cond_broadcast(&COND_refresh));
  error=0;

 end:
  if (!locked)
    VOID(pthread_mutex_unlock(&LOCK_open));
  DBUG_RETURN(error);
}


/*
  Used with ALTER TABLE:
  Close all instanses of table when LOCK TABLES is in used;
  Close first all instances of table and then reopen them
 */

bool close_data_tables(THD *thd,const char *db, const char *table_name)
{
  TABLE *table;
  for (table=thd->open_tables; table ; table=table->next)
  {
    if (!strcmp(table->real_name,table_name) &&
	!strcmp(table->table_cache_key,db))
    {
      mysql_lock_remove(thd, thd->locked_tables,table);
      table->file->close();
      table->db_stat=0;
    }
  }
  return 0;					// For the future
}


/*
  Reopen all tables with closed data files
  One should have lock on LOCK_open when calling this
*/

bool reopen_tables(THD *thd,bool get_locks,bool in_refresh)
{
  DBUG_ENTER("reopen_tables");
1114 1115
  safe_mutex_assert_owner(&LOCK_open);

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183
  if (!thd->open_tables)
    DBUG_RETURN(0);

  TABLE *table,*next,**prev;
  TABLE **tables,**tables_ptr;			// For locks
  bool error=0;
  if (get_locks)
  {
    /* The ptr is checked later */
    uint opens=0;
    for (table=thd->open_tables; table ; table=table->next) opens++;
    tables= (TABLE**) my_alloca(sizeof(TABLE*)*opens);
  }
  else
    tables= &thd->open_tables;
  tables_ptr =tables;

  prev= &thd->open_tables;
  for (table=thd->open_tables; table ; table=next)
  {
    uint db_stat=table->db_stat;
    next=table->next;
    if (!tables || (!db_stat && reopen_table(table,1)))
    {
      my_error(ER_CANT_REOPEN_TABLE,MYF(0),table->table_name);
      VOID(hash_delete(&open_cache,(byte*) table));
      error=1;
    }
    else
    {
      *prev= table;
      prev= &table->next;
      if (get_locks && !db_stat)
	*tables_ptr++= table;			// need new lock on this
      if (in_refresh)
      {
	table->version=0;
	table->locked_by_flush=0;
      }
    }
  }
  if (tables != tables_ptr)			// Should we get back old locks
  {
    MYSQL_LOCK *lock;
    /* We should always get these locks */
    thd->some_tables_deleted=0;
    if ((lock=mysql_lock_tables(thd,tables,(uint) (tables_ptr-tables))))
    {
      thd->locked_tables=mysql_lock_merge(thd->locked_tables,lock);
    }
    else
      error=1;
  }
  if (get_locks && tables)
  {
    my_afree((gptr) tables);
  }
  VOID(pthread_cond_broadcast(&COND_refresh)); // Signal to refresh
  *prev=0;
  DBUG_RETURN(error);
}

/*
  Close handlers for tables in list, but leave the TABLE structure
  intact so that we can re-open these quickly
  abort_locks is set if called from flush_tables.
*/

1184 1185
void close_old_data_files(THD *thd, TABLE *table, bool abort_locks,
			  bool send_refresh)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1186
{
1187 1188
  DBUG_ENTER("close_old_data_files");
  bool found=send_refresh;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210
  for (; table ; table=table->next)
  {
    if (table->version != refresh_version)
    {
      found=1;
      if (!abort_locks)				// If not from flush tables
	table->version = refresh_version;	// Let other threads use table
      if (table->db_stat)
      {
	if (abort_locks)
	{
	  mysql_lock_abort(thd,table);		// Close waiting threads
	  mysql_lock_remove(thd, thd->locked_tables,table);
	  table->locked_by_flush=1;		// Will be reopened with locks
	}
	table->file->close();
	table->db_stat=0;
      }
    }
  }
  if (found)
    VOID(pthread_cond_broadcast(&COND_refresh)); // Signal to refresh
1211
  DBUG_VOID_RETURN;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1212 1213 1214 1215 1216 1217 1218 1219 1220
}


/*
  Wait until all threads has closed the tables in the list
  We have also to wait if there is thread that has a lock on this table even
  if the table is closed
*/

1221
bool table_is_used(TABLE *table, bool wait_for_name_lock)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1222 1223 1224 1225 1226
{
  do
  {
    char *key= table->table_cache_key;
    uint key_length=table->key_length;
1227 1228
    for (TABLE *search=(TABLE*) hash_search(&open_cache,
					    (byte*) key,key_length) ;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1229 1230 1231 1232
	 search ;
	 search = (TABLE*) hash_next(&open_cache,(byte*) key,key_length))
    {
      if (search->locked_by_flush ||
1233
	  search->locked_by_name && wait_for_name_lock ||
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250
	  search->db_stat && search->version < refresh_version)
	return 1;				// Table is used
    }
  } while ((table=table->next));
  return 0;
}


/* Wait until all used tables are refreshed */

bool wait_for_tables(THD *thd)
{
  bool result;
  DBUG_ENTER("wait_for_tables");

  thd->proc_info="Waiting for tables";
  pthread_mutex_lock(&LOCK_open);
1251
  while (!thd->killed)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1252
  {
1253 1254
    thd->some_tables_deleted=0;
    close_old_data_files(thd,thd->open_tables,0,dropping_tables != 0);
1255
    mysql_ha_flush(thd, (TABLE_LIST*) NULL, MYSQL_HA_REOPEN_ON_USAGE);
1256 1257
    if (!table_is_used(thd->open_tables,1))
      break;
1258
    (void) pthread_cond_wait(&COND_refresh,&LOCK_open);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1259 1260 1261 1262 1263 1264 1265
  }
  if (thd->killed)
    result= 1;					// aborted
  else
  {
    /* Now we can open all tables without any interference */
    thd->proc_info="Reopen tables";
1266
    result=reopen_tables(thd,0,0);
1267
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308
  pthread_mutex_unlock(&LOCK_open);
  thd->proc_info=0;
  DBUG_RETURN(result);
}


/* drop tables from locked list */

bool drop_locked_tables(THD *thd,const char *db, const char *table_name)
{
  TABLE *table,*next,**prev;
  bool found=0;
  prev= &thd->open_tables;
  for (table=thd->open_tables; table ; table=next)
  {
    next=table->next;
    if (!strcmp(table->real_name,table_name) &&
	!strcmp(table->table_cache_key,db))
    {
      mysql_lock_remove(thd, thd->locked_tables,table);
      VOID(hash_delete(&open_cache,(byte*) table));
      found=1;
    }
    else
    {
      *prev=table;
      prev= &table->next;
    }
  }
  *prev=0;
  if (found)
    VOID(pthread_cond_broadcast(&COND_refresh)); // Signal to refresh
  if (thd->locked_tables && thd->locked_tables->table_count == 0)
  {
    my_free((gptr) thd->locked_tables,MYF(0));
    thd->locked_tables=0;
  }
  return found;
}


1309 1310 1311 1312 1313
/*
  If we have the table open, which only happens when a LOCK TABLE has been
  done on the table, change the lock type to a lock that will abort all
  other threads trying to get the lock.
*/
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1314 1315 1316 1317

void abort_locked_tables(THD *thd,const char *db, const char *table_name)
{
  TABLE *table;
1318
  for (table= thd->open_tables; table ; table= table->next)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1319 1320 1321
  {
    if (!strcmp(table->real_name,table_name) &&
	!strcmp(table->table_cache_key,db))
1322
    {
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1323
      mysql_lock_abort(thd,table);
1324 1325
      break;
    }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1326 1327 1328
  }
}

1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346

/*
  Load a table definition from file and open unireg table

  SYNOPSIS
    open_unireg_entry()
    thd			Thread handle
    entry		Store open table definition here
    db			Database name
    name		Table name
    alias		Alias name

  NOTES
   Extra argument for open is taken from thd->open_options

  RETURN
    0	ok
    #	Error
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1347 1348
*/

1349
static int open_unireg_entry(THD *thd, TABLE *entry, const char *db,
1350
			     const char *name, const char *alias)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1351 1352
{
  char path[FN_REFLEN];
1353
  int error;
1354
  uint discover_retry_count= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1355 1356
  DBUG_ENTER("open_unireg_entry");

1357
  strxmov(path, mysql_data_home, "/", db, "/", name, NullS);
1358
  while (openfrm(path,alias,
1359 1360 1361
	       (uint) (HA_OPEN_KEYFILE | HA_OPEN_RNDFILE | HA_GET_INDEX |
		       HA_TRY_READ_ONLY),
	       READ_KEYINFO | COMPUTE_TYPES | EXTRA_RECORD,
1362
	      thd->open_options, entry))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1363
  {
1364
    if (!entry->crashed)
1365 1366 1367 1368 1369 1370 1371 1372 1373
    {
      /*
       Frm file could not be found on disk
       Since it does not exist, no one can be using it
       LOCK_open has been locked to protect from someone else
       trying to discover the table at the same time.
      */
      if (discover_retry_count++ != 0)
       goto err;
1374
      if (ha_create_table_from_engine(thd, db, name, TRUE) != 0)
1375 1376 1377 1378 1379
       goto err;

      thd->clear_error(); // Clear error message
      continue;
    }
1380

1381
    // Code below is for repairing a crashed file
1382
    TABLE_LIST table_list;
1383
    bzero((char*) &table_list, sizeof(table_list)); // just for safe
1384
    table_list.db=(char*) db;
1385
    table_list.real_name=(char*) name;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
1386

1387 1388
    safe_mutex_assert_owner(&LOCK_open);

1389 1390 1391 1392 1393 1394 1395 1396 1397 1398 1399 1400 1401
    if ((error=lock_table_name(thd,&table_list)))
    {
      if (error < 0)
      {
	goto err;
      }
      if (wait_for_locked_table_names(thd,&table_list))
      {
	unlock_table_name(thd,&table_list);
	goto err;
      }
    }
    pthread_mutex_unlock(&LOCK_open);
1402 1403
    thd->clear_error();				// Clear error message
    error= 0;
1404 1405 1406 1407 1408
    if (openfrm(path,alias,
		(uint) (HA_OPEN_KEYFILE | HA_OPEN_RNDFILE | HA_GET_INDEX |
			 HA_TRY_READ_ONLY),
		READ_KEYINFO | COMPUTE_TYPES | EXTRA_RECORD,
		ha_open_options | HA_OPEN_FOR_REPAIR,
1409
		entry) || ! entry->file ||
1410 1411
	(entry->file->is_crashed() && entry->file->check_and_repair(thd)))
    {
1412
      /* Give right error message */
1413
      thd->clear_error();
1414
      my_error(ER_NOT_KEYFILE, MYF(0), name, my_errno);
serg@serg.mylan's avatar
serg@serg.mylan committed
1415
      sql_print_error("Couldn't repair table: %s.%s",db,name);
1416 1417
      if (entry->file)
	closefrm(entry);
1418 1419
      error=1;
    }
1420
    else
1421
      thd->clear_error();			// Clear error message
1422
    pthread_mutex_lock(&LOCK_open);
1423
    unlock_table_name(thd,&table_list);
1424

1425 1426
    if (error)
      goto err;
1427
    break;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1428
  }
1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443
  /*
    If we are here, there was no fatal error (but error may be still
    unitialized).
  */
  if (unlikely(entry->file->implicit_emptied))
  {
    entry->file->implicit_emptied= 0;
    if (mysql_bin_log.is_open())
    {
      char *query, *end;
      uint query_buf_size= 20 + 2*NAME_LEN + 1;
      if ((query= (char*)my_malloc(query_buf_size,MYF(MY_WME))))
      {
        end = strxmov(strmov(query, "DELETE FROM `"),
                      db,"`.`",name,"`", NullS);
1444
        Query_log_event qinfo(thd, query, (ulong)(end-query), 0, FALSE);
1445 1446 1447 1448 1449 1450 1451 1452 1453 1454
        mysql_bin_log.write(&qinfo);
        my_free(query, MYF(0));
      }
      else
      {
        /*
          As replication is maybe going to be corrupted, we need to warn the
          DBA on top of warning the client (which will automatically be done
          because of MYF(MY_WME) in my_malloc() above).
        */
serg@serg.mylan's avatar
serg@serg.mylan committed
1455
        sql_print_error("When opening HEAP table, could not allocate \
1456 1457 1458 1459 1460 1461 1462
memory to write 'DELETE FROM `%s`.`%s`' to the binary log",db,name);
        if (entry->file)
          closefrm(entry);
        goto err;
      }
    }
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1463
  DBUG_RETURN(0);
1464 1465
err:
  DBUG_RETURN(1);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1466 1467
}

1468 1469
/*
  Open all tables in list
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1470

1471 1472 1473 1474 1475 1476 1477 1478 1479 1480 1481 1482
  SYNOPSIS
    open_tables()
    thd - thread handler
    start - list of tables
    counter - number of opened tables will be return using this parameter

  RETURN
    0  - OK
    -1 - error
*/

int open_tables(THD *thd, TABLE_LIST *start, uint *counter)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1483 1484 1485 1486 1487 1488
{
  TABLE_LIST *tables;
  bool refresh;
  int result=0;
  DBUG_ENTER("open_tables");

1489
  thd->current_tablenr= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1490
 restart:
monty@mysql.com's avatar
monty@mysql.com committed
1491
  *counter= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1492 1493 1494
  thd->proc_info="Opening tables";
  for (tables=start ; tables ; tables=tables->next)
  {
1495 1496 1497 1498
    /*
      Ignore placeholders for derived tables. After derived tables
      processing, link to created temporary table will be put here.
     */
1499 1500
    if (tables->derived)
      continue;
1501
    (*counter)++;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1502
    if (!tables->table &&
1503 1504 1505 1506
	!(tables->table= open_table(thd,
				    tables->db,
				    tables->real_name,
				    tables->alias, &refresh)))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1507 1508 1509 1510 1511
    {
      if (refresh)				// Refresh in progress
      {
	/* close all 'old' tables used by this thread */
	pthread_mutex_lock(&LOCK_open);
1512 1513
	// if query_id is not reset, we will get an error
	// re-opening a temp table
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1514 1515 1516 1517 1518
	thd->version=refresh_version;
	TABLE **prev_table= &thd->open_tables;
	bool found=0;
	for (TABLE_LIST *tmp=start ; tmp ; tmp=tmp->next)
	{
1519 1520
	  /* Close normal (not temporary) changed tables */
	  if (tmp->table && ! tmp->table->tmp_table)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1521 1522 1523 1524 1525 1526 1527 1528 1529 1530 1531 1532 1533 1534 1535 1536
	  {
	    if (tmp->table->version != refresh_version ||
		! tmp->table->db_stat)
	    {
	      VOID(hash_delete(&open_cache,(byte*) tmp->table));
	      tmp->table=0;
	      found=1;
	    }
	    else
	    {
	      *prev_table= tmp->table;		// Relink open list
	      prev_table= &tmp->table->next;
	    }
	  }
	}
	*prev_table=0;
1537
	pthread_mutex_unlock(&LOCK_open);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1538 1539 1540 1541 1542 1543 1544
	if (found)
	  VOID(pthread_cond_broadcast(&COND_refresh)); // Signal to refresh
	goto restart;
      }
      result= -1;				// Fatal error
      break;
    }
1545
    if (tables->lock_type != TL_UNLOCK && ! thd->locked_tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1546 1547 1548 1549 1550 1551 1552 1553
      tables->table->reginfo.lock_type=tables->lock_type;
    tables->table->grant= tables->grant;
  }
  thd->proc_info=0;
  DBUG_RETURN(result);
}


1554 1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566 1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608
/*
  Check that lock is ok for tables; Call start stmt if ok

  SYNOPSIS
    check_lock_and_start_stmt()
    thd			Thread handle
    table_list		Table to check
    lock_type		Lock used for table

  RETURN VALUES
  0	ok
  1	error
*/

static bool check_lock_and_start_stmt(THD *thd, TABLE *table,
				      thr_lock_type lock_type)
{
  int error;
  DBUG_ENTER("check_lock_and_start_stmt");

  if ((int) lock_type >= (int) TL_WRITE_ALLOW_READ &&
      (int) table->reginfo.lock_type < (int) TL_WRITE_ALLOW_READ)
  {
    my_printf_error(ER_TABLE_NOT_LOCKED_FOR_WRITE,
		    ER(ER_TABLE_NOT_LOCKED_FOR_WRITE),
		    MYF(0),table->table_name);
    DBUG_RETURN(1);
  }
  if ((error=table->file->start_stmt(thd)))
  {
    table->file->print_error(error,MYF(0));
    DBUG_RETURN(1);
  }
  DBUG_RETURN(0);
}


/*
  Open and lock one table

  SYNOPSIS
    open_ltable()
    thd			Thread handler
    table_list		Table to open is first table in this list
    lock_type		Lock to use for open

  RETURN VALUES
    table		Opened table
    0			Error
  
    If ok, the following are also set:
      table_list->lock_type 	lock_type
      table_list->table		table
*/

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1609 1610 1611 1612 1613 1614 1615
TABLE *open_ltable(THD *thd, TABLE_LIST *table_list, thr_lock_type lock_type)
{
  TABLE *table;
  bool refresh;
  DBUG_ENTER("open_ltable");

  thd->proc_info="Opening table";
1616
  thd->current_tablenr= 0;
1617
  while (!(table=open_table(thd,table_list->db,
1618
			    table_list->real_name,table_list->alias,
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1619
			    &refresh)) && refresh) ;
1620

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1621 1622
  if (table)
  {
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
1623
#if defined( __WIN__) || defined(OS2)
1624
    /* Win32 can't drop a file that is open */
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
1625
    if (lock_type == TL_WRITE_ALLOW_READ)
1626 1627 1628
    {
      lock_type= TL_WRITE;
    }
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
1629
#endif /* __WIN__ || OS2 */
1630 1631
    table_list->lock_type= lock_type;
    table_list->table=	   table;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1632 1633 1634
    table->grant= table_list->grant;
    if (thd->locked_tables)
    {
1635 1636 1637 1638 1639
      if (check_lock_and_start_stmt(thd, table, lock_type))
	table= 0;
    }
    else
    {
1640
      DBUG_ASSERT(thd->lock == 0);	// You must lock everything at once
1641 1642 1643
      if ((table->reginfo.lock_type= lock_type) != TL_UNLOCK)
	if (!(thd->lock=mysql_lock_tables(thd,&table_list->table,1)))
	  table= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1644 1645 1646 1647 1648 1649
    }
  }
  thd->proc_info=0;
  DBUG_RETURN(table);
}

1650

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1651
/*
1652 1653 1654 1655 1656 1657 1658 1659
  Open all tables in list and locks them for read without derived
  tables processing.

  SYNOPSIS
    simple_open_n_lock_tables()
    thd		- thread handler
    tables	- list of tables for open&locking

1660 1661 1662 1663
  RETURN
    0  - ok
    -1 - error

1664 1665
  NOTE
    The lock will automaticly be freed by close_thread_tables()
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1666 1667
*/

1668
int simple_open_n_lock_tables(THD *thd, TABLE_LIST *tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1669
{
1670 1671 1672
  DBUG_ENTER("simple_open_n_lock_tables");
  uint counter;
  if (open_tables(thd, tables, &counter) || lock_tables(thd, tables, counter))
1673 1674 1675 1676 1677 1678 1679 1680 1681 1682
    DBUG_RETURN(-1);				/* purecov: inspected */
  DBUG_RETURN(0);
}


/*
  Open all tables in list, locks them and process derived tables
  tables processing.

  SYNOPSIS
1683
    open_and_lock_tables()
1684 1685 1686
    thd		- thread handler
    tables	- list of tables for open&locking

1687 1688 1689 1690
  RETURN
    0  - ok
    -1 - error

1691 1692 1693 1694 1695 1696 1697
  NOTE
    The lock will automaticly be freed by close_thread_tables()
*/

int open_and_lock_tables(THD *thd, TABLE_LIST *tables)
{
  DBUG_ENTER("open_and_lock_tables");
1698 1699
  uint counter;
  if (open_tables(thd, tables, &counter) || lock_tables(thd, tables, counter))
1700
    DBUG_RETURN(-1);				/* purecov: inspected */
monty@mysql.com's avatar
monty@mysql.com committed
1701 1702 1703 1704 1705
  relink_tables_for_derived(thd);
  DBUG_RETURN(mysql_handle_derived(thd->lex));
}


1706 1707 1708 1709 1710 1711 1712 1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733
/*
  Open all tables in list and process derived tables

  SYNOPSIS
    open_normal_and_derived_tables
    thd		- thread handler
    tables	- list of tables for open&locking

  RETURN
    FALSE - ok
    TRUE  - error

  NOTE 
    This is to be used on prepare stage when you don't read any
    data from the tables.
*/

int open_normal_and_derived_tables(THD *thd, TABLE_LIST *tables)
{
  uint counter;
  DBUG_ENTER("open_normal_and_derived_tables");
  if (open_tables(thd, tables, &counter))
    DBUG_RETURN(-1);				/* purecov: inspected */
  relink_tables_for_derived(thd);
  DBUG_RETURN(mysql_handle_derived(thd->lex));
}


monty@mysql.com's avatar
monty@mysql.com committed
1734 1735 1736 1737 1738 1739 1740
/*
  Let us propagate pointers to open tables from global table list
  to table lists in particular selects if needed.
*/

void relink_tables_for_derived(THD *thd)
{
1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752
  if (thd->lex->all_selects_list->next_select_in_list() ||
      thd->lex->time_zone_tables_used)
  {
    for (SELECT_LEX *sl= thd->lex->all_selects_list;
	 sl;
	 sl= sl->next_select_in_list())
      for (TABLE_LIST *cursor= (TABLE_LIST *) sl->table_list.first;
           cursor;
           cursor=cursor->next)
        if (cursor->table_list)
          cursor->table= cursor->table_list->table;
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1753 1754
}

1755 1756 1757 1758 1759 1760 1761 1762

/*
  Lock all tables in list

  SYNOPSIS
    lock_tables()
    thd			Thread handler
    tables		Tables to lock
1763
    count		umber of opened tables
1764

1765 1766 1767 1768 1769
  NOTES
    You can't call lock_tables twice, as this would break the dead-lock-free
    handling thr_lock gives us.  You most always get all needed locks at
    once.

1770 1771 1772 1773 1774
  RETURN VALUES
   0	ok
   -1	Error
*/

1775
int lock_tables(THD *thd, TABLE_LIST *tables, uint count)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1776
{
1777
  TABLE_LIST *table;
1778 1779 1780 1781
  if (!tables)
    return 0;

  if (!thd->locked_tables)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1782
  {
1783
    DBUG_ASSERT(thd->lock == 0);	// You must lock everything at once
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1784 1785 1786 1787
    TABLE **start,**ptr;
    if (!(ptr=start=(TABLE**) sql_alloc(sizeof(TABLE*)*count)))
      return -1;
    for (table = tables ; table ; table=table->next)
1788 1789 1790 1791
    {
      if (!table->derived)
	*(ptr++)= table->table;
    }
monty@mysql.com's avatar
monty@mysql.com committed
1792
    if (!(thd->lock=mysql_lock_tables(thd,start, (uint) (ptr - start))))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1793 1794
      return -1;				/* purecov: inspected */
  }
1795 1796 1797 1798
  else
  {
    for (table = tables ; table ; table=table->next)
    {
1799 1800
      if (!table->derived && 
	  check_lock_and_start_stmt(thd, table->table, table->lock_type))
1801
      {
1802
	ha_rollback_stmt(thd);
1803 1804 1805 1806
	return -1;
      }
    }
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1807 1808 1809
  return 0;
}

1810

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1811
/*
1812 1813 1814
  Open a single table without table caching and don't set it in open_list
  Used by alter_table to open a temporary table and when creating
  a temporary table with CREATE TEMPORARY ...
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1815 1816 1817 1818 1819 1820 1821
*/

TABLE *open_temporary_table(THD *thd, const char *path, const char *db,
			    const char *table_name, bool link_in_list)
{
  TABLE *tmp_table;
  DBUG_ENTER("open_temporary_table");
1822

1823 1824 1825 1826 1827 1828 1829
  /*
    The extra size in my_malloc() is for table_cache_key
    4 bytes for master thread id if we are in the slave
    1 byte to terminate db
    1 byte to terminate table_name
    total of 6 extra bytes in my_malloc in addition to table/db stuff
  */
1830
  if (!(tmp_table=(TABLE*) my_malloc(sizeof(*tmp_table)+(uint) strlen(db)+
guilhem@mysql.com's avatar
guilhem@mysql.com committed
1831
				     (uint) strlen(table_name)+6+4,
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1832 1833 1834 1835
				     MYF(MY_WME))))
    DBUG_RETURN(0);				/* purecov: inspected */

  if (openfrm(path, table_name,
1836
	      (uint) (HA_OPEN_KEYFILE | HA_OPEN_RNDFILE | HA_GET_INDEX),
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1837
	      READ_KEYINFO | COMPUTE_TYPES | EXTRA_RECORD,
1838
	      ha_open_options,
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1839 1840
	      tmp_table))
  {
1841
    my_free((char*) tmp_table,MYF(0));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1842 1843 1844
    DBUG_RETURN(0);
  }

1845
  tmp_table->reginfo.lock_type=TL_WRITE;	 // Simulate locked
1846
  tmp_table->in_use= thd;
1847 1848
  tmp_table->tmp_table = (tmp_table->file->has_transactions() ? 
			  TRANSACTIONAL_TMP_TABLE : TMP_TABLE);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1849
  tmp_table->table_cache_key=(char*) (tmp_table+1);
1850 1851 1852
  tmp_table->key_length= (uint) (strmov((tmp_table->real_name=
					 strmov(tmp_table->table_cache_key,db)
					 +1), table_name)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1853
				 - tmp_table->table_cache_key)+1;
guilhem@mysql.com's avatar
guilhem@mysql.com committed
1854 1855 1856
  int4store(tmp_table->table_cache_key + tmp_table->key_length,
	    thd->server_id);
  tmp_table->key_length += 4;
1857
  int4store(tmp_table->table_cache_key + tmp_table->key_length,
1858
	    thd->variables.pseudo_thread_id);
1859
  tmp_table->key_length += 4;
1860

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1861 1862 1863 1864
  if (link_in_list)
  {
    tmp_table->next=thd->temporary_tables;
    thd->temporary_tables=tmp_table;
1865 1866
    if (thd->slave_thread)
      slave_open_temp_tables++;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1867 1868 1869 1870 1871 1872 1873 1874
  }
  DBUG_RETURN(tmp_table);
}


bool rm_temporary_table(enum db_type base, char *path)
{
  bool error=0;
1875 1876
  DBUG_ENTER("rm_temporary_table");

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1877 1878 1879 1880 1881 1882 1883
  fn_format(path, path,"",reg_ext,4);
  unpack_filename(path,path);
  if (my_delete(path,MYF(0)))
    error=1; /* purecov: inspected */
  *fn_ext(path)='\0';				// remove extension
  handler *file=get_new_handler((TABLE*) 0, base);
  if (file && file->delete_table(path))
1884
  {
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1885
    error=1;
serg@serg.mylan's avatar
serg@serg.mylan committed
1886 1887
    sql_print_warning("Could not remove tmp table: '%s', error: %d",
                      path, my_errno);
1888
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1889
  delete file;
1890
  DBUG_RETURN(error);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1891 1892 1893 1894 1895 1896 1897 1898 1899 1900 1901
}


/*****************************************************************************
** find field in list or tables. if field is unqualifed and unique,
** return unique field
******************************************************************************/

#define WRONG_GRANT (Field*) -1

Field *find_field_in_table(THD *thd,TABLE *table,const char *name,uint length,
1902
                           bool check_grants, bool allow_rowid, 
1903
                           uint *cached_field_index_ptr)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1904
{
1905 1906
  Field **field_ptr, *field;
  uint cached_field_index= *cached_field_index_ptr;
1907

1908 1909
  /* We assume here that table->field < NO_CACHED_FIELD_INDEX = UINT_MAX */
  if (cached_field_index < table->fields &&
1910 1911
      !my_strcasecmp(system_charset_info, 
                     table->field[cached_field_index]->field_name, name))
1912
    field_ptr= table->field + cached_field_index;
1913 1914 1915
  else if (table->name_hash.records)
    field_ptr= (Field**)hash_search(&table->name_hash,(byte*) name,
                                    length);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1916 1917
  else
  {
1918
    if (!(field_ptr= table->field))
1919
      return (Field *)0;
1920
    for (; *field_ptr; ++field_ptr)
1921 1922
      if (!my_strcasecmp(system_charset_info, (*field_ptr)->field_name, name))
        break;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1923 1924
  }

1925
  if (field_ptr && *field_ptr)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1926
  {
1927 1928
    *cached_field_index_ptr= field_ptr - table->field;
    field= *field_ptr;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1929 1930 1931
  }
  else
  {
1932 1933 1934 1935
    if (!allow_rowid ||
        my_strcasecmp(system_charset_info, name, "_rowid") ||
        !(field=table->rowid_field))
      return (Field*) 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1936 1937 1938 1939 1940 1941 1942
  }

  if (thd->set_query_id)
  {
    if (field->query_id != thd->query_id)
    {
      field->query_id=thd->query_id;
1943
      table->used_fields++;
1944
      table->used_keys.intersect(field->part_of_key);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1945 1946 1947 1948
    }
    else
      thd->dupp_field=field;
  }
hf@deer.(none)'s avatar
hf@deer.(none) committed
1949
#ifndef NO_EMBEDDED_ACCESS_CHECKS
1950
  if (check_grants && check_grant_column(thd,table,name,length))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1951
    return WRONG_GRANT;
hf@deer.(none)'s avatar
hf@deer.(none) committed
1952
#endif
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1953 1954 1955
  return field;
}

1956

1957 1958 1959 1960 1961
/*
  Find field in table list.

  SYNOPSIS
    find_field_in_tables()
1962 1963
    thd			Pointer to current thread structure
    item		Field item that should be found
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
1964
    tables		Tables for scanning
1965 1966 1967 1968
    where		Table where field found will be returned via
			this parameter
    report_error	If FALSE then do not report error if item not found
			and return not_found_field
1969 1970

  RETURN VALUES
1971 1972
    0			Field is not found or field is not unique- error
			message is reported
1973
    not_found_field	Function was called with report_error == FALSE and
1974
			field was not found. no error message reported.
1975 1976
    found field
*/
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1977

1978 1979 1980
// Special Field pointer for find_field_in_tables returning
const Field *not_found_field= (Field*) 0x1;

bk@work.mysql.com's avatar
bk@work.mysql.com committed
1981
Field *
1982
find_field_in_tables(THD *thd, Item_ident *item, TABLE_LIST *tables,
1983
		     TABLE_LIST **where, bool report_error)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
1984 1985 1986 1987 1988
{
  Field *found=0;
  const char *db=item->db_name;
  const char *table_name=item->table_name;
  const char *name=item->field_name;
1989
  uint length=(uint) strlen(name);
1990 1991
  char name_buff[NAME_LEN+1];

1992

1993
  if (item->cached_table)
1994 1995 1996 1997 1998 1999 2000 2001 2002 2003
  {
    /*
      This shortcut is used by prepared statements. We assuming that 
      TABLE_LIST *tables is not changed during query execution (which 
      is true for all queries except RENAME but luckily RENAME doesn't 
      use fields...) so we can rely on reusing pointer to its member.
      With this optimisation we also miss case when addition of one more
      field makes some prepared query ambiguous and so erronous, but we 
      accept this trade off.
    */
2004 2005 2006
    found= find_field_in_table(thd, item->cached_table->table, name, length,
                               test(item->cached_table->
				    table->grant.want_privilege),
2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017
                               1, &(item->cached_field_index));

    if (found)
    {
      (*where)= tables;
      if (found == WRONG_GRANT)
        return (Field*) 0;
      return found;
    }
  }

2018 2019 2020 2021 2022 2023 2024 2025
  if (db && lower_case_table_names)
  {
    /*
      convert database to lower case for comparision.
      We can't do this in Item_field as this would change the
      'name' of the item which may be used in the select list
    */
    strmake(name_buff, db, sizeof(name_buff)-1);
monty@mysql.com's avatar
monty@mysql.com committed
2026
    my_casedn_str(files_charset_info, name_buff);
2027 2028
    db= name_buff;
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2029

Sinisa@sinisa.nasamreza.org's avatar
Sinisa@sinisa.nasamreza.org committed
2030
  if (table_name && table_name[0])
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2031 2032 2033 2034
  {						/* Qualified field */
    bool found_table=0;
    for (; tables ; tables=tables->next)
    {
2035
      if (!my_strcasecmp(table_alias_charset, tables->alias, table_name) &&
Sinisa@sinisa.nasamreza.org's avatar
Sinisa@sinisa.nasamreza.org committed
2036
	  (!db || !tables->db ||  !tables->db[0] || !strcmp(db,tables->db)))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2037 2038 2039
      {
	found_table=1;
	Field *find=find_field_in_table(thd,tables->table,name,length,
2040 2041
					test(tables->table->grant.
					     want_privilege),
2042
					1, &(item->cached_field_index));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2043 2044
	if (find)
	{
2045
	  (*where)= item->cached_table= tables;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2046
	  if (!tables->cacheable_table)
2047
	    item->cached_table= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2048 2049 2050 2051 2052 2053 2054 2055 2056 2057 2058 2059 2060 2061 2062 2063
	  if (find == WRONG_GRANT)
	    return (Field*) 0;
	  if (db || !thd->where)
	    return find;
	  if (found)
	  {
	    my_printf_error(ER_NON_UNIQ_ERROR,ER(ER_NON_UNIQ_ERROR),MYF(0),
			    item->full_name(),thd->where);
	    return (Field*) 0;
	  }
	  found=find;
	}
      }
    }
    if (found)
      return found;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2064
    if (!found_table && report_error)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2065 2066
    {
      char buff[NAME_LEN*2+1];
2067
      if (db && db[0])
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2068
      {
2069
	strxnmov(buff,sizeof(buff)-1,db,".",table_name,NullS);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2070 2071
	table_name=buff;
      }
2072 2073
      my_printf_error(ER_UNKNOWN_TABLE, ER(ER_UNKNOWN_TABLE), MYF(0),
                      table_name, thd->where);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2074 2075
    }
    else
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2076 2077 2078
      if (report_error)
	my_printf_error(ER_BAD_FIELD_ERROR,ER(ER_BAD_FIELD_ERROR),MYF(0),
			item->full_name(),thd->where);
2079 2080
      else
	return (Field*) not_found_field;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2081 2082 2083 2084 2085
    return (Field*) 0;
  }
  bool allow_rowid= tables && !tables->next;	// Only one table
  for (; tables ; tables=tables->next)
  {
2086 2087 2088 2089 2090 2091 2092 2093
    if (!tables->table)
    {
      if (report_error)
	my_printf_error(ER_BAD_FIELD_ERROR,ER(ER_BAD_FIELD_ERROR),MYF(0),
			item->full_name(),thd->where);
      return (Field*) not_found_field;
    }

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2094
    Field *field=find_field_in_table(thd,tables->table,name,length,
2095
				     test(tables->table->grant.want_privilege),
2096
				     allow_rowid, &(item->cached_field_index));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2097 2098 2099 2100
    if (field)
    {
      if (field == WRONG_GRANT)
	return (Field*) 0;
2101
      (*where)= item->cached_table= tables;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2102
      if (!tables->cacheable_table)
2103
	item->cached_table= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2104 2105
      if (found)
      {
2106
	if (!thd->where)			// Returns first found
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2107
	  break;
2108 2109
	my_printf_error(ER_NON_UNIQ_ERROR,ER(ER_NON_UNIQ_ERROR),MYF(0),
			name,thd->where);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2110 2111 2112 2113 2114 2115 2116
	return (Field*) 0;
      }
      found=field;
    }
  }
  if (found)
    return found;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2117 2118 2119
  if (report_error)
    my_printf_error(ER_BAD_FIELD_ERROR, ER(ER_BAD_FIELD_ERROR),
		    MYF(0), item->full_name(), thd->where);
2120 2121
  else
    return (Field*) not_found_field;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2122 2123 2124
  return (Field*) 0;
}

2125 2126 2127

/*
  Find Item in list of items (find_field_in_tables analog)
2128 2129 2130 2131

  TODO
    is it better return only counter?

2132 2133
  SYNOPSIS
    find_item_in_list()
2134 2135 2136
    find			Item to find
    items			List of items
    counter			To return number of found item
2137
    report_error
2138 2139 2140 2141 2142
      REPORT_ALL_ERRORS		report errors, return 0 if error
      REPORT_EXCEPT_NOT_FOUND	Do not report 'not found' error and
				return not_found_item, report other errors,
				return 0
      IGNORE_ERRORS		Do not report errors, return 0 if error
2143 2144 2145 2146
    unaliased                   Set to true if item is field which was found
                                by original field name and not by its alias
                                in item list. Set to false otherwise.

2147
  RETURN VALUES
2148 2149 2150 2151 2152
    0			Item is not found or item is not unique,
			error message is reported
    not_found_item	Function was called with
			report_error == REPORT_EXCEPT_NOT_FOUND and
			item was not found. No error message was reported
2153
                        found field
2154 2155
*/

2156 2157 2158 2159
// Special Item pointer for find_item_in_list returning
const Item **not_found_item= (const Item**) 0x1;


bk@work.mysql.com's avatar
bk@work.mysql.com committed
2160
Item **
2161
find_item_in_list(Item *find, List<Item> &items, uint *counter,
2162
                  find_item_error_report_type report_error, bool *unaliased)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2163 2164
{
  List_iterator<Item> li(items);
2165
  Item **found=0, **found_unaliased= 0, *item;
2166
  const char *db_name=0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2167 2168
  const char *field_name=0;
  const char *table_name=0;
2169 2170
  bool found_unaliased_non_uniq= 0;
  uint unaliased_counter;
2171 2172 2173

  *unaliased= FALSE;

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2174 2175 2176 2177
  if (find->type() == Item::FIELD_ITEM	|| find->type() == Item::REF_ITEM)
  {
    field_name= ((Item_ident*) find)->field_name;
    table_name= ((Item_ident*) find)->table_name;
2178
    db_name=    ((Item_ident*) find)->db_name;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2179 2180
  }

2181
  for (uint i= 0; (item=li++); i++)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2182 2183 2184
  {
    if (field_name && item->type() == Item::FIELD_ITEM)
    {
2185
      Item_field *item_field= (Item_field*) item;
2186

2187 2188 2189 2190 2191 2192
      /*
	In case of group_concat() with ORDER BY condition in the QUERY
	item_field can be field of temporary table without item name 
	(if this field created from expression argument of group_concat()),
	=> we have to check presence of name before compare
      */ 
2193 2194 2195 2196
      if (!item_field->name)
        continue;

      if (table_name)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2197
      {
2198 2199 2200
        /*
          If table name is specified we should find field 'field_name' in
          table 'table_name'. According to SQL-standard we should ignore
2201 2202 2203 2204 2205
          aliases in this case.

          Since we should NOT prefer fields from the select list over
          other fields from the tables participating in this select in
          case of ambiguity we have to do extra check outside this function.
2206

2207
          We use strcmp for table names and database names as these may be
2208 2209
          case sensitive. In cases where they are not case sensitive, they
          are always in lower case.
2210 2211

	  item_field->field_name and item_field->table_name can be 0x0 if
2212
	  item is not fix_field()'ed yet.
2213
        */
2214 2215
        if (item_field->field_name && item_field->table_name &&
	    !my_strcasecmp(system_charset_info, item_field->field_name,
2216 2217 2218 2219 2220
                           field_name) &&
            !strcmp(item_field->table_name, table_name) &&
            (!db_name || (item_field->db_name &&
                          !strcmp(item_field->db_name, db_name))))
        {
2221
          if (found_unaliased)
2222
          {
2223 2224 2225 2226 2227 2228 2229
            if ((*found_unaliased)->eq(item, 0))
              continue;
            /*
              Two matching fields in select list.
              We already can bail out because we are searching through
              unaliased names only and will have duplicate error anyway.
            */
2230 2231 2232 2233 2234
            if (report_error != IGNORE_ERRORS)
              my_printf_error(ER_NON_UNIQ_ERROR, ER(ER_NON_UNIQ_ERROR),
                              MYF(0), find->full_name(), current_thd->where);
            return (Item**) 0;
          }
2235 2236
          found_unaliased= li.ref();
          unaliased_counter= i;
2237 2238
          if (db_name)
            break;                              // Perfect match
2239 2240 2241
        }
      }
      else if (!my_strcasecmp(system_charset_info, item_field->name,
2242
                              field_name))
2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259 2260 2261 2262 2263 2264 2265 2266 2267 2268 2269 2270 2271 2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282
      {
        /*
          If table name was not given we should scan through aliases
          (or non-aliased fields) first. We are also checking unaliased
          name of the field in then next else-if, to be able to find
          instantly field (hidden by alias) if no suitable alias (or
          non-aliased field) was found.
        */
        if (found)
        {
          if ((*found)->eq(item, 0))
            continue;                           // Same field twice
          if (report_error != IGNORE_ERRORS)
            my_printf_error(ER_NON_UNIQ_ERROR, ER(ER_NON_UNIQ_ERROR),
                            MYF(0), find->full_name(), current_thd->where);
          return (Item**) 0;
        }
        found= li.ref();
        *counter= i;
      }
      else if (!my_strcasecmp(system_charset_info, item_field->field_name,
                              field_name))
      {
        /*
          We will use un-aliased field or react on such ambiguities only if
          we won't be able to find aliased field.
          Again if we have ambiguity with field outside of select list
          we should prefer fields from select list.
        */
        if (found_unaliased)
        {
          if ((*found_unaliased)->eq(item, 0))
            continue;                           // Same field twice
          found_unaliased_non_uniq= 1;
        }
        else
        {
          found_unaliased= li.ref();
          unaliased_counter= i;
        }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2283 2284
      }
    }
2285
    else if (!table_name && (item->eq(find,0) ||
2286
			     find->name && item->name &&
2287 2288
			     !my_strcasecmp(system_charset_info, 
					    item->name,find->name)))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2289
    {
2290 2291
      found= li.ref();
      *counter= i;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2292 2293 2294
      break;
    }
  }
2295 2296 2297 2298 2299 2300 2301 2302 2303 2304 2305 2306 2307
  if (!found)
  {
    if (found_unaliased_non_uniq)
    {
      if (report_error != IGNORE_ERRORS)
        my_printf_error(ER_NON_UNIQ_ERROR, ER(ER_NON_UNIQ_ERROR), MYF(0),
                        find->full_name(), current_thd->where);
      return (Item **) 0;
    }
    if (found_unaliased)
    {
      found= found_unaliased;
      *counter= unaliased_counter;
2308
      *unaliased= TRUE;
2309 2310
    }
  }
2311 2312
  if (found)
    return found;
2313
  if (report_error != REPORT_EXCEPT_NOT_FOUND)
2314 2315 2316 2317 2318 2319 2320 2321
  {
    if (report_error == REPORT_ALL_ERRORS)
      my_printf_error(ER_BAD_FIELD_ERROR, ER(ER_BAD_FIELD_ERROR), MYF(0),
		      find->full_name(), current_thd->where);
    return (Item **) 0;
  }
  else
    return (Item **) not_found_item;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2322 2323 2324
}

/****************************************************************************
2325
** Expand all '*' in given fields
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2326 2327
****************************************************************************/

2328 2329 2330
int setup_wild(THD *thd, TABLE_LIST *tables, List<Item> &fields,
	       List<Item> *sum_func_list,
	       uint wild_num)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2331
{
2332
  DBUG_ENTER("setup_wild");
2333
  if (!wild_num)
2334 2335
    DBUG_RETURN(0);

2336 2337 2338
  reg2 Item *item;
  List_iterator<Item> it(fields);
  Item_arena *arena, backup;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2339 2340 2341 2342
  /*
    If we are in preparing prepared statement phase then we have change
    temporary mem_root to statement mem root to save changes of SELECT list
  */
2343
  arena= thd->change_arena_if_needed(&backup);
2344

2345
  while (wild_num && (item= it++))
2346
  {    
2347 2348
    if (item->type() == Item::FIELD_ITEM &&
        ((Item_field*) item)->field_name &&
2349 2350
	((Item_field*) item)->field_name[0] == '*' &&
	!((Item_field*) item)->field)
2351
    {
2352
      uint elem= fields.elements;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2353 2354 2355 2356 2357 2358 2359 2360 2361 2362 2363 2364 2365
      Item_subselect *subsel= thd->lex->current_select->master_unit()->item;
      if (subsel &&
          subsel->substype() == Item_subselect::EXISTS_SUBS)
      {
        /*
          It is EXISTS(SELECT * ...) and we can replace * by any constant.

          Item_int do not need fix_fields() because it is basic constant.
        */
        it.replace(new Item_int("Not_used", (longlong) 1, 21));
      }
      else if (insert_fields(thd,tables,((Item_field*) item)->db_name,
                             ((Item_field*) item)->table_name, &it))
2366
      {
2367
        if (arena)
2368
	  thd->restore_backup_item_arena(arena, &backup);
2369
	DBUG_RETURN(-1);
2370
      }
2371
      if (sum_func_list)
2372 2373 2374 2375 2376 2377 2378 2379
      {
	/*
	  sum_func_list is a list that has the fields list as a tail.
	  Because of this we have to update the element count also for this
	  list after expanding the '*' entry.
	*/
	sum_func_list->elements+= fields.elements - elem;
      }
2380
      wild_num--;
2381 2382
    }
  }
2383
  if (arena)
2384 2385
    thd->restore_backup_item_arena(arena, &backup);
  DBUG_RETURN(0);
2386 2387
}

2388 2389 2390 2391 2392 2393 2394 2395 2396 2397 2398 2399 2400 2401 2402 2403
/****************************************************************************
** Check that all given fields exists and fill struct with current data
****************************************************************************/

int setup_fields(THD *thd, Item **ref_pointer_array, TABLE_LIST *tables, 
		 List<Item> &fields, bool set_query_id,
		 List<Item> *sum_func_list, bool allow_sum_func)
{
  reg2 Item *item;
  List_iterator<Item> it(fields);
  DBUG_ENTER("setup_fields");

  thd->set_query_id=set_query_id;
  thd->allow_sum_func= allow_sum_func;
  thd->where="field list";

2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417
  /*
    To prevent fail on forward lookup we fill it with zerows,
    then if we got pointer on zero after find_item_in_list we will know
    that it is forward lookup.

    There is other way to solve problem: fill array with pointers to list,
    but it will be slower.

    TODO: remove it when (if) we made one list for allfields and
    ref_pointer_array
  */
  if (ref_pointer_array)
    bzero(ref_pointer_array, sizeof(Item *) * fields.elements);

2418 2419
  Item **ref= ref_pointer_array;
  while ((item= it++))
2420
  {
2421
    if (!item->fixed && item->fix_fields(thd, tables, it.ref()) ||
2422
	(item= *(it.ref()))->check_cols(1))
2423
      DBUG_RETURN(-1); /* purecov: inspected */
2424 2425
    if (ref)
      *(ref++)= item;
2426 2427
    if (item->with_sum_func && item->type() != Item::SUM_FUNC_ITEM &&
	sum_func_list)
2428
      item->split_sum_func(thd, ref_pointer_array, *sum_func_list);
2429 2430
    thd->used_tables|=item->used_tables();
  }
2431
  DBUG_RETURN(test(thd->net.report_error));
2432
}
2433

2434

2435
/*
2436
  prepare tables
2437

2438 2439
  SYNOPSIS
    setup_tables()
2440
    tables	table list
2441 2442


2443 2444 2445 2446 2447 2448 2449
 NOTE
   Remap table numbers if INSERT ... SELECT
   Check also that the 'used keys' and 'ignored keys' exists and set up the
   table structure accordingly

   This has to be called for all tables that are used by items, as otherwise
   table->map is not set and all Item_field will be regarded as const items.
2450

2451 2452 2453
 RETURN
   0	ok;  In this case *map will includes the choosed index
   1	error
2454 2455
*/

2456
bool setup_tables(TABLE_LIST *tables)
2457 2458
{
  DBUG_ENTER("setup_tables");
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2459
  uint tablenr=0;
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
2460 2461
  for (TABLE_LIST *table_list=tables ; table_list ;
       table_list=table_list->next,tablenr++)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2462
  {
2463 2464
    TABLE *table= table_list->table;
    setup_table_map(table, table_list, tablenr);
2465
    table->used_keys= table->keys_for_keyread;
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
2466
    if (table_list->use_index)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2467
    {
2468 2469 2470
      key_map map;
      get_key_map_from_key_list(&map, table, table_list->use_index);
      if (map.is_set_all())
2471
	DBUG_RETURN(1);
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
2472
      table->keys_in_use_for_query=map;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2473
    }
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
2474
    if (table_list->ignore_index)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2475
    {
2476 2477 2478
      key_map map;
      get_key_map_from_key_list(&map, table, table_list->ignore_index);
      if (map.is_set_all())
2479
	DBUG_RETURN(1);
2480
      table->keys_in_use_for_query.subtract(map);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2481
    }
2482
    table->used_keys.intersect(table->keys_in_use_for_query);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2483 2484 2485 2486
  }
  if (tablenr > MAX_TABLES)
  {
    my_error(ER_TOO_MANY_TABLES,MYF(0),MAX_TABLES);
2487
    DBUG_RETURN(1);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2488
  }
2489
  DBUG_RETURN(0);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2490
}
2491

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2492

2493 2494 2495 2496 2497 2498 2499 2500 2501 2502 2503 2504 2505 2506 2507
/*
   Create a key_map from a list of index names

   SYNOPSIS
     get_key_map_from_key_list()
     map		key_map to fill in
     table		Table
     index_list		List of index names

   RETURN
     0	ok;  In this case *map will includes the choosed index
     1	error
*/

bool get_key_map_from_key_list(key_map *map, TABLE *table,
2508
                               List<String> *index_list)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2509
{
monty@tik.mysql.fi's avatar
monty@tik.mysql.fi committed
2510
  List_iterator_fast<String> it(*index_list);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2511 2512
  String *name;
  uint pos;
2513 2514

  map->clear_all();
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2515 2516
  while ((name=it++))
  {
2517 2518
    if ((pos= find_type(&table->keynames, name->ptr(), name->length(), 1)) <=
	0)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2519 2520 2521
    {
      my_error(ER_KEY_COLUMN_DOES_NOT_EXITS, MYF(0), name->c_ptr(),
	       table->real_name);
2522
      map->set_all();
2523
      return 1;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2524
    }
2525
    map->set_bit(pos-1);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2526
  }
2527
  return 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2528 2529
}

2530

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2531
/****************************************************************************
2532 2533
  This just drops in all fields instead of current '*' field
  Returns pointer to last inserted field if ok
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2534 2535
****************************************************************************/

sasha@mysql.sashanet.com's avatar
sasha@mysql.sashanet.com committed
2536
bool
2537 2538
insert_fields(THD *thd,TABLE_LIST *tables, const char *db_name,
	      const char *table_name, List_iterator<Item> *it)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2539
{
2540
  char name_buff[NAME_LEN+1];
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2541 2542 2543
  uint found;
  DBUG_ENTER("insert_fields");

2544 2545
  if (db_name && lower_case_table_names)
  {
monty@mysql.com's avatar
monty@mysql.com committed
2546 2547 2548 2549 2550
    /*
      convert database to lower case for comparison
      We can't do this in Item_field as this would change the
      'name' of the item which may be used in the select list
    */
serg@serg.mylan's avatar
serg@serg.mylan committed
2551
    strmake(name_buff, db_name, sizeof(name_buff)-1);
monty@mysql.com's avatar
monty@mysql.com committed
2552
    my_casedn_str(files_charset_info, name_buff);
monty@mysql.com's avatar
monty@mysql.com committed
2553
    db_name= name_buff;
2554 2555 2556
  }


bk@work.mysql.com's avatar
bk@work.mysql.com committed
2557
  found=0;
2558
  for (; tables ; tables=tables->next)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2559
  {
2560
    TABLE *table=tables->table;
2561 2562
    if (!table_name || (!my_strcasecmp(table_alias_charset, table_name,
				       tables->alias) &&
2563
			(!db_name || !strcmp(tables->db,db_name))))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2564
    {
hf@deer.(none)'s avatar
hf@deer.(none) committed
2565
#ifndef NO_EMBEDDED_ACCESS_CHECKS
2566
      /* Ensure that we have access right to all columns */
2567
      if (!(table->grant.privilege & SELECT_ACL) &&
2568
	  check_grant_all_columns(thd,SELECT_ACL,table))
2569
	DBUG_RETURN(-1);
hf@deer.(none)'s avatar
hf@deer.(none) committed
2570
#endif
2571
      Field **ptr=table->field,*field;
monty@mysql.com's avatar
monty@mysql.com committed
2572 2573
      TABLE *natural_join_table= 0;

2574
      thd->used_tables|=table->map;
monty@mysql.com's avatar
monty@mysql.com committed
2575 2576 2577
      if (!table->outer_join &&
          tables->natural_join &&
          !tables->natural_join->table->outer_join)
ram@gw.mysql.r18.ru's avatar
ram@gw.mysql.r18.ru committed
2578
        natural_join_table= tables->natural_join->table;
monty@mysql.com's avatar
monty@mysql.com committed
2579

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2580 2581
      while ((field = *ptr++))
      {
2582
        uint not_used_field_index= NO_CACHED_FIELD_INDEX;
2583
        /* Skip duplicate field names if NATURAL JOIN is used */
ram@gw.mysql.r18.ru's avatar
ram@gw.mysql.r18.ru committed
2584 2585
        if (!natural_join_table ||
            !find_field_in_table(thd, natural_join_table, field->field_name, 
2586 2587
                                 strlen(field->field_name), 0, 0,
                                 &not_used_field_index))
2588
        {
2589
          Item_field *item= new Item_field(thd, field);
2590 2591 2592 2593 2594
          if (!found++)
            (void) it->replace(item);		// Replace '*'
          else
            it->after(item);
        }
2595 2596 2597 2598
	/*
	  Mark if field used before in this select.
	  Used by 'insert' to verify if a field name is used twice
	*/
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2599 2600 2601
	if (field->query_id == thd->query_id)
	  thd->dupp_field=field;
	field->query_id=thd->query_id;
2602
	table->used_keys.intersect(field->part_of_key);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2603 2604
      }
      /* All fields are used */
2605
      table->used_fields=table->fields;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2606 2607 2608 2609 2610 2611 2612 2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624
    }
  }
  if (!found)
  {
    if (!table_name)
      my_error(ER_NO_TABLES_USED,MYF(0));
    else
      my_error(ER_BAD_TABLE_ERROR,MYF(0),table_name);
  }
  DBUG_RETURN(!found);
}


/*
** Fix all conditions and outer join expressions
*/

int setup_conds(THD *thd,TABLE_LIST *tables,COND **conds)
{
2625
  table_map not_null_tables= 0;
2626
  Item_arena *arena= 0, backup;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2627
  DBUG_ENTER("setup_conds");
2628

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2629
  thd->set_query_id=1;
2630
  thd->lex->current_select->cond_count= 0;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2631 2632 2633
  if (*conds)
  {
    thd->where="where clause";
2634 2635
    if (!(*conds)->fixed && (*conds)->fix_fields(thd, tables, conds) ||
	(*conds)->check_cols(1))
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2636
      DBUG_RETURN(1);
2637
    not_null_tables= (*conds)->not_null_tables();
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2638 2639
  }

2640

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2641 2642 2643
  /* Check if we are using outer joins */
  for (TABLE_LIST *table=tables ; table ; table=table->next)
  {
2644 2645 2646 2647
    if (table->on_expr)
    {
      /* Make a join an a expression */
      thd->where="on clause";
2648 2649 2650
      
      if (!table->on_expr->fixed &&
	  table->on_expr->fix_fields(thd, tables, &table->on_expr) ||
2651
	  table->on_expr->check_cols(1))
2652
	DBUG_RETURN(1);
2653
      thd->lex->current_select->cond_count++;
2654

2655 2656 2657 2658 2659 2660 2661
      /*
	If it's a normal join or a LEFT JOIN which can be optimized away
	add the ON/USING expression to the WHERE
      */
      if (!table->outer_join ||
	  ((table->table->map & not_null_tables) &&
	   !(specialflag & SPECIAL_NO_NEW_FUNC)))
2662
      {
2663
	table->outer_join= 0;
2664
        arena= thd->change_arena_if_needed(&backup);
2665
	*conds= and_conds(*conds, table->on_expr);
2666
	table->on_expr=0;
2667 2668
	if (arena)
        {
2669
	  thd->restore_backup_item_arena(arena, &backup);
2670 2671
          arena= 0;                             // Safety if goto err
        }
2672 2673 2674
	if ((*conds) && !(*conds)->fixed &&
	    (*conds)->fix_fields(thd, tables, conds))
	  DBUG_RETURN(1);
2675 2676
      }
    }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2677 2678
    if (table->natural_join)
    {
2679
      arena= thd->change_arena_if_needed(&backup);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2680
      /* Make a join of all fields with have the same name */
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2681 2682 2683
      TABLE *t1= table->table;
      TABLE *t2= table->natural_join->table;
      Item_cond_and *cond_and= new Item_cond_and();
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2684
      if (!cond_and)				// If not out of memory
2685
	goto err;
2686
      cond_and->top_level_item();
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2687

2688 2689
      Field **t1_field, *t2_field;
      for (t1_field= t1->field; (*t1_field); t1_field++)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2690
      {
2691
        const char *t1_field_name= (*t1_field)->field_name;
2692
        uint not_used_field_index= NO_CACHED_FIELD_INDEX;
2693 2694

        if ((t2_field= find_field_in_table(thd, t2, t1_field_name,
2695 2696
                                           strlen(t1_field_name), 0, 0,
                                           &not_used_field_index)))
2697
        {
2698 2699
          Item_func_eq *tmp=new Item_func_eq(new Item_field(thd, *t1_field),
                                             new Item_field(thd, t2_field));
2700
          if (!tmp)
2701
            goto err;
2702 2703 2704 2705 2706 2707
          /* Mark field used for table cache */
          (*t1_field)->query_id= t2_field->query_id= thd->query_id;
          cond_and->list.push_back(tmp);
          t1->used_keys.intersect((*t1_field)->part_of_key);
          t2->used_keys.intersect(t2_field->part_of_key);
        }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2708
      }
2709
      thd->lex->current_select->cond_count+= cond_and->list.elements;
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2710

2711 2712 2713
      // to prevent natural join processing during PS re-execution
      table->natural_join= 0;

2714
      if (cond_and->list.elements)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2715
      {
2716 2717 2718 2719
        if (!table->outer_join)			// Not left join
        {
          *conds= and_conds(*conds, cond_and);
          // fix_fields() should be made with temporary memory pool
2720
          if (arena)
2721
            thd->restore_backup_item_arena(arena, &backup);
2722 2723
          if (*conds && !(*conds)->fixed)
          {
2724 2725
            if (!(*conds)->fixed && 
                (*conds)->fix_fields(thd, tables, conds))
2726 2727 2728 2729 2730 2731 2732
              DBUG_RETURN(1);
          }
        }
        else
        {
          table->on_expr= and_conds(table->on_expr, cond_and);
          // fix_fields() should be made with temporary memory pool
2733
          if (arena)
2734
            thd->restore_backup_item_arena(arena, &backup);
2735 2736
          if (table->on_expr && !table->on_expr->fixed)
          {
2737 2738
            if (!table->on_expr->fixed && 
                table->on_expr->fix_fields(thd, tables, &table->on_expr))
2739 2740 2741
             DBUG_RETURN(1);
          }
        }
bell@sanja.is.com.ua's avatar
bell@sanja.is.com.ua committed
2742
      }
2743 2744
      else if (arena)
      {
2745
        thd->restore_backup_item_arena(arena, &backup);
2746 2747
        arena= 0;                               // Safety if goto err
      }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2748 2749
    }
  }
2750

2751
  if (thd->current_arena->is_stmt_prepare())
2752 2753 2754 2755 2756 2757 2758 2759 2760
  {
    /*
      We are in prepared statement preparation code => we should store
      WHERE clause changing for next executions.

      We do this ON -> WHERE transformation only once per PS statement.
    */
    thd->lex->current_select->where= *conds;
  }
2761
  DBUG_RETURN(test(thd->net.report_error));
2762 2763

err:
2764 2765
  if (arena)
    thd->restore_backup_item_arena(arena, &backup);
2766
  DBUG_RETURN(1);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2767 2768 2769 2770 2771 2772 2773 2774 2775
}


/******************************************************************************
** Fill a record with data (for INSERT or UPDATE)
** Returns : 1 if some field has wrong type
******************************************************************************/

int
2776
fill_record(List<Item> &fields,List<Item> &values, bool ignore_errors)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2777
{
monty@tik.mysql.fi's avatar
monty@tik.mysql.fi committed
2778
  List_iterator_fast<Item> f(fields),v(values);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2779 2780 2781 2782 2783 2784 2785
  Item *value;
  Item_field *field;
  DBUG_ENTER("fill_record");

  while ((field=(Item_field*) f++))
  {
    value=v++;
2786 2787
    Field *rfield= field->field;
    TABLE *table= rfield->table;
2788
    if (rfield == table->next_number_field)
2789
      table->auto_increment_field_not_null= TRUE;
2790
    if ((value->save_in_field(rfield, 0) < 0) && !ignore_errors)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2791 2792 2793 2794 2795 2796 2797
      DBUG_RETURN(1);
  }
  DBUG_RETURN(0);
}


int
2798
fill_record(Field **ptr,List<Item> &values, bool ignore_errors)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2799
{
monty@tik.mysql.fi's avatar
monty@tik.mysql.fi committed
2800
  List_iterator_fast<Item> v(values);
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2801 2802 2803 2804 2805 2806 2807
  Item *value;
  DBUG_ENTER("fill_record");

  Field *field;
  while ((field = *ptr++))
  {
    value=v++;
2808
    TABLE *table= field->table;
2809
    if (field == table->next_number_field)
2810
      table->auto_increment_field_not_null= TRUE;
2811
    if ((value->save_in_field(field, 0) < 0) && !ignore_errors)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2812 2813 2814 2815 2816 2817 2818 2819
      DBUG_RETURN(1);
  }
  DBUG_RETURN(0);
}


static void mysql_rm_tmp_tables(void)
{
2820 2821
  uint i, idx;
  char	filePath[FN_REFLEN], *tmpdir;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2822 2823 2824 2825
  MY_DIR *dirp;
  FILEINFO *file;
  DBUG_ENTER("mysql_rm_tmp_tables");

2826 2827 2828
  for (i=0; i<=mysql_tmpdir_list.max; i++)
  {
    tmpdir=mysql_tmpdir_list.list[i];
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2829
  /* See if the directory exists */
2830 2831
    if (!(dirp = my_dir(tmpdir,MYF(MY_WME | MY_DONT_SORT))))
      continue;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2832

2833
    /* Remove all SQLxxx tables from directory */
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2834

2835
  for (idx=0 ; idx < (uint) dirp->number_off_files ; idx++)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2836 2837
  {
    file=dirp->dir_entry+idx;
2838 2839 2840 2841 2842 2843

    /* skiping . and .. */
    if (file->name[0] == '.' && (!file->name[1] ||
       (file->name[1] == '.' &&  !file->name[2])))
      continue;

bk@work.mysql.com's avatar
bk@work.mysql.com committed
2844 2845
    if (!bcmp(file->name,tmp_file_prefix,tmp_file_prefix_length))
    {
2846 2847
        sprintf(filePath,"%s%s",tmpdir,file->name);
        VOID(my_delete(filePath,MYF(MY_WME)));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2848 2849 2850
    }
  }
  my_dirend(dirp);
2851
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2852 2853 2854 2855 2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866
  DBUG_VOID_RETURN;
}



/*****************************************************************************
	unireg support functions
*****************************************************************************/

/*
** Invalidate any cache entries that are for some DB
** We can't use hash_delete when looping hash_elements. We mark them first
** and afterwards delete those marked unused.
*/

2867
void remove_db_from_cache(const char *db)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2868
{
2869 2870 2871 2872 2873 2874 2875 2876 2877 2878
  char name_buff[NAME_LEN+1];
  if (db && lower_case_table_names)
  {
    /*
      convert database to lower case for comparision.
    */
    strmake(name_buff, db, sizeof(name_buff)-1);
    my_casedn_str(files_charset_info, name_buff);
    db= name_buff;
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907
  for (uint idx=0 ; idx < open_cache.records ; idx++)
  {
    TABLE *table=(TABLE*) hash_element(&open_cache,idx);
    if (!strcmp(table->table_cache_key,db))
    {
      table->version=0L;			/* Free when thread is ready */
      if (!table->in_use)
	relink_unused(table);
    }
  }
  while (unused_tables && !unused_tables->version)
    VOID(hash_delete(&open_cache,(byte*) unused_tables));
}


/*
** free all unused tables
*/

void flush_tables()
{
  (void) pthread_mutex_lock(&LOCK_open);
  while (unused_tables)
    hash_delete(&open_cache,(byte*) unused_tables);
  (void) pthread_mutex_unlock(&LOCK_open);
}


/*
2908 2909 2910 2911 2912 2913 2914 2915 2916
  Mark all entries with the table as deleted to force an reopen of the table

  The table will be closed (not stored in cache) by the current thread when
  close_thread_tables() is called.

  RETURN
    0  This thread now have exclusive access to this table and no other thread
       can access the table until close_thread_tables() is called.
    1  Table is in use by another thread
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2917 2918
*/

2919 2920
bool remove_table_from_cache(THD *thd, const char *db, const char *table_name,
			     bool return_if_owned_by_thd)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2921 2922 2923 2924 2925 2926 2927 2928 2929 2930 2931 2932
{
  char key[MAX_DBKEY_LENGTH];
  uint key_length;
  TABLE *table;
  bool result=0;
  DBUG_ENTER("remove_table_from_cache");

  key_length=(uint) (strmov(strmov(key,db)+1,table_name)-key)+1;
  for (table=(TABLE*) hash_search(&open_cache,(byte*) key,key_length) ;
       table;
       table = (TABLE*) hash_next(&open_cache,(byte*) key,key_length))
  {
2933
    THD *in_use;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2934
    table->version=0L;			/* Free when thread is ready */
2935
    if (!(in_use=table->in_use))
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
2936 2937
    {
      DBUG_PRINT("info",("Table was not in use"));
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2938
      relink_unused(table);
monty@hundin.mysql.fi's avatar
monty@hundin.mysql.fi committed
2939
    }
2940
    else if (in_use != thd)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2941 2942 2943 2944 2945
    {
      in_use->some_tables_deleted=1;
      if (table->db_stat)
	result=1;
      /* Kill delayed insert threads */
guilhem@mysql.com's avatar
guilhem@mysql.com committed
2946 2947
      if ((in_use->system_thread & SYSTEM_THREAD_DELAYED_INSERT) &&
          ! in_use->killed)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2948 2949 2950
      {
	in_use->killed=1;
	pthread_mutex_lock(&in_use->mysys_var->mutex);
2951
	if (in_use->mysys_var->current_cond)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2952 2953 2954 2955 2956 2957 2958
	{
	  pthread_mutex_lock(in_use->mysys_var->current_mutex);
	  pthread_cond_broadcast(in_use->mysys_var->current_cond);
	  pthread_mutex_unlock(in_use->mysys_var->current_mutex);
	}
	pthread_mutex_unlock(&in_use->mysys_var->mutex);
      }
2959 2960 2961 2962 2963 2964 2965 2966 2967 2968 2969
      /*
	Now we must abort all tables locks used by this thread
	as the thread may be waiting to get a lock for another table
      */
      for (TABLE *thd_table= in_use->open_tables;
	   thd_table ;
	   thd_table= thd_table->next)
      {
	if (thd_table->db_stat)			// If table is open
	  mysql_lock_abort_for_thread(thd, thd_table);
      }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2970
    }
2971 2972
    else
      result= result || return_if_owned_by_thd;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2973 2974 2975 2976 2977 2978
  }
  while (unused_tables && !unused_tables->version)
    VOID(hash_delete(&open_cache,(byte*) unused_tables));
  DBUG_RETURN(result);
}

2979
int setup_ftfuncs(SELECT_LEX *select_lex)
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2980
{
2981 2982
  List_iterator<Item_func_match> li(*(select_lex->ftfunc_list)),
                                 lj(*(select_lex->ftfunc_list));
2983
  Item_func_match *ftf, *ftf2;
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2984 2985

  while ((ftf=li++))
2986
  {
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2987 2988
    if (ftf->fix_index())
      return 1;
2989 2990
    lj.rewind();
    while ((ftf2=lj++) != ftf)
2991
    {
2992
      if (ftf->eq(ftf2,1) && !ftf2->master)
2993 2994 2995
        ftf2->master=ftf;
    }
  }
bk@work.mysql.com's avatar
bk@work.mysql.com committed
2996 2997 2998

  return 0;
}
2999

3000

3001
int init_ftfuncs(THD *thd, SELECT_LEX *select_lex, bool no_order)
3002
{
3003
  if (select_lex->ftfunc_list->elements)
3004
  {
3005
    List_iterator<Item_func_match> li(*(select_lex->ftfunc_list));
3006 3007 3008
    Item_func_match *ifm;
    DBUG_PRINT("info",("Performing FULLTEXT search"));
    thd->proc_info="FULLTEXT initialization";
3009

3010 3011 3012
    while ((ifm=li++))
      ifm->init_search(no_order);
  }
3013 3014
  return 0;
}