sql_base.cc 80.1 KB
Newer Older
unknown's avatar
unknown committed
1
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
2

unknown's avatar
unknown 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

unknown's avatar
unknown 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

unknown's avatar
unknown 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 */
unknown's avatar
unknown committed
19 20 21

#include "mysql_priv.h"
#include "sql_acl.h"
22
#include "sql_select.h"
unknown's avatar
unknown committed
23 24 25 26 27 28 29 30 31 32
#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 */
unknown's avatar
unknown committed
33
HASH assign_cache;
unknown's avatar
unknown committed
34

35
static int open_unireg_entry(THD *thd,TABLE *entry,const char *db,
unknown's avatar
unknown committed
36
			     const char *name, const char *alias);
unknown's avatar
unknown committed
37 38 39 40
static void free_cache_entry(TABLE *entry);
static void mysql_rm_tmp_tables(void);


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

49
bool table_cache_init(void)
unknown's avatar
unknown committed
50 51
{
  mysql_rm_tmp_tables();
unknown's avatar
unknown committed
52
  return hash_init(&open_cache, &my_charset_bin, table_cache_size+16,
unknown's avatar
unknown committed
53
		   0, 0,table_cache_key,
unknown's avatar
unknown committed
54
		   (hash_free_key) free_cache_entry, 0) != 0;
unknown's avatar
unknown committed
55 56 57 58 59
}

void table_cache_free(void)
{
  DBUG_ENTER("table_cache_free");
unknown's avatar
unknown committed
60
  close_cached_tables((THD*) 0,0,(TABLE_LIST*) 0);
unknown's avatar
unknown committed
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 100
  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)
  {
101
    DBUG_PRINT("error",("Unused_links doesn't match open_cache: diff: %d", /* purecov: inspected */
unknown's avatar
unknown committed
102 103 104 105 106 107 108
			count)); /* purecov: inspected */
  }
}
#else
#define check_unused()
#endif

unknown's avatar
unknown committed
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
/*
  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.
*/

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

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

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

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

155 156 157 158
    /* 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;
unknown's avatar
unknown committed
159

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

/*****************************************************************************
 *	 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
}

208 209 210 211 212 213 214 215 216 217
/*
  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
*/
unknown's avatar
unknown committed
218 219 220 221

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

  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;
}

241
/* Free resources allocated by filesort() and read_record() */
unknown's avatar
unknown committed
242 243 244

void free_io_cache(TABLE *table)
{
245
  DBUG_ENTER("free_io_cache");
unknown's avatar
unknown committed
246
  if (table->sort.io_cache)
unknown's avatar
unknown committed
247
  {
unknown's avatar
unknown committed
248 249 250
    close_cached_file(table->sort.io_cache);
    my_free((gptr) table->sort.io_cache,MYF(0));
    table->sort.io_cache=0;
unknown's avatar
unknown committed
251
  }
252
  DBUG_VOID_RETURN;
unknown's avatar
unknown committed
253 254 255 256
}

	/* Close all tables which aren't in use by any thread */

unknown's avatar
unknown committed
257 258
bool close_cached_tables(THD *thd, bool if_wait_for_refresh,
			 TABLE_LIST *tables)
unknown's avatar
unknown committed
259 260 261 262 263
{
  bool result=0;
  DBUG_ENTER("close_cached_tables");

  VOID(pthread_mutex_lock(&LOCK_open));
unknown's avatar
unknown committed
264
  if (!tables)
unknown's avatar
unknown committed
265
  {
unknown's avatar
unknown committed
266 267
    while (unused_tables)
    {
unknown's avatar
unknown committed
268
#ifdef EXTRA_DEBUG
unknown's avatar
unknown committed
269 270
      if (hash_delete(&open_cache,(byte*) unused_tables))
	printf("Warning: Couldn't delete open table from hash\n");
unknown's avatar
unknown committed
271
#else
unknown's avatar
unknown committed
272
      VOID(hash_delete(&open_cache,(byte*) unused_tables));
unknown's avatar
unknown committed
273
#endif
unknown's avatar
unknown committed
274 275
    }
    refresh_version++;				// Force close of open tables
unknown's avatar
unknown committed
276
  }
unknown's avatar
unknown committed
277
  else
unknown's avatar
unknown committed
278
  {
unknown's avatar
unknown committed
279 280 281
    bool found=0;
    for (TABLE_LIST *table=tables ; table ; table=table->next)
    {
282
      if (remove_table_from_cache(thd, table->db, table->real_name, 1))
unknown's avatar
unknown committed
283 284 285 286
	found=1;
    }
    if (!found)
      if_wait_for_refresh=0;			// Nothing to wait for
unknown's avatar
unknown committed
287
  }
288
#ifndef EMBEDDED_LIBRARY
289 290
  if (!tables)
    kill_delayed_threads();
291
#endif
unknown's avatar
unknown committed
292 293 294 295 296 297 298 299 300 301
  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";

unknown's avatar
unknown committed
302
    close_old_data_files(thd,thd->open_tables,1,1);
303
    mysql_ha_flush(thd, tables, MYSQL_HA_REOPEN_ON_USAGE | MYSQL_HA_FLUSH_ALL);
unknown's avatar
unknown committed
304 305
    bool found=1;
    /* Wait until all threads has closed all the tables we had locked */
306 307
    DBUG_PRINT("info",
	       ("Waiting for others threads to close their open tables"));
unknown's avatar
unknown committed
308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329
    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;
330 331 332
    /* Set version for table */
    for (TABLE *table=thd->open_tables; table ; table=table->next)
      table->version=refresh_version;
unknown's avatar
unknown committed
333 334 335 336 337 338 339 340 341 342 343 344 345 346 347
  }
  VOID(pthread_mutex_unlock(&LOCK_open));
  if (if_wait_for_refresh)
  {
    THD *thd=current_thd;
    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);
}


348 349
/*
  Close all tables used by thread
unknown's avatar
unknown committed
350

351 352 353 354 355 356 357 358 359 360 361 362
  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.
*/
unknown's avatar
unknown committed
363

364
void close_thread_tables(THD *thd, bool lock_in_use, bool skip_derived)
unknown's avatar
unknown committed
365
{
366
  bool found_old_table;
unknown's avatar
unknown committed
367 368
  DBUG_ENTER("close_thread_tables");

369 370 371 372 373 374 375 376 377 378 379 380 381 382
  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;
  }
unknown's avatar
unknown committed
383
  if (thd->locked_tables)
384 385
  {
    ha_commit_stmt(thd);			// If select statement
unknown's avatar
unknown committed
386
    DBUG_VOID_RETURN;				// LOCK TABLES in use
387
  }
unknown's avatar
unknown committed
388 389 390

  if (thd->lock)
  {
391 392
    mysql_unlock_tables(thd, thd->lock);
    thd->lock=0;
unknown's avatar
unknown committed
393 394
  }
  /* VOID(pthread_sigmask(SIG_SETMASK,&thd->block_signals,NULL)); */
395
  if (!lock_in_use)
unknown's avatar
unknown committed
396
    VOID(pthread_mutex_lock(&LOCK_open));
397
  safe_mutex_assert_owner(&LOCK_open);
unknown's avatar
unknown committed
398 399

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

401
 found_old_table= 0;
402 403
  while (thd->open_tables)
    found_old_table|=close_thread_table(thd, &thd->open_tables);
404
  thd->some_tables_deleted=0;
405

unknown's avatar
unknown committed
406
  /* Free tables to hold down open files */
407
  while (open_cache.records > table_cache_size && unused_tables)
unknown's avatar
unknown committed
408 409 410 411 412 413 414
    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));
  }
415
  if (!lock_in_use)
unknown's avatar
unknown committed
416 417 418 419 420
    VOID(pthread_mutex_unlock(&LOCK_open));
  /*  VOID(pthread_sigmask(SIG_SETMASK,&thd->signals,NULL)); */
  DBUG_VOID_RETURN;
}

421 422 423 424 425 426
/* move one table to free list */

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

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

  *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
448
      table->file->reset();
449 450 451 452 453 454 455 456 457 458 459 460 461 462 463
    }
    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);
}

