~drizzle-trunk/drizzle/development

1 by brian
clean slate
1
/* Copyright (C) 2000-2003 MySQL AB
2
3
   This program is free software; you can redistribute it and/or modify
4
   it under the terms of the GNU General Public License as published by
5
   the Free Software Foundation; version 2 of the License.
6
7
   This program is distributed in the hope that it will be useful,
8
   but WITHOUT ANY WARRANTY; without even the implied warranty of
9
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10
   GNU General Public License for more details.
11
12
   You should have received a copy of the GNU General Public License
13
   along with this program; if not, write to the Free Software
14
   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
15
16
17
/* create and drop of databases */
549 by Monty Taylor
Took gettext.h out of header files.
18
#include <config.h>
481.1.9 by Monty Taylor
Added autoconf tests for location of cstdint and cinttypes. Use those in C++ programs now, so that we don't have to define _STDC_LIMIT_MACROS, etc by hand. Stop, in fact, defining those by hand.
19
#include CSTDINT_H
20
#include CINTTYPES_H
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
21
#include <string>
22
#include <fstream>
316 by Brian Aker
First pass of new sql_db.cc work
23
#include <drizzled/serialize/serialize.h>
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
24
using namespace std;
243.1.17 by Jay Pipes
FINAL PHASE removal of mysql_priv.h (Bye, bye my friend.)
25
#include <drizzled/server_includes.h>
212.5.13 by Monty Taylor
Moved my_sys/my_pthread/my_nosys and mysys_err to mysys.
26
#include <mysys/mysys_err.h>
212.5.38 by Monty Taylor
Moved my_dir (and removed references to a lot of places)
27
#include <mysys/my_dir.h>
1 by brian
clean slate
28
#include "log.h"
549 by Monty Taylor
Took gettext.h out of header files.
29
#include <drizzled/error.h>
538 by Monty Taylor
Moved gettext.h into drizzled in anticipation of the new client lib.
30
#include <drizzled/gettext.h>
1 by brian
clean slate
31
316 by Brian Aker
First pass of new sql_db.cc work
32
1 by brian
clean slate
33
#define MAX_DROP_TABLE_Q_LEN      1024
34
461 by Monty Taylor
Removed NullS. bu-bye.
35
const char *del_exts[]= {".frm", ".BAK", ".TMD",".opt", NULL};
1 by brian
clean slate
36
static TYPELIB deletable_extentions=
37
{array_elements(del_exts)-1,"del_exts", del_exts, NULL};
38
520.1.22 by Brian Aker
Second pass of thd cleanup
39
static long mysql_rm_known_files(Session *session, MY_DIR *dirp,
482 by Brian Aker
Remove uint.
40
				 const char *db, const char *path, uint32_t level, 
327.2.4 by Brian Aker
Refactoring table.h
41
                                 TableList **dropped_tables);
1 by brian
clean slate
42
         
253 by Brian Aker
Removed final my_bool from sql_db.
43
static bool rm_dir_w_symlink(const char *org_path, bool send_error);
520.1.22 by Brian Aker
Second pass of thd cleanup
44
static void mysql_change_db_impl(Session *session,
1 by brian
clean slate
45
                                 LEX_STRING *new_db_name,
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
46
                                 const CHARSET_INFO * const new_db_charset);
1 by brian
clean slate
47
48
49
/* Database lock hash */
50
HASH lock_db_cache;
51
pthread_mutex_t LOCK_lock_db;
52
int creating_database= 0;  // how many database locks are made
53
54
55
/* Structure for database lock */
56
typedef struct my_dblock_st
57
{
58
  char *name;        /* Database name        */
482 by Brian Aker
Remove uint.
59
  uint32_t name_length;  /* Database length name */
1 by brian
clean slate
60
} my_dblock_t;
61
62
63
/*
64
  lock_db key.
65
*/
66
481 by Brian Aker
Remove all of uchar.
67
extern "C" unsigned char* lock_db_get_key(my_dblock_t *, size_t *, bool not_used);
1 by brian
clean slate
68
481 by Brian Aker
Remove all of uchar.
69
unsigned char* lock_db_get_key(my_dblock_t *ptr, size_t *length,
253 by Brian Aker
Removed final my_bool from sql_db.
70
                       bool not_used __attribute__((unused)))
1 by brian
clean slate
71
{
72
  *length= ptr->name_length;
481 by Brian Aker
Remove all of uchar.
73
  return (unsigned char*) ptr->name;
1 by brian
clean slate
74
}
75
76
77
/*
78
  Free lock_db hash element.
79
*/
80
81
extern "C" void lock_db_free_element(void *ptr);
82
83
void lock_db_free_element(void *ptr)
84
{
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
85
  free(ptr);
1 by brian
clean slate
86
}
87
88
89
/*
90
  Delete a database lock entry from hash.
91
*/
92
482 by Brian Aker
Remove uint.
93
void lock_db_delete(const char *name, uint32_t length)
1 by brian
clean slate
94
{
95
  my_dblock_t *opt;
96
  safe_mutex_assert_owner(&LOCK_lock_db);
97
  if ((opt= (my_dblock_t *)hash_search(&lock_db_cache,
481 by Brian Aker
Remove all of uchar.
98
                                       (const unsigned char*) name, length)))
99
    hash_delete(&lock_db_cache, (unsigned char*) opt);
1 by brian
clean slate
100
}
101
102
103
/* Database options hash */
104
static HASH dboptions;
253 by Brian Aker
Removed final my_bool from sql_db.
105
static bool dboptions_init= 0;
1 by brian
clean slate
106
static rw_lock_t LOCK_dboptions;
107
108
/* Structure for database options */
109
typedef struct my_dbopt_st
110
{
111
  char *name;			/* Database name                  */
482 by Brian Aker
Remove uint.
112
  uint32_t name_length;		/* Database length name           */
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
113
  const CHARSET_INFO *charset;	/* Database default character set */
1 by brian
clean slate
114
} my_dbopt_t;
115
116
117
/*
118
  Function we use in the creation of our hash to get key.
119
*/
120
481 by Brian Aker
Remove all of uchar.
121
extern "C" unsigned char* dboptions_get_key(my_dbopt_t *opt, size_t *length,
253 by Brian Aker
Removed final my_bool from sql_db.
122
                                    bool not_used);
1 by brian
clean slate
123
481 by Brian Aker
Remove all of uchar.
124
unsigned char* dboptions_get_key(my_dbopt_t *opt, size_t *length,
253 by Brian Aker
Removed final my_bool from sql_db.
125
                         bool not_used __attribute__((unused)))
1 by brian
clean slate
126
{
127
  *length= opt->name_length;
481 by Brian Aker
Remove all of uchar.
128
  return (unsigned char*) opt->name;
1 by brian
clean slate
129
}
130
131
132
/*
133
  Helper function to write a query to binlog used by mysql_rm_db()
134
*/
135
520.1.22 by Brian Aker
Second pass of thd cleanup
136
static inline void write_to_binlog(Session *session, char *query, uint32_t q_len,
482 by Brian Aker
Remove uint.
137
                                   char *db, uint32_t db_len)