unknown's avatar
unknown committed
464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483
	/* 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;
484 485
  char *query, *end;
  uint query_buf_size; 
486
  bool found_user_tables = 0;
unknown's avatar
unknown committed
487

488 489 490
  if (!thd->temporary_tables)
    return;
  
491
  LINT_INIT(end);
492
  query_buf_size= 50;   // Enough for DROP ... TABLE IF EXISTS
493

494
  for (table=thd->temporary_tables ; table ; table=table->next)
495 496 497 498 499
    /*
      We are going to add 4 ` around the db/table names, so 1 does not look
      enough; indeed it is enough, because table->key_length is greater (by 8,
      because of server_id and thread_id) than db||table.
    */
unknown's avatar
unknown committed
500
    query_buf_size+= table->key_length+1;
501

unknown's avatar
unknown committed
502
  if ((query = alloc_root(thd->mem_root, query_buf_size)))
503 504
    // Better add "if exists", in case a RESET MASTER has been done
    end=strmov(query, "DROP /*!40005 TEMPORARY */ TABLE IF EXISTS ");
505

unknown's avatar
unknown committed
506 507
  for (table=thd->temporary_tables ; table ; table=next)
  {
508 509 510
    if (query) // we might be out of memory, but this is not fatal
    {
      // skip temporary tables not created directly by the user
unknown's avatar
unknown committed
511
      if (table->real_name[0] != '#')
512
	found_user_tables = 1;
513 514 515 516
      /*
        Here we assume table_cache_key always starts
        with \0 terminated db name
      */
517 518
      end = strxmov(end,"`",table->table_cache_key,"`.`",
                    table->real_name,"`,", NullS);
519
    }
unknown's avatar
unknown committed
520 521 522
    next=table->next;
    close_temporary(table);
  }
523
  if (query && found_user_tables && mysql_bin_log.is_open())
524
  {
525
    /* The -1 is to remove last ',' */
unknown's avatar
unknown committed
526
    thd->clear_error();
527
    Query_log_event qinfo(thd, query, (ulong)(end-query)-1, 0, FALSE);
528 529 530 531 532 533 534 535 536 537
    /*
      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.
      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;
538 539
    mysql_bin_log.write(&qinfo);
  }
unknown's avatar
unknown committed
540 541 542
  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;
}
unknown's avatar
unknown committed
567

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

  SYNOPSIS
unknown's avatar
unknown committed
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;
}

unknown's avatar
unknown 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;

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

unknown's avatar
unknown 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;
unknown's avatar
unknown committed
626 627 628
  return 0;
}

unknown's avatar
unknown 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,
unknown's avatar
unknown committed
637 638 639 640
			    const char *table_name)
{
  char *key;
  if (!(key=(char*) alloc_root(&table->mem_root,
unknown's avatar
unknown committed
641
			       (uint) strlen(db)+
unknown's avatar
unknown committed
642
			       (uint) strlen(table_name)+6+4)))
unknown's avatar
unknown 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;
unknown's avatar
unknown committed
648 649
  int4store(key+table->key_length,thd->server_id);
  table->key_length += 4;
unknown's avatar
unknown committed
650
  int4store(key+table->key_length,thd->variables.pseudo_thread_id);
651
  table->key_length += 4;
unknown's avatar
unknown 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
/*
unknown's avatar
unknown committed
710
   When we call the following function we must have a lock on
711
   LOCK_open ; This lock will be unlocked on return.
unknown's avatar
unknown committed
712 713 714 715
*/

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

unknown's avatar
unknown 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";
unknown's avatar
unknown committed
724 725
  if (!thd->killed)
    (void) pthread_cond_wait(&COND_refresh,&LOCK_open);
unknown's avatar
unknown 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;
unknown's avatar
unknown committed
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);
unknown's avatar
unknown committed
752
  if (open_unireg_entry(thd, table, db, table_name, table_name) ||
unknown's avatar
unknown committed
753 754
      !(table->table_cache_key =memdup_root(&table->mem_root,(char*) key,
					    key_length)))
unknown's avatar
unknown committed
755 756 757 758 759
  {
    closefrm(table);
    pthread_mutex_unlock(&LOCK_open);
    DBUG_RETURN(0);
  }
760

761
  table->key_length=key_length;
unknown's avatar
unknown committed
762 763
  table->version=0;
  table->flush_version=0;
764 765 766
  table->in_use = thd;
  check_unused();
  pthread_mutex_unlock(&LOCK_open);
unknown's avatar
unknown 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
}

unknown's avatar
unknown 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;
unknown's avatar
unknown committed
804
  int4store(key + key_length, thd->server_id);
unknown's avatar
unknown committed
805
  int4store(key + key_length + 4, thd->variables.pseudo_thread_id);
806

unknown's avatar
unknown committed
807 808
  for (table=thd->temporary_tables; table ; table=table->next)
  {
unknown's avatar
unknown 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))
unknown's avatar
unknown 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;
unknown's avatar
unknown committed
820
      table->clear_query_id=1;
unknown's avatar
unknown committed
821
      thd->tmp_table_used= 1;
unknown's avatar
unknown committed
822
      DBUG_PRINT("info",("Using temporary table"));
unknown's avatar
unknown 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) &&
unknown's avatar
unknown 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;
unknown's avatar
unknown committed
837
        DBUG_PRINT("info",("Using locked table"));
unknown's avatar
unknown committed
838
	goto reset;
839
      }
unknown's avatar
unknown committed
840 841 842 843
    }
    my_printf_error(ER_TABLE_NOT_LOCKED,ER(ER_TABLE_NOT_LOCKED),MYF(0),alias);
    DBUG_RETURN(0);
  }
unknown's avatar
unknown committed
844

unknown's avatar
unknown 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

unknown's avatar
unknown 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.
      */
unknown's avatar
unknown committed
870
      close_old_data_files(thd,thd->open_tables,0,0);
unknown's avatar
unknown 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;
unknown's avatar
unknown committed
890

unknown's avatar
unknown committed
891 892 893 894
  }
  else
  {
    /* Free cache if too big */
895
    while (open_cache.records > table_cache_size && unused_tables)
unknown's avatar
unknown 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));
unknown's avatar
unknown committed
902
      DBUG_RETURN(NULL);
903
    }
unknown's avatar
unknown committed
904
    if (open_unireg_entry(thd, table,db,table_name,alias) ||
unknown's avatar
unknown 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));
unknown's avatar
SCRUM  
unknown committed
917
    VOID(my_hash_insert(&open_cache,(byte*) table));
unknown's avatar
unknown committed
918 919 920
  }

  table->in_use=thd;
921
  check_unused();				// Debugging call
unknown's avatar
unknown committed
922
       
unknown's avatar
unknown 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))
  {
unknown's avatar
unknown committed
935
    uint length=(uint) strlen(alias)+1;
unknown's avatar
unknown 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
unknown's avatar
unknown 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;
unknown's avatar
unknown 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();
unknown's avatar
unknown committed
977
  DBUG_ASSERT(table->key_read == 0);
unknown's avatar
unknown 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);
unknown's avatar
unknown committed
1022

unknown's avatar
unknown committed
1023
  if (open_unireg_entry(current_thd,&tmp,db,table_name,table->table_name))
unknown's avatar
unknown 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 */
unknown's avatar
unknown 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;
unknown's avatar
unknown 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;
unknown's avatar
unknown committed
1052 1053
  tmp.grant=		table->grant;

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

unknown's avatar
unknown 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);
unknown's avatar
unknown committed
1065 1066
  for (field=table->field ; *field ; field++)
  {
1067
    (*field)->table= (*field)->orig_table= table;
unknown's avatar
unknown 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;
unknown's avatar
unknown 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);

unknown's avatar
unknown 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.
*/

unknown's avatar
unknown committed
1184 1185
void close_old_data_files(THD *thd, TABLE *table, bool abort_locks,
			  bool send_refresh)
unknown's avatar
unknown committed
1186
{
unknown's avatar
unknown committed
1187 1188
  DBUG_ENTER("close_old_data_files");
  bool found=send_refresh;
unknown's avatar
unknown 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
unknown's avatar
unknown committed
1211
  DBUG_VOID_RETURN;
unknown's avatar
unknown 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
*/

unknown's avatar
unknown committed
1221
bool table_is_used(TABLE *table, bool wait_for_name_lock)
unknown's avatar
unknown committed
1222 1223 1224 1225 1226
{
  do
  {
    char *key= table->table_cache_key;
    uint key_length=table->key_length;
unknown's avatar
unknown committed
1227 1228
    for (TABLE *search=(TABLE*) hash_search(&open_cache,
					    (byte*) key,key_length) ;
unknown's avatar
unknown committed
1229 1230 1231 1232
	 search ;
	 search = (TABLE*) hash_next(&open_cache,(byte*) key,key_length))
    {
      if (search->locked_by_flush ||
unknown's avatar
unknown committed
1233
	  search->locked_by_name && wait_for_name_lock ||
unknown's avatar
unknown 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);
unknown's avatar
unknown committed
1251
  while (!thd->killed)
unknown's avatar
unknown committed
1252
  {
unknown's avatar
unknown committed
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);
unknown's avatar
unknown committed
1256 1257
    if (!table_is_used(thd->open_tables,1))
      break;
1258
    (void) pthread_cond_wait(&COND_refresh,&LOCK_open);
unknown's avatar
unknown 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
  }
unknown's avatar
unknown 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.
*/
unknown's avatar
unknown 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)
unknown's avatar
unknown committed
1319 1320 1321
  {
    if (!strcmp(table->real_name,table_name) &&
	!strcmp(table->table_cache_key,db))
1322
    {
unknown's avatar
unknown committed
1323
      mysql_lock_abort(thd,table);
1324 1325
      break;
    }
unknown's avatar
unknown 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
unknown's avatar
unknown committed
1347 1348
*/

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

1357
  strxmov(path, mysql_data_home, "/", db, "/", name, NullS);
unknown's avatar
unknown committed
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))
unknown's avatar
unknown committed
1363
  {
1364
    if (!entry->crashed)
unknown's avatar
unknown committed
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;
unknown's avatar
unknown committed
1374
      if (ha_create_table_from_engine(thd, db, name, TRUE) != 0)
unknown's avatar
unknown committed
1375 1376 1377 1378 1379
       goto err;

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

unknown's avatar
unknown committed
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;
unknown's avatar
unknown 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);
unknown's avatar
unknown committed
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)))
    {
unknown's avatar
unknown committed
1412
      /* Give right error message */
unknown's avatar
unknown committed
1413
      thd->clear_error();
1414
      my_error(ER_NOT_KEYFILE, MYF(0), name, my_errno);
unknown's avatar
unknown 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
unknown's avatar
unknown committed
1421
      thd->clear_error();			// Clear error message
unknown's avatar
unknown committed
1422
    pthread_mutex_lock(&LOCK_open);
1423
    unlock_table_name(thd,&table_list);
unknown's avatar
unknown committed
1424

1425 1426
    if (error)
      goto err;
unknown's avatar
unknown committed
1427
    break;
unknown's avatar
unknown 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).
        */
unknown's avatar
unknown 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;
      }
    }
  }
unknown's avatar
unknown committed
1463
  DBUG_RETURN(0);
1464 1465
err:
  DBUG_RETURN(1);
unknown's avatar
unknown committed
1466 1467
}

unknown's avatar
unknown committed
1468 1469
/*
  Open all tables in list
unknown's avatar
unknown committed
1470

unknown's avatar
unknown committed
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)
unknown's avatar
unknown committed
1483 1484 1485 1486 1487 1488
{
  TABLE_LIST *tables;
  bool refresh;
  int result=0;
  DBUG_ENTER("open_tables");

1489
  thd->current_tablenr= 0;
unknown's avatar
unknown committed
1490
 restart:
unknown's avatar
unknown committed
1491
  *counter= 0;
unknown's avatar
unknown committed
1492 1493 1494
  thd->proc_info="Opening tables";
  for (tables=start ; tables ; tables=tables->next)
  {
unknown's avatar
unknown committed
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;
unknown's avatar
unknown committed
1501
    (*counter)++;
unknown's avatar
unknown committed
1502
    if (!tables->table &&
1503 1504 1505 1506
	!(tables->table= open_table(thd,
				    tables->db,
				    tables->real_name,
				    tables->alias, &refresh)))
unknown's avatar
unknown 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
unknown's avatar
unknown 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)
unknown's avatar
unknown 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);
unknown's avatar
unknown 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)
unknown's avatar
unknown 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);
}


unknown's avatar
unknown committed
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
*/

unknown's avatar
unknown 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,
unknown's avatar
unknown committed
1619
			    &refresh)) && refresh) ;
unknown's avatar
unknown committed
1620

unknown's avatar
unknown committed
1621 1622
  if (table)
  {
unknown's avatar
unknown committed
1623
#if defined( __WIN__) || defined(OS2)
1624
    /* Win32 can't drop a file that is open */
unknown's avatar
unknown committed
1625
    if (lock_type == TL_WRITE_ALLOW_READ)
1626 1627 1628
    {
      lock_type= TL_WRITE;
    }
unknown's avatar
unknown committed
1629
#endif /* __WIN__ || OS2 */
unknown's avatar
unknown committed
1630 1631
    table_list->lock_type= lock_type;
    table_list->table=	   table;
unknown's avatar
unknown committed
1632 1633 1634
    table->grant= table_list->grant;
    if (thd->locked_tables)
    {
unknown's avatar
unknown committed
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
unknown's avatar
unknown committed
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;
unknown's avatar
unknown committed
1644 1645 1646 1647 1648 1649
    }
  }
  thd->proc_info=0;
  DBUG_RETURN(table);
}

unknown's avatar
unknown committed
1650

unknown's avatar
unknown 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

unknown's avatar
unknown committed
1660 1661 1662 1663
  RETURN
    0  - ok
    -1 - error

1664 1665
  NOTE
    The lock will automaticly be freed by close_thread_tables()
unknown's avatar
unknown committed
1666 1667
*/

1668
int simple_open_n_lock_tables(THD *thd, TABLE_LIST *tables)
unknown's avatar
unknown committed
1669
{
unknown's avatar
unknown committed
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
unknown's avatar
unknown committed
1683
    open_and_lock_tables()
1684 1685 1686
    thd		- thread handler
    tables	- list of tables for open&locking

unknown's avatar
unknown committed
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");
unknown's avatar
unknown committed
1698 1699
  uint counter;
  if (open_tables(thd, tables, &counter) || lock_tables(thd, tables, counter))
1700
    DBUG_RETURN(-1);				/* purecov: inspected */
unknown's avatar
unknown committed
1701 1702 1703 1704 1705 1706 1707 1708 1709 1710 1711 1712
  relink_tables_for_derived(thd);
  DBUG_RETURN(mysql_handle_derived(thd->lex));
}


/*
  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)
{
1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724
  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;
  }
unknown's avatar
unknown committed
1725 1726
}

unknown's avatar
unknown committed
1727 1728 1729 1730 1731 1732 1733 1734

/*
  Lock all tables in list

  SYNOPSIS
    lock_tables()
    thd			Thread handler
    tables		Tables to lock
unknown's avatar
unknown committed
1735
    count		umber of opened tables
unknown's avatar
unknown committed
1736

1737 1738 1739 1740 1741
  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.

unknown's avatar
unknown committed
1742 1743 1744 1745 1746
  RETURN VALUES
   0	ok
   -1	Error
*/

unknown's avatar
unknown committed
1747
int lock_tables(THD *thd, TABLE_LIST *tables, uint count)
unknown's avatar
unknown committed
1748
{
unknown's avatar
unknown committed
1749
  TABLE_LIST *table;
unknown's avatar
unknown committed
1750 1751 1752 1753
  if (!tables)
    return 0;

  if (!thd->locked_tables)
unknown's avatar
unknown committed
1754
  {
1755
    DBUG_ASSERT(thd->lock == 0);	// You must lock everything at once
unknown's avatar
unknown committed
1756 1757 1758 1759
    TABLE **start,**ptr;
    if (!(ptr=start=(TABLE**) sql_alloc(sizeof(TABLE*)*count)))
      return -1;
    for (table = tables ; table ; table=table->next)
1760 1761 1762 1763
    {
      if (!table->derived)
	*(ptr++)= table->table;
    }
unknown's avatar
unknown committed
1764
    if (!(thd->lock=mysql_lock_tables(thd,start, (uint) (ptr - start))))
unknown's avatar
unknown committed
1765 1766
      return -1;				/* purecov: inspected */
  }
unknown's avatar
unknown committed
1767 1768 1769 1770
  else
  {
    for (table = tables ; table ; table=table->next)
    {
1771 1772
      if (!table->derived && 
	  check_lock_and_start_stmt(thd, table->table, table->lock_type))
unknown's avatar
unknown committed
1773
      {
unknown's avatar
unknown committed
1774
	ha_rollback_stmt(thd);
unknown's avatar
unknown committed
1775 1776 1777 1778
	return -1;
      }
    }
  }
unknown's avatar
unknown committed
1779 1780 1781
  return 0;
}