1 by brian
clean slate
138
{
520.1.22 by Brian Aker
Second pass of thd cleanup
139
  Query_log_event qinfo(session, query, q_len, 0, 0);
1 by brian
clean slate
140
  qinfo.error_code= 0;
141
  qinfo.db= db;
142
  qinfo.db_len= db_len;
143
  mysql_bin_log.write(&qinfo);
144
}  
145
146
147
/*
148
  Function to free dboptions hash element
149
*/
150
151
extern "C" void free_dbopt(void *dbopt);
152
153
void free_dbopt(void *dbopt)
154
{
481 by Brian Aker
Remove all of uchar.
155
  free((unsigned char*) dbopt);
1 by brian
clean slate
156
}
157
158
159
/* 
160
  Initialize database option hash and locked database hash.
161
162
  SYNOPSIS
163
    my_database_names()
164
165
  NOTES
166
    Must be called before any other database function is called.
167
168
  RETURN
169
    0	ok
170
    1	Fatal error
171
*/
172
173
bool my_database_names_init(void)
174
{
253 by Brian Aker
Removed final my_bool from sql_db.
175
  bool error= false;
1 by brian
clean slate
176
  (void) my_rwlock_init(&LOCK_dboptions, NULL);
177
  if (!dboptions_init)
178
  {
179
    dboptions_init= 1;
180
    error= hash_init(&dboptions, lower_case_table_names ? 
181
                     &my_charset_bin : system_charset_info,
182
                     32, 0, 0, (hash_get_key) dboptions_get_key,
183
                     free_dbopt,0) ||
184
           hash_init(&lock_db_cache, lower_case_table_names ? 
185
                     &my_charset_bin : system_charset_info,
186
                     32, 0, 0, (hash_get_key) lock_db_get_key,
187
                     lock_db_free_element,0);
188
189
  }
190
  return error;
191
}
192
193
194
195
/* 
196
  Free database option hash and locked databases hash.
197
*/
198
199
void my_database_names_free(void)
200
{
201
  if (dboptions_init)
202
  {
203
    dboptions_init= 0;
204
    hash_free(&dboptions);
205
    (void) rwlock_destroy(&LOCK_dboptions);
206
    hash_free(&lock_db_cache);
207
  }
208
}
209
210
211
/*
212
  Cleanup cached options
213
*/
214
215
void my_dbopt_cleanup(void)
216
{
217
  rw_wrlock(&LOCK_dboptions);
218
  hash_free(&dboptions);
219
  hash_init(&dboptions, lower_case_table_names ? 
220
            &my_charset_bin : system_charset_info,
221
            32, 0, 0, (hash_get_key) dboptions_get_key,
222
            free_dbopt,0);
223
  rw_unlock(&LOCK_dboptions);
224
}
225
226
227
/*
228
  Find database options in the hash.
229
  
230
  DESCRIPTION
231
    Search a database options in the hash, usings its path.
232
    Fills "create" on success.
233
  
234
  RETURN VALUES
235
    0 on success.
236
    1 on error.
237
*/
238
253 by Brian Aker
Removed final my_bool from sql_db.
239
static bool get_dbopt(const char *dbname, HA_CREATE_INFO *create)
1 by brian
clean slate
240
{
241
  my_dbopt_t *opt;
482 by Brian Aker
Remove uint.
242
  uint32_t length;
253 by Brian Aker
Removed final my_bool from sql_db.
243
  bool error= true;
1 by brian
clean slate
244
  
245
  length= (uint) strlen(dbname);
246
  
247
  rw_rdlock(&LOCK_dboptions);
481 by Brian Aker
Remove all of uchar.
248
  if ((opt= (my_dbopt_t*) hash_search(&dboptions, (unsigned char*) dbname, length)))
1 by brian
clean slate
249
  {
250
    create->default_table_charset= opt->charset;
253 by Brian Aker
Removed final my_bool from sql_db.
251
    error= true;
1 by brian
clean slate
252
  }
253
  rw_unlock(&LOCK_dboptions);
254
  return error;
255
}
256
257
258
/*
259
  Writes database options into the hash.
260
  
261
  DESCRIPTION
262
    Inserts database options into the hash, or updates
263
    options if they are already in the hash.
264
  
265
  RETURN VALUES
266
    0 on success.
267
    1 on error.
268
*/
269
253 by Brian Aker
Removed final my_bool from sql_db.
270
static bool put_dbopt(const char *dbname, HA_CREATE_INFO *create)
1 by brian
clean slate
271
{
272
  my_dbopt_t *opt;
482 by Brian Aker
Remove uint.
273
  uint32_t length;
253 by Brian Aker
Removed final my_bool from sql_db.
274
  bool error= false;
1 by brian
clean slate
275
276
  length= (uint) strlen(dbname);
277
  
278
  rw_wrlock(&LOCK_dboptions);
481 by Brian Aker
Remove all of uchar.
279
  if (!(opt= (my_dbopt_t*) hash_search(&dboptions, (unsigned char*) dbname, length)))
1 by brian
clean slate
280
  { 
281
    /* Options are not in the hash, insert them */
282
    char *tmp_name;
283
    if (!my_multi_malloc(MYF(MY_WME | MY_ZEROFILL),
284
                         &opt, (uint) sizeof(*opt), &tmp_name, (uint) length+1,
461 by Monty Taylor
Removed NullS. bu-bye.
285
                         NULL))
1 by brian
clean slate
286
    {
253 by Brian Aker
Removed final my_bool from sql_db.
287
      error= true;
1 by brian
clean slate
288
      goto end;
289
    }
290
    
291
    opt->name= tmp_name;
411.1.1 by Brian Aker
Work on removing GNU specific calls.
292
    my_stpcpy(opt->name, dbname);
1 by brian
clean slate
293
    opt->name_length= length;
294
    
481 by Brian Aker
Remove all of uchar.
295
    if ((error= my_hash_insert(&dboptions, (unsigned char*) opt)))
1 by brian
clean slate
296
    {
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
297
      free(opt);
1 by brian
clean slate
298
      goto end;
299
    }
300
  }
301
302
  /* Update / write options in hash */
303
  opt->charset= create->default_table_charset;
304
305
end:
306
  rw_unlock(&LOCK_dboptions);  
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
307
  return(error);
1 by brian
clean slate
308
}
309
310
311
/*
312
  Deletes database options from the hash.
313
*/
314
315
void del_dbopt(const char *path)
316
{
317
  my_dbopt_t *opt;
318
  rw_wrlock(&LOCK_dboptions);
481 by Brian Aker
Remove all of uchar.
319
  if ((opt= (my_dbopt_t *)hash_search(&dboptions, (const unsigned char*) path,
1 by brian
clean slate
320
                                      strlen(path))))
481 by Brian Aker
Remove all of uchar.
321
    hash_delete(&dboptions, (unsigned char*) opt);
1 by brian
clean slate
322
  rw_unlock(&LOCK_dboptions);
323
}
324
325
326
/*
327
  Create database options file:
328
329
  DESCRIPTION
330
    Currently database default charset is only stored there.
331
332
  RETURN VALUES
333
  0	ok
334
  1	Could not create file or write to it.  Error sent through my_error()
335
*/
336
520.1.22 by Brian Aker
Second pass of thd cleanup
337
static bool write_db_opt(Session *session, const char *path, const char *name, HA_CREATE_INFO *create)
1 by brian
clean slate
338
{
253 by Brian Aker
Removed final my_bool from sql_db.
339
  bool error= true;
323 by Brian Aker
Updated proto file for table (not FRM work).
340
  drizzle::Schema db;
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
341
342
  assert(name);
1 by brian
clean slate
343
344
  if (!create->default_table_charset)
520.1.22 by Brian Aker
Second pass of thd cleanup
345
    create->default_table_charset= session->variables.collation_server;
1 by brian
clean slate
346
347
  if (put_dbopt(path, create))
348
    return 1;
349
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
350
  db.set_name(name);
351
  db.set_characterset(create->default_table_charset->csname);
352
  db.set_collation(create->default_table_charset->name);
353
354
  fstream output(path, ios::out | ios::trunc | ios::binary);
355
  if (!db.SerializeToOstream(&output)) 
356
    error= false;
357
1 by brian
clean slate
358
  return error;
359
}
360
361
362
/*
363
  Load database options file
364
365
  load_db_opt()
366
  path		Path for option file
367
  create	Where to store the read options
368
369
  DESCRIPTION
370
371
  RETURN VALUES
372
  0	File found
373
  1	No database file or could not open it
374
375
*/
376
520.1.22 by Brian Aker
Second pass of thd cleanup
377
bool load_db_opt(Session *session, const char *path, HA_CREATE_INFO *create)
1 by brian
clean slate
378
{
379
  bool error=1;
323 by Brian Aker
Updated proto file for table (not FRM work).
380
  drizzle::Schema db;
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
381
  string buffer;
1 by brian
clean slate
382
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
383
  memset(create, 0, sizeof(*create));
520.1.22 by Brian Aker
Second pass of thd cleanup
384
  create->default_table_charset= session->variables.collation_server;
1 by brian
clean slate
385
386
  /* Check if options for this database are already in the hash */
387
  if (!get_dbopt(path, create))
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
388
    return(0);
1 by brian
clean slate
389
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
390
  fstream input(path, ios::in | ios::binary);
391
  if (!input)
392
    goto err1;
393
  else if (!db.ParseFromIstream(&input))
394
    goto err1;
395
396
  buffer= db.characterset();
397
  if (!(create->default_table_charset= get_charset_by_csname(buffer.c_str(), MY_CS_PRIMARY, MYF(0))))
398
  {
338 by Monty Taylor
Tagged more strings.
399
    sql_print_error(_("Error while loading database options: '%s':"),path);
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
400
    sql_print_error(ER(ER_UNKNOWN_COLLATION), buffer.c_str());
401
    create->default_table_charset= default_charset_info;
402
  }
403
404
  buffer= db.collation();
405
  if (!(create->default_table_charset= get_charset_by_name(buffer.c_str(), MYF(0))))
406
  {
338 by Monty Taylor
Tagged more strings.
407
    sql_print_error(_("Error while loading database options: '%s':"),path);
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
408
    sql_print_error(ER(ER_UNKNOWN_COLLATION), buffer.c_str());
409
    create->default_table_charset= default_charset_info;
410
  }
411
1 by brian
clean slate
412
  /*
413
    Put the loaded value into the hash.
414
    Note that another thread could've added the same
415
    entry to the hash after we called get_dbopt(),
416
    but it's not an error, as put_dbopt() takes this
417
    possibility into account.
418
  */
419
  error= put_dbopt(path, create);
420
421
err1:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
422
  return(error);
1 by brian
clean slate
423
}
424
425
426
/*
427
  Retrieve database options by name. Load database options file or fetch from
428
  cache.
429
430
  SYNOPSIS
431
    load_db_opt_by_name()
432
    db_name         Database name
433
    db_create_info  Where to store the database options
434
435
  DESCRIPTION
436
    load_db_opt_by_name() is a shortcut for load_db_opt().
437
438
  NOTE
439
    Although load_db_opt_by_name() (and load_db_opt()) returns status of
440
    the operation, it is useless usually and should be ignored. The problem
441
    is that there are 1) system databases ("mysql") and 2) virtual
442
    databases ("information_schema"), which do not contain options file.
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
443
    So, load_db_opt[_by_name]() returns false for these databases, but this
1 by brian
clean slate
444
    is not an error.
445
446
    load_db_opt[_by_name]() clears db_create_info structure in any case, so
447
    even on failure it contains valid data. So, common use case is just
448
    call load_db_opt[_by_name]() without checking return value and use
449
    db_create_info right after that.
450
451
  RETURN VALUES (read NOTE!)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
452
    false   Success
453
    true    Failed to retrieve options
1 by brian
clean slate
454
*/
455
520.1.22 by Brian Aker
Second pass of thd cleanup
456
bool load_db_opt_by_name(Session *session, const char *db_name,
1 by brian
clean slate
457
                         HA_CREATE_INFO *db_create_info)