unknown's avatar
unknown committed
1782

unknown's avatar
unknown committed
1783
/*
unknown's avatar
unknown committed
1784 1785 1786
  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 ...
unknown's avatar
unknown committed
1787 1788 1789 1790 1791 1792 1793
*/

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");
1794

unknown's avatar
unknown committed
1795 1796 1797 1798 1799 1800 1801
  /*
    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
  */
unknown's avatar
unknown committed
1802
  if (!(tmp_table=(TABLE*) my_malloc(sizeof(*tmp_table)+(uint) strlen(db)+
unknown's avatar
unknown committed
1803
				     (uint) strlen(table_name)+6+4,
unknown's avatar
unknown committed
1804 1805 1806 1807
				     MYF(MY_WME))))
    DBUG_RETURN(0);				/* purecov: inspected */

  if (openfrm(path, table_name,
unknown's avatar
unknown committed
1808
	      (uint) (HA_OPEN_KEYFILE | HA_OPEN_RNDFILE | HA_GET_INDEX),
unknown's avatar
unknown committed
1809
	      READ_KEYINFO | COMPUTE_TYPES | EXTRA_RECORD,
1810
	      ha_open_options,
unknown's avatar
unknown committed
1811 1812
	      tmp_table))
  {
1813
    my_free((char*) tmp_table,MYF(0));
unknown's avatar
unknown committed
1814 1815 1816
    DBUG_RETURN(0);
  }

unknown's avatar
unknown committed
1817
  tmp_table->reginfo.lock_type=TL_WRITE;	 // Simulate locked
1818
  tmp_table->in_use= thd;
unknown's avatar
unknown committed
1819 1820
  tmp_table->tmp_table = (tmp_table->file->has_transactions() ? 
			  TRANSACTIONAL_TMP_TABLE : TMP_TABLE);
unknown's avatar
unknown committed
1821
  tmp_table->table_cache_key=(char*) (tmp_table+1);
1822 1823 1824
  tmp_table->key_length= (uint) (strmov((tmp_table->real_name=
					 strmov(tmp_table->table_cache_key,db)
					 +1), table_name)
unknown's avatar
unknown committed
1825
				 - tmp_table->table_cache_key)+1;
unknown's avatar
unknown committed
1826 1827 1828
  int4store(tmp_table->table_cache_key + tmp_table->key_length,
	    thd->server_id);
  tmp_table->key_length += 4;
unknown's avatar
unknown committed
1829
  int4store(tmp_table->table_cache_key + tmp_table->key_length,
unknown's avatar
unknown committed
1830
	    thd->variables.pseudo_thread_id);
1831
  tmp_table->key_length += 4;
1832

unknown's avatar
unknown committed
1833 1834 1835 1836
  if (link_in_list)
  {
    tmp_table->next=thd->temporary_tables;
    thd->temporary_tables=tmp_table;
unknown's avatar
unknown committed
1837 1838
    if (thd->slave_thread)
      slave_open_temp_tables++;
unknown's avatar
unknown committed
1839 1840 1841 1842 1843 1844 1845 1846
  }
  DBUG_RETURN(tmp_table);
}


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

unknown's avatar
unknown committed
1849 1850 1851 1852 1853 1854 1855
  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))
1856
  {
unknown's avatar
unknown committed
1857
    error=1;
unknown's avatar
unknown committed
1858 1859
    sql_print_warning("Could not remove tmp table: '%s', error: %d",
                      path, my_errno);
1860
  }
unknown's avatar
unknown committed
1861
  delete file;
1862
  DBUG_RETURN(error);
unknown's avatar
unknown committed
1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873
}


/*****************************************************************************
** 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,
1874
                           bool check_grants, bool allow_rowid, 
1875
                           uint *cached_field_index_ptr)
unknown's avatar
unknown committed
1876
{
1877 1878
  Field **field_ptr, *field;
  uint cached_field_index= *cached_field_index_ptr;
1879

1880 1881
  /* We assume here that table->field < NO_CACHED_FIELD_INDEX = UINT_MAX */
  if (cached_field_index < table->fields &&
1882 1883
      !my_strcasecmp(system_charset_info, 
                     table->field[cached_field_index]->field_name, name))
1884
    field_ptr= table->field + cached_field_index;
1885 1886 1887
  else if (table->name_hash.records)
    field_ptr= (Field**)hash_search(&table->name_hash,(byte*) name,
                                    length);
unknown's avatar
unknown committed
1888 1889
  else
  {
1890
    if (!(field_ptr= table->field))
1891
      return (Field *)0;
1892
    for (; *field_ptr; ++field_ptr)
1893 1894
      if (!my_strcasecmp(system_charset_info, (*field_ptr)->field_name, name))
        break;
unknown's avatar
unknown committed
1895 1896
  }

1897
  if (field_ptr && *field_ptr)
unknown's avatar
unknown committed
1898
  {
1899 1900
    *cached_field_index_ptr= field_ptr - table->field;
    field= *field_ptr;
unknown's avatar
unknown committed
1901 1902 1903
  }
  else
  {
1904 1905 1906 1907
    if (!allow_rowid ||
        my_strcasecmp(system_charset_info, name, "_rowid") ||
        !(field=table->rowid_field))
      return (Field*) 0;
unknown's avatar
unknown committed
1908 1909 1910 1911 1912 1913 1914
  }

  if (thd->set_query_id)
  {
    if (field->query_id != thd->query_id)
    {
      field->query_id=thd->query_id;
1915
      table->used_fields++;
1916
      table->used_keys.intersect(field->part_of_key);
unknown's avatar
unknown committed
1917 1918 1919 1920
    }
    else
      thd->dupp_field=field;
  }
unknown's avatar
unknown committed
1921
#ifndef NO_EMBEDDED_ACCESS_CHECKS
1922
  if (check_grants && check_grant_column(thd,table,name,length))
unknown's avatar
unknown committed
1923
    return WRONG_GRANT;
unknown's avatar
unknown committed
1924
#endif
unknown's avatar
unknown committed
1925 1926 1927
  return field;
}

1928

1929 1930 1931 1932 1933
/*
  Find field in table list.

  SYNOPSIS
    find_field_in_tables()
1934 1935
    thd			Pointer to current thread structure
    item		Field item that should be found
unknown's avatar
unknown committed
1936
    tables		Tables for scanning
1937 1938 1939 1940
    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
1941 1942

  RETURN VALUES
1943 1944
    0			Field is not found or field is not unique- error
			message is reported
1945
    not_found_field	Function was called with report_error == FALSE and
1946
			field was not found. no error message reported.
1947 1948
    found field
*/
unknown's avatar
unknown committed
1949

1950 1951 1952
// Special Field pointer for find_field_in_tables returning
const Field *not_found_field= (Field*) 0x1;

unknown's avatar
unknown committed
1953
Field *
1954
find_field_in_tables(THD *thd, Item_ident *item, TABLE_LIST *tables,
1955
		     TABLE_LIST **where, bool report_error)