458
{
459
  char db_opt_path[FN_REFLEN];
460
461
  /*
462
    Pass an empty file name, and the database options file name as extension
463
    to avoid table name to file name encoding.
464
  */
465
  (void) build_table_filename(db_opt_path, sizeof(db_opt_path),
466
                              db_name, "", MY_DB_OPT_FILE, 0);
467
520.1.22 by Brian Aker
Second pass of thd cleanup
468
  return load_db_opt(session, db_opt_path, db_create_info);
1 by brian
clean slate
469
}
470
471
472
/**
473
  Return default database collation.
474
520.1.22 by Brian Aker
Second pass of thd cleanup
475
  @param session     Thread context.
1 by brian
clean slate
476
  @param db_name Database name.
477
478
  @return CHARSET_INFO object. The operation always return valid character
479
    set, even if the database does not exist.
480
*/
481
520.1.22 by Brian Aker
Second pass of thd cleanup
482
const CHARSET_INFO *get_default_db_collation(Session *session, const char *db_name)
1 by brian
clean slate
483
{
484
  HA_CREATE_INFO db_info;
485
520.1.22 by Brian Aker
Second pass of thd cleanup
486
  if (session->db != NULL && strcmp(db_name, session->db) == 0)
487
    return session->db_charset;
1 by brian
clean slate
488
520.1.22 by Brian Aker
Second pass of thd cleanup
489
  load_db_opt_by_name(session, db_name, &db_info);
1 by brian
clean slate
490
491
  /*
492
    NOTE: even if load_db_opt_by_name() fails,
493
    db_info.default_table_charset contains valid character set
494
    (collation_server). We should not fail if load_db_opt_by_name() fails,
495
    because it is valid case. If a database has been created just by
496
    "mkdir", it does not contain db.opt file, but it is valid database.
497
  */
498
499
  return db_info.default_table_charset;
500
}
501
502
503
/*
504
  Create a database
505
506
  SYNOPSIS
507
  mysql_create_db()
520.1.22 by Brian Aker
Second pass of thd cleanup
508
  session		Thread handler
1 by brian
clean slate
509
  db		Name of database to create
510
		Function assumes that this is already validated.
511
  create_info	Database create options (like character set)
512
  silent	Used by replication when internally creating a database.
513
		In this case the entry should not be logged.
514
515
  SIDE-EFFECTS
516
   1. Report back to client that command succeeded (my_ok)
517
   2. Report errors to client
518
   3. Log event to binary log
519
   (The 'silent' flags turns off 1 and 3.)
520
521
  RETURN VALUES
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
522
  false ok
523
  true  Error
1 by brian
clean slate
524
525
*/
526
520.1.22 by Brian Aker
Second pass of thd cleanup
527
int mysql_create_db(Session *session, char *db, HA_CREATE_INFO *create_info, bool silent)
1 by brian
clean slate
528
{
529
  char	 path[FN_REFLEN+16];
530
  char	 tmp_query[FN_REFLEN+16];
531
  long result= 1;
532
  int error= 0;
15 by brian
Fix for stat, NETWARE removal
533
  struct stat stat_info;
482 by Brian Aker
Remove uint.
534
  uint32_t create_options= create_info ? create_info->options : 0;
535
  uint32_t path_len;
1 by brian
clean slate
536
537
  /* do not create 'information_schema' db */
575.4.7 by Monty Taylor
More header cleanup.
538
  if (!my_strcasecmp(system_charset_info, db, INFORMATION_SCHEMA_NAME.c_str()))
1 by brian
clean slate
539
  {
540
    my_error(ER_DB_CREATE_EXISTS, MYF(0), db);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
541
    return(-1);
1 by brian
clean slate
542
  }
543
544
  /*
545
    Do not create database if another thread is holding read lock.
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
546
    Wait for global read lock before acquiring LOCK_drizzle_create_db.
1 by brian
clean slate
547
    After wait_if_global_read_lock() we have protection against another
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
548
    global read lock. If we would acquire LOCK_drizzle_create_db first,
1 by brian
clean slate
549
    another thread could step in and get the global read lock before we
550
    reach wait_if_global_read_lock(). If this thread tries the same as we
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
551
    (admin a db), it would then go and wait on LOCK_drizzle_create_db...
1 by brian
clean slate
552
    Furthermore wait_if_global_read_lock() checks if the current thread
553
    has the global read lock and refuses the operation with
554
    ER_CANT_UPDATE_WITH_READLOCK if applicable.
555
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
556
  if (wait_if_global_read_lock(session, 0, 1))
1 by brian
clean slate
557
  {
558
    error= -1;
559
    goto exit2;
560
  }
561
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
562
  pthread_mutex_lock(&LOCK_drizzle_create_db);
1 by brian
clean slate
563
564
  /* Check directory */
565
  path_len= build_table_filename(path, sizeof(path), db, "", "", 0);
566
  path[path_len-1]= 0;                    // Remove last '/' from path
567
15 by brian
Fix for stat, NETWARE removal
568
  if (!stat(path,&stat_info))
1 by brian
clean slate
569
  {
570
    if (!(create_options & HA_LEX_CREATE_IF_NOT_EXISTS))
571
    {
572
      my_error(ER_DB_CREATE_EXISTS, MYF(0), db);
573
      error= -1;
574
      goto exit;
575
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
576
    push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
577
			ER_DB_CREATE_EXISTS, ER(ER_DB_CREATE_EXISTS), db);
578
    if (!silent)
520.1.22 by Brian Aker
Second pass of thd cleanup
579
      my_ok(session);
1 by brian
clean slate
580
    error= 0;
581
    goto exit;
582
  }
583
  else
584
  {
15 by brian
Fix for stat, NETWARE removal
585
    if (errno != ENOENT)
1 by brian
clean slate
586
    {
15 by brian
Fix for stat, NETWARE removal
587
      my_error(EE_STAT, MYF(0), path, errno);
1 by brian
clean slate
588
      goto exit;
589
    }
590
    if (my_mkdir(path,0777,MYF(0)) < 0)
591
    {
592
      my_error(ER_CANT_CREATE_DB, MYF(0), db, my_errno);
593
      error= -1;
594
      goto exit;
595
    }
596
  }
597
598
  path[path_len-1]= FN_LIBCHAR;
599
  strmake(path+path_len, MY_DB_OPT_FILE, sizeof(path)-path_len-1);
520.1.22 by Brian Aker
Second pass of thd cleanup
600
  if (write_db_opt(session, path, db, create_info))
1 by brian
clean slate
601
  {
602
    /*
603
      Could not create options file.
604
      Restore things to beginning.
605
    */
606
    path[path_len]= 0;
607
    if (rmdir(path) >= 0)
608
    {
609
      error= -1;
610
      goto exit;
611
    }
612
    /*
613
      We come here when we managed to create the database, but not the option
614
      file.  In this case it's best to just continue as if nothing has
615
      happened.  (This is a very unlikely senario)
616
    */
617
  }
618
  
619
  if (!silent)
620
  {
621
    char *query;
482 by Brian Aker
Remove uint.
622
    uint32_t query_length;
1 by brian
clean slate
623
520.1.22 by Brian Aker
Second pass of thd cleanup
624
    if (!session->query)				// Only in replication
1 by brian
clean slate
625
    {
626
      query= 	     tmp_query;
627
      query_length= (uint) (strxmov(tmp_query,"create database `",
461 by Monty Taylor
Removed NullS. bu-bye.
628
                                    db, "`", NULL) - tmp_query);
1 by brian
clean slate
629
    }
630
    else
631
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
632
      query= 	    session->query;
633
      query_length= session->query_length;
1 by brian
clean slate
634
    }
635
636
    if (mysql_bin_log.is_open())
637
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
638
      Query_log_event qinfo(session, query, query_length, 0, 
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
639
			    /* suppress_use */ true);
1 by brian
clean slate
640
641
      /*
642
	Write should use the database being created as the "current
643
        database" and not the threads current database, which is the
644
        default. If we do not change the "current database" to the
645
        database being created, the CREATE statement will not be
646
        replicated when using --binlog-do-db to select databases to be
647
        replicated. 
648
649
	An example (--binlog-do-db=sisyfos):
650
       
651
          CREATE DATABASE bob;        # Not replicated
652
          USE bob;                    # 'bob' is the current database
653
          CREATE DATABASE sisyfos;    # Not replicated since 'bob' is
654
                                      # current database.
655
          USE sisyfos;                # Will give error on slave since
656
                                      # database does not exist.
657
      */
658
      qinfo.db     = db;
659
      qinfo.db_len = strlen(db);
660
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
661
      /* These DDL methods and logging protected with LOCK_drizzle_create_db */
1 by brian
clean slate
662
      mysql_bin_log.write(&qinfo);
663
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
664
    my_ok(session, result);
1 by brian
clean slate
665
  }
666
667
exit:
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
668
  pthread_mutex_unlock(&LOCK_drizzle_create_db);
520.1.22 by Brian Aker
Second pass of thd cleanup
669
  start_waiting_global_read_lock(session);
1 by brian
clean slate
670
exit2:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
671
  return(error);
1 by brian
clean slate
672
}
673
674
675
/* db-name is already validated when we come here */
676
520.1.22 by Brian Aker
Second pass of thd cleanup
677
bool mysql_alter_db(Session *session, const char *db, HA_CREATE_INFO *create_info)
1 by brian
clean slate
678
{
679
  char path[FN_REFLEN+16];
680
  long result=1;
253 by Brian Aker
Removed final my_bool from sql_db.
681
  int error= false;
1 by brian
clean slate
682
683
  /*
684
    Do not alter database if another thread is holding read lock.
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
685
    Wait for global read lock before acquiring LOCK_drizzle_create_db.
1 by brian
clean slate
686
    After wait_if_global_read_lock() we have protection against another
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
687
    global read lock. If we would acquire LOCK_drizzle_create_db first,
1 by brian
clean slate
688
    another thread could step in and get the global read lock before we
689
    reach wait_if_global_read_lock(). If this thread tries the same as we
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
690
    (admin a db), it would then go and wait on LOCK_drizzle_create_db...
1 by brian
clean slate
691
    Furthermore wait_if_global_read_lock() checks if the current thread
692
    has the global read lock and refuses the operation with
693
    ER_CANT_UPDATE_WITH_READLOCK if applicable.
694
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
695
  if ((error=wait_if_global_read_lock(session,0,1)))
1 by brian
clean slate
696
    goto exit2;
697
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
698
  pthread_mutex_lock(&LOCK_drizzle_create_db);
1 by brian
clean slate
699
700
  /* 
701
     Recreate db options file: /dbpath/.db.opt
702
     We pass MY_DB_OPT_FILE as "extension" to avoid
703
     "table name to file name" encoding.
704
  */
705
  build_table_filename(path, sizeof(path), db, "", MY_DB_OPT_FILE, 0);
520.1.22 by Brian Aker
Second pass of thd cleanup
706
  if ((error=write_db_opt(session, path, db, create_info)))
1 by brian
clean slate
707
    goto exit;
708
709
  /* Change options if current database is being altered. */
710
520.1.22 by Brian Aker
Second pass of thd cleanup
711
  if (session->db && !strcmp(session->db,db))
1 by brian
clean slate
712
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
713
    session->db_charset= create_info->default_table_charset ?
1 by brian
clean slate
714
		     create_info->default_table_charset :
520.1.22 by Brian Aker
Second pass of thd cleanup
715
		     session->variables.collation_server;
716
    session->variables.collation_database= session->db_charset;
1 by brian
clean slate
717
  }
718
719
  if (mysql_bin_log.is_open())
720
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
721
    Query_log_event qinfo(session, session->query, session->query_length, 0,
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
722
			  /* suppress_use */ true);
1 by brian
clean slate
723
724
    /*
725
      Write should use the database being created as the "current
726
      database" and not the threads current database, which is the
727
      default.
728
    */
729
    qinfo.db     = db;
730
    qinfo.db_len = strlen(db);
731
520.1.22 by Brian Aker
Second pass of thd cleanup
732
    session->clear_error();
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
733
    /* These DDL methods and logging protected with LOCK_drizzle_create_db */
1 by brian
clean slate
734
    mysql_bin_log.write(&qinfo);
735
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
736
  my_ok(session, result);
1 by brian
clean slate
737
738
exit:
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
739
  pthread_mutex_unlock(&LOCK_drizzle_create_db);
520.1.22 by Brian Aker
Second pass of thd cleanup
740
  start_waiting_global_read_lock(session);
1 by brian
clean slate
741
exit2:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
742
  return(error);