unknown's avatar
unknown committed
1956 1957 1958 1959 1960
{
  Field *found=0;
  const char *db=item->db_name;
  const char *table_name=item->table_name;
  const char *name=item->field_name;
unknown's avatar
unknown committed
1961
  uint length=(uint) strlen(name);
1962 1963
  char name_buff[NAME_LEN+1];

1964

1965
  if (item->cached_table)
1966 1967 1968 1969 1970 1971 1972 1973 1974 1975
  {
    /*
      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.
    */
1976 1977 1978
    found= find_field_in_table(thd, item->cached_table->table, name, length,
                               test(item->cached_table->
				    table->grant.want_privilege),
1979 1980 1981 1982 1983 1984 1985 1986 1987 1988 1989
                               1, &(item->cached_field_index));

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

1990 1991 1992 1993 1994 1995 1996 1997
  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);
unknown's avatar
unknown committed
1998
    my_casedn_str(files_charset_info, name_buff);
1999 2000
    db= name_buff;
  }
unknown's avatar
unknown committed
2001

unknown's avatar
unknown committed
2002
  if (table_name && table_name[0])
unknown's avatar
unknown committed
2003 2004 2005 2006
  {						/* Qualified field */
    bool found_table=0;
    for (; tables ; tables=tables->next)
    {
2007
      if (!my_strcasecmp(table_alias_charset, tables->alias, table_name) &&
unknown's avatar
unknown committed
2008
	  (!db || !tables->db ||  !tables->db[0] || !strcmp(db,tables->db)))
unknown's avatar
unknown committed
2009 2010 2011
      {
	found_table=1;
	Field *find=find_field_in_table(thd,tables->table,name,length,
2012 2013
					test(tables->table->grant.
					     want_privilege),
2014
					1, &(item->cached_field_index));
unknown's avatar
unknown committed
2015 2016
	if (find)
	{
2017
	  (*where)= item->cached_table= tables;
unknown's avatar
unknown committed
2018
	  if (!tables->cacheable_table)
2019
	    item->cached_table= 0;
unknown's avatar
unknown committed
2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031 2032 2033 2034 2035
	  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;
unknown's avatar
unknown committed
2036
    if (!found_table && report_error)
unknown's avatar
unknown committed
2037 2038
    {
      char buff[NAME_LEN*2+1];
unknown's avatar
unknown committed
2039
      if (db && db[0])
unknown's avatar
unknown committed
2040
      {
2041
	strxnmov(buff,sizeof(buff)-1,db,".",table_name,NullS);
unknown's avatar
unknown committed
2042 2043
	table_name=buff;
      }
2044
      if (report_error)
2045
      {
unknown's avatar
unknown committed
2046 2047
	my_printf_error(ER_UNKNOWN_TABLE, ER(ER_UNKNOWN_TABLE), MYF(0),
			table_name, thd->where);
2048
      }
2049 2050
      else
	return (Field*) not_found_field;
unknown's avatar
unknown committed
2051 2052
    }
    else
unknown's avatar
unknown committed
2053 2054 2055
      if (report_error)
	my_printf_error(ER_BAD_FIELD_ERROR,ER(ER_BAD_FIELD_ERROR),MYF(0),
			item->full_name(),thd->where);
2056 2057
      else
	return (Field*) not_found_field;
unknown's avatar
unknown committed
2058 2059 2060 2061 2062
    return (Field*) 0;
  }
  bool allow_rowid= tables && !tables->next;	// Only one table
  for (; tables ; tables=tables->next)
  {
2063 2064 2065 2066 2067 2068 2069 2070
    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;
    }

unknown's avatar
unknown committed
2071
    Field *field=find_field_in_table(thd,tables->table,name,length,
2072
				     test(tables->table->grant.want_privilege),
2073
				     allow_rowid, &(item->cached_field_index));
unknown's avatar
unknown committed
2074 2075 2076 2077
    if (field)
    {
      if (field == WRONG_GRANT)
	return (Field*) 0;
2078
      (*where)= item->cached_table= tables;
unknown's avatar
unknown committed
2079
      if (!tables->cacheable_table)
2080
	item->cached_table= 0;
unknown's avatar
unknown committed
2081 2082
      if (found)
      {
2083
	if (!thd->where)			// Returns first found
unknown's avatar
unknown committed
2084
	  break;
2085 2086
	my_printf_error(ER_NON_UNIQ_ERROR,ER(ER_NON_UNIQ_ERROR),MYF(0),
			name,thd->where);
unknown's avatar
unknown committed
2087 2088 2089 2090 2091 2092 2093
	return (Field*) 0;
      }
      found=field;
    }
  }
  if (found)
    return found;
unknown's avatar
unknown committed
2094 2095 2096
  if (report_error)
    my_printf_error(ER_BAD_FIELD_ERROR, ER(ER_BAD_FIELD_ERROR),
		    MYF(0), item->full_name(), thd->where);
2097 2098
  else
    return (Field*) not_found_field;
unknown's avatar
unknown committed
2099 2100 2101
  return (Field*) 0;
}

2102 2103 2104

/*
  Find Item in list of items (find_field_in_tables analog)
2105 2106 2107 2108

  TODO
    is it better return only counter?

2109 2110
  SYNOPSIS
    find_item_in_list()
2111 2112 2113
    find			Item to find
    items			List of items
    counter			To return number of found item
2114
    report_error
2115 2116 2117 2118 2119
      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
2120 2121 2122 2123
    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.

2124
  RETURN VALUES
2125 2126 2127 2128 2129
    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
2130
                        found field
2131 2132
*/

2133 2134 2135 2136
// Special Item pointer for find_item_in_list returning
const Item **not_found_item= (const Item**) 0x1;


unknown's avatar
unknown committed
2137
Item **
2138
find_item_in_list(Item *find, List<Item> &items, uint *counter,
2139
                  find_item_error_report_type report_error, bool *unaliased)
unknown's avatar
unknown committed
2140 2141
{
  List_iterator<Item> li(items);
2142
  Item **found=0, **found_unaliased= 0, *item;
2143
  const char *db_name=0;
unknown's avatar
unknown committed
2144 2145
  const char *field_name=0;
  const char *table_name=0;
2146 2147
  bool found_unaliased_non_uniq= 0;
  uint unaliased_counter;
2148 2149 2150

  *unaliased= FALSE;

unknown's avatar
unknown committed
2151 2152 2153 2154
  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;
2155
    db_name=    ((Item_ident*) find)->db_name;
unknown's avatar
unknown committed
2156 2157
  }

2158
  for (uint i= 0; (item=li++); i++)
unknown's avatar
unknown committed
2159 2160 2161
  {
    if (field_name && item->type() == Item::FIELD_ITEM)
    {
2162
      Item_field *item_field= (Item_field*) item;
2163

2164 2165 2166 2167 2168 2169
      /*
	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
      */ 
2170 2171 2172 2173
      if (!item_field->name)
        continue;

      if (table_name)
unknown's avatar
unknown committed
2174
      {
2175 2176 2177
        /*
          If table name is specified we should find field 'field_name' in
          table 'table_name'. According to SQL-standard we should ignore
2178 2179 2180 2181 2182
          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.
2183

2184
          We use strcmp for table names and database names as these may be
2185 2186
          case sensitive. In cases where they are not case sensitive, they
          are always in lower case.
2187 2188

	  item_field->field_name and item_field->table_name can be 0x0 if
2189
	  item is not fix_field()'ed yet.
2190
        */
2191 2192
        if (item_field->field_name && item_field->table_name &&
	    !my_strcasecmp(system_charset_info, item_field->field_name,
2193 2194 2195 2196 2197
                           field_name) &&
            !strcmp(item_field->table_name, table_name) &&
            (!db_name || (item_field->db_name &&
                          !strcmp(item_field->db_name, db_name))))
        {
2198
          if (found_unaliased)
2199
          {
2200 2201 2202 2203 2204 2205 2206
            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.
            */
2207 2208 2209 2210 2211
            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;
          }
2212 2213
          found_unaliased= li.ref();
          unaliased_counter= i;
2214 2215
          if (db_name)
            break;                              // Perfect match
2216 2217 2218
        }
      }
      else if (!my_strcasecmp(system_charset_info, item_field->name,
2219
                              field_name))
2220 2221 2222 2223 2224 2225 2226 2227 2228 2229 2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256 2257 2258 2259
      {
        /*
          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;
        }
unknown's avatar
unknown committed
2260 2261
      }
    }
2262
    else if (!table_name && (item->eq(find,0) ||
2263
			     find->name && item->name &&
2264 2265
			     !my_strcasecmp(system_charset_info, 
					    item->name,find->name)))
unknown's avatar
unknown committed
2266
    {
2267 2268
      found= li.ref();
      *counter= i;
unknown's avatar
unknown committed
2269 2270 2271
      break;
    }
  }
2272 2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284
  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;
2285
      *unaliased= TRUE;
2286 2287
    }
  }
2288 2289
  if (found)
    return found;
2290
  if (report_error != REPORT_EXCEPT_NOT_FOUND)
2291 2292 2293 2294 2295 2296 2297 2298
  {
    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;
unknown's avatar
unknown committed
2299 2300 2301
}

/****************************************************************************
2302
** Expand all '*' in given fields
unknown's avatar
unknown committed
2303 2304
****************************************************************************/

2305 2306 2307
int setup_wild(THD *thd, TABLE_LIST *tables, List<Item> &fields,
	       List<Item> *sum_func_list,
	       uint wild_num)
unknown's avatar
unknown committed
2308
{
2309 2310
  bool is_stmt_prepare;
  DBUG_ENTER("setup_wild");
2311
  if (!wild_num)
2312 2313
    DBUG_RETURN(0);

unknown's avatar
unknown committed
2314 2315 2316
  reg2 Item *item;
  List_iterator<Item> it(fields);
  Item_arena *arena, backup;
unknown's avatar
unknown committed
2317 2318 2319 2320
  /*
    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
  */
unknown's avatar
unknown committed
2321
  arena= thd->change_arena_if_needed(&backup);
2322

unknown's avatar
unknown committed
2323
  while (wild_num && (item= it++))
2324
  {    
2325 2326
    if (item->type() == Item::FIELD_ITEM &&
        ((Item_field*) item)->field_name &&
2327 2328
	((Item_field*) item)->field_name[0] == '*' &&
	!((Item_field*) item)->field)
2329
    {
2330
      uint elem= fields.elements;
unknown's avatar
unknown committed
2331 2332 2333 2334 2335 2336 2337 2338 2339 2340 2341 2342 2343
      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))
2344
      {
unknown's avatar
unknown committed
2345
        if (arena)
2346
	  thd->restore_backup_item_arena(arena, &backup);
2347
	DBUG_RETURN(-1);
2348
      }
2349
      if (sum_func_list)
2350 2351 2352 2353 2354 2355 2356 2357
      {
	/*
	  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;
      }
2358
      wild_num--;
2359 2360
    }
  }
unknown's avatar
unknown committed
2361
  if (arena)
2362 2363
    thd->restore_backup_item_arena(arena, &backup);
  DBUG_RETURN(0);
2364 2365
}

2366 2367 2368 2369 2370 2371 2372 2373 2374 2375 2376 2377 2378 2379 2380 2381
/****************************************************************************
** 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";

unknown's avatar
unknown committed
2382 2383
  Item **ref= ref_pointer_array;
  while ((item= it++))
2384
  {
2385
    if (!item->fixed && item->fix_fields(thd, tables, it.ref()) ||
2386
	(item= *(it.ref()))->check_cols(1))
2387
      DBUG_RETURN(-1); /* purecov: inspected */
unknown's avatar
unknown committed
2388 2389
    if (ref)
      *(ref++)= item;
2390 2391
    if (item->with_sum_func && item->type() != Item::SUM_FUNC_ITEM &&
	sum_func_list)
2392
      item->split_sum_func(thd, ref_pointer_array, *sum_func_list);
2393 2394
    thd->used_tables|=item->used_tables();
  }
2395
  DBUG_RETURN(test(thd->net.report_error));
2396
}
2397

2398

2399
/*
2400
  prepare tables
2401

2402 2403
  SYNOPSIS
    setup_tables()
2404
    tables	table list
2405 2406


2407 2408 2409 2410 2411 2412 2413
 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.
2414

2415 2416 2417
 RETURN
   0	ok;  In this case *map will includes the choosed index
   1	error
2418 2419
*/

2420
bool setup_tables(TABLE_LIST *tables)
2421 2422
{
  DBUG_ENTER("setup_tables");
unknown's avatar
unknown committed
2423
  uint tablenr=0;
unknown's avatar
unknown committed
2424 2425
  for (TABLE_LIST *table_list=tables ; table_list ;
       table_list=table_list->next,tablenr++)
unknown's avatar
unknown committed
2426
  {
unknown's avatar
unknown committed
2427 2428
    TABLE *table= table_list->table;
    setup_table_map(table, table_list, tablenr);
2429
    table->used_keys= table->keys_for_keyread;
unknown's avatar
unknown committed
2430
    if (table_list->use_index)
unknown's avatar
unknown committed
2431
    {
2432 2433 2434
      key_map map;
      get_key_map_from_key_list(&map, table, table_list->use_index);
      if (map.is_set_all())
2435
	DBUG_RETURN(1);
unknown's avatar
unknown committed
2436
      table->keys_in_use_for_query=map;
unknown's avatar
unknown committed
2437
    }
unknown's avatar
unknown committed
2438
    if (table_list->ignore_index)
unknown's avatar
unknown committed
2439
    {
2440 2441 2442
      key_map map;
      get_key_map_from_key_list(&map, table, table_list->ignore_index);
      if (map.is_set_all())
2443
	DBUG_RETURN(1);
2444
      table->keys_in_use_for_query.subtract(map);
unknown's avatar
unknown committed
2445
    }
2446
    table->used_keys.intersect(table->keys_in_use_for_query);
unknown's avatar
unknown committed
2447 2448 2449 2450
  }
  if (tablenr > MAX_TABLES)
  {
    my_error(ER_TOO_MANY_TABLES,MYF(0),MAX_TABLES);
2451
    DBUG_RETURN(1);
unknown's avatar
unknown committed
2452
  }
2453
  DBUG_RETURN(0);
unknown's avatar
unknown committed
2454
}
2455

unknown's avatar
unknown committed
2456

2457 2458 2459 2460 2461 2462 2463 2464 2465 2466 2467 2468 2469 2470 2471
/*
   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,
2472
                               List<String> *index_list)
unknown's avatar
unknown committed
2473
{
unknown's avatar
unknown committed
2474
  List_iterator_fast<String> it(*index_list);
unknown's avatar
unknown committed
2475 2476
  String *name;
  uint pos;
2477 2478

  map->clear_all();
unknown's avatar
unknown committed
2479 2480
  while ((name=it++))
  {
2481 2482
    if ((pos= find_type(&table->keynames, name->ptr(), name->length(), 1)) <=
	0)
unknown's avatar
unknown committed
2483 2484 2485
    {
      my_error(ER_KEY_COLUMN_DOES_NOT_EXITS, MYF(0), name->c_ptr(),
	       table->real_name);
2486
      map->set_all();
2487
      return 1;
unknown's avatar
unknown committed
2488
    }
2489
    map->set_bit(pos-1);
unknown's avatar
unknown committed
2490
  }
2491
  return 0;
unknown's avatar
unknown committed
2492 2493
}

2494

unknown's avatar
unknown committed
2495
/****************************************************************************
unknown's avatar
unknown committed
2496 2497
  This just drops in all fields instead of current '*' field
  Returns pointer to last inserted field if ok
unknown's avatar
unknown committed
2498 2499
****************************************************************************/

unknown's avatar
unknown committed
2500
bool
2501 2502
insert_fields(THD *thd,TABLE_LIST *tables, const char *db_name,
	      const char *table_name, List_iterator<Item> *it)
unknown's avatar
unknown committed
2503
{
2504
  char name_buff[NAME_LEN+1];
unknown's avatar
unknown committed
2505 2506 2507
  uint found;
  DBUG_ENTER("insert_fields");

2508 2509
  if (db_name && lower_case_table_names)
  {
unknown's avatar
unknown committed
2510 2511 2512 2513 2514
    /*
      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
    */
unknown's avatar
unknown committed
2515
    strmake(name_buff, db_name, sizeof(name_buff)-1);
unknown's avatar
unknown committed
2516
    my_casedn_str(files_charset_info, name_buff);
unknown's avatar
unknown committed
2517
    db_name= name_buff;
2518 2519 2520
  }


unknown's avatar
unknown committed
2521
  found=0;
2522
  for (; tables ; tables=tables->next)
unknown's avatar
unknown committed
2523
  {
2524
    TABLE *table=tables->table;
2525 2526
    if (!table_name || (!my_strcasecmp(table_alias_charset, table_name,
				       tables->alias) &&
2527
			(!db_name || !strcmp(tables->db,db_name))))
unknown's avatar
unknown committed
2528
    {
unknown's avatar
unknown committed
2529
#ifndef NO_EMBEDDED_ACCESS_CHECKS
unknown's avatar
unknown committed
2530
      /* Ensure that we have access right to all columns */
unknown's avatar
unknown committed
2531
      if (!(table->grant.privilege & SELECT_ACL) &&
unknown's avatar
unknown committed
2532
	  check_grant_all_columns(thd,SELECT_ACL,table))
unknown's avatar
unknown committed
2533
	DBUG_RETURN(-1);
unknown's avatar
unknown committed
2534
#endif
2535
      Field **ptr=table->field,*field;
unknown's avatar
unknown committed
2536 2537
      TABLE *natural_join_table= 0;

unknown's avatar
unknown committed
2538
      thd->used_tables|=table->map;
unknown's avatar
unknown committed
2539 2540 2541
      if (!table->outer_join &&
          tables->natural_join &&
          !tables->natural_join->table->outer_join)
unknown's avatar
unknown committed
2542
        natural_join_table= tables->natural_join->table;
unknown's avatar
unknown committed
2543

unknown's avatar
unknown committed
2544 2545
      while ((field = *ptr++))
      {
2546
        uint not_used_field_index= NO_CACHED_FIELD_INDEX;
2547
        /* Skip duplicate field names if NATURAL JOIN is used */
unknown's avatar
unknown committed
2548 2549
        if (!natural_join_table ||
            !find_field_in_table(thd, natural_join_table, field->field_name, 
2550 2551
                                 strlen(field->field_name), 0, 0,
                                 &not_used_field_index))
2552
        {
2553
          Item_field *item= new Item_field(thd, field);
2554 2555 2556 2557 2558
          if (!found++)
            (void) it->replace(item);		// Replace '*'
          else
            it->after(item);
        }
unknown's avatar
unknown committed
2559 2560 2561 2562
	/*
	  Mark if field used before in this select.
	  Used by 'insert' to verify if a field name is used twice
	*/
unknown's avatar
unknown committed
2563 2564 2565
	if (field->query_id == thd->query_id)
	  thd->dupp_field=field;
	field->query_id=thd->query_id;
2566
	table->used_keys.intersect(field->part_of_key);
unknown's avatar
unknown committed
2567 2568
      }
      /* All fields are used */
2569
      table->used_fields=table->fields;
unknown's avatar
unknown committed
2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585 2586 2587 2588
    }
  }
  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)
{
2589
  table_map not_null_tables= 0;
unknown's avatar
unknown committed
2590
  Item_arena *arena= 0, backup;
unknown's avatar
unknown committed
2591
  DBUG_ENTER("setup_conds");
unknown's avatar
unknown committed
2592

unknown's avatar
unknown committed
2593
  thd->set_query_id=1;
2594
  thd->lex->current_select->cond_count= 0;
unknown's avatar
unknown committed
2595 2596 2597
  if (*conds)
  {
    thd->where="where clause";
2598 2599
    if (!(*conds)->fixed && (*conds)->fix_fields(thd, tables, conds) ||
	(*conds)->check_cols(1))
unknown's avatar
unknown committed
2600
      DBUG_RETURN(1);
2601
    not_null_tables= (*conds)->not_null_tables();
unknown's avatar
unknown committed
2602 2603
  }

2604

unknown's avatar
unknown committed
2605 2606 2607
  /* Check if we are using outer joins */
  for (TABLE_LIST *table=tables ; table ; table=table->next)
  {
2608 2609 2610 2611
    if (table->on_expr)
    {
      /* Make a join an a expression */
      thd->where="on clause";
2612 2613 2614
      
      if (!table->on_expr->fixed &&
	  table->on_expr->fix_fields(thd, tables, &table->on_expr) ||
unknown's avatar
unknown committed
2615
	  table->on_expr->check_cols(1))
unknown's avatar
unknown committed
2616
	DBUG_RETURN(1);
2617
      thd->lex->current_select->cond_count++;
2618

2619 2620 2621 2622 2623 2624 2625
      /*
	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)))
2626
      {
2627
	table->outer_join= 0;
unknown's avatar
unknown committed
2628
        arena= thd->change_arena_if_needed(&backup);
unknown's avatar
unknown committed
2629
	*conds= and_conds(*conds, table->on_expr);
2630
	table->on_expr=0;
unknown's avatar
unknown committed
2631 2632
	if (arena)
        {
2633
	  thd->restore_backup_item_arena(arena, &backup);
unknown's avatar
unknown committed
2634 2635
          arena= 0;                             // Safety if goto err
        }
unknown's avatar
unknown committed
2636 2637 2638
	if ((*conds) && !(*conds)->fixed &&
	    (*conds)->fix_fields(thd, tables, conds))
	  DBUG_RETURN(1);
2639 2640
      }
    }
unknown's avatar
unknown committed
2641 2642
    if (table->natural_join)
    {
unknown's avatar
unknown committed
2643
      arena= thd->change_arena_if_needed(&backup);
unknown's avatar
unknown committed
2644
      /* Make a join of all fields with have the same name */
unknown's avatar
unknown committed
2645 2646 2647
      TABLE *t1= table->table;
      TABLE *t2= table->natural_join->table;
      Item_cond_and *cond_and= new Item_cond_and();
unknown's avatar
unknown committed
2648
      if (!cond_and)				// If not out of memory
2649
	goto err;
2650
      cond_and->top_level_item();
unknown's avatar
unknown committed
2651

2652 2653
      Field **t1_field, *t2_field;
      for (t1_field= t1->field; (*t1_field); t1_field++)
unknown's avatar
unknown committed
2654
      {
2655
        const char *t1_field_name= (*t1_field)->field_name;
2656
        uint not_used_field_index= NO_CACHED_FIELD_INDEX;
2657 2658

        if ((t2_field= find_field_in_table(thd, t2, t1_field_name,
2659 2660
                                           strlen(t1_field_name), 0, 0,
                                           &not_used_field_index)))
2661
        {
2662 2663
          Item_func_eq *tmp=new Item_func_eq(new Item_field(thd, *t1_field),
                                             new Item_field(thd, t2_field));
2664
          if (!tmp)
2665
            goto err;
2666 2667 2668 2669 2670 2671
          /* 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);
        }
unknown's avatar
unknown committed
2672
      }
2673
      thd->lex->current_select->cond_count+= cond_and->list.elements;
unknown's avatar
unknown committed
2674

2675 2676 2677
      // to prevent natural join processing during PS re-execution
      table->natural_join= 0;

2678
      if (cond_and->list.elements)
unknown's avatar
unknown committed
2679
      {
2680 2681 2682 2683
        if (!table->outer_join)			// Not left join
        {
          *conds= and_conds(*conds, cond_and);
          // fix_fields() should be made with temporary memory pool
unknown's avatar
unknown committed
2684
          if (arena)
2685
            thd->restore_backup_item_arena(arena, &backup);
2686 2687 2688 2689 2690 2691 2692 2693 2694 2695
          if (*conds && !(*conds)->fixed)
          {
            if ((*conds)->fix_fields(thd, tables, conds))
              DBUG_RETURN(1);
          }
        }
        else
        {
          table->on_expr= and_conds(table->on_expr, cond_and);
          // fix_fields() should be made with temporary memory pool
unknown's avatar
unknown committed
2696
          if (arena)
2697
            thd->restore_backup_item_arena(arena, &backup);
2698 2699 2700 2701 2702 2703
          if (table->on_expr && !table->on_expr->fixed)
          {
            if (table->on_expr->fix_fields(thd, tables, &table->on_expr))
             DBUG_RETURN(1);
          }
        }
unknown's avatar
unknown committed
2704
      }
unknown's avatar
unknown committed
2705 2706
      else if (arena)
      {
2707
        thd->restore_backup_item_arena(arena, &backup);
unknown's avatar
unknown committed
2708 2709
        arena= 0;                               // Safety if goto err
      }
unknown's avatar
unknown committed
2710 2711
    }
  }
2712

unknown's avatar
unknown committed
2713
  if (thd->current_arena->is_stmt_prepare())
2714 2715 2716 2717 2718 2719 2720 2721 2722
  {
    /*
      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;
  }
2723
  DBUG_RETURN(test(thd->net.report_error));
2724 2725

err:
unknown's avatar
unknown committed
2726 2727
  if (arena)
    thd->restore_backup_item_arena(arena, &backup);
2728
  DBUG_RETURN(1);
unknown's avatar
unknown committed
2729 2730 2731 2732 2733 2734 2735 2736 2737
}


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

int
2738
fill_record(List<Item> &fields,List<Item> &values, bool ignore_errors)
unknown's avatar
unknown committed
2739
{
unknown's avatar
unknown committed
2740
  List_iterator_fast<Item> f(fields),v(values);
unknown's avatar
unknown committed
2741 2742 2743 2744 2745 2746 2747
  Item *value;
  Item_field *field;
  DBUG_ENTER("fill_record");

  while ((field=(Item_field*) f++))
  {
    value=v++;
2748 2749
    Field *rfield= field->field;
    TABLE *table= rfield->table;
unknown's avatar
unknown committed
2750
    if (rfield == table->next_number_field)
unknown's avatar
unknown committed
2751
      table->auto_increment_field_not_null= TRUE;
unknown's avatar
unknown committed
2752
    if ((value->save_in_field(rfield, 0) < 0) && !ignore_errors)
unknown's avatar
unknown committed
2753 2754 2755 2756 2757 2758 2759
      DBUG_RETURN(1);
  }
  DBUG_RETURN(0);
}


int
2760
fill_record(Field **ptr,List<Item> &values, bool ignore_errors)
unknown's avatar
unknown committed
2761
{
unknown's avatar
unknown committed
2762
  List_iterator_fast<Item> v(values);
unknown's avatar
unknown committed
2763 2764 2765 2766 2767 2768 2769
  Item *value;
  DBUG_ENTER("fill_record");

  Field *field;
  while ((field = *ptr++))
  {
    value=v++;
2770
    TABLE *table= field->table;
unknown's avatar
unknown committed
2771
    if (field == table->next_number_field)
unknown's avatar
unknown committed
2772
      table->auto_increment_field_not_null= TRUE;
unknown's avatar
unknown committed
2773
    if ((value->save_in_field(field, 0) < 0) && !ignore_errors)
unknown's avatar
unknown committed
2774 2775 2776 2777 2778 2779 2780 2781
      DBUG_RETURN(1);
  }
  DBUG_RETURN(0);
}


static void mysql_rm_tmp_tables(void)
{
unknown's avatar
unknown committed
2782 2783
  uint i, idx;
  char	filePath[FN_REFLEN], *tmpdir;
unknown's avatar
unknown committed
2784 2785 2786 2787
  MY_DIR *dirp;
  FILEINFO *file;
  DBUG_ENTER("mysql_rm_tmp_tables");

unknown's avatar
unknown committed
2788 2789 2790
  for (i=0; i<=mysql_tmpdir_list.max; i++)
  {
    tmpdir=mysql_tmpdir_list.list[i];
unknown's avatar
unknown committed
2791
  /* See if the directory exists */
unknown's avatar
unknown committed
2792 2793
    if (!(dirp = my_dir(tmpdir,MYF(MY_WME | MY_DONT_SORT))))
      continue;
unknown's avatar
unknown committed
2794

unknown's avatar
unknown committed
2795
    /* Remove all SQLxxx tables from directory */
unknown's avatar
unknown committed
2796

2797
  for (idx=0 ; idx < (uint) dirp->number_off_files ; idx++)
unknown's avatar
unknown committed
2798 2799
  {
    file=dirp->dir_entry+idx;
2800 2801 2802 2803 2804 2805

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

unknown's avatar
unknown committed
2806 2807
    if (!bcmp(file->name,tmp_file_prefix,tmp_file_prefix_length))
    {
unknown's avatar
unknown committed
2808 2809
        sprintf(filePath,"%s%s",tmpdir,file->name);
        VOID(my_delete(filePath,MYF(MY_WME)));
unknown's avatar
unknown committed
2810 2811 2812
    }
  }
  my_dirend(dirp);
unknown's avatar
unknown committed
2813
  }
unknown's avatar
unknown committed
2814 2815 2816 2817 2818 2819 2820 2821 2822 2823 2824 2825 2826 2827 2828 2829 2830 2831 2832 2833 2834 2835 2836 2837 2838 2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850 2851 2852 2853 2854 2855 2856 2857 2858 2859
  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.
*/

void remove_db_from_cache(const my_string db)
{
  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);
}


/*
unknown's avatar
unknown committed
2860 2861 2862 2863 2864 2865 2866 2867 2868
  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
unknown's avatar
unknown committed
2869 2870
*/

2871 2872
bool remove_table_from_cache(THD *thd, const char *db, const char *table_name,
			     bool return_if_owned_by_thd)
unknown's avatar
unknown committed
2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884
{
  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))
  {
2885
    THD *in_use;
unknown's avatar
unknown committed
2886
    table->version=0L;			/* Free when thread is ready */
2887
    if (!(in_use=table->in_use))
unknown's avatar
unknown committed
2888 2889
    {
      DBUG_PRINT("info",("Table was not in use"));
unknown's avatar
unknown committed
2890
      relink_unused(table);
unknown's avatar
unknown committed
2891
    }
2892
    else if (in_use != thd)
unknown's avatar
unknown committed
2893 2894 2895 2896 2897
    {
      in_use->some_tables_deleted=1;
      if (table->db_stat)
	result=1;
      /* Kill delayed insert threads */
unknown's avatar
unknown committed
2898 2899
      if ((in_use->system_thread & SYSTEM_THREAD_DELAYED_INSERT) &&
          ! in_use->killed)
unknown's avatar
unknown committed
2900 2901 2902
      {
	in_use->killed=1;
	pthread_mutex_lock(&in_use->mysys_var->mutex);
unknown's avatar
unknown committed
2903
	if (in_use->mysys_var->current_cond)
unknown's avatar
unknown committed
2904 2905 2906 2907 2908 2909 2910
	{
	  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);
      }
2911 2912 2913 2914 2915 2916 2917 2918 2919 2920 2921
      /*
	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);
      }
unknown's avatar
unknown committed
2922
    }
2923 2924
    else
      result= result || return_if_owned_by_thd;
unknown's avatar
unknown committed
2925 2926 2927 2928 2929 2930
  }
  while (unused_tables && !unused_tables->version)
    VOID(hash_delete(&open_cache,(byte*) unused_tables));
  DBUG_RETURN(result);
}

2931
int setup_ftfuncs(SELECT_LEX *select_lex)
unknown's avatar
unknown committed
2932
{
2933 2934
  List_iterator<Item_func_match> li(*(select_lex->ftfunc_list)),
                                 lj(*(select_lex->ftfunc_list));
2935
  Item_func_match *ftf, *ftf2;
unknown's avatar
unknown committed
2936 2937

  while ((ftf=li++))
2938
  {
unknown's avatar
unknown committed
2939 2940
    if (ftf->fix_index())
      return 1;
2941 2942
    lj.rewind();
    while ((ftf2=lj++) != ftf)
2943
    {
2944
      if (ftf->eq(ftf2,1) && !ftf2->master)
2945 2946 2947
        ftf2->master=ftf;
    }
  }
unknown's avatar
unknown committed
2948 2949 2950

  return 0;
}
2951

2952

2953
int init_ftfuncs(THD *thd, SELECT_LEX *select_lex, bool no_order)
2954
{
2955
  if (select_lex->ftfunc_list->elements)
2956
  {
2957
    List_iterator<Item_func_match> li(*(select_lex->ftfunc_list));
unknown's avatar
unknown committed
2958 2959 2960
    Item_func_match *ifm;
    DBUG_PRINT("info",("Performing FULLTEXT search"));
    thd->proc_info="FULLTEXT initialization";
2961

unknown's avatar
unknown committed
2962 2963 2964
    while ((ifm=li++))
      ifm->init_search(no_order);
  }
2965 2966
  return 0;
}