1 by brian
clean slate
743
}
744
745
746
/*
747
  Drop all tables in a database and the database itself
748
749
  SYNOPSIS
750
    mysql_rm_db()
520.1.22 by Brian Aker
Second pass of thd cleanup
751
    session			Thread handle
1 by brian
clean slate
752
    db			Database name in the case given by user
753
		        It's already validated and set to lower case
754
                        (if needed) when we come here
755
    if_exists		Don't give error if database doesn't exists
756
    silent		Don't generate errors
757
758
  RETURN
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
759
    false ok (Database dropped)
1 by brian
clean slate
760
    ERROR Error
761
*/
762
520.1.22 by Brian Aker
Second pass of thd cleanup
763
bool mysql_rm_db(Session *session,char *db,bool if_exists, bool silent)
1 by brian
clean slate
764
{
765
  long deleted=0;
253 by Brian Aker
Removed final my_bool from sql_db.
766
  int error= false;
1 by brian
clean slate
767
  char	path[FN_REFLEN+16];
768
  MY_DIR *dirp;
482 by Brian Aker
Remove uint.
769
  uint32_t length;
327.2.4 by Brian Aker
Refactoring table.h
770
  TableList* dropped_tables= 0;
1 by brian
clean slate
771
772
  if (db && (strcmp(db, "information_schema") == 0))
773
  {
575.4.7 by Monty Taylor
More header cleanup.
774
    my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), "", "", INFORMATION_SCHEMA_NAME.c_str());
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
775
    return(true);
1 by brian
clean slate
776
  }
777
778
  /*
779
    Do not drop database if another thread is holding read lock.
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
780
    Wait for global read lock before acquiring LOCK_drizzle_create_db.
1 by brian
clean slate
781
    After wait_if_global_read_lock() we have protection against another
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
782
    global read lock. If we would acquire LOCK_drizzle_create_db first,
1 by brian
clean slate
783
    another thread could step in and get the global read lock before we
784
    reach wait_if_global_read_lock(). If this thread tries the same as we
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
785
    (admin a db), it would then go and wait on LOCK_drizzle_create_db...
1 by brian
clean slate
786
    Furthermore wait_if_global_read_lock() checks if the current thread
787
    has the global read lock and refuses the operation with
788
    ER_CANT_UPDATE_WITH_READLOCK if applicable.
789
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
790
  if (wait_if_global_read_lock(session, 0, 1))
1 by brian
clean slate
791
  {
792
    error= -1;
793
    goto exit2;
794
  }
795
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
796
  pthread_mutex_lock(&LOCK_drizzle_create_db);
1 by brian
clean slate
797
798
  /*
799
    This statement will be replicated as a statement, even when using
800
    row-based replication. The flag will be reset at the end of the
801
    statement.
802
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
803
  session->clear_current_stmt_binlog_row_based();
1 by brian
clean slate
804
805
  length= build_table_filename(path, sizeof(path), db, "", "", 0);
411.1.1 by Brian Aker
Work on removing GNU specific calls.
806
  my_stpcpy(path+length, MY_DB_OPT_FILE);		// Append db option file name
1 by brian
clean slate
807
  del_dbopt(path);				// Remove dboption hash entry
808
  path[length]= '\0';				// Remove file name
809
810
  /* See if the directory exists */
811
  if (!(dirp= my_dir(path,MYF(MY_DONT_SORT))))
812
  {
813
    if (!if_exists)
814
    {
815
      error= -1;
816
      my_error(ER_DB_DROP_EXISTS, MYF(0), db);
817
      goto exit;
818
    }
819
    else
520.1.22 by Brian Aker
Second pass of thd cleanup
820
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
821
			  ER_DB_DROP_EXISTS, ER(ER_DB_DROP_EXISTS), db);
822
  }
823
  else
824
  {
825
    pthread_mutex_lock(&LOCK_open);
826
    remove_db_from_cache(db);
827
    pthread_mutex_unlock(&LOCK_open);
828
829
    
830
    error= -1;
520.1.22 by Brian Aker
Second pass of thd cleanup
831
    if ((deleted= mysql_rm_known_files(session, dirp, db, path, 0,
1 by brian
clean slate
832
                                       &dropped_tables)) >= 0)
833
    {
834
      ha_drop_database(path);
835
      error = 0;
836
    }
837
  }
838
  if (!silent && deleted>=0)
839
  {
840
    const char *query;
308 by Brian Aker
ulong conversion
841
    uint32_t query_length;
520.1.22 by Brian Aker
Second pass of thd cleanup
842
    if (!session->query)
1 by brian
clean slate
843
    {
844
      /* The client used the old obsolete mysql_drop_db() call */
845
      query= path;
846
      query_length= (uint) (strxmov(path, "drop database `", db, "`",
461 by Monty Taylor
Removed NullS. bu-bye.
847
                                     NULL) - path);
1 by brian
clean slate
848
    }
849
    else
850
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
851
      query =session->query;
852
      query_length= session->query_length;
1 by brian
clean slate
853
    }
854
    if (mysql_bin_log.is_open())
855
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
856
      Query_log_event qinfo(session, query, query_length, 0, 
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
857
			    /* suppress_use */ true);
1 by brian
clean slate
858
      /*
859
        Write should use the database being created as the "current
860
        database" and not the threads current database, which is the
861
        default.
862
      */
863
      qinfo.db     = db;
864
      qinfo.db_len = strlen(db);
865
520.1.22 by Brian Aker
Second pass of thd cleanup
866
      session->clear_error();
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
867
      /* These DDL methods and logging protected with LOCK_drizzle_create_db */
1 by brian
clean slate
868
      mysql_bin_log.write(&qinfo);
869
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
870
    session->clear_error();
871
    session->server_status|= SERVER_STATUS_DB_DROPPED;
872
    my_ok(session, (uint32_t) deleted);
873
    session->server_status&= ~SERVER_STATUS_DB_DROPPED;
1 by brian
clean slate
874
  }
875
  else if (mysql_bin_log.is_open())
876
  {
877
    char *query, *query_pos, *query_end, *query_data_start;
327.2.4 by Brian Aker
Refactoring table.h
878
    TableList *tbl;
482 by Brian Aker
Remove uint.
879
    uint32_t db_len;
1 by brian
clean slate
880
520.1.22 by Brian Aker
Second pass of thd cleanup
881
    if (!(query= (char*) session->alloc(MAX_DROP_TABLE_Q_LEN)))
1 by brian
clean slate
882
      goto exit; /* not much else we can do */
411.1.1 by Brian Aker
Work on removing GNU specific calls.
883
    query_pos= query_data_start= my_stpcpy(query,"drop table ");
1 by brian
clean slate
884
    query_end= query + MAX_DROP_TABLE_Q_LEN;
885
    db_len= strlen(db);
886
887
    for (tbl= dropped_tables; tbl; tbl= tbl->next_local)
888
    {
482 by Brian Aker
Remove uint.
889
      uint32_t tbl_name_len;
1 by brian
clean slate
890
891
      /* 3 for the quotes and the comma*/
892
      tbl_name_len= strlen(tbl->table_name) + 3;
893
      if (query_pos + tbl_name_len + 1 >= query_end)
894
      {
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
895
        /* These DDL methods and logging protected with LOCK_drizzle_create_db */
520.1.22 by Brian Aker
Second pass of thd cleanup
896
        write_to_binlog(session, query, query_pos -1 - query, db, db_len);
1 by brian
clean slate
897
        query_pos= query_data_start;
898
      }
899
900
      *query_pos++ = '`';
411.1.1 by Brian Aker
Work on removing GNU specific calls.
901
      query_pos= my_stpcpy(query_pos,tbl->table_name);
1 by brian
clean slate
902
      *query_pos++ = '`';
903
      *query_pos++ = ',';
904
    }
905
906
    if (query_pos != query_data_start)
907
    {
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
908
      /* These DDL methods and logging protected with LOCK_drizzle_create_db */
520.1.22 by Brian Aker
Second pass of thd cleanup
909
      write_to_binlog(session, query, query_pos -1 - query, db, db_len);
1 by brian
clean slate
910
    }
911
  }
912
913
exit:
914
  /*
915
    If this database was the client's selected database, we silently
916
    change the client's selected database to nothing (to have an empty
520.1.22 by Brian Aker
Second pass of thd cleanup
917
    SELECT DATABASE() in the future). For this we free() session->db and set
1 by brian
clean slate
918
    it to 0.
919
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
920
  if (session->db && !strcmp(session->db, db))
921
    mysql_change_db_impl(session, NULL, session->variables.collation_server);
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
922
  pthread_mutex_unlock(&LOCK_drizzle_create_db);
520.1.22 by Brian Aker
Second pass of thd cleanup
923
  start_waiting_global_read_lock(session);
1 by brian
clean slate
924
exit2:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
925
  return(error);
1 by brian
clean slate
926
}
927
928
/*
255 by Brian Aker
Removed RAID table delete point.
929
  Removes files with known extensions plus.
520.1.22 by Brian Aker
Second pass of thd cleanup
930
  session MUST be set when calling this function!
1 by brian
clean slate
931
*/
932
520.1.22 by Brian Aker
Second pass of thd cleanup
933
static long mysql_rm_known_files(Session *session, MY_DIR *dirp, const char *db,
482 by Brian Aker
Remove uint.
934
				 const char *org_path, uint32_t level,
327.2.4 by Brian Aker
Refactoring table.h
935
                                 TableList **dropped_tables)
1 by brian
clean slate
936
{
937
  long deleted=0;
308 by Brian Aker
ulong conversion
938
  uint32_t found_other_files=0;
1 by brian
clean slate
939
  char filePath[FN_REFLEN];
327.2.4 by Brian Aker
Refactoring table.h
940
  TableList *tot_list=0, **tot_list_next;
1 by brian
clean slate
941
942
  tot_list_next= &tot_list;
943
482 by Brian Aker
Remove uint.
944
  for (uint32_t idx=0 ;
520.1.22 by Brian Aker
Second pass of thd cleanup
945
       idx < (uint) dirp->number_off_files && !session->killed ;
1 by brian
clean slate
946
       idx++)
947
  {
948
    FILEINFO *file=dirp->dir_entry+idx;
949
    char *extension;
950
951
    /* skiping . and .. */
952
    if (file->name[0] == '.' && (!file->name[1] ||
953
       (file->name[1] == '.' &&  !file->name[2])))
954
      continue;
955
956
    if (!(extension= strrchr(file->name, '.')))
376 by Brian Aker
strend remove
957
      extension= strchr(file->name, '\0');
1 by brian
clean slate
958
    if (find_type(extension, &deletable_extentions,1+2) <= 0)
959
    {
960
      if (find_type(extension, ha_known_exts(),1+2) <= 0)
961
	found_other_files++;
962
      continue;
963
    }
964
    /* just for safety we use files_charset_info */
965
    if (db && !my_strcasecmp(files_charset_info,
966
                             extension, reg_ext))
967
    {
968
      /* Drop the table nicely */
969
      *extension= 0;			// Remove extension
327.2.4 by Brian Aker
Refactoring table.h
970
      TableList *table_list=(TableList*)
520.1.22 by Brian Aker
Second pass of thd cleanup
971
                              session->calloc(sizeof(*table_list) + 
1 by brian
clean slate
972
                                          strlen(db) + 1 +
973
                                          MYSQL50_TABLE_NAME_PREFIX_LENGTH + 
974
                                          strlen(file->name) + 1);
975
976
      if (!table_list)
977
        goto err;
978
      table_list->db= (char*) (table_list+1);
411.1.1 by Brian Aker
Work on removing GNU specific calls.
979
      table_list->table_name= my_stpcpy(table_list->db, db) + 1;
398.1.10 by Monty Taylor
Actually removed VOID() this time.
980
      filename_to_tablename(file->name, table_list->table_name,
981
                            MYSQL50_TABLE_NAME_PREFIX_LENGTH +
982
                            strlen(file->name) + 1);
1 by brian
clean slate
983
      table_list->alias= table_list->table_name;	// If lower_case_table_names=2
984
      table_list->internal_tmp_table= is_prefix(file->name, tmp_file_prefix);
985
      /* Link into list */
986
      (*tot_list_next)= table_list;
987
      tot_list_next= &table_list->next_local;
988
      deleted++;
989
    }
990
    else
991
    {
461 by Monty Taylor
Removed NullS. bu-bye.
992
      strxmov(filePath, org_path, "/", file->name, NULL);
1 by brian
clean slate
993
      if (my_delete_with_symlink(filePath,MYF(MY_WME)))
994
      {
995
	goto err;
996
      }
997
    }
998
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
999
  if (session->killed ||
1000
      (tot_list && mysql_rm_table_part2(session, tot_list, 1, 0, 1, 1)))
1 by brian
clean slate
1001
    goto err;
1002
1003
  my_dirend(dirp);  
1004
  
1005
  if (dropped_tables)
1006
    *dropped_tables= tot_list;
1007
  
1008
  /*
1009
    If the directory is a symbolic link, remove the link first, then
1010
    remove the directory the symbolic link pointed at
1011
  */
1012
  if (found_other_files)
1013
  {
1014
    my_error(ER_DB_DROP_RMDIR, MYF(0), org_path, EEXIST);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1015
    return(-1);
1 by brian
clean slate
1016
  }
1017
  else
1018
  {
1019
    /* Don't give errors if we can't delete 'RAID' directory */
1020
    if (rm_dir_w_symlink(org_path, level == 0))
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1021
      return(-1);
1 by brian
clean slate
1022
  }
1023
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1024
  return(deleted);
1 by brian
clean slate
1025
1026
err:
1027
  my_dirend(dirp);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1028
  return(-1);
1 by brian
clean slate
1029
}
1030
1031
1032
/*
1033
  Remove directory with symlink
1034
1035
  SYNOPSIS
1036
    rm_dir_w_symlink()
1037
    org_path    path of derictory
1038
    send_error  send errors
1039
  RETURN
1040
    0 OK
1041
    1 ERROR
1042
*/
1043
253 by Brian Aker
Removed final my_bool from sql_db.
1044
static bool rm_dir_w_symlink(const char *org_path, bool send_error)
1 by brian
clean slate
1045
{
1046
  char tmp_path[FN_REFLEN], *pos;
1047
  char *path= tmp_path;
1048
  unpack_filename(tmp_path, org_path);
1049
#ifdef HAVE_READLINK
1050
  int error;
1051
  char tmp2_path[FN_REFLEN];
1052
1053
  /* Remove end FN_LIBCHAR as this causes problem on Linux in readlink */
376 by Brian Aker
strend remove
1054
  pos= strchr(path, '\0');
1 by brian
clean slate
1055
  if (pos > path && pos[-1] == FN_LIBCHAR)
1056
    *--pos=0;
1057
1058
  if ((error= my_readlink(tmp2_path, path, MYF(MY_WME))) < 0)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1059
    return(1);
1 by brian
clean slate
1060
  if (!error)
1061
  {
1062
    if (my_delete(path, MYF(send_error ? MY_WME : 0)))
1063
    {
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1064
      return(send_error);
1 by brian
clean slate
1065
    }
1066
    /* Delete directory symbolic link pointed at */
1067
    path= tmp2_path;
1068
  }
1069
#endif
1070
  /* Remove last FN_LIBCHAR to not cause a problem on OS/2 */
376 by Brian Aker
strend remove
1071
  pos= strchr(path, '\0');
1 by brian
clean slate
1072
1073
  if (pos > path && pos[-1] == FN_LIBCHAR)
1074
    *--pos=0;
1075
  if (rmdir(path) < 0 && send_error)
1076
  {
1077
    my_error(ER_DB_DROP_RMDIR, MYF(0), path, errno);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1078
    return(1);
1 by brian
clean slate
1079
  }
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1080
  return(0);
1 by brian
clean slate
1081
}
1082
1083
1084
/**
1085
  @brief Internal implementation: switch current database to a valid one.
1086
520.1.22 by Brian Aker
Second pass of thd cleanup
1087
  @param session            Thread context.
1 by brian
clean slate
1088
  @param new_db_name    Name of the database to switch to. The function will
1089
                        take ownership of the name (the caller must not free
1090
                        the allocated memory). If the name is NULL, we're
1091
                        going to switch to NULL db.
1092
  @param new_db_charset Character set of the new database.
1093
*/
1094
520.1.22 by Brian Aker
Second pass of thd cleanup
1095
static void mysql_change_db_impl(Session *session,
1 by brian
clean slate
1096
                                 LEX_STRING *new_db_name,
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
1097
                                 const CHARSET_INFO * const new_db_charset)
1 by brian
clean slate
1098
{
520.1.21 by Brian Aker
THD -> Session rename
1099
  /* 1. Change current database in Session. */
1 by brian
clean slate
1100
1101
  if (new_db_name == NULL)
1102
  {
1103
    /*
520.1.21 by Brian Aker
THD -> Session rename
1104
      Session::set_db() does all the job -- it frees previous database name and
1 by brian
clean slate
1105
      sets the new one.
1106
    */
1107
520.1.22 by Brian Aker
Second pass of thd cleanup
1108
    session->set_db(NULL, 0);
1 by brian
clean slate
1109
  }
575.4.7 by Monty Taylor
More header cleanup.
1110
  else if (my_strcasecmp(system_charset_info, new_db_name->str,
1111
                         INFORMATION_SCHEMA_NAME.c_str()) == 0)
1 by brian
clean slate
1112
  {
1113
    /*
520.1.21 by Brian Aker
THD -> Session rename
1114
      Here we must use Session::set_db(), because we want to copy
1 by brian
clean slate
1115
      INFORMATION_SCHEMA_NAME constant.
1116
    */
1117
575.4.7 by Monty Taylor
More header cleanup.
1118
    session->set_db(INFORMATION_SCHEMA_NAME.c_str(),
1119
                    INFORMATION_SCHEMA_NAME.length());
1 by brian
clean slate
1120
  }
1121
  else
1122
  {
1123
    /*
520.1.21 by Brian Aker
THD -> Session rename
1124
      Here we already have a copy of database name to be used in Session. So,
1125
      we just call Session::reset_db(). Since Session::reset_db() does not releases
1 by brian
clean slate
1126
      the previous database name, we should do it explicitly.
1127
    */
1128
520.1.22 by Brian Aker
Second pass of thd cleanup
1129
    if (session->db)
1130
      free(session->db);
1 by brian
clean slate
1131
520.1.22 by Brian Aker
Second pass of thd cleanup
1132
    session->reset_db(new_db_name->str, new_db_name->length);
1 by brian
clean slate
1133
  }
1134
1135
  /* 3. Update db-charset environment variables. */
1136
520.1.22 by Brian Aker
Second pass of thd cleanup
1137
  session->db_charset= new_db_charset;
1138
  session->variables.collation_database= new_db_charset;
1 by brian
clean slate
1139
}
1140
1141
1142
1143
/**
1144
  Backup the current database name before switch.
1145
520.1.22 by Brian Aker
Second pass of thd cleanup
1146
  @param[in]      session             thread handle
1 by brian
clean slate
1147
  @param[in, out] saved_db_name   IN: "str" points to a buffer where to store
1148
                                  the old database name, "length" contains the
1149
                                  buffer size
1150
                                  OUT: if the current (default) database is
1151
                                  not NULL, its name is copied to the
1152
                                  buffer pointed at by "str"
1153
                                  and "length" is updated accordingly.
1154
                                  Otherwise "str" is set to NULL and
1155
                                  "length" is set to 0.
1156
*/
1157
520.1.22 by Brian Aker
Second pass of thd cleanup
1158
static void backup_current_db_name(Session *session,
1 by brian
clean slate
1159
                                   LEX_STRING *saved_db_name)
1160
{
520.1.22 by Brian Aker
Second pass of thd cleanup
1161
  if (!session->db)
1 by brian
clean slate
1162
  {
1163
    /* No current (default) database selected. */
1164
1165
    saved_db_name->str= NULL;
1166
    saved_db_name->length= 0;
1167
  }
1168
  else
1169
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
1170
    strmake(saved_db_name->str, session->db, saved_db_name->length - 1);
1171
    saved_db_name->length= session->db_length;
1 by brian
clean slate
1172
  }
1173
}
1174
1175
1176
/**
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1177
  Return true if db1_name is equal to db2_name, false otherwise.
1 by brian
clean slate
1178
1179
  The function allows to compare database names according to the MySQL
1180
  rules. The database names db1 and db2 are equal if:
1181
     - db1 is NULL and db2 is NULL;
1182
     or
1183
     - db1 is not-NULL, db2 is not-NULL, db1 is equal (ignoring case) to
1184
       db2 in system character set (UTF8).
1185
*/
1186
1187
static inline bool
1188
cmp_db_names(const char *db1_name,
1189
             const char *db2_name)
1190
{
1191
  return
1192
         /* db1 is NULL and db2 is NULL */
1193
         (!db1_name && !db2_name) ||
1194
1195
         /* db1 is not-NULL, db2 is not-NULL, db1 == db2. */
1196
         (db1_name && db2_name && my_strcasecmp(system_charset_info, db1_name, db2_name) == 0);
1197
}
1198
1199
1200
/**
1201
  @brief Change the current database and its attributes unconditionally.
1202
520.1.22 by Brian Aker
Second pass of thd cleanup
1203
  @param session          thread handle
1 by brian
clean slate
1204
  @param new_db_name  database name
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1205
  @param force_switch if force_switch is false, then the operation will fail if
1 by brian
clean slate
1206
1207
                        - new_db_name is NULL or empty;
1208
1209
                        - OR new database name is invalid
1210
                          (check_db_name() failed);
1211
1212
                        - OR user has no privilege on the new database;
1213
1214
                        - OR new database does not exist;
1215
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1216
                      if force_switch is true, then
1 by brian
clean slate
1217
1218
                        - if new_db_name is NULL or empty, the current
1219
                          database will be NULL, @@collation_database will
1220
                          be set to @@collation_server, the operation will
1221
                          succeed.
1222
1223
                        - if new database name is invalid
1224
                          (check_db_name() failed), the current database
1225
                          will be NULL, @@collation_database will be set to
1226
                          @@collation_server, but the operation will fail;
1227
1228
                        - user privileges will not be checked
520.1.21 by Brian Aker
THD -> Session rename
1229
                          (Session::db_access however is updated);
1 by brian
clean slate
1230
1231
                          TODO: is this really the intention?
1232
                                (see sp-security.test).
1233
1234
                        - if new database does not exist,the current database
1235
                          will be NULL, @@collation_database will be set to
1236
                          @@collation_server, a warning will be thrown, the
1237
                          operation will succeed.
1238
1239
  @details The function checks that the database name corresponds to a
1240
  valid and existent database, checks access rights and changes the current
1241
  database with database attributes (@@collation_database session variable,
520.1.21 by Brian Aker
THD -> Session rename
1242
  Session::db_access).
1 by brian
clean slate
1243
1244
  This function is not the only way to switch the database that is
1245
  currently employed. When the replication slave thread switches the
520.1.22 by Brian Aker
Second pass of thd cleanup
1246
  database before executing a query, it calls session->set_db directly.
1 by brian
clean slate
1247
  However, if the query, in turn, uses a stored routine, the stored routine
1248
  will use this function, even if it's run on the slave.
1249
1250
  This function allocates the name of the database on the system heap: this
1251
  is necessary to be able to uniformly change the database from any module
1252
  of the server. Up to 5.0 different modules were using different memory to
1253
  store the name of the database, and this led to memory corruption:
1254
  a stack pointer set by Stored Procedures was used by replication after
1255
  the stack address was long gone.
1256
1257
  @return Operation status
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1258
    @retval false Success
1259
    @retval true  Error
1 by brian
clean slate
1260
*/
1261
520.1.22 by Brian Aker
Second pass of thd cleanup
1262
bool mysql_change_db(Session *session, const LEX_STRING *new_db_name, bool force_switch)
1 by brian
clean slate
1263
{
1264
  LEX_STRING new_db_file_name;
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
1265
  const CHARSET_INFO *db_default_cl;
1 by brian
clean slate
1266
1267
  if (new_db_name == NULL ||
1268
      new_db_name->length == 0)
1269
  {
1270
    if (force_switch)
1271
    {
1272
      /*
1273
        This can happen only if we're switching the current database back
1274
        after loading stored program. The thing is that loading of stored
1275
        program can happen when there is no current database.
1276
1277
        TODO: actually, new_db_name and new_db_name->str seem to be always
1278
        non-NULL. In case of stored program, new_db_name->str == "" and
1279
        new_db_name->length == 0.
1280
      */
1281
520.1.22 by Brian Aker
Second pass of thd cleanup
1282
      mysql_change_db_impl(session, NULL, session->variables.collation_server);
1 by brian
clean slate
1283
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1284
      return(false);
1 by brian
clean slate
1285
    }
1286
    else
1287
    {
1288
      my_message(ER_NO_DB_ERROR, ER(ER_NO_DB_ERROR), MYF(0));
1289
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1290
      return(true);
1 by brian
clean slate
1291
    }
1292
  }
1293
1294
  if (my_strcasecmp(system_charset_info, new_db_name->str,
575.4.7 by Monty Taylor
More header cleanup.
1295
                    INFORMATION_SCHEMA_NAME.c_str()) == 0)
1 by brian
clean slate
1296
  {
1297
    /* Switch the current database to INFORMATION_SCHEMA. */
575.4.7 by Monty Taylor
More header cleanup.
1298
    /* const_cast<> is safe here: mysql_change_db_impl does a copy */
1299
    LEX_STRING is_name= { const_cast<char *>(INFORMATION_SCHEMA_NAME.c_str()),
1300
                          INFORMATION_SCHEMA_NAME.length() };
1301
    mysql_change_db_impl(session, &is_name, system_charset_info);
1 by brian
clean slate
1302
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1303
    return(false);
1 by brian
clean slate
1304
  }
1305
1306
  /*
1307
    Now we need to make a copy because check_db_name requires a
1308
    non-constant argument. Actually, it takes database file name.
1309
1310
    TODO: fix check_db_name().
1311
  */
1312
1313
  new_db_file_name.str= my_strndup(new_db_name->str, new_db_name->length,
1314
                                   MYF(MY_WME));
1315
  new_db_file_name.length= new_db_name->length;
1316
1317
  if (new_db_file_name.str == NULL)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1318
    return(true);                             /* the error is set */
1 by brian
clean slate
1319
1320
  /*
1321
    NOTE: if check_db_name() fails, we should throw an error in any case,
1322
    even if we are called from sp_head::execute().
1323
1324
    It's next to impossible however to get this error when we are called
1325
    from sp_head::execute(). But let's switch the current database to NULL
1326
    in this case to be sure.
1327
  */
1328
1329
  if (check_db_name(&new_db_file_name))
1330
  {
1331
    my_error(ER_WRONG_DB_NAME, MYF(0), new_db_file_name.str);
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
1332
    free(new_db_file_name.str);
1 by brian
clean slate
1333
1334
    if (force_switch)
520.1.22 by Brian Aker
Second pass of thd cleanup
1335
      mysql_change_db_impl(session, NULL, session->variables.collation_server);
1 by brian
clean slate
1336
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1337
    return(true);
1 by brian
clean slate
1338
  }
1339
1340
  if (check_db_dir_existence(new_db_file_name.str))
1341
  {
1342
    if (force_switch)
1343
    {
1344
      /* Throw a warning and free new_db_file_name. */
1345
520.1.22 by Brian Aker
Second pass of thd cleanup
1346
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
1347
                          ER_BAD_DB_ERROR, ER(ER_BAD_DB_ERROR),
1348
                          new_db_file_name.str);
1349
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
1350
      free(new_db_file_name.str);
1 by brian
clean slate
1351
1352
      /* Change db to NULL. */
1353
520.1.22 by Brian Aker
Second pass of thd cleanup
1354
      mysql_change_db_impl(session, NULL, session->variables.collation_server);
1 by brian
clean slate
1355
1356
      /* The operation succeed. */
1357
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1358
      return(false);
1 by brian
clean slate
1359
    }
1360
    else
1361
    {
1362
      /* Report an error and free new_db_file_name. */
1363
1364
      my_error(ER_BAD_DB_ERROR, MYF(0), new_db_file_name.str);
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
1365
      free(new_db_file_name.str);
1 by brian
clean slate
1366
1367
      /* The operation failed. */
1368
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1369
      return(true);
1 by brian
clean slate
1370
    }
1371
  }
1372
1373
  /*
520.1.21 by Brian Aker
THD -> Session rename
1374
    NOTE: in mysql_change_db_impl() new_db_file_name is assigned to Session
1375
    attributes and will be freed in Session::~Session().
1 by brian
clean slate
1376
  */
1377
520.1.22 by Brian Aker
Second pass of thd cleanup
1378
  db_default_cl= get_default_db_collation(session, new_db_file_name.str);
1 by brian
clean slate
1379
520.1.22 by Brian Aker
Second pass of thd cleanup
1380
  mysql_change_db_impl(session, &new_db_file_name, db_default_cl);
1 by brian
clean slate
1381
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1382
  return(false);
1 by brian
clean slate
1383
}
1384
1385
1386
/**
1387
  Change the current database and its attributes if needed.
1388
520.1.22 by Brian Aker
Second pass of thd cleanup
1389
  @param          session             thread handle
1 by brian
clean slate
1390
  @param          new_db_name     database name
1391
  @param[in, out] saved_db_name   IN: "str" points to a buffer where to store
1392
                                  the old database name, "length" contains the
1393
                                  buffer size
1394
                                  OUT: if the current (default) database is
1395
                                  not NULL, its name is copied to the
1396
                                  buffer pointed at by "str"
1397
                                  and "length" is updated accordingly.
1398
                                  Otherwise "str" is set to NULL and
1399
                                  "length" is set to 0.
1400
  @param          force_switch    @see mysql_change_db()
1401
  @param[out]     cur_db_changed  out-flag to indicate whether the current
1402
                                  database has been changed (valid only if
1403
                                  the function suceeded)
1404
*/
1405
520.1.22 by Brian Aker
Second pass of thd cleanup
1406
bool mysql_opt_change_db(Session *session,
1 by brian
clean slate
1407
                         const LEX_STRING *new_db_name,
1408
                         LEX_STRING *saved_db_name,
1409
                         bool force_switch,
1410
                         bool *cur_db_changed)
1411
{
520.1.22 by Brian Aker
Second pass of thd cleanup
1412
  *cur_db_changed= !cmp_db_names(session->db, new_db_name->str);
1 by brian
clean slate
1413
1414
  if (!*cur_db_changed)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1415
    return false;
1 by brian
clean slate
1416
520.1.22 by Brian Aker
Second pass of thd cleanup
1417
  backup_current_db_name(session, saved_db_name);
1 by brian
clean slate
1418
520.1.22 by Brian Aker
Second pass of thd cleanup
1419
  return mysql_change_db(session, new_db_name, force_switch);
1 by brian
clean slate
1420
}
1421
1422
1423
/*
1424
  Check if there is directory for the database name.
1425
1426
  SYNOPSIS
1427
    check_db_dir_existence()
1428
    db_name   database name
1429
1430
  RETURN VALUES
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1431
    false   There is directory for the specified database name.
1432
    true    The directory does not exist.
1 by brian
clean slate
1433
*/
1434
1435
bool check_db_dir_existence(const char *db_name)
1436
{
1437
  char db_dir_path[FN_REFLEN];
482 by Brian Aker
Remove uint.
1438
  uint32_t db_dir_path_len;
1 by brian
clean slate
1439
1440
  db_dir_path_len= build_table_filename(db_dir_path, sizeof(db_dir_path),
1441
                                        db_name, "", "", 0);
1442
1443
  if (db_dir_path_len && db_dir_path[db_dir_path_len - 1] == FN_LIBCHAR)
1444
    db_dir_path[db_dir_path_len - 1]= 0;
1445
1446
  /* Check access. */
1447
1448
  return my_access(db_dir_path, F_OK);
1449
}