~drizzle-trunk/drizzle/development

1 by brian
clean slate
1
/* Copyright (C) 2000-2004 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
/* drop and alter of tables */
17
243.1.17 by Jay Pipes
FINAL PHASE removal of mysql_priv.h (Bye, bye my friend.)
18
#include <drizzled/server_includes.h>
992.1.25 by Monty Taylor
Moved myisam to new plugin system.
19
#include <plugin/myisam/myisam.h>
575.4.7 by Monty Taylor
More header cleanup.
20
#include <drizzled/show.h>
549 by Monty Taylor
Took gettext.h out of header files.
21
#include <drizzled/error.h>
538 by Monty Taylor
Moved gettext.h into drizzled in anticipation of the new client lib.
22
#include <drizzled/gettext.h>
520.6.7 by Monty Taylor
Moved a bunch of crap out of common_includes.
23
#include <drizzled/data_home.h>
520.8.2 by Monty Taylor
Moved sql_parse.h and sql_error.h out of common_includes.
24
#include <drizzled/sql_parse.h>
584.1.15 by Monty Taylor
The mega-patch from hell. Renamed sql_class to session (since that's what it is) and removed it and field and table from common_includes.
25
#include <mysys/hash.h>
26
#include <drizzled/sql_lex.h>
27
#include <drizzled/session.h>
28
#include <drizzled/sql_base.h>
29
#include <drizzled/db.h>
670.2.4 by Monty Taylor
Removed more stuff from the headers.
30
#include <drizzled/lock.h>
31
#include <drizzled/unireg.h>
642.1.21 by Lee
header file clean up
32
#include <drizzled/item/int.h>
675 by Brian Aker
Cleanup around item includes.
33
#include <drizzled/item/empty_string.h>
988.1.6 by Jay Pipes
Removed old protobuf_replicator plugin, fixed up db.cc and other files to use new
34
#include <drizzled/transaction_services.h>
35
1001.1.3 by Andrew Ettinger
Revert bad spacing
36
670.3.3 by Toru Maesaka
Added namespacing for std to .cc files that needed it
37
using namespace std;
988.1.6 by Jay Pipes
Removed old protobuf_replicator plugin, fixed up db.cc and other files to use new
38
extern drizzled::TransactionServices transaction_services;
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
39
40
bool is_primary_key(KEY *key_info)
41
{
42
  static const char * primary_key_name="PRIMARY";
43
  return (strcmp(key_info->name, primary_key_name)==0);
44
}
45
46
const char* is_primary_key_name(const char* key_name)
47
{
48
  static const char * primary_key_name="PRIMARY";
49
  if (strcmp(key_name, primary_key_name)==0)
50
    return key_name;
51
  else
52
    return NULL;
53
}
1 by brian
clean slate
54
55
static bool check_if_keyname_exists(const char *name,KEY *start, KEY *end);
56
static char *make_unique_key_name(const char *field_name,KEY *start,KEY *end);
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
57
static int copy_data_between_tables(Table *from,Table *to,
1 by brian
clean slate
58
                                    List<Create_field> &create, bool ignore,
482 by Brian Aker
Remove uint.
59
                                    uint32_t order_num, order_st *order,
1 by brian
clean slate
60
                                    ha_rows *copied,ha_rows *deleted,
61
                                    enum enum_enable_or_disable keys_onoff,
62
                                    bool error_if_not_empty);
63
520.1.22 by Brian Aker
Second pass of thd cleanup
64
static bool prepare_blob_field(Session *session, Create_field *sql_field);
520.1.21 by Brian Aker
THD -> Session rename
65
static bool check_engine(Session *, const char *, HA_CREATE_INFO *);
1 by brian
clean slate
66
static int
520.1.22 by Brian Aker
Second pass of thd cleanup
67
mysql_prepare_create_table(Session *session, HA_CREATE_INFO *create_info,
1 by brian
clean slate
68
                           Alter_info *alter_info,
69
                           bool tmp_table,
482 by Brian Aker
Remove uint.
70
                               uint32_t *db_options,
1 by brian
clean slate
71
                               handler *file, KEY **key_info_buffer,
482 by Brian Aker
Remove uint.
72
                               uint32_t *key_count, int select_field_count);
1 by brian
clean slate
73
static bool
520.1.22 by Brian Aker
Second pass of thd cleanup
74
mysql_prepare_alter_table(Session *session, Table *table,
1 by brian
clean slate
75
                          HA_CREATE_INFO *create_info,
76
                          Alter_info *alter_info);
77
1014.3.1 by Brian Aker
Simplify the calling stack for getting schema collation. We need to extend
78
static void set_table_default_charset(HA_CREATE_INFO *create_info, char *db)
820.1.11 by Stewart Smith
re-introduce db.opt, but with parsing it from disk instead of in process cache with mutex.
79
{
80
  /*
81
    If the table character set was not given explicitly,
82
    let's fetch the database default character set and
83
    apply it to the table.
84
  */
1014.3.1 by Brian Aker
Simplify the calling stack for getting schema collation. We need to extend
85
  if (create_info->default_table_charset == NULL)
86
    create_info->default_table_charset= get_default_db_collation(db);
820.1.11 by Stewart Smith
re-introduce db.opt, but with parsing it from disk instead of in process cache with mutex.
87
}
88
1 by brian
clean slate
89
/*
90
  Translate a file name to a table name (WL #1324).
91
92
  SYNOPSIS
93
    filename_to_tablename()
94
      from                      The file name in my_charset_filename.
95
      to                OUT     The table name in system_charset_info.
96
      to_length                 The size of the table name buffer.
97
98
  RETURN
99
    Table name length.
100
*/
482 by Brian Aker
Remove uint.
101
uint32_t filename_to_tablename(const char *from, char *to, uint32_t to_length)
1 by brian
clean slate
102
{
482 by Brian Aker
Remove uint.
103
  uint32_t errors;
104
  uint32_t res;
1 by brian
clean slate
105
584.2.1 by Stewart Smith
convert tmp_file_prefix to TMP_FILE_PREFIX to indicate it's a macro
106
  if (!memcmp(from, TMP_FILE_PREFIX, TMP_FILE_PREFIX_LENGTH))
1 by brian
clean slate
107
  {
108
    /* Temporary table name. */
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
109
    res= strlen(strncpy(to, from, to_length));
1 by brian
clean slate
110
  }
111
  else
112
  {
113
    res= strconvert(&my_charset_filename, from,
114
                    system_charset_info,  to, to_length, &errors);
115
    if (errors) // Old 5.0 name
116
    {
896.4.12 by Stewart Smith
Remove MYSQL50_TABLE_NAME_PREFIX.
117
      to[0]='\0';
118
      strncat(to, from, to_length-1);
534 by Monty Taylor
Removed stxnmov. Also deleted strstr which had already been removed.
119
      res= strlen(to);
755.2.1 by Mark Atwood
replace sql_print_error etc with errmsg_print
120
      errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid (old?) table or database name '%s'"), from);
1 by brian
clean slate
121
    }
122
  }
123
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
124
  return(res);
1 by brian
clean slate
125
}
126
127
128
/*
129
  Translate a table name to a file name (WL #1324).
130
131
  SYNOPSIS
132
    tablename_to_filename()
133
      from                      The table name in system_charset_info.
134
      to                OUT     The file name in my_charset_filename.
135
      to_length                 The size of the file name buffer.
136
137
  RETURN
138
    File name length.
139
*/
482 by Brian Aker
Remove uint.
140
uint32_t tablename_to_filename(const char *from, char *to, uint32_t to_length)
1 by brian
clean slate
141
{
482 by Brian Aker
Remove uint.
142
  uint32_t errors, length;
1 by brian
clean slate
143
144
  length= strconvert(system_charset_info, from,
145
                     &my_charset_filename, to, to_length, &errors);
146
  if (check_if_legal_tablename(to) &&
147
      length + 4 < to_length)
148
  {
149
    memcpy(to + length, "@@@", 4);
150
    length+= 3;
151
  }
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
152
  return(length);
1 by brian
clean slate
153
}
154
155
156
/*
575.4.1 by ysano
Rename mysql to drizzle.
157
  Creates path to a file: drizzle_data_dir/db/table.ext
1 by brian
clean slate
158
159
  SYNOPSIS
160
   build_table_filename()
161
     buff                       Where to write result in my_charset_filename.
162
                                This may be the same as table_name.
163
     bufflen                    buff size
164
     db                         Database name in system_charset_info.
165
     table_name                 Table name in system_charset_info.
166
     ext                        File extension.
167
     flags                      FN_FROM_IS_TMP or FN_TO_IS_TMP or FN_IS_TMP
168
                                table_name is temporary, do not change.
169
170
  NOTES
171
172
    Uses database and table name, and extension to create
575.4.1 by ysano
Rename mysql to drizzle.
173
    a file name in drizzle_data_dir. Database and table
1 by brian
clean slate
174
    names are converted from system_charset_info into "fscs".
175
    Unless flags indicate a temporary table name.
176
    'db' is always converted.
177
    'ext' is not converted.
178
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
179
    The conversion suppression is required for ALTER Table. This
1 by brian
clean slate
180
    statement creates intermediate tables. These are regular
181
    (non-temporary) tables with a temporary name. Their path names must
182
    be derivable from the table name. So we cannot use
183
    build_tmptable_filename() for them.
184
185
  RETURN
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
186
    path length on success, 0 on failure
1 by brian
clean slate
187
*/
188
1039.1.6 by Brian Aker
Refactor for build_table_filename()
189
size_t build_table_filename(char *buff, size_t bufflen, const char *db, const char *table_name, bool is_tmp)
1 by brian
clean slate
190
{
670.3.3 by Toru Maesaka
Added namespacing for std to .cc files that needed it
191
  string table_path;
1 by brian
clean slate
192
  char dbbuff[FN_REFLEN];
193
  char tbbuff[FN_REFLEN];
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
194
  int rootdir_len= strlen(FN_ROOTDIR);
1 by brian
clean slate
195
1039.1.6 by Brian Aker
Refactor for build_table_filename()
196
  if (is_tmp) // FN_FROM_IS_TMP | FN_TO_IS_TMP
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
197
    strncpy(tbbuff, table_name, sizeof(tbbuff));
1 by brian
clean slate
198
  else
398.1.10 by Monty Taylor
Actually removed VOID() this time.
199
    tablename_to_filename(table_name, tbbuff, sizeof(tbbuff));
1 by brian
clean slate
200
398.1.10 by Monty Taylor
Actually removed VOID() this time.
201
  tablename_to_filename(db, dbbuff, sizeof(dbbuff));
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
202
  table_path= drizzle_data_home;
203
  int without_rootdir= table_path.length()-rootdir_len;
1 by brian
clean slate
204
575.4.1 by ysano
Rename mysql to drizzle.
205
  /* Don't add FN_ROOTDIR if dirzzle_data_home already includes it */
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
206
  if (without_rootdir >= 0)
207
  {
208
    char *tmp= (char*)table_path.c_str()+without_rootdir;
209
    if (memcmp(tmp, FN_ROOTDIR, rootdir_len) != 0)
210
      table_path.append(FN_ROOTDIR);
211
  }
212
213
  table_path.append(dbbuff);
214
  table_path.append(FN_ROOTDIR);
1 by brian
clean slate
215
#ifdef USE_SYMDIR
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
216
  table_path.clear();
1 by brian
clean slate
217
#endif
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
218
  table_path.append(tbbuff);
219
220
  if (bufflen < table_path.length())
221
    return 0;
222
223
  strcpy(buff, table_path.c_str());
1039.1.6 by Brian Aker
Refactor for build_table_filename()
224
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
225
  return table_path.length();
1 by brian
clean slate
226
}
227
228
229
/*
575.4.3 by ysano
Rename mysql to drizzle.
230
  Creates path to a file: drizzle_tmpdir/#sql1234_12_1.ext
1 by brian
clean slate
231
232
  SYNOPSIS
233
   build_tmptable_filename()
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
234
     session                    The thread handle.
1 by brian
clean slate
235
     buff                       Where to write result in my_charset_filename.
236
     bufflen                    buff size
237
238
  NOTES
239
240
    Uses current_pid, thread_id, and tmp_table counter to create
575.4.3 by ysano
Rename mysql to drizzle.
241
    a file name in drizzle_tmpdir.
1 by brian
clean slate
242
243
  RETURN
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
244
    path length on success, 0 on failure
1 by brian
clean slate
245
*/
246
520.1.22 by Brian Aker
Second pass of thd cleanup
247
uint32_t build_tmptable_filename(Session* session, char *buff, size_t bufflen)
1 by brian
clean slate
248
{
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
249
  uint32_t length;
670.3.3 by Toru Maesaka
Added namespacing for std to .cc files that needed it
250
  ostringstream path_str, post_tmpdir_str;
251
  string tmp;
1 by brian
clean slate
252
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
253
  path_str << drizzle_tmpdir;
254
  post_tmpdir_str << "/" << TMP_FILE_PREFIX << current_pid;
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
255
  post_tmpdir_str << session->thread_id << session->tmp_table++;
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
256
  tmp= post_tmpdir_str.str();
1 by brian
clean slate
257
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
258
  transform(tmp.begin(), tmp.end(), tmp.begin(), ::tolower);
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
259
260
  path_str << tmp;
261
262
  if (bufflen < path_str.str().length())
263
    length= 0;
264
  else
265
    length= unpack_filename(buff, path_str.str().c_str());
266
267
  return length;
1 by brian
clean slate
268
}
269
270
/*
271
  SYNOPSIS
272
    write_bin_log()
520.1.22 by Brian Aker
Second pass of thd cleanup
273
    session                           Thread object
1 by brian
clean slate
274
    clear_error                   is clear_error to be called
275
    query                         Query to log
276
    query_length                  Length of query
277
278
  RETURN VALUES
279
    NONE
280
281
  DESCRIPTION
282
    Write the binlog if open, routine used in multiple places in this
283
    file
284
*/
285
666 by Brian Aker
Updating new replication positions in code.
286
void write_bin_log(Session *session, bool,
287
                   char const *query, size_t query_length)
1 by brian
clean slate
288
{
988.1.6 by Jay Pipes
Removed old protobuf_replicator plugin, fixed up db.cc and other files to use new
289
  transaction_services.rawStatement(session, query, query_length);
1 by brian
clean slate
290
}
291
292
293
/*
294
 delete (drop) tables.
295
296
  SYNOPSIS
297
   mysql_rm_table()
520.1.22 by Brian Aker
Second pass of thd cleanup
298
   session			Thread handle
1 by brian
clean slate
299
   tables		List of tables to delete
300
   if_exists		If 1, don't give error if one table doesn't exists
301
302
  NOTES
303
    Will delete all tables that can be deleted and give a compact error
304
    messages for tables that could not be deleted.
305
    If a table is in use, we will wait for all users to free the table
306
    before dropping it
307
308
    Wait if global_read_lock (FLUSH TABLES WITH READ LOCK) is set, but
309
    not if under LOCK TABLES.
310
311
  RETURN
55 by brian
Update for using real bool types.
312
    false OK.  In this case ok packet is sent to user
313
    true  Error
1 by brian
clean slate
314
315
*/
316
520.1.22 by Brian Aker
Second pass of thd cleanup
317
bool mysql_rm_table(Session *session,TableList *tables, bool if_exists, bool drop_temporary)
1 by brian
clean slate
318
{
55 by brian
Update for using real bool types.
319
  bool error, need_start_waiting= false;
1 by brian
clean slate
320
321
  if (tables && tables->schema_table)
322
  {
575.4.7 by Monty Taylor
More header cleanup.
323
    my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), "", "", INFORMATION_SCHEMA_NAME.c_str());
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
324
    return(true);
1 by brian
clean slate
325
  }
326
327
  /* mark for close and remove all cached entries */
328
329
  if (!drop_temporary)
330
  {
1054.1.8 by Brian Aker
Remove lock_tables list from session.
331
    if (!(need_start_waiting= !wait_if_global_read_lock(session, 0, 1)))
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
332
      return(true);
1 by brian
clean slate
333
  }
334
335
  /*
336
    Acquire LOCK_open after wait_if_global_read_lock(). If we would hold
337
    LOCK_open during wait_if_global_read_lock(), other threads could not
338
    close their tables. This would make a pretty deadlock.
339
  */
590.1.7 by Stewart Smith
remove mysql_frm_type
340
  error= mysql_rm_table_part2(session, tables, if_exists, drop_temporary, 0);
1 by brian
clean slate
341
342
  if (need_start_waiting)
520.1.22 by Brian Aker
Second pass of thd cleanup
343
    start_waiting_global_read_lock(session);
1 by brian
clean slate
344
345
  if (error)
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
346
    return(true);
836 by Brian Aker
Fixed session call from function to method.
347
  session->my_ok();
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
348
  return(false);
1 by brian
clean slate
349
}
350
351
/*
352
  Execute the drop of a normal or temporary table
353
354
  SYNOPSIS
355
    mysql_rm_table_part2()
520.1.22 by Brian Aker
Second pass of thd cleanup
356
    session			Thread handler
1 by brian
clean slate
357
    tables		Tables to drop
358
    if_exists		If set, don't give an error if table doesn't exists.
359
			In this case we give an warning of level 'NOTE'
360
    drop_temporary	Only drop temporary tables
361
    drop_view		Allow to delete VIEW .frm
362
    dont_log_query	Don't write query to log files. This will also not
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
363
			generate warnings if the handler files doesn't exists
1 by brian
clean slate
364
365
  TODO:
366
    When logging to the binary log, we should log
367
    tmp_tables and transactional tables as separate statements if we
368
    are in a transaction;  This is needed to get these tables into the
369
    cached binary log that is only written on COMMIT.
370
371
   The current code only writes DROP statements that only uses temporary
372
   tables to the cache binary log.  This should be ok on most cases, but
373
   not all.
374
375
 RETURN
376
   0	ok
377
   1	Error
378
   -1	Thread was killed
379
*/
380
520.1.22 by Brian Aker
Second pass of thd cleanup
381
int mysql_rm_table_part2(Session *session, TableList *tables, bool if_exists,
590.1.7 by Stewart Smith
remove mysql_frm_type
382
                         bool drop_temporary, bool dont_log_query)
1 by brian
clean slate
383
{
327.2.4 by Brian Aker
Refactoring table.h
384
  TableList *table;
1039.1.4 by Brian Aker
Modified alias to being const.
385
  char path[FN_REFLEN];
802 by Brian Aker
Fix uninit of path length for OSX.
386
  uint32_t path_length= 0;
1 by brian
clean slate
387
  String wrong_tables;
388
  int error= 0;
389
  int non_temp_tables_count= 0;
390
  bool some_tables_deleted=0, tmp_table_deleted=0, foreign_key_error=0;
391
  String built_query;
392
581 by Brian Aker
Second pass through on replication row patch
393
  if (!dont_log_query)
1 by brian
clean slate
394
  {
395
    built_query.set_charset(system_charset_info);
396
    if (if_exists)
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
397
      built_query.append("DROP Table IF EXISTS ");
1 by brian
clean slate
398
    else
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
399
      built_query.append("DROP Table ");
1 by brian
clean slate
400
  }
401
1046.1.2 by Brian Aker
Comments on LOCK_open
402
  pthread_mutex_lock(&LOCK_open); /* Part 2 of rm a table */
1 by brian
clean slate
403
404
  /*
405
    If we have the table in the definition cache, we don't have to check the
406
    .frm file to find if the table is a normal table (not view) and what
407
    engine to use.
408
  */
409
410
  for (table= tables; table; table= table->next_local)
411
  {
1000.1.3 by Brian Aker
Renamed TABLE_SHARE to TableShare
412
    TableShare *share;
1 by brian
clean slate
413
    table->db_type= NULL;
414
    if ((share= get_cached_table_share(table->db, table->table_name)))
415
      table->db_type= share->db_type();
416
  }
417
520.1.22 by Brian Aker
Second pass of thd cleanup
418
  if (!drop_temporary && lock_table_names_exclusively(session, tables))
1 by brian
clean slate
419
  {
420
    pthread_mutex_unlock(&LOCK_open);
1046.1.10 by Brian Aker
Formatting around return (style)
421
    return 1;
1 by brian
clean slate
422
  }
423
424
  /* Don't give warnings for not found errors, as we already generate notes */
520.1.22 by Brian Aker
Second pass of thd cleanup
425
  session->no_warnings_for_error= 1;
1 by brian
clean slate
426
427
  for (table= tables; table; table= table->next_local)
428
  {
429
    char *db=table->db;
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
430
    StorageEngine *table_type;
1 by brian
clean slate
431
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
432
    error= session->drop_temporary_table(table);
1 by brian
clean slate
433
434
    switch (error) {
435
    case  0:
436
      // removed temporary table
437
      tmp_table_deleted= 1;
438
      continue;
439
    case -1:
440
      error= 1;
441
      goto err_with_placeholders;
442
    default:
443
      // temporary table not found
444
      error= 0;
445
    }
446
447
    /*
448
      If row-based replication is used and the table is not a
449
      temporary table, we add the table name to the drop statement
450
      being built.  The string always end in a comma and the comma
451
      will be chopped off before being written to the binary log.
452
      */
581 by Brian Aker
Second pass through on replication row patch
453
    if (!dont_log_query)
1 by brian
clean slate
454
    {
455
      non_temp_tables_count++;
456
      /*
457
        Don't write the database name if it is the current one (or if
520.1.22 by Brian Aker
Second pass of thd cleanup
458
        session->db is NULL).
1 by brian
clean slate
459
      */
460
      built_query.append("`");
520.1.22 by Brian Aker
Second pass of thd cleanup
461
      if (session->db == NULL || strcmp(db,session->db) != 0)
1 by brian
clean slate
462
      {
463
        built_query.append(db);
464
        built_query.append("`.`");
465
      }
466
467
      built_query.append(table->table_name);
468
      built_query.append("`,");
469
    }
470
471
    table_type= table->db_type;
472
    if (!drop_temporary)
473
    {
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
474
      Table *locked_table;
520.1.22 by Brian Aker
Second pass of thd cleanup
475
      abort_locked_tables(session, db, table->table_name);
476
      remove_table_from_cache(session, db, table->table_name,
1 by brian
clean slate
477
                              RTFC_WAIT_OTHER_THREAD_FLAG |
478
                              RTFC_CHECK_KILLED_FLAG);
479
      /*
480
        If the table was used in lock tables, remember it so that
481
        unlock_table_names can free it
482
      */
520.1.22 by Brian Aker
Second pass of thd cleanup
483
      if ((locked_table= drop_locked_tables(session, db, table->table_name)))
1 by brian
clean slate
484
        table->table= locked_table;
485
520.1.22 by Brian Aker
Second pass of thd cleanup
486
      if (session->killed)
1 by brian
clean slate
487
      {
488
        error= -1;
489
        goto err_with_placeholders;
490
      }
491
      /* remove .frm file and engine files */
1039.1.6 by Brian Aker
Refactor for build_table_filename()
492
      path_length= build_table_filename(path, sizeof(path), db, table->table_name, table->internal_tmp_table);
1 by brian
clean slate
493
    }
494
    if (drop_temporary ||
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
495
        ((table_type == NULL && (table_proto_exists(path)!=EEXIST))))
1 by brian
clean slate
496
    {
497
      // Table was not found on disk and table can't be created from engine
498
      if (if_exists)
520.1.22 by Brian Aker
Second pass of thd cleanup
499
        push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
500
                            ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR),
501
                            table->table_name);
502
      else
503
        error= 1;
504
    }
505
    else
506
    {
590.1.7 by Stewart Smith
remove mysql_frm_type
507
      error= ha_delete_table(session, path, db, table->table_name,
1 by brian
clean slate
508
                             !dont_log_query);
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
509
      if ((error == ENOENT || error == HA_ERR_NO_SUCH_TABLE) &&
590.1.7 by Stewart Smith
remove mysql_frm_type
510
	  if_exists)
1 by brian
clean slate
511
      {
512
	error= 0;
520.1.22 by Brian Aker
Second pass of thd cleanup
513
        session->clear_error();
1 by brian
clean slate
514
      }
515
      if (error == HA_ERR_ROW_IS_REFERENCED)
516
      {
517
        /* the table is referenced by a foreign key constraint */
518
        foreign_key_error=1;
519
      }
520
      if (!error || error == ENOENT || error == HA_ERR_NO_SUCH_TABLE)
521
      {
522
        int new_error;
584.2.7 by Stewart Smith
rename and delete tabledefinition protobuf file
523
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
524
        if (!(new_error= delete_table_proto_file(path)))
1 by brian
clean slate
525
        {
526
          some_tables_deleted=1;
527
          new_error= 0;
528
        }
529
        error|= new_error;
530
      }
531
    }
532
    if (error)
533
    {
534
      if (wrong_tables.length())
535
        wrong_tables.append(',');
536
      wrong_tables.append(String(table->table_name,system_charset_info));
537
    }
538
  }
539
  /*
540
    It's safe to unlock LOCK_open: we have an exclusive lock
541
    on the table name.
542
  */
543
  pthread_mutex_unlock(&LOCK_open);
544
  error= 0;
545
  if (wrong_tables.length())
546
  {
547
    if (!foreign_key_error)
548
      my_printf_error(ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR), MYF(0),
549
                      wrong_tables.c_ptr());
550
    else
186 by Brian Aker
Partial fix for alter table
551
    {
1 by brian
clean slate
552
      my_message(ER_ROW_IS_REFERENCED, ER(ER_ROW_IS_REFERENCED), MYF(0));
186 by Brian Aker
Partial fix for alter table
553
    }
1 by brian
clean slate
554
    error= 1;
555
  }
556
557
  if (some_tables_deleted || tmp_table_deleted || !error)
558
  {
559
    if (!dont_log_query)
560
    {
581 by Brian Aker
Second pass through on replication row patch
561
      if ((non_temp_tables_count > 0 && !tmp_table_deleted))
1 by brian
clean slate
562
      {
563
        /*
564
          In this case, we are either using statement-based
565
          replication or using row-based replication but have only
566
          deleted one or more non-temporary tables (and no temporary
567
          tables).  In this case, we can write the original query into
568
          the binary log.
569
         */
520.1.22 by Brian Aker
Second pass of thd cleanup
570
        write_bin_log(session, !error, session->query, session->query_length);
1 by brian
clean slate
571
      }
581 by Brian Aker
Second pass through on replication row patch
572
      else if (non_temp_tables_count > 0 &&
1 by brian
clean slate
573
               tmp_table_deleted)
574
      {
575
        /*
576
          In this case we have deleted both temporary and
577
          non-temporary tables, so:
578
          - since we have deleted a non-temporary table we have to
579
            binlog the statement, but
580
          - since we have deleted a temporary table we cannot binlog
581
            the statement (since the table has not been created on the
582
            slave, this might cause the slave to stop).
583
584
          Instead, we write a built statement, only containing the
585
          non-temporary tables, to the binary log
586
        */
587
        built_query.chop();                  // Chop of the last comma
588
        built_query.append(" /* generated by server */");
520.1.22 by Brian Aker
Second pass of thd cleanup
589
        write_bin_log(session, !error, built_query.ptr(), built_query.length());
1 by brian
clean slate
590
      }
591
      /*
592
        The remaining cases are:
593
        - no tables where deleted and
594
        - only temporary tables where deleted and row-based
595
          replication is used.
596
        In both these cases, nothing should be written to the binary
597
        log.
598
      */
599
    }
600
  }
1046.1.2 by Brian Aker
Comments on LOCK_open
601
  pthread_mutex_lock(&LOCK_open); /* final bit in rm table lock */
1 by brian
clean slate
602
err_with_placeholders:
1034.1.7 by Brian Aker
Remove dead bits to the end of functions.
603
  unlock_table_names(tables, NULL);
1 by brian
clean slate
604
  pthread_mutex_unlock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
605
  session->no_warnings_for_error= 0;
1034.1.7 by Brian Aker
Remove dead bits to the end of functions.
606
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
607
  return(error);
1 by brian
clean slate
608
}
609
610
611
/*
612
  Quickly remove a table.
613
614
  SYNOPSIS
615
    quick_rm_table()
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
616
      base                      The StorageEngine handle.
1 by brian
clean slate
617
      db                        The database name.
618
      table_name                The table name.
1039.1.6 by Brian Aker
Refactor for build_table_filename()
619
      is_tmp                    If the table is temp.
1 by brian
clean slate
620
621
  RETURN
622
    0           OK
623
    != 0        Error
624
*/
625
1039.1.6 by Brian Aker
Refactor for build_table_filename()
626
bool quick_rm_table(StorageEngine *, const char *db,
627
                    const char *table_name, bool is_tmp)
1 by brian
clean slate
628
{
629
  char path[FN_REFLEN];
630
  bool error= 0;
631
1039.1.6 by Brian Aker
Refactor for build_table_filename()
632
  build_table_filename(path, sizeof(path), db, table_name, is_tmp);
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
633
634
  error= delete_table_proto_file(path);
584.2.7 by Stewart Smith
rename and delete tabledefinition protobuf file
635
590.1.7 by Stewart Smith
remove mysql_frm_type
636
  return(ha_delete_table(current_session, path, db, table_name, 0) ||
1 by brian
clean slate
637
              error);
638
}
639
640
/*
641
  Sort keys in the following order:
642
  - PRIMARY KEY
643
  - UNIQUE keys where all column are NOT NULL
644
  - UNIQUE keys that don't contain partial segments
645
  - Other UNIQUE keys
646
  - Normal keys
647
  - Fulltext keys
648
649
  This will make checking for duplicated keys faster and ensure that
650
  PRIMARY keys are prioritized.
651
*/
652
653
static int sort_keys(KEY *a, KEY *b)
654
{
655
  ulong a_flags= a->flags, b_flags= b->flags;
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
656
1 by brian
clean slate
657
  if (a_flags & HA_NOSAME)
658
  {
659
    if (!(b_flags & HA_NOSAME))
660
      return -1;
53.2.14 by Monty Taylor
Removed HA_END_SPACE_KEY and references to it. It was _supposed_ to be gone anyway, but the ifdef around it was broken (MYSQL_VERSION_ID was actually undefined.)
661
    if ((a_flags ^ b_flags) & (HA_NULL_PART_KEY))
1 by brian
clean slate
662
    {
663
      /* Sort NOT NULL keys before other keys */
53.2.14 by Monty Taylor
Removed HA_END_SPACE_KEY and references to it. It was _supposed_ to be gone anyway, but the ifdef around it was broken (MYSQL_VERSION_ID was actually undefined.)
664
      return (a_flags & (HA_NULL_PART_KEY)) ? 1 : -1;
1 by brian
clean slate
665
    }
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
666
    if (is_primary_key(a))
1 by brian
clean slate
667
      return -1;
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
668
    if (is_primary_key(b))
1 by brian
clean slate
669
      return 1;
670
    /* Sort keys don't containing partial segments before others */
671
    if ((a_flags ^ b_flags) & HA_KEY_HAS_PART_KEY_SEG)
672
      return (a_flags & HA_KEY_HAS_PART_KEY_SEG) ? 1 : -1;
673
  }
674
  else if (b_flags & HA_NOSAME)
675
    return 1;					// Prefer b
676
677
  /*
678
    Prefer original key order.	usable_key_parts contains here
679
    the original key position.
680
  */
681
  return ((a->usable_key_parts < b->usable_key_parts) ? -1 :
682
          (a->usable_key_parts > b->usable_key_parts) ? 1 :
683
          0);
684
}
685
686
/*
687
  Check TYPELIB (set or enum) for duplicates
688
689
  SYNOPSIS
690
    check_duplicates_in_interval()
691
    set_or_name   "SET" or "ENUM" string for warning message
692
    name	  name of the checked column
693
    typelib	  list of values for the column
694
    dup_val_count  returns count of duplicate elements
695
696
  DESCRIPTION
697
    This function prints an warning for each value in list
698
    which has some duplicates on its right
699
700
  RETURN VALUES
701
    0             ok
702
    1             Error
703
*/
704
705
bool check_duplicates_in_interval(const char *set_or_name,
706
                                  const char *name, TYPELIB *typelib,
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
707
                                  const CHARSET_INFO * const cs, unsigned int *dup_val_count)
1 by brian
clean slate
708
{
709
  TYPELIB tmp= *typelib;
710
  const char **cur_value= typelib->type_names;
711
  unsigned int *cur_length= typelib->type_lengths;
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
712
  *dup_val_count= 0;
713
1 by brian
clean slate
714
  for ( ; tmp.count > 1; cur_value++, cur_length++)
715
  {
716
    tmp.type_names++;
717
    tmp.type_lengths++;
718
    tmp.count--;
719
    if (find_type2(&tmp, (const char*)*cur_value, *cur_length, cs))
720
    {
721
      my_error(ER_DUPLICATED_VALUE_IN_TYPE, MYF(0),
722
               name,*cur_value,set_or_name);
723
      return 1;
724
    }
725
  }
726
  return 0;
727
}
728
729
730
/*
731
  Check TYPELIB (set or enum) max and total lengths
732
733
  SYNOPSIS
734
    calculate_interval_lengths()
735
    cs            charset+collation pair of the interval
736
    typelib       list of values for the column
737
    max_length    length of the longest item
738
    tot_length    sum of the item lengths
739
740
  DESCRIPTION
741
    After this function call:
742
    - ENUM uses max_length
743
    - SET uses tot_length.
744
745
  RETURN VALUES
746
    void
747
*/
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
748
void calculate_interval_lengths(const CHARSET_INFO * const cs, TYPELIB *interval,
205 by Brian Aker
uint32 -> uin32_t
749
                                uint32_t *max_length, uint32_t *tot_length)
1 by brian
clean slate
750
{
751
  const char **pos;
482 by Brian Aker
Remove uint.
752
  uint32_t *len;
1 by brian
clean slate
753
  *max_length= *tot_length= 0;
754
  for (pos= interval->type_names, len= interval->type_lengths;
755
       *pos ; pos++, len++)
756
  {
482 by Brian Aker
Remove uint.
757
    uint32_t length= cs->cset->numchars(cs, *pos, *pos + *len);
1 by brian
clean slate
758
    *tot_length+= length;
205 by Brian Aker
uint32 -> uin32_t
759
    set_if_bigger(*max_length, (uint32_t)length);
1 by brian
clean slate
760
  }
761
}
762
763
764
/*
765
  Prepare a create_table instance for packing
766
767
  SYNOPSIS
768
    prepare_create_field()
769
    sql_field     field to prepare for packing
770
    blob_columns  count for BLOBs
771
    timestamps    count for timestamps
772
    table_flags   table flags
773
774
  DESCRIPTION
775
    This function prepares a Create_field instance.
776
    Fields such as pack_flag are valid after this call.
777
778
  RETURN VALUES
779
   0	ok
780
   1	Error
781
*/
782
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
783
int prepare_create_field(Create_field *sql_field,
482 by Brian Aker
Remove uint.
784
                         uint32_t *blob_columns,
1 by brian
clean slate
785
                         int *timestamps, int *timestamps_with_niladic,
779.1.27 by Monty Taylor
Got rid of __attribute__((unused)) and the like from the .cc files.
786
                         int64_t )
1 by brian
clean slate
787
{
788
  unsigned int dup_val_count;
789
790
  /*
791
    This code came from mysql_prepare_create_table.
792
    Indent preserved to make patching easier
793
  */
51.2.1 by Patrick Galbraith
Removed DBUG_PRINTs, DBUG_ASSERTs, DBUG_EXECUTE_IFs from
794
  assert(sql_field->charset);
1 by brian
clean slate
795
796
  switch (sql_field->sql_type) {
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
797
  case DRIZZLE_TYPE_BLOB:
1 by brian
clean slate
798
    sql_field->pack_flag=FIELDFLAG_BLOB |
799
      pack_length_to_packflag(sql_field->pack_length -
800
                              portable_sizeof_char_ptr);
801
    if (sql_field->charset->state & MY_CS_BINSORT)
802
      sql_field->pack_flag|=FIELDFLAG_BINARY;
803
    sql_field->length=8;			// Unireg field length
804
    (*blob_columns)++;
805
    break;
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
806
  case DRIZZLE_TYPE_VARCHAR:
1 by brian
clean slate
807
    sql_field->pack_flag=0;
808
    if (sql_field->charset->state & MY_CS_BINSORT)
809
      sql_field->pack_flag|=FIELDFLAG_BINARY;
810
    break;
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
811
  case DRIZZLE_TYPE_ENUM:
1 by brian
clean slate
812
    sql_field->pack_flag=pack_length_to_packflag(sql_field->pack_length) |
813
      FIELDFLAG_INTERVAL;
814
    if (sql_field->charset->state & MY_CS_BINSORT)
815
      sql_field->pack_flag|=FIELDFLAG_BINARY;
816
    if (check_duplicates_in_interval("ENUM",sql_field->field_name,
817
                                 sql_field->interval,
818
                                     sql_field->charset, &dup_val_count))
1046.1.10 by Brian Aker
Formatting around return (style)
819
      return 1;
1 by brian
clean slate
820
    break;
575.5.1 by David Axmark
Changed NEWDATE to DATE. One failing test but I think its somewhere else in the code
821
  case DRIZZLE_TYPE_DATE:  // Rest of string types
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
822
  case DRIZZLE_TYPE_DATETIME:
823
  case DRIZZLE_TYPE_NULL:
895 by Brian Aker
Completion (?) of uint conversion.
824
    sql_field->pack_flag=f_settype((uint32_t) sql_field->sql_type);
1 by brian
clean slate
825
    break;
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
826
  case DRIZZLE_TYPE_NEWDECIMAL:
1 by brian
clean slate
827
    sql_field->pack_flag=(FIELDFLAG_NUMBER |
828
                          (sql_field->flags & UNSIGNED_FLAG ? 0 :
829
                           FIELDFLAG_DECIMAL) |
216 by Brian Aker
Remove completely ZEROFILL
830
                          (sql_field->flags & DECIMAL_FLAG ?  FIELDFLAG_DECIMAL_POSITION : 0) |
1 by brian
clean slate
831
                          (sql_field->decimals << FIELDFLAG_DEC_SHIFT));
832
    break;
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
833
  case DRIZZLE_TYPE_TIMESTAMP:
1 by brian
clean slate
834
    /* We should replace old TIMESTAMP fields with their newer analogs */
835
    if (sql_field->unireg_check == Field::TIMESTAMP_OLD_FIELD)
836
    {
837
      if (!*timestamps)
838
      {
839
        sql_field->unireg_check= Field::TIMESTAMP_DNUN_FIELD;
840
        (*timestamps_with_niladic)++;
841
      }
842
      else
843
        sql_field->unireg_check= Field::NONE;
844
    }
845
    else if (sql_field->unireg_check != Field::NONE)
846
      (*timestamps_with_niladic)++;
847
848
    (*timestamps)++;
849
    /* fall-through */
850
  default:
851
    sql_field->pack_flag=(FIELDFLAG_NUMBER |
852
                          (sql_field->flags & UNSIGNED_FLAG ? 0 :
853
                           FIELDFLAG_DECIMAL) |
895 by Brian Aker
Completion (?) of uint conversion.
854
                          f_settype((uint32_t) sql_field->sql_type) |
1 by brian
clean slate
855
                          (sql_field->decimals << FIELDFLAG_DEC_SHIFT));
856
    break;
857
  }
998.1.2 by Brian Aker
First pass on removing virt columns
858
  if (!(sql_field->flags & NOT_NULL_FLAG))
1 by brian
clean slate
859
    sql_field->pack_flag|= FIELDFLAG_MAYBE_NULL;
860
  if (sql_field->flags & NO_DEFAULT_VALUE_FLAG)
861
    sql_field->pack_flag|= FIELDFLAG_NO_DEFAULT;
1046.1.10 by Brian Aker
Formatting around return (style)
862
  return 0;
1 by brian
clean slate
863
}
864
865
/*
866
  Preparation for table creation
867
868
  SYNOPSIS
869
    mysql_prepare_create_table()
520.1.22 by Brian Aker
Second pass of thd cleanup
870
      session                       Thread object.
1 by brian
clean slate
871
      create_info               Create information (like MAX_ROWS).
872
      alter_info                List of columns and indexes to create
873
      tmp_table                 If a temporary table is to be created.
874
      db_options          INOUT Table options (like HA_OPTION_PACK_RECORD).
875
      file                      The handler for the new table.
876
      key_info_buffer     OUT   An array of KEY structs for the indexes.
877
      key_count           OUT   The number of elements in the array.
878
      select_field_count        The number of fields coming from a select table.
879
880
  DESCRIPTION
881
    Prepares the table and key structures for table creation.
882
883
  NOTES
884
    sets create_info->varchar if the table has a varchar
885
886
  RETURN VALUES
55 by brian
Update for using real bool types.
887
    false    OK
888
    true     error
1 by brian
clean slate
889
*/
890
891
static int
520.1.22 by Brian Aker
Second pass of thd cleanup
892
mysql_prepare_create_table(Session *session, HA_CREATE_INFO *create_info,
1 by brian
clean slate
893
                           Alter_info *alter_info,
894
                           bool tmp_table,
482 by Brian Aker
Remove uint.
895
                               uint32_t *db_options,
1 by brian
clean slate
896
                               handler *file, KEY **key_info_buffer,
482 by Brian Aker
Remove uint.
897
                               uint32_t *key_count, int select_field_count)
1 by brian
clean slate
898
{
899
  const char	*key_name;
900
  Create_field	*sql_field,*dup_field;
901
  uint		field,null_fields,blob_columns,max_key_length;
902
  ulong		record_offset= 0;
903
  KEY		*key_info;
904
  KEY_PART_INFO *key_part_info;
905
  int		timestamps= 0, timestamps_with_niladic= 0;
906
  int		field_no,dup_no;
907
  int		select_field_pos,auto_increment=0;
908
  List_iterator<Create_field> it(alter_info->create_list);
909
  List_iterator<Create_field> it2(alter_info->create_list);
482 by Brian Aker
Remove uint.
910
  uint32_t total_uneven_bit_length= 0;
1 by brian
clean slate
911
912
  select_field_pos= alter_info->create_list.elements - select_field_count;
913
  null_fields=blob_columns=0;
914
  create_info->varchar= 0;
915
  max_key_length= file->max_key_length();
916
917
  for (field_no=0; (sql_field=it++) ; field_no++)
918
  {
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
919
    const CHARSET_INFO *save_cs;
1 by brian
clean slate
920
921
    /*
922
      Initialize length from its original value (number of characters),
923
      which was set in the parser. This is necessary if we're
924
      executing a prepared statement for the second time.
925
    */
926
    sql_field->length= sql_field->char_length;
927
    if (!sql_field->charset)
928
      sql_field->charset= create_info->default_table_charset;
929
    /*
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
930
      table_charset is set in ALTER Table if we want change character set
1 by brian
clean slate
931
      for all varchar/char columns.
932
      But the table charset must not affect the BLOB fields, so don't
933
      allow to change my_charset_bin to somethig else.
934
    */
935
    if (create_info->table_charset && sql_field->charset != &my_charset_bin)
936
      sql_field->charset= create_info->table_charset;
937
938
    save_cs= sql_field->charset;
939
    if ((sql_field->flags & BINCMP_FLAG) &&
862 by Brian Aker
Remove charset directory code.
940
        !(sql_field->charset= get_charset_by_csname(sql_field->charset->csname, MY_CS_BINSORT)))
1 by brian
clean slate
941
    {
942
      char tmp[64];
629.5.3 by Toru Maesaka
Third pass of replacing MySQL's strmake() with libc calls
943
      char *tmp_pos= tmp;
944
      strncpy(tmp_pos, save_cs->csname, sizeof(tmp)-4);
945
      tmp_pos+= strlen(tmp);
946
      strncpy(tmp_pos, STRING_WITH_LEN("_bin"));
1 by brian
clean slate
947
      my_error(ER_UNKNOWN_COLLATION, MYF(0), tmp);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
948
      return(true);
1 by brian
clean slate
949
    }
950
951
    /*
952
      Convert the default value from client character
953
      set into the column character set if necessary.
954
    */
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
955
    if (sql_field->def &&
1 by brian
clean slate
956
        save_cs != sql_field->def->collation.collation &&
325 by Brian Aker
Remove SET
957
        (sql_field->sql_type == DRIZZLE_TYPE_ENUM))
1 by brian
clean slate
958
    {
959
      /*
960
        Starting from 5.1 we work here with a copy of Create_field
961
        created by the caller, not with the instance that was
962
        originally created during parsing. It's OK to create
963
        a temporary item and initialize with it a member of the
964
        copy -- this item will be thrown away along with the copy
965
        at the end of execution, and thus not introduce a dangling
966
        pointer in the parsed tree of a prepared statement or a
967
        stored procedure statement.
968
      */
969
      sql_field->def= sql_field->def->safe_charset_converter(save_cs);
970
971
      if (sql_field->def == NULL)
972
      {
973
        /* Could not convert */
974
        my_error(ER_INVALID_DEFAULT, MYF(0), sql_field->field_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
975
        return(true);
1 by brian
clean slate
976
      }
977
    }
978
325 by Brian Aker
Remove SET
979
    if (sql_field->sql_type == DRIZZLE_TYPE_ENUM)
1 by brian
clean slate
980
    {
205 by Brian Aker
uint32 -> uin32_t
981
      uint32_t dummy;
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
982
      const CHARSET_INFO * const cs= sql_field->charset;
1 by brian
clean slate
983
      TYPELIB *interval= sql_field->interval;
984
985
      /*
986
        Create typelib from interval_list, and if necessary
987
        convert strings from client character set to the
988
        column character set.
989
      */
990
      if (!interval)
991
      {
992
        /*
993
          Create the typelib in runtime memory - we will free the
994
          occupied memory at the same time when we free this
995
          sql_field -- at the end of execution.
996
        */
520.1.22 by Brian Aker
Second pass of thd cleanup
997
        interval= sql_field->interval= typelib(session->mem_root,
1 by brian
clean slate
998
                                               sql_field->interval_list);
999
        List_iterator<String> int_it(sql_field->interval_list);
1000
        String conv, *tmp;
1001
        char comma_buf[4];
481 by Brian Aker
Remove all of uchar.
1002
        int comma_length= cs->cset->wc_mb(cs, ',', (unsigned char*) comma_buf,
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1003
                                          (unsigned char*) comma_buf +
1 by brian
clean slate
1004
                                          sizeof(comma_buf));
51.2.1 by Patrick Galbraith
Removed DBUG_PRINTs, DBUG_ASSERTs, DBUG_EXECUTE_IFs from
1005
        assert(comma_length > 0);
482 by Brian Aker
Remove uint.
1006
        for (uint32_t i= 0; (tmp= int_it++); i++)
1 by brian
clean slate
1007
        {
482 by Brian Aker
Remove uint.
1008
          uint32_t lengthsp;
1 by brian
clean slate
1009
          if (String::needs_conversion(tmp->length(), tmp->charset(),
1010
                                       cs, &dummy))
1011
          {
482 by Brian Aker
Remove uint.
1012
            uint32_t cnv_errs;
1 by brian
clean slate
1013
            conv.copy(tmp->ptr(), tmp->length(), tmp->charset(), cs, &cnv_errs);
520.1.22 by Brian Aker
Second pass of thd cleanup
1014
            interval->type_names[i]= strmake_root(session->mem_root, conv.ptr(),
1 by brian
clean slate
1015
                                                  conv.length());
1016
            interval->type_lengths[i]= conv.length();
1017
          }
1018
1019
          // Strip trailing spaces.
1020
          lengthsp= cs->cset->lengthsp(cs, interval->type_names[i],
1021
                                       interval->type_lengths[i]);
1022
          interval->type_lengths[i]= lengthsp;
481 by Brian Aker
Remove all of uchar.
1023
          ((unsigned char *)interval->type_names[i])[lengthsp]= '\0';
1 by brian
clean slate
1024
        }
1025
        sql_field->interval_list.empty(); // Don't need interval_list anymore
1026
      }
1027
325 by Brian Aker
Remove SET
1028
      /* DRIZZLE_TYPE_ENUM */
1 by brian
clean slate
1029
      {
205 by Brian Aker
uint32 -> uin32_t
1030
        uint32_t field_length;
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
1031
        assert(sql_field->sql_type == DRIZZLE_TYPE_ENUM);
1 by brian
clean slate
1032
        if (sql_field->def != NULL)
1033
        {
1034
          String str, *def= sql_field->def->val_str(&str);
1035
          if (def == NULL) /* SQL "NULL" maps to NULL */
1036
          {
1037
            if ((sql_field->flags & NOT_NULL_FLAG) != 0)
1038
            {
1039
              my_error(ER_INVALID_DEFAULT, MYF(0), sql_field->field_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1040
              return(true);
1 by brian
clean slate
1041
            }
1042
1043
            /* else, the defaults yield the correct length for NULLs. */
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1044
          }
1 by brian
clean slate
1045
          else /* not NULL */
1046
          {
1047
            def->length(cs->cset->lengthsp(cs, def->ptr(), def->length()));
1048
            if (find_type2(interval, def->ptr(), def->length(), cs) == 0) /* not found */
1049
            {
1050
              my_error(ER_INVALID_DEFAULT, MYF(0), sql_field->field_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1051
              return(true);
1 by brian
clean slate
1052
            }
1053
          }
1054
        }
1055
        calculate_interval_lengths(cs, interval, &field_length, &dummy);
1056
        sql_field->length= field_length;
1057
      }
937.2.6 by Stewart Smith
make set_if_bigger typesafe for C and C++. Fix up everywhere.
1058
      set_if_smaller(sql_field->length, (uint32_t)MAX_FIELD_WIDTH-1);
1 by brian
clean slate
1059
    }
1060
1061
    sql_field->create_length_to_internal_length();
520.1.22 by Brian Aker
Second pass of thd cleanup
1062
    if (prepare_blob_field(session, sql_field))
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1063
      return(true);
1 by brian
clean slate
1064
1065
    if (!(sql_field->flags & NOT_NULL_FLAG))
1066
      null_fields++;
1067
1068
    if (check_column_name(sql_field->field_name))
1069
    {
1070
      my_error(ER_WRONG_COLUMN_NAME, MYF(0), sql_field->field_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1071
      return(true);
1 by brian
clean slate
1072
    }
1073
1074
    /* Check if we have used the same field name before */
1075
    for (dup_no=0; (dup_field=it2++) != sql_field; dup_no++)
1076
    {
1077
      if (my_strcasecmp(system_charset_info,
1078
                        sql_field->field_name,
1079
                        dup_field->field_name) == 0)
1080
      {
1081
	/*
1082
	  If this was a CREATE ... SELECT statement, accept a field
1083
	  redefinition if we are changing a field in the SELECT part
1084
	*/
1085
	if (field_no < select_field_pos || dup_no >= select_field_pos)
1086
	{
1087
	  my_error(ER_DUP_FIELDNAME, MYF(0), sql_field->field_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1088
	  return(true);
1 by brian
clean slate
1089
	}
1090
	else
1091
	{
1092
	  /* Field redefined */
1093
	  sql_field->def=		dup_field->def;
1094
	  sql_field->sql_type=		dup_field->sql_type;
1095
	  sql_field->charset=		(dup_field->charset ?
1096
					 dup_field->charset :
1097
					 create_info->default_table_charset);
1098
	  sql_field->length=		dup_field->char_length;
1099
          sql_field->pack_length=	dup_field->pack_length;
1100
          sql_field->key_length=	dup_field->key_length;
1101
	  sql_field->decimals=		dup_field->decimals;
1102
	  sql_field->create_length_to_internal_length();
1103
	  sql_field->unireg_check=	dup_field->unireg_check;
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1104
          /*
1 by brian
clean slate
1105
            We're making one field from two, the result field will have
1106
            dup_field->flags as flags. If we've incremented null_fields
1107
            because of sql_field->flags, decrement it back.
1108
          */
1109
          if (!(sql_field->flags & NOT_NULL_FLAG))
1110
            null_fields--;
1111
	  sql_field->flags=		dup_field->flags;
1112
          sql_field->interval=          dup_field->interval;
1113
	  it2.remove();			// Remove first (create) definition
1114
	  select_field_pos--;
1115
	  break;
1116
	}
1117
      }
1118
    }
1119
    /* Don't pack rows in old tables if the user has requested this */
1120
    if ((sql_field->flags & BLOB_FLAG) ||
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
1121
	(sql_field->sql_type == DRIZZLE_TYPE_VARCHAR && create_info->row_type != ROW_TYPE_FIXED))
1 by brian
clean slate
1122
      (*db_options)|= HA_OPTION_PACK_RECORD;
1123
    it2.rewind();
1124
  }
1125
1126
  /* record_offset will be increased with 'length-of-null-bits' later */
1127
  record_offset= 0;
1128
  null_fields+= total_uneven_bit_length;
1129
1130
  it.rewind();
1131
  while ((sql_field=it++))
1132
  {
51.2.1 by Patrick Galbraith
Removed DBUG_PRINTs, DBUG_ASSERTs, DBUG_EXECUTE_IFs from
1133
    assert(sql_field->charset != 0);
1 by brian
clean slate
1134
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1135
    if (prepare_create_field(sql_field, &blob_columns,
1 by brian
clean slate
1136
			     &timestamps, &timestamps_with_niladic,
1137
			     file->ha_table_flags()))
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1138
      return(true);
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
1139
    if (sql_field->sql_type == DRIZZLE_TYPE_VARCHAR)
55 by brian
Update for using real bool types.
1140
      create_info->varchar= true;
1 by brian
clean slate
1141
    sql_field->offset= record_offset;
1142
    if (MTYP_TYPENR(sql_field->unireg_check) == Field::NEXT_NUMBER)
1143
      auto_increment++;
1144
  }
1145
  if (timestamps_with_niladic > 1)
1146
  {
1147
    my_message(ER_TOO_MUCH_AUTO_TIMESTAMP_COLS,
1148
               ER(ER_TOO_MUCH_AUTO_TIMESTAMP_COLS), MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1149
    return(true);
1 by brian
clean slate
1150
  }
1151
  if (auto_increment > 1)
1152
  {
1153
    my_message(ER_WRONG_AUTO_KEY, ER(ER_WRONG_AUTO_KEY), MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1154
    return(true);
1 by brian
clean slate
1155
  }
1156
  if (auto_increment &&
1157
      (file->ha_table_flags() & HA_NO_AUTO_INCREMENT))
1158
  {
1159
    my_message(ER_TABLE_CANT_HANDLE_AUTO_INCREMENT,
1160
               ER(ER_TABLE_CANT_HANDLE_AUTO_INCREMENT), MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1161
    return(true);
1 by brian
clean slate
1162
  }
1163
1164
  if (blob_columns && (file->ha_table_flags() & HA_NO_BLOBS))
1165
  {
1166
    my_message(ER_TABLE_CANT_HANDLE_BLOB, ER(ER_TABLE_CANT_HANDLE_BLOB),
1167
               MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1168
    return(true);
1 by brian
clean slate
1169
  }
1170
1171
  /* Create keys */
1172
1173
  List_iterator<Key> key_iterator(alter_info->key_list);
1174
  List_iterator<Key> key_iterator2(alter_info->key_list);
482 by Brian Aker
Remove uint.
1175
  uint32_t key_parts=0, fk_key_count=0;
1 by brian
clean slate
1176
  bool primary_key=0,unique_key=0;
1177
  Key *key, *key2;
482 by Brian Aker
Remove uint.
1178
  uint32_t tmp, key_number;
1 by brian
clean slate
1179
  /* special marker for keys to be ignored */
1180
  static char ignore_key[1];
1181
1182
  /* Calculate number of key segements */
1183
  *key_count= 0;
1184
1185
  while ((key=key_iterator++))
1186
  {
1187
    if (key->type == Key::FOREIGN_KEY)
1188
    {
1189
      fk_key_count++;
383.7.1 by Andrey Zhakov
Initial submit of code and tests
1190
      if (((Foreign_key *)key)->validate(alter_info->create_list))
1191
        return true;
1 by brian
clean slate
1192
      Foreign_key *fk_key= (Foreign_key*) key;
1193
      if (fk_key->ref_columns.elements &&
1194
	  fk_key->ref_columns.elements != fk_key->columns.elements)
1195
      {
1196
        my_error(ER_WRONG_FK_DEF, MYF(0),
1197
                 (fk_key->name.str ? fk_key->name.str :
1198
                                     "foreign key without name"),
1199
                 ER(ER_KEY_REF_DO_NOT_MATCH_TABLE_REF));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1200
	return(true);
1 by brian
clean slate
1201
      }
1202
      continue;
1203
    }
1204
    (*key_count)++;
1205
    tmp=file->max_key_parts();
1206
    if (key->columns.elements > tmp)
1207
    {
1208
      my_error(ER_TOO_MANY_KEY_PARTS,MYF(0),tmp);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1209
      return(true);
1 by brian
clean slate
1210
    }
1211
    if (check_identifier_name(&key->name, ER_TOO_LONG_IDENT))
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1212
      return(true);
1 by brian
clean slate
1213
    key_iterator2.rewind ();
1214
    if (key->type != Key::FOREIGN_KEY)
1215
    {
1216
      while ((key2 = key_iterator2++) != key)
1217
      {
1218
	/*
1219
          foreign_key_prefix(key, key2) returns 0 if key or key2, or both, is
1220
          'generated', and a generated key is a prefix of the other key.
1221
          Then we do not need the generated shorter key.
1222
        */
1223
        if ((key2->type != Key::FOREIGN_KEY &&
1224
             key2->name.str != ignore_key &&
1225
             !foreign_key_prefix(key, key2)))
1226
        {
1227
          /* TODO: issue warning message */
1228
          /* mark that the generated key should be ignored */
1229
          if (!key2->generated ||
1230
              (key->generated && key->columns.elements <
1231
               key2->columns.elements))
1232
            key->name.str= ignore_key;
1233
          else
1234
          {
1235
            key2->name.str= ignore_key;
1236
            key_parts-= key2->columns.elements;
1237
            (*key_count)--;
1238
          }
1239
          break;
1240
        }
1241
      }
1242
    }
1243
    if (key->name.str != ignore_key)
1244
      key_parts+=key->columns.elements;
1245
    else
1246
      (*key_count)--;
1247
    if (key->name.str && !tmp_table && (key->type != Key::PRIMARY) &&
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
1248
        is_primary_key_name(key->name.str))
1 by brian
clean slate
1249
    {
1250
      my_error(ER_WRONG_NAME_FOR_INDEX, MYF(0), key->name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1251
      return(true);
1 by brian
clean slate
1252
    }
1253
  }
1254
  tmp=file->max_keys();
1255
  if (*key_count > tmp)
1256
  {
1257
    my_error(ER_TOO_MANY_KEYS,MYF(0),tmp);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1258
    return(true);
1 by brian
clean slate
1259
  }
1260
1261
  (*key_info_buffer)= key_info= (KEY*) sql_calloc(sizeof(KEY) * (*key_count));
1262
  key_part_info=(KEY_PART_INFO*) sql_calloc(sizeof(KEY_PART_INFO)*key_parts);
1263
  if (!*key_info_buffer || ! key_part_info)
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1264
    return(true);				// Out of memory
1 by brian
clean slate
1265
1266
  key_iterator.rewind();
1267
  key_number=0;
1268
  for (; (key=key_iterator++) ; key_number++)
1269
  {
482 by Brian Aker
Remove uint.
1270
    uint32_t key_length=0;
1 by brian
clean slate
1271
    Key_part_spec *column;
1272
1273
    if (key->name.str == ignore_key)
1274
    {
1275
      /* ignore redundant keys */
1276
      do
1277
	key=key_iterator++;
1278
      while (key && key->name.str == ignore_key);
1279
      if (!key)
1280
	break;
1281
    }
1282
1283
    switch (key->type) {
1284
    case Key::MULTIPLE:
1285
	key_info->flags= 0;
1286
	break;
1287
    case Key::FOREIGN_KEY:
1288
      key_number--;				// Skip this key
1289
      continue;
1290
    default:
1291
      key_info->flags = HA_NOSAME;
1292
      break;
1293
    }
1294
    if (key->generated)
1295
      key_info->flags|= HA_GENERATED_KEY;
1296
206 by Brian Aker
Removed final uint dead types.
1297
    key_info->key_parts=(uint8_t) key->columns.elements;
1 by brian
clean slate
1298
    key_info->key_part=key_part_info;
1299
    key_info->usable_key_parts= key_number;
1300
    key_info->algorithm= key->key_create_info.algorithm;
1301
1302
    /* Take block size from key part or table part */
1303
    /*
1304
      TODO: Add warning if block size changes. We can't do it here, as
1305
      this may depend on the size of the key
1306
    */
1307
    key_info->block_size= (key->key_create_info.block_size ?
1308
                           key->key_create_info.block_size :
1309
                           create_info->key_block_size);
1310
1311
    if (key_info->block_size)
1312
      key_info->flags|= HA_USES_BLOCK_SIZE;
1313
482 by Brian Aker
Remove uint.
1314
    uint32_t tmp_len= system_charset_info->cset->charpos(system_charset_info,
1 by brian
clean slate
1315
                                           key->key_create_info.comment.str,
1316
                                           key->key_create_info.comment.str +
1317
                                           key->key_create_info.comment.length,
1318
                                           INDEX_COMMENT_MAXLEN);
1319
1320
    if (tmp_len < key->key_create_info.comment.length)
1321
    {
1322
      my_error(ER_WRONG_STRING_LENGTH, MYF(0),
1323
               key->key_create_info.comment.str,"INDEX COMMENT",
895 by Brian Aker
Completion (?) of uint conversion.
1324
               (uint32_t) INDEX_COMMENT_MAXLEN);
1046.1.10 by Brian Aker
Formatting around return (style)
1325
      return -1;
1 by brian
clean slate
1326
    }
1327
1328
    key_info->comment.length= key->key_create_info.comment.length;
1329
    if (key_info->comment.length > 0)
1330
    {
1331
      key_info->flags|= HA_USES_COMMENT;
1332
      key_info->comment.str= key->key_create_info.comment.str;
1333
    }
1334
1335
    List_iterator<Key_part_spec> cols(key->columns), cols2(key->columns);
482 by Brian Aker
Remove uint.
1336
    for (uint32_t column_nr=0 ; (column=cols++) ; column_nr++)
1 by brian
clean slate
1337
    {
482 by Brian Aker
Remove uint.
1338
      uint32_t length;
1 by brian
clean slate
1339
      Key_part_spec *dup_column;
1340
1341
      it.rewind();
1342
      field=0;
1343
      while ((sql_field=it++) &&
1344
	     my_strcasecmp(system_charset_info,
1345
			   column->field_name.str,
1346
			   sql_field->field_name))
1347
	field++;
1348
      if (!sql_field)
1349
      {
1350
	my_error(ER_KEY_COLUMN_DOES_NOT_EXITS, MYF(0), column->field_name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1351
	return(true);
1 by brian
clean slate
1352
      }
1353
      while ((dup_column= cols2++) != column)
1354
      {
1355
        if (!my_strcasecmp(system_charset_info,
1356
	     	           column->field_name.str, dup_column->field_name.str))
1357
	{
1358
	  my_printf_error(ER_DUP_FIELDNAME,
1359
			  ER(ER_DUP_FIELDNAME),MYF(0),
1360
			  column->field_name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1361
	  return(true);
1 by brian
clean slate
1362
	}
1363
      }
1364
      cols2.rewind();
1365
      {
1366
	column->length*= sql_field->charset->mbmaxlen;
1367
1368
	if (f_is_blob(sql_field->pack_flag))
1369
	{
1370
	  if (!(file->ha_table_flags() & HA_CAN_INDEX_BLOBS))
1371
	  {
1372
	    my_error(ER_BLOB_USED_AS_KEY, MYF(0), column->field_name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1373
	    return(true);
1 by brian
clean slate
1374
	  }
1375
	  if (!column->length)
1376
	  {
1377
	    my_error(ER_BLOB_KEY_WITHOUT_LENGTH, MYF(0), column->field_name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1378
	    return(true);
1 by brian
clean slate
1379
	  }
1380
	}
1381
	if (!(sql_field->flags & NOT_NULL_FLAG))
1382
	{
1383
	  if (key->type == Key::PRIMARY)
1384
	  {
1385
	    /* Implicitly set primary key fields to NOT NULL for ISO conf. */
1386
	    sql_field->flags|= NOT_NULL_FLAG;
1387
	    sql_field->pack_flag&= ~FIELDFLAG_MAYBE_NULL;
1388
            null_fields--;
1389
	  }
1390
	  else
1391
          {
1392
            key_info->flags|= HA_NULL_PART_KEY;
1393
            if (!(file->ha_table_flags() & HA_NULL_IN_KEY))
1394
            {
1395
              my_error(ER_NULL_COLUMN_IN_INDEX, MYF(0), column->field_name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1396
              return(true);
1 by brian
clean slate
1397
            }
1398
          }
1399
	}
1400
	if (MTYP_TYPENR(sql_field->unireg_check) == Field::NEXT_NUMBER)
1401
	{
1402
	  if (column_nr == 0 || (file->ha_table_flags() & HA_AUTO_PART_KEY))
1403
	    auto_increment--;			// Field is used
1404
	}
1405
      }
1406
1407
      key_part_info->fieldnr= field;
206 by Brian Aker
Removed final uint dead types.
1408
      key_part_info->offset=  (uint16_t) sql_field->offset;
1 by brian
clean slate
1409
      key_part_info->key_type=sql_field->pack_flag;
1410
      length= sql_field->key_length;
1411
1412
      if (column->length)
1413
      {
1414
	if (f_is_blob(sql_field->pack_flag))
1415
	{
1416
	  if ((length=column->length) > max_key_length ||
1417
	      length > file->max_key_part_length())
1418
	  {
398.1.4 by Monty Taylor
Renamed max/min.
1419
	    length=cmin(max_key_length, file->max_key_part_length());
1 by brian
clean slate
1420
	    if (key->type == Key::MULTIPLE)
1421
	    {
1422
	      /* not a critical problem */
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
1423
	      char warn_buff[DRIZZLE_ERRMSG_SIZE];
77.1.18 by Monty Taylor
Removed my_vsnprintf and my_snprintf.
1424
	      snprintf(warn_buff, sizeof(warn_buff), ER(ER_TOO_LONG_KEY),
1425
                       length);
520.1.22 by Brian Aker
Second pass of thd cleanup
1426
	      push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
1 by brian
clean slate
1427
			   ER_TOO_LONG_KEY, warn_buff);
1428
              /* Align key length to multibyte char boundary */
1429
              length-= length % sql_field->charset->mbmaxlen;
1430
	    }
1431
	    else
1432
	    {
1433
	      my_error(ER_TOO_LONG_KEY,MYF(0),length);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1434
	      return(true);
1 by brian
clean slate
1435
	    }
1436
	  }
1437
	}
1438
	else if ((column->length > length ||
1439
                   !Field::type_can_have_key_part (sql_field->sql_type) ||
1440
		   ((f_is_packed(sql_field->pack_flag) ||
1441
		     ((file->ha_table_flags() & HA_NO_PREFIX_CHAR_KEYS) &&
1442
		      (key_info->flags & HA_NOSAME))) &&
1443
		    column->length != length)))
1444
	{
1445
	  my_message(ER_WRONG_SUB_KEY, ER(ER_WRONG_SUB_KEY), MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1446
	  return(true);
1 by brian
clean slate
1447
	}
1448
	else if (!(file->ha_table_flags() & HA_NO_PREFIX_CHAR_KEYS))
1449
	  length=column->length;
1450
      }
1451
      else if (length == 0)
1452
      {
1453
	my_error(ER_WRONG_KEY_COLUMN, MYF(0), column->field_name.str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1454
	  return(true);
1 by brian
clean slate
1455
      }
1456
      if (length > file->max_key_part_length())
1457
      {
1458
        length= file->max_key_part_length();
1459
	if (key->type == Key::MULTIPLE)
1460
	{
1461
	  /* not a critical problem */
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
1462
	  char warn_buff[DRIZZLE_ERRMSG_SIZE];
77.1.18 by Monty Taylor
Removed my_vsnprintf and my_snprintf.
1463
	  snprintf(warn_buff, sizeof(warn_buff), ER(ER_TOO_LONG_KEY),
1464
                   length);
520.1.22 by Brian Aker
Second pass of thd cleanup
1465
	  push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
1 by brian
clean slate
1466
		       ER_TOO_LONG_KEY, warn_buff);
1467
          /* Align key length to multibyte char boundary */
1468
          length-= length % sql_field->charset->mbmaxlen;
1469
	}
1470
	else
1471
	{
1472
	  my_error(ER_TOO_LONG_KEY,MYF(0),length);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1473
	  return(true);
1 by brian
clean slate
1474
	}
1475
      }
206 by Brian Aker
Removed final uint dead types.
1476
      key_part_info->length=(uint16_t) length;
1 by brian
clean slate
1477
      /* Use packed keys for long strings on the first column */
1478
      if (!((*db_options) & HA_OPTION_NO_PACK_KEYS) &&
1479
	  (length >= KEY_DEFAULT_PACK_LENGTH &&
241 by Brian Aker
First pass of CHAR removal.
1480
	   (sql_field->sql_type == DRIZZLE_TYPE_VARCHAR ||
1 by brian
clean slate
1481
	    sql_field->pack_flag & FIELDFLAG_BLOB)))
1482
      {
1483
	if ((column_nr == 0 && (sql_field->pack_flag & FIELDFLAG_BLOB)) ||
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
1484
            sql_field->sql_type == DRIZZLE_TYPE_VARCHAR)
1 by brian
clean slate
1485
	  key_info->flags|= HA_BINARY_PACK_KEY | HA_VAR_LENGTH_KEY;
1486
	else
1487
	  key_info->flags|= HA_PACK_KEY;
1488
      }
1489
      /* Check if the key segment is partial, set the key flag accordingly */
1490
      if (length != sql_field->key_length)
1491
        key_info->flags|= HA_KEY_HAS_PART_KEY_SEG;
1492
1493
      key_length+=length;
1494
      key_part_info++;
1495
1496
      /* Create the key name based on the first column (if not given) */
1497
      if (column_nr == 0)
1498
      {
1499
	if (key->type == Key::PRIMARY)
1500
	{
1501
	  if (primary_key)
1502
	  {
1503
	    my_message(ER_MULTIPLE_PRI_KEY, ER(ER_MULTIPLE_PRI_KEY),
1504
                       MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1505
	    return(true);
1 by brian
clean slate
1506
	  }
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
1507
          static const char pkey_name[]= "PRIMARY";
1508
	  key_name=pkey_name;
1 by brian
clean slate
1509
	  primary_key=1;
1510
	}
1511
	else if (!(key_name= key->name.str))
1512
	  key_name=make_unique_key_name(sql_field->field_name,
1513
					*key_info_buffer, key_info);
1514
	if (check_if_keyname_exists(key_name, *key_info_buffer, key_info))
1515
	{
1516
	  my_error(ER_DUP_KEYNAME, MYF(0), key_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1517
	  return(true);
1 by brian
clean slate
1518
	}
1519
	key_info->name=(char*) key_name;
1520
      }
1521
    }
1522
    if (!key_info->name || check_column_name(key_info->name))
1523
    {
1524
      my_error(ER_WRONG_NAME_FOR_INDEX, MYF(0), key_info->name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1525
      return(true);
1 by brian
clean slate
1526
    }
1527
    if (!(key_info->flags & HA_NULL_PART_KEY))
1528
      unique_key=1;
206 by Brian Aker
Removed final uint dead types.
1529
    key_info->key_length=(uint16_t) key_length;
1 by brian
clean slate
1530
    if (key_length > max_key_length)
1531
    {
1532
      my_error(ER_TOO_LONG_KEY,MYF(0),max_key_length);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1533
      return(true);
1 by brian
clean slate
1534
    }
1535
    key_info++;
1536
  }
1537
  if (!unique_key && !primary_key &&
1538
      (file->ha_table_flags() & HA_REQUIRE_PRIMARY_KEY))
1539
  {
1540
    my_message(ER_REQUIRES_PRIMARY_KEY, ER(ER_REQUIRES_PRIMARY_KEY), MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1541
    return(true);
1 by brian
clean slate
1542
  }
1543
  if (auto_increment > 0)
1544
  {
1545
    my_message(ER_WRONG_AUTO_KEY, ER(ER_WRONG_AUTO_KEY), MYF(0));
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1546
    return(true);
1 by brian
clean slate
1547
  }
1548
  /* Sort keys in optimized order */
481 by Brian Aker
Remove all of uchar.
1549
  my_qsort((unsigned char*) *key_info_buffer, *key_count, sizeof(KEY),
1 by brian
clean slate
1550
	   (qsort_cmp) sort_keys);
1551
  create_info->null_bits= null_fields;
1552
1553
  /* Check fields. */
1554
  it.rewind();
1555
  while ((sql_field=it++))
1556
  {
1557
    Field::utype type= (Field::utype) MTYP_TYPENR(sql_field->unireg_check);
1558
520.1.22 by Brian Aker
Second pass of thd cleanup
1559
    if (session->variables.sql_mode & MODE_NO_ZERO_DATE &&
1 by brian
clean slate
1560
        !sql_field->def &&
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
1561
        sql_field->sql_type == DRIZZLE_TYPE_TIMESTAMP &&
1 by brian
clean slate
1562
        (sql_field->flags & NOT_NULL_FLAG) &&
1563
        (type == Field::NONE || type == Field::TIMESTAMP_UN_FIELD))
1564
    {
1565
      /*
1566
        An error should be reported if:
1567
          - NO_ZERO_DATE SQL mode is active;
1568
          - there is no explicit DEFAULT clause (default column value);
1569
          - this is a TIMESTAMP column;
1570
          - the column is not NULL;
1571
          - this is not the DEFAULT CURRENT_TIMESTAMP column.
1572
1573
        In other words, an error should be reported if
1574
          - NO_ZERO_DATE SQL mode is active;
1575
          - the column definition is equivalent to
1576
            'column_name TIMESTAMP DEFAULT 0'.
1577
      */
1578
1579
      my_error(ER_INVALID_DEFAULT, MYF(0), sql_field->field_name);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1580
      return(true);
1 by brian
clean slate
1581
    }
1582
  }
1583
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
1584
  return(false);
1 by brian
clean slate
1585
}
1586
1587
/*
1588
  Extend long VARCHAR fields to blob & prepare field if it's a blob
1589
1590
  SYNOPSIS
1591
    prepare_blob_field()
1592
    sql_field		Field to check
1593
1594
  RETURN
1595
    0	ok
1596
    1	Error (sql_field can't be converted to blob)
1597
        In this case the error is given
1598
*/
1599
779.1.27 by Monty Taylor
Got rid of __attribute__((unused)) and the like from the .cc files.
1600
static bool prepare_blob_field(Session *,
77.1.46 by Monty Taylor
Finished the warnings work!
1601
                               Create_field *sql_field)
1 by brian
clean slate
1602
{
1603
1604
  if (sql_field->length > MAX_FIELD_VARCHARLENGTH &&
1605
      !(sql_field->flags & BLOB_FLAG))
1606
  {
1607
    my_error(ER_TOO_BIG_FIELDLENGTH, MYF(0), sql_field->field_name,
1608
             MAX_FIELD_VARCHARLENGTH / sql_field->charset->mbmaxlen);
1046.1.10 by Brian Aker
Formatting around return (style)
1609
    return 1;
1 by brian
clean slate
1610
  }
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1611
1 by brian
clean slate
1612
  if ((sql_field->flags & BLOB_FLAG) && sql_field->length)
1613
  {
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
1614
    if (sql_field->sql_type == DRIZZLE_TYPE_BLOB)
1 by brian
clean slate
1615
    {
1616
      /* The user has given a length to the blob column */
1617
      sql_field->sql_type= get_blob_type_from_length(sql_field->length);
1618
      sql_field->pack_length= calc_pack_length(sql_field->sql_type, 0);
1619
    }
1620
    sql_field->length= 0;
1621
  }
1046.1.10 by Brian Aker
Formatting around return (style)
1622
  return 0;
1 by brian
clean slate
1623
}
1624
1625
1626
/*
1039.1.11 by Brian Aker
Refactor function to make sense.
1627
  Ignore the name of this function... it locks :(
1628
1 by brian
clean slate
1629
  Create a table
1630
1631
  SYNOPSIS
1632
    mysql_create_table_no_lock()
520.1.22 by Brian Aker
Second pass of thd cleanup
1633
    session			Thread object
1 by brian
clean slate
1634
    db			Database
1635
    table_name		Table name
1636
    create_info	        Create information (like MAX_ROWS)
1637
    fields		List of fields to create
1638
    keys		List of keys to create
1639
    internal_tmp_table  Set to 1 if this is an internal temporary table
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
1640
			(From ALTER Table)
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1641
    select_field_count
1 by brian
clean slate
1642
1643
  DESCRIPTION
1644
    If one creates a temporary table, this is automatically opened
1645
1646
    Note that this function assumes that caller already have taken
1647
    name-lock on table being created or used some other way to ensure
1648
    that concurrent operations won't intervene. mysql_create_table()
1649
    is a wrapper that can be used for this.
1650
1651
  RETURN VALUES
55 by brian
Update for using real bool types.
1652
    false OK
1653
    true  error
1 by brian
clean slate
1654
*/
1655
520.1.22 by Brian Aker
Second pass of thd cleanup
1656
bool mysql_create_table_no_lock(Session *session,
1 by brian
clean slate
1657
                                const char *db, const char *table_name,
1658
                                HA_CREATE_INFO *create_info,
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
1659
				drizzled::message::Table *table_proto,
1 by brian
clean slate
1660
                                Alter_info *alter_info,
1661
                                bool internal_tmp_table,
1039.1.11 by Brian Aker
Refactor function to make sense.
1662
                                uint32_t select_field_count)
1 by brian
clean slate
1663
{
1664
  char		path[FN_REFLEN];
482 by Brian Aker
Remove uint.
1665
  uint32_t          path_length;
1 by brian
clean slate
1666
  uint		db_options, key_count;
1667
  KEY		*key_info_buffer;
1668
  handler	*file;
55 by brian
Update for using real bool types.
1669
  bool		error= true;
1 by brian
clean slate
1670
  /* Check for duplicate fields and check type of table to create */
1671
  if (!alter_info->create_list.elements)
1672
  {
1673
    my_message(ER_TABLE_MUST_HAVE_COLUMNS, ER(ER_TABLE_MUST_HAVE_COLUMNS),
1674
               MYF(0));
1039.1.11 by Brian Aker
Refactor function to make sense.
1675
    return true;
1 by brian
clean slate
1676
  }
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
1677
  assert(strcmp(table_name,table_proto->name().c_str())==0);
520.1.22 by Brian Aker
Second pass of thd cleanup
1678
  if (check_engine(session, table_name, create_info))
1039.1.11 by Brian Aker
Refactor function to make sense.
1679
    return true;
1 by brian
clean slate
1680
  db_options= create_info->table_options;
1681
  if (create_info->row_type == ROW_TYPE_DYNAMIC)
1682
    db_options|=HA_OPTION_PACK_RECORD;
1000.1.3 by Brian Aker
Renamed TABLE_SHARE to TableShare
1683
  if (!(file= get_new_handler((TableShare*) 0, session->mem_root,
1 by brian
clean slate
1684
                              create_info->db_type)))
1685
  {
1686
    my_error(ER_OUTOFMEMORY, MYF(0), sizeof(handler));
1039.1.11 by Brian Aker
Refactor function to make sense.
1687
    return true;
1 by brian
clean slate
1688
  }
1689
1014.3.1 by Brian Aker
Simplify the calling stack for getting schema collation. We need to extend
1690
  set_table_default_charset(create_info, (char*) db);
1 by brian
clean slate
1691
520.1.22 by Brian Aker
Second pass of thd cleanup
1692
  if (mysql_prepare_create_table(session, create_info, alter_info,
1 by brian
clean slate
1693
                                 internal_tmp_table,
1694
                                 &db_options, file,
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
1695
                                 &key_info_buffer, &key_count,
1696
                                 select_field_count))
1 by brian
clean slate
1697
    goto err;
1698
1699
      /* Check if table exists */
1700
  if (create_info->options & HA_LEX_CREATE_TMP_TABLE)
1701
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
1702
    path_length= build_tmptable_filename(session, path, sizeof(path));
1 by brian
clean slate
1703
    create_info->table_options|=HA_CREATE_DELAY_KEY_WRITE;
1704
  }
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
1705
  else
1 by brian
clean slate
1706
  {
1707
 #ifdef FN_DEVCHAR
1708
    /* check if the table name contains FN_DEVCHAR when defined */
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
1709
    if (strchr(table_name, FN_DEVCHAR))
1 by brian
clean slate
1710
    {
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
1711
      my_error(ER_WRONG_TABLE_NAME, MYF(0), table_name);
1039.1.11 by Brian Aker
Refactor function to make sense.
1712
      return true;
1 by brian
clean slate
1713
    }
1714
#endif
1039.1.6 by Brian Aker
Refactor for build_table_filename()
1715
    path_length= build_table_filename(path, sizeof(path), db, table_name, internal_tmp_table);
1 by brian
clean slate
1716
  }
1717
1718
  /* Check if table already exists */
1719
  if ((create_info->options & HA_LEX_CREATE_TMP_TABLE) &&
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
1720
      session->find_temporary_table(db, table_name))
1 by brian
clean slate
1721
  {
1722
    if (create_info->options & HA_LEX_CREATE_IF_NOT_EXISTS)
1723
    {
1724
      create_info->table_existed= 1;		// Mark that table existed
520.1.22 by Brian Aker
Second pass of thd cleanup
1725
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
1726
                          ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
1727
                          table_name);
1 by brian
clean slate
1728
      error= 0;
1729
      goto err;
1730
    }
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
1731
    my_error(ER_TABLE_EXISTS_ERROR, MYF(0), table_name);
1 by brian
clean slate
1732
    goto err;
1733
  }
1734
1046.1.2 by Brian Aker
Comments on LOCK_open
1735
  pthread_mutex_lock(&LOCK_open); /* CREATE TABLE (some confussion on naming, double check) */
1 by brian
clean slate
1736
  if (!internal_tmp_table && !(create_info->options & HA_LEX_CREATE_TMP_TABLE))
1737
  {
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
1738
    if (table_proto_exists(path)==EEXIST)
1 by brian
clean slate
1739
    {
1740
      if (create_info->options & HA_LEX_CREATE_IF_NOT_EXISTS)
1039.1.11 by Brian Aker
Refactor function to make sense.
1741
      {
1742
        error= false;
1743
        push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1744
                            ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1745
                            table_name);
1746
        create_info->table_existed= 1;		// Mark that table existed
1747
      }
1748
      else 
1749
        my_error(ER_TABLE_EXISTS_ERROR,MYF(0),table_name);
1750
1 by brian
clean slate
1751
      goto unlock_and_end;
1752
    }
1753
    /*
1754
      We don't assert here, but check the result, because the table could be
1755
      in the table definition cache and in the same time the .frm could be
1756
      missing from the disk, in case of manual intervention which deletes
1757
      the .frm file. The user has to use FLUSH TABLES; to clear the cache.
1758
      Then she could create the table. This case is pretty obscure and
1759
      therefore we don't introduce a new error message only for it.
1760
    */
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
1761
    if (get_cached_table_share(db, table_name))
1 by brian
clean slate
1762
    {
1763
      my_error(ER_TABLE_EXISTS_ERROR, MYF(0), table_name);
1764
      goto unlock_and_end;
1765
    }
1766
  }
1767
1768
  /*
1769
    Check that table with given name does not already
1770
    exist in any storage engine. In such a case it should
1771
    be discovered and the error ER_TABLE_EXISTS_ERROR be returned
1772
    unless user specified CREATE TABLE IF EXISTS
1773
    The LOCK_open mutex has been locked to make sure no
1774
    one else is attempting to discover the table. Since
1775
    it's not on disk as a frm file, no one could be using it!
1776
  */
1777
  if (!(create_info->options & HA_LEX_CREATE_TMP_TABLE))
1778
  {
1779
    bool create_if_not_exists =
1780
      create_info->options & HA_LEX_CREATE_IF_NOT_EXISTS;
520.1.22 by Brian Aker
Second pass of thd cleanup
1781
    int retcode = ha_table_exists_in_engine(session, db, table_name);
1 by brian
clean slate
1782
    switch (retcode)
1783
    {
1784
      case HA_ERR_NO_SUCH_TABLE:
1785
        /* Normal case, no table exists. we can go and create it */
1786
        break;
1787
      case HA_ERR_TABLE_EXIST:
892.2.4 by Monty Taylor
Fixed more warnings.
1788
        if (create_if_not_exists)
1039.1.11 by Brian Aker
Refactor function to make sense.
1789
        {
1790
          error= false;
1791
          push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1792
                              ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1793
                              table_name);
1794
          create_info->table_existed= 1;		// Mark that table existed
1795
          goto unlock_and_end;
1796
        }
892.2.4 by Monty Taylor
Fixed more warnings.
1797
        my_error(ER_TABLE_EXISTS_ERROR,MYF(0),table_name);
1798
        goto unlock_and_end;
1 by brian
clean slate
1799
      default:
1800
        my_error(retcode, MYF(0),table_name);
1801
        goto unlock_and_end;
1802
    }
1803
  }
1804
520.1.22 by Brian Aker
Second pass of thd cleanup
1805
  session->set_proc_info("creating table");
1 by brian
clean slate
1806
  create_info->table_existed= 0;		// Mark that table is created
1807
1808
#ifdef HAVE_READLINK
1809
  if (test_if_data_home_dir(create_info->data_file_name))
1810
  {
1811
    my_error(ER_WRONG_ARGUMENTS, MYF(0), "DATA DIRECTORY");
1812
    goto unlock_and_end;
1813
  }
1814
  if (test_if_data_home_dir(create_info->index_file_name))
1815
  {
1816
    my_error(ER_WRONG_ARGUMENTS, MYF(0), "INDEX DIRECTORY");
1817
    goto unlock_and_end;
1818
  }
1819
1820
  if (!my_use_symdir)
1821
#endif /* HAVE_READLINK */
1822
  {
1823
    if (create_info->data_file_name)
520.1.22 by Brian Aker
Second pass of thd cleanup
1824
      push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN, 0,
1 by brian
clean slate
1825
                   "DATA DIRECTORY option ignored");
1826
    if (create_info->index_file_name)
520.1.22 by Brian Aker
Second pass of thd cleanup
1827
      push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN, 0,
1 by brian
clean slate
1828
                   "INDEX DIRECTORY option ignored");
1829
    create_info->data_file_name= create_info->index_file_name= 0;
1830
  }
1831
  create_info->table_options=db_options;
1832
520.1.22 by Brian Aker
Second pass of thd cleanup
1833
  if (rea_create_table(session, path, db, table_name,
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
1834
		       table_proto,
1 by brian
clean slate
1835
                       create_info, alter_info->create_list,
1008.3.23 by Stewart Smith
remove unused (originally only in ha_partition, but should really be part of ha_create there too) ha_create_handler_files
1836
                       key_count, key_info_buffer, false))
1 by brian
clean slate
1837
    goto unlock_and_end;
1838
1839
  if (create_info->options & HA_LEX_CREATE_TMP_TABLE)
1840
  {
1841
    /* Open table and put in temporary table list */
520.1.22 by Brian Aker
Second pass of thd cleanup
1842
    if (!(open_temporary_table(session, path, db, table_name, 1, OTM_OPEN)))
1 by brian
clean slate
1843
    {
590.1.3 by Stewart Smith
remove frm_only create_info option
1844
      (void) rm_temporary_table(create_info->db_type, path);
1 by brian
clean slate
1845
      goto unlock_and_end;
1846
    }
1847
  }
1848
1849
  /*
1850
    Don't write statement if:
1851
    - It is an internal temporary table,
1852
    - Row-based logging is used and it we are creating a temporary table, or
1853
    - The binary log is not open.
1854
    Otherwise, the statement shall be binlogged.
1855
   */
1856
  if (!internal_tmp_table &&
581 by Brian Aker
Second pass through on replication row patch
1857
      ((!(create_info->options & HA_LEX_CREATE_TMP_TABLE))))
520.1.22 by Brian Aker
Second pass of thd cleanup
1858
    write_bin_log(session, true, session->query, session->query_length);
55 by brian
Update for using real bool types.
1859
  error= false;
1 by brian
clean slate
1860
unlock_and_end:
1039.1.11 by Brian Aker
Refactor function to make sense.
1861
  pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
1862
1863
err:
520.1.22 by Brian Aker
Second pass of thd cleanup
1864
  session->set_proc_info("After create");
1 by brian
clean slate
1865
  delete file;
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
1866
  return(error);
1 by brian
clean slate
1867
}
1868
1869
1870
/*
1871
  Database locking aware wrapper for mysql_create_table_no_lock(),
1872
*/
1873
520.1.22 by Brian Aker
Second pass of thd cleanup
1874
bool mysql_create_table(Session *session, const char *db, const char *table_name,
1 by brian
clean slate
1875
                        HA_CREATE_INFO *create_info,
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
1876
			drizzled::message::Table *table_proto,
1 by brian
clean slate
1877
                        Alter_info *alter_info,
1878
                        bool internal_tmp_table,
482 by Brian Aker
Remove uint.
1879
                        uint32_t select_field_count)
1 by brian
clean slate
1880
{
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
1881
  Table *name_lock= NULL;
1 by brian
clean slate
1882
  bool result;
1883
1884
  if (!(create_info->options & HA_LEX_CREATE_TMP_TABLE))
1885
  {
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
1886
    if (session->lock_table_name_if_not_cached(db, table_name, &name_lock))
1 by brian
clean slate
1887
    {
55 by brian
Update for using real bool types.
1888
      result= true;
1 by brian
clean slate
1889
      goto unlock;
1890
    }
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
1891
    if (name_lock == NULL)
1 by brian
clean slate
1892
    {
1893
      if (create_info->options & HA_LEX_CREATE_IF_NOT_EXISTS)
1894
      {
520.1.22 by Brian Aker
Second pass of thd cleanup
1895
        push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
1896
                            ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1897
                            table_name);
1898
        create_info->table_existed= 1;
55 by brian
Update for using real bool types.
1899
        result= false;
1 by brian
clean slate
1900
      }
1901
      else
1902
      {
1903
        my_error(ER_TABLE_EXISTS_ERROR,MYF(0),table_name);
55 by brian
Update for using real bool types.
1904
        result= true;
1 by brian
clean slate
1905
      }
1906
      goto unlock;
1907
    }
1908
  }
1909
520.1.22 by Brian Aker
Second pass of thd cleanup
1910
  result= mysql_create_table_no_lock(session, db, table_name, create_info,
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
1911
				     table_proto,
1 by brian
clean slate
1912
                                     alter_info,
1913
                                     internal_tmp_table,
1039.1.11 by Brian Aker
Refactor function to make sense.
1914
                                     select_field_count);
1 by brian
clean slate
1915
1916
unlock:
1917
  if (name_lock)
1918
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
1919
    pthread_mutex_lock(&LOCK_open); /* Lock for removing name_lock during table create */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
1920
    session->unlink_open_table(name_lock);
1 by brian
clean slate
1921
    pthread_mutex_unlock(&LOCK_open);
1922
  }
1053 by Brian Aker
Merge Stewart, fix LOCK (it was dead, only for dead RENAME SCHEMA code), and
1923
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
1924
  return(result);
1 by brian
clean slate
1925
}
1926
1927
1928
/*
1929
** Give the key name after the first field with an optional '_#' after
1930
**/
1931
1932
static bool
1933
check_if_keyname_exists(const char *name, KEY *start, KEY *end)
1934
{
1935
  for (KEY *key=start ; key != end ; key++)
1936
    if (!my_strcasecmp(system_charset_info,name,key->name))
1937
      return 1;
1938
  return 0;
1939
}
1940
1941
1942
static char *
1943
make_unique_key_name(const char *field_name,KEY *start,KEY *end)
1944
{
1945
  char buff[MAX_FIELD_NAME],*buff_end;
1946
1947
  if (!check_if_keyname_exists(field_name,start,end) &&
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
1948
      !is_primary_key_name(field_name))
1 by brian
clean slate
1949
    return (char*) field_name;			// Use fieldname
629.5.3 by Toru Maesaka
Third pass of replacing MySQL's strmake() with libc calls
1950
1951
  buff_end= strncpy(buff, field_name, sizeof(buff)-4);
1952
  buff_end+= strlen(buff);
1 by brian
clean slate
1953
1954
  /*
1955
    Only 3 chars + '\0' left, so need to limit to 2 digit
1956
    This is ok as we can't have more than 100 keys anyway
1957
  */
482 by Brian Aker
Remove uint.
1958
  for (uint32_t i=2 ; i< 100; i++)
1 by brian
clean slate
1959
  {
1960
    *buff_end= '_';
1961
    int10_to_str(i, buff_end+1, 10);
1962
    if (!check_if_keyname_exists(buff,start,end))
1963
      return sql_strdup(buff);
1964
  }
1965
  return (char*) "not_specified";		// Should never happen
1966
}
1967
1968
1969
/****************************************************************************
1970
** Alter a table definition
1971
****************************************************************************/
1972
1973
1974
/*
1975
  Rename a table.
1976
1977
  SYNOPSIS
1978
    mysql_rename_table()
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
1979
      base                      The StorageEngine handle.
1 by brian
clean slate
1980
      old_db                    The old database name.
1981
      old_name                  The old table name.
1982
      new_db                    The new database name.
1983
      new_name                  The new table name.
1984
      flags                     flags for build_table_filename().
1985
                                FN_FROM_IS_TMP old_name is temporary.
1986
                                FN_TO_IS_TMP   new_name is temporary.
1987
                                NO_FRM_RENAME  Don't rename the FRM file
1988
                                but only the table in the storage engine.
1989
1990
  RETURN
55 by brian
Update for using real bool types.
1991
    false   OK
1992
    true    Error
1 by brian
clean slate
1993
*/
1994
1995
bool
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
1996
mysql_rename_table(StorageEngine *base, const char *old_db,
1 by brian
clean slate
1997
                   const char *old_name, const char *new_db,
482 by Brian Aker
Remove uint.
1998
                   const char *new_name, uint32_t flags)
1 by brian
clean slate
1999
{
520.1.22 by Brian Aker
Second pass of thd cleanup
2000
  Session *session= current_session;
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
2001
  char from[FN_REFLEN], to[FN_REFLEN];
1 by brian
clean slate
2002
  char *from_base= from, *to_base= to;
1039.1.11 by Brian Aker
Refactor function to make sense.
2003
  int error= 0;
1 by brian
clean slate
2004
1039.3.10 by Stewart Smith
move ha_rename_table to just be StorageEngine::renameTable with engines implementing renameTableImpl.
2005
  assert(base);
1 by brian
clean slate
2006
1039.1.6 by Brian Aker
Refactor for build_table_filename()
2007
  build_table_filename(from, sizeof(from), old_db, old_name,
1 by brian
clean slate
2008
                       flags & FN_FROM_IS_TMP);
1039.1.6 by Brian Aker
Refactor for build_table_filename()
2009
  build_table_filename(to, sizeof(to), new_db, new_name,
1 by brian
clean slate
2010
                       flags & FN_TO_IS_TMP);
2011
1039.3.10 by Stewart Smith
move ha_rename_table to just be StorageEngine::renameTable with engines implementing renameTableImpl.
2012
  if (!(error=base->renameTable(session, from_base, to_base)))
1 by brian
clean slate
2013
  {
584.2.7 by Stewart Smith
rename and delete tabledefinition protobuf file
2014
    if(!(flags & NO_FRM_RENAME)
2015
       && rename_table_proto_file(from_base, to_base))
2016
    {
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
2017
      error= my_errno;
1039.3.10 by Stewart Smith
move ha_rename_table to just be StorageEngine::renameTable with engines implementing renameTableImpl.
2018
      base->renameTable(session, to_base, from_base);
584.2.7 by Stewart Smith
rename and delete tabledefinition protobuf file
2019
    }
1 by brian
clean slate
2020
  }
1039.3.11 by Stewart Smith
fix build error in mysql_rename_table: forgot to remove old dead code.
2021
1 by brian
clean slate
2022
  if (error == HA_ERR_WRONG_COMMAND)
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2023
    my_error(ER_NOT_SUPPORTED_YET, MYF(0), "ALTER Table");
1 by brian
clean slate
2024
  else if (error)
2025
    my_error(ER_ERROR_ON_RENAME, MYF(0), from, to, error);
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
2026
  return(error != 0);
1 by brian
clean slate
2027
}
2028
2029
2030
/*
2031
  Force all other threads to stop using the table
2032
2033
  SYNOPSIS
2034
    wait_while_table_is_used()
520.1.22 by Brian Aker
Second pass of thd cleanup
2035
    session			Thread handler
1 by brian
clean slate
2036
    table		Table to remove from cache
2037
    function            HA_EXTRA_PREPARE_FOR_DROP if table is to be deleted
2038
                        HA_EXTRA_FORCE_REOPEN if table is not be used
2039
                        HA_EXTRA_PREPARE_FOR_RENAME if table is to be renamed
2040
  NOTES
2041
   When returning, the table will be unusable for other threads until
2042
   the table is closed.
2043
2044
  PREREQUISITES
2045
    Lock on LOCK_open
2046
    Win32 clients must also have a WRITE LOCK on the table !
2047
*/
2048
520.1.22 by Brian Aker
Second pass of thd cleanup
2049
void wait_while_table_is_used(Session *session, Table *table,
1 by brian
clean slate
2050
                              enum ha_extra_function function)
2051
{
2052
2053
  safe_mutex_assert_owner(&LOCK_open);
2054
398.1.10 by Monty Taylor
Actually removed VOID() this time.
2055
  table->file->extra(function);
1 by brian
clean slate
2056
  /* Mark all tables that are in use as 'old' */
520.1.22 by Brian Aker
Second pass of thd cleanup
2057
  mysql_lock_abort(session, table, true);	/* end threads waiting on lock */
1 by brian
clean slate
2058
2059
  /* Wait until all there are no other threads that has this table open */
520.1.22 by Brian Aker
Second pass of thd cleanup
2060
  remove_table_from_cache(session, table->s->db.str,
1 by brian
clean slate
2061
                          table->s->table_name.str,
2062
                          RTFC_WAIT_OTHER_THREAD_FLAG);
2063
}
2064
2065
/*
2066
  Close a cached table
2067
2068
  SYNOPSIS
2069
    close_cached_table()
520.1.22 by Brian Aker
Second pass of thd cleanup
2070
    session			Thread handler
1 by brian
clean slate
2071
    table		Table to remove from cache
2072
2073
  NOTES
2074
    Function ends by signaling threads waiting for the table to try to
2075
    reopen the table.
2076
2077
  PREREQUISITES
2078
    Lock on LOCK_open
2079
    Win32 clients must also have a WRITE LOCK on the table !
2080
*/
2081
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2082
void Session::close_cached_table(Table *table)
1 by brian
clean slate
2083
{
2084
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2085
  wait_while_table_is_used(this, table, HA_EXTRA_FORCE_REOPEN);
1 by brian
clean slate
2086
  /* Close lock if this is not got with LOCK TABLES */
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2087
  if (lock)
1 by brian
clean slate
2088
  {
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2089
    mysql_unlock_tables(this, lock);
2090
    lock= NULL;			// Start locked threads
1 by brian
clean slate
2091
  }
2092
  /* Close all copies of 'table'.  This also frees all LOCK TABLES lock */
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2093
  unlink_open_table(table);
1 by brian
clean slate
2094
2095
  /* When lock on LOCK_open is freed other threads can continue */
2096
  broadcast_refresh();
2097
}
2098
520.1.22 by Brian Aker
Second pass of thd cleanup
2099
static int send_check_errmsg(Session *session, TableList* table,
1 by brian
clean slate
2100
			     const char* operator_name, const char* errmsg)
2101
2102
{
520.1.22 by Brian Aker
Second pass of thd cleanup
2103
  Protocol *protocol= session->protocol;
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2104
  protocol->prepareForResend();
1 by brian
clean slate
2105
  protocol->store(table->alias, system_charset_info);
2106
  protocol->store((char*) operator_name, system_charset_info);
2107
  protocol->store(STRING_WITH_LEN("error"), system_charset_info);
2108
  protocol->store(errmsg, system_charset_info);
520.1.22 by Brian Aker
Second pass of thd cleanup
2109
  session->clear_error();
1 by brian
clean slate
2110
  if (protocol->write())
2111
    return -1;
2112
  return 1;
2113
}
2114
2115
520.1.22 by Brian Aker
Second pass of thd cleanup
2116
static int prepare_for_repair(Session *session, TableList *table_list,
1 by brian
clean slate
2117
			      HA_CHECK_OPT *check_opt)
2118
{
2119
  int error= 0;
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2120
  Table tmp_table, *table;
1000.1.3 by Brian Aker
Renamed TABLE_SHARE to TableShare
2121
  TableShare *share;
1 by brian
clean slate
2122
  char from[FN_REFLEN],tmp[FN_REFLEN+32];
2123
  const char **ext;
15 by brian
Fix for stat, NETWARE removal
2124
  struct stat stat_info;
1 by brian
clean slate
2125
792 by Brian Aker
Refactor for myisam specifics in handler
2126
  if (!(check_opt->use_frm))
1046.1.10 by Brian Aker
Formatting around return (style)
2127
    return 0;
1 by brian
clean slate
2128
2129
  if (!(table= table_list->table))		/* if open_ltable failed */
2130
  {
2131
    char key[MAX_DBKEY_LENGTH];
482 by Brian Aker
Remove uint.
2132
    uint32_t key_length;
1 by brian
clean slate
2133
1054.1.7 by Brian Aker
Refactor TableList methods.
2134
    key_length= table_list->create_table_def_key(key);
1046.1.2 by Brian Aker
Comments on LOCK_open
2135
    pthread_mutex_lock(&LOCK_open); /* Lock table for repair */
520.1.22 by Brian Aker
Second pass of thd cleanup
2136
    if (!(share= (get_table_share(session, table_list, key, key_length, 0,
1 by brian
clean slate
2137
                                  &error))))
2138
    {
2139
      pthread_mutex_unlock(&LOCK_open);
1039.1.10 by Brian Aker
Minor formating, change of one name to make grep easier :)
2140
1046.1.10 by Brian Aker
Formatting around return (style)
2141
      return 0;				// Can't open frm file
1 by brian
clean slate
2142
    }
2143
520.1.22 by Brian Aker
Second pass of thd cleanup
2144
    if (open_table_from_share(session, share, "", 0, 0, 0, &tmp_table, OTM_OPEN))
1 by brian
clean slate
2145
    {
1039.1.17 by Brian Aker
Factored out dead enum.
2146
      release_table_share(share);
1 by brian
clean slate
2147
      pthread_mutex_unlock(&LOCK_open);
1039.1.10 by Brian Aker
Minor formating, change of one name to make grep easier :)
2148
1046.1.10 by Brian Aker
Formatting around return (style)
2149
      return 0;                           // Out of memory
1 by brian
clean slate
2150
    }
2151
    table= &tmp_table;
2152
    pthread_mutex_unlock(&LOCK_open);
2153
  }
2154
2155
  /*
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2156
    REPAIR Table ... USE_FRM for temporary tables makes little sense.
1 by brian
clean slate
2157
  */
2158
  if (table->s->tmp_table)
2159
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
2160
    error= send_check_errmsg(session, table_list, "repair",
1 by brian
clean slate
2161
			     "Cannot repair temporary table from .frm file");
2162
    goto end;
2163
  }
2164
2165
  /*
2166
    User gave us USE_FRM which means that the header in the index file is
2167
    trashed.
2168
    In this case we will try to fix the table the following way:
2169
    - Rename the data file to a temporary name
2170
    - Truncate the table
2171
    - Replace the new data file with the old one
2172
    - Run a normal repair using the new index file and the old data file
2173
  */
2174
2175
  /*
2176
    Check if this is a table type that stores index and data separately,
2177
    like ISAM or MyISAM. We assume fixed order of engine file name
2178
    extentions array. First element of engine file name extentions array
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
2179
    is meta/index file extention. Second element - data file extention.
1 by brian
clean slate
2180
  */
1039.3.1 by Stewart Smith
move bas_ext to StorageEngine instead of handler
2181
  ext= table->file->engine->bas_ext();
1 by brian
clean slate
2182
  if (!ext[0] || !ext[1])
2183
    goto end;					// No data file
2184
2185
  // Name of data file
673.2.1 by Toru Maesaka
First pass of replacing MySQL's strxmov with libc alternatives
2186
  sprintf(from,"%s%s", table->s->normalized_path.str, ext[1]);
15 by brian
Fix for stat, NETWARE removal
2187
  if (stat(from, &stat_info))
1 by brian
clean slate
2188
    goto end;				// Can't use USE_FRM flag
2189
555 by Monty
Fixed 32-bit issues.
2190
  snprintf(tmp, sizeof(tmp), "%s-%lx_%"PRIx64,
606 by Brian Aker
Remove dead lex structure and clean up use of pid_t
2191
           from, (unsigned long)current_pid, session->thread_id);
1 by brian
clean slate
2192
2193
  /* If we could open the table, close it */
2194
  if (table_list->table)
2195
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
2196
    pthread_mutex_lock(&LOCK_open); /* Close for repair table */
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2197
    session->close_cached_table(table);
1 by brian
clean slate
2198
    pthread_mutex_unlock(&LOCK_open);
2199
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
2200
  if (lock_and_wait_for_table_name(session,table_list))
1 by brian
clean slate
2201
  {
2202
    error= -1;
2203
    goto end;
2204
  }
2205
  if (my_rename(from, tmp, MYF(MY_WME)))
2206
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
2207
    pthread_mutex_lock(&LOCK_open); /* Lock during rename of table (aka we go to unlock ) */
1017 by Brian Aker
Drop dead session pass
2208
    unlock_table_name(table_list);
1 by brian
clean slate
2209
    pthread_mutex_unlock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
2210
    error= send_check_errmsg(session, table_list, "repair",
1 by brian
clean slate
2211
			     "Failed renaming data file");
2212
    goto end;
2213
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
2214
  if (mysql_truncate(session, table_list, 1))
1 by brian
clean slate
2215
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
2216
    pthread_mutex_lock(&LOCK_open); /* Lock during truncate of table during repair operation. */
1017 by Brian Aker
Drop dead session pass
2217
    unlock_table_name(table_list);
1 by brian
clean slate
2218
    pthread_mutex_unlock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
2219
    error= send_check_errmsg(session, table_list, "repair",
1 by brian
clean slate
2220
			     "Failed generating table from .frm file");
2221
    goto end;
2222
  }
2223
  if (my_rename(tmp, from, MYF(MY_WME)))
2224
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
2225
    pthread_mutex_lock(&LOCK_open); /* Final repair of table for rename */
1017 by Brian Aker
Drop dead session pass
2226
    unlock_table_name(table_list);
1 by brian
clean slate
2227
    pthread_mutex_unlock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
2228
    error= send_check_errmsg(session, table_list, "repair",
1 by brian
clean slate
2229
			     "Failed restoring .MYD file");
2230
    goto end;
2231
  }
2232
2233
  /*
2234
    Now we should be able to open the partially repaired table
2235
    to finish the repair in the handler later on.
2236
  */
1046.1.2 by Brian Aker
Comments on LOCK_open
2237
  pthread_mutex_lock(&LOCK_open); /* Lock for opening partially repaired table */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2238
  if (session->reopen_name_locked_table(table_list, true))
1 by brian
clean slate
2239
  {
1017 by Brian Aker
Drop dead session pass
2240
    unlock_table_name(table_list);
1 by brian
clean slate
2241
    pthread_mutex_unlock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
2242
    error= send_check_errmsg(session, table_list, "repair",
1 by brian
clean slate
2243
                             "Failed to open partially repaired table");
2244
    goto end;
2245
  }
2246
  pthread_mutex_unlock(&LOCK_open);
2247
2248
end:
2249
  if (table == &tmp_table)
2250
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
2251
    pthread_mutex_lock(&LOCK_open); /* Lock to close table after repair operation */
793 by Brian Aker
Pass through on refactoring functions to clases.
2252
    table->closefrm(true);				// Free allocated memory
1 by brian
clean slate
2253
    pthread_mutex_unlock(&LOCK_open);
2254
  }
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
2255
  return(error);
1 by brian
clean slate
2256
}
2257
2258
2259
2260
/*
2261
  RETURN VALUES
55 by brian
Update for using real bool types.
2262
    false Message sent to net (admin operation went ok)
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
2263
    true  Message should be sent by caller
1 by brian
clean slate
2264
          (admin operation or network communication failed)
2265
*/
520.1.22 by Brian Aker
Second pass of thd cleanup
2266
static bool mysql_admin_table(Session* session, TableList* tables,
1 by brian
clean slate
2267
                              HA_CHECK_OPT* check_opt,
2268
                              const char *operator_name,
2269
                              thr_lock_type lock_type,
2270
                              bool open_for_modify,
2271
                              bool no_warnings_for_error,
482 by Brian Aker
Remove uint.
2272
                              uint32_t extra_open_options,
520.1.21 by Brian Aker
THD -> Session rename
2273
                              int (*prepare_func)(Session *, TableList *,
1 by brian
clean slate
2274
                                                  HA_CHECK_OPT *),
520.1.21 by Brian Aker
THD -> Session rename
2275
                              int (handler::*operator_func)(Session *,
1 by brian
clean slate
2276
                                                            HA_CHECK_OPT *))
2277
{
327.2.4 by Brian Aker
Refactoring table.h
2278
  TableList *table;
846 by Brian Aker
Removing on typedeffed class.
2279
  Select_Lex *select= &session->lex->select_lex;
1 by brian
clean slate
2280
  List<Item> field_list;
2281
  Item *item;
520.1.22 by Brian Aker
Second pass of thd cleanup
2282
  Protocol *protocol= session->protocol;
2283
  LEX *lex= session->lex;
1 by brian
clean slate
2284
  int result_code= 0;
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
2285
  const CHARSET_INFO * const cs= system_charset_info;
1 by brian
clean slate
2286
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
2287
  if (! session->endActiveTransaction())
1046.1.10 by Brian Aker
Formatting around return (style)
2288
    return 1;
1 by brian
clean slate
2289
  field_list.push_back(item = new Item_empty_string("Table",
2290
                                                    NAME_CHAR_LEN * 2,
2291
                                                    cs));
2292
  item->maybe_null = 1;
2293
  field_list.push_back(item = new Item_empty_string("Op", 10, cs));
2294
  item->maybe_null = 1;
2295
  field_list.push_back(item = new Item_empty_string("Msg_type", 10, cs));
2296
  item->maybe_null = 1;
2297
  field_list.push_back(item = new Item_empty_string("Msg_text", 255, cs));
2298
  item->maybe_null = 1;
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2299
  if (protocol->sendFields(&field_list,
2300
                           Protocol::SEND_NUM_ROWS | Protocol::SEND_EOF))
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
2301
    return(true);
1 by brian
clean slate
2302
2303
  for (table= tables; table; table= table->next_local)
2304
  {
2305
    char table_name[NAME_LEN*2+2];
2306
    char* db = table->db;
2307
    bool fatal_error=0;
2308
673.2.1 by Toru Maesaka
First pass of replacing MySQL's strxmov with libc alternatives
2309
    sprintf(table_name,"%s.%s",db,table->table_name);
520.1.22 by Brian Aker
Second pass of thd cleanup
2310
    session->open_options|= extra_open_options;
1 by brian
clean slate
2311
    table->lock_type= lock_type;
2312
    /* open only one table from local list of command */
2313
    {
327.2.4 by Brian Aker
Refactoring table.h
2314
      TableList *save_next_global, *save_next_local;
1 by brian
clean slate
2315
      save_next_global= table->next_global;
2316
      table->next_global= 0;
2317
      save_next_local= table->next_local;
2318
      table->next_local= 0;
481 by Brian Aker
Remove all of uchar.
2319
      select->table_list.first= (unsigned char*)table;
1 by brian
clean slate
2320
      /*
2321
        Time zone tables and SP tables can be add to lex->query_tables list,
2322
        so it have to be prepared.
2323
        TODO: Investigate if we can put extra tables into argument instead of
2324
        using lex->query_tables
2325
      */
2326
      lex->query_tables= table;
2327
      lex->query_tables_last= &table->next_global;
2328
      lex->query_tables_own_last= 0;
520.1.22 by Brian Aker
Second pass of thd cleanup
2329
      session->no_warnings_for_error= no_warnings_for_error;
1 by brian
clean slate
2330
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2331
      session->open_and_lock_tables(table);
520.1.22 by Brian Aker
Second pass of thd cleanup
2332
      session->no_warnings_for_error= 0;
1 by brian
clean slate
2333
      table->next_global= save_next_global;
2334
      table->next_local= save_next_local;
520.1.22 by Brian Aker
Second pass of thd cleanup
2335
      session->open_options&= ~extra_open_options;
1 by brian
clean slate
2336
    }
2337
2338
    if (prepare_func)
2339
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
2340
      switch ((*prepare_func)(session, table, check_opt)) {
1 by brian
clean slate
2341
      case  1:           // error, message written to net
520.1.22 by Brian Aker
Second pass of thd cleanup
2342
        ha_autocommit_or_rollback(session, 1);
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
2343
        session->endTransaction(ROLLBACK);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2344
        session->close_thread_tables();
1 by brian
clean slate
2345
        continue;
2346
      case -1:           // error, message could be written to net
2347
        /* purecov: begin inspected */
2348
        goto err;
2349
        /* purecov: end */
2350
      default:           // should be 0 otherwise
2351
        ;
2352
      }
2353
    }
2354
2355
    /*
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2356
      CHECK Table command is only command where VIEW allowed here and this
1 by brian
clean slate
2357
      command use only temporary teble method for VIEWs resolving => there
2358
      can't be VIEW tree substitition of join view => if opening table
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2359
      succeed then table->table will have real Table pointer as value (in
1 by brian
clean slate
2360
      case of join view substitution table->table can be 0, but here it is
2361
      impossible)
2362
    */
2363
    if (!table->table)
2364
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
2365
      if (!session->warn_list.elements)
2366
        push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_ERROR,
1 by brian
clean slate
2367
                     ER_CHECK_NO_SUCH_TABLE, ER(ER_CHECK_NO_SUCH_TABLE));
1001.1.6 by Andrew Ettinger
Match style guide
2368
      result_code= HA_ADMIN_CORRUPT;
1 by brian
clean slate
2369
      goto send_result;
2370
    }
2371
2372
    if ((table->table->db_stat & HA_READ_ONLY) && open_for_modify)
2373
    {
2374
      /* purecov: begin inspected */
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
2375
      char buff[FN_REFLEN + DRIZZLE_ERRMSG_SIZE];
482 by Brian Aker
Remove uint.
2376
      uint32_t length;
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2377
      protocol->prepareForResend();
1 by brian
clean slate
2378
      protocol->store(table_name, system_charset_info);
2379
      protocol->store(operator_name, system_charset_info);
2380
      protocol->store(STRING_WITH_LEN("error"), system_charset_info);
77.1.18 by Monty Taylor
Removed my_vsnprintf and my_snprintf.
2381
      length= snprintf(buff, sizeof(buff), ER(ER_OPEN_AS_READONLY),
2382
                       table_name);
1 by brian
clean slate
2383
      protocol->store(buff, length, system_charset_info);
520.1.22 by Brian Aker
Second pass of thd cleanup
2384
      ha_autocommit_or_rollback(session, 0);
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
2385
      session->endTransaction(COMMIT);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2386
      session->close_thread_tables();
55 by brian
Update for using real bool types.
2387
      lex->reset_query_tables_list(false);
1 by brian
clean slate
2388
      table->table=0;				// For query cache
2389
      if (protocol->write())
2390
	goto err;
2391
      continue;
2392
      /* purecov: end */
2393
    }
2394
2395
    /* Close all instances of the table to allow repair to rename files */
2396
    if (lock_type == TL_WRITE && table->table->s->version)
2397
    {
1046.1.2 by Brian Aker
Comments on LOCK_open
2398
      pthread_mutex_lock(&LOCK_open); /* Lock type is TL_WRITE and we lock to repair the table */
520.1.22 by Brian Aker
Second pass of thd cleanup
2399
      const char *old_message=session->enter_cond(&COND_refresh, &LOCK_open,
1 by brian
clean slate
2400
					      "Waiting to get writelock");
520.1.22 by Brian Aker
Second pass of thd cleanup
2401
      mysql_lock_abort(session,table->table, true);
2402
      remove_table_from_cache(session, table->table->s->db.str,
1 by brian
clean slate
2403
                              table->table->s->table_name.str,
2404
                              RTFC_WAIT_OTHER_THREAD_FLAG |
2405
                              RTFC_CHECK_KILLED_FLAG);
520.1.22 by Brian Aker
Second pass of thd cleanup
2406
      session->exit_cond(old_message);
2407
      if (session->killed)
1 by brian
clean slate
2408
	goto err;
2409
      open_for_modify= 0;
2410
    }
2411
2412
    if (table->table->s->crashed && operator_func == &handler::ha_check)
2413
    {
2414
      /* purecov: begin inspected */
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2415
      protocol->prepareForResend();
1 by brian
clean slate
2416
      protocol->store(table_name, system_charset_info);
2417
      protocol->store(operator_name, system_charset_info);
2418
      protocol->store(STRING_WITH_LEN("warning"), system_charset_info);
2419
      protocol->store(STRING_WITH_LEN("Table is marked as crashed"),
2420
                      system_charset_info);
2421
      if (protocol->write())
2422
        goto err;
2423
      /* purecov: end */
2424
    }
2425
792 by Brian Aker
Refactor for myisam specifics in handler
2426
    if (operator_func == &handler::ha_repair && !(check_opt->use_frm))
1 by brian
clean slate
2427
    {
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2428
      if ((table->table->file->check_old_types() == HA_ADMIN_NEEDS_ALTER))
1001.1.3 by Andrew Ettinger
Revert bad spacing
2429
      {
520.1.22 by Brian Aker
Second pass of thd cleanup
2430
        ha_autocommit_or_rollback(session, 1);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2431
        session->close_thread_tables();
1001.1.3 by Andrew Ettinger
Revert bad spacing
2432
        result_code= mysql_recreate_table(session, table);
1 by brian
clean slate
2433
        /*
2434
          mysql_recreate_table() can push OK or ERROR.
2435
          Clear 'OK' status. If there is an error, keep it:
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
2436
          we will store the error message in a result set row
1 by brian
clean slate
2437
          and then clear.
2438
        */
520.1.22 by Brian Aker
Second pass of thd cleanup
2439
        if (session->main_da.is_ok())
2440
          session->main_da.reset_diagnostics_area();
1 by brian
clean slate
2441
        goto send_result;
2442
      }
2443
    }
1001.1.3 by Andrew Ettinger
Revert bad spacing
2444
520.1.22 by Brian Aker
Second pass of thd cleanup
2445
    result_code = (table->table->file->*operator_func)(session, check_opt);
1001.1.3 by Andrew Ettinger
Revert bad spacing
2446
1 by brian
clean slate
2447
send_result:
2448
2449
    lex->cleanup_after_one_table_open();
520.1.22 by Brian Aker
Second pass of thd cleanup
2450
    session->clear_error();  // these errors shouldn't get client
1 by brian
clean slate
2451
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
2452
      List_iterator_fast<DRIZZLE_ERROR> it(session->warn_list);
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
2453
      DRIZZLE_ERROR *err;
1 by brian
clean slate
2454
      while ((err= it++))
2455
      {
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2456
        protocol->prepareForResend();
1 by brian
clean slate
2457
        protocol->store(table_name, system_charset_info);
2458
        protocol->store((char*) operator_name, system_charset_info);
2459
        protocol->store(warning_level_names[err->level].str,
2460
                        warning_level_names[err->level].length,
2461
                        system_charset_info);
2462
        protocol->store(err->msg, system_charset_info);
2463
        if (protocol->write())
2464
          goto err;
2465
      }
520.1.22 by Brian Aker
Second pass of thd cleanup
2466
      drizzle_reset_errors(session, true);
1 by brian
clean slate
2467
    }
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2468
    protocol->prepareForResend();
1 by brian
clean slate
2469
    protocol->store(table_name, system_charset_info);
2470
    protocol->store(operator_name, system_charset_info);
2471
2472
send_result_message:
2473
2474
    switch (result_code) {
2475
    case HA_ADMIN_NOT_IMPLEMENTED:
1001.1.3 by Andrew Ettinger
Revert bad spacing
2476
      {
2477
	char buf[ERRMSGSIZE+20];
2478
	uint32_t length=snprintf(buf, ERRMSGSIZE,
2479
                             ER(ER_CHECK_NOT_IMPLEMENTED), operator_name);
2480
	protocol->store(STRING_WITH_LEN("note"), system_charset_info);
2481
	protocol->store(buf, length, system_charset_info);
2482
      }
2483
      break;
1 by brian
clean slate
2484
2485
    case HA_ADMIN_OK:
2486
      protocol->store(STRING_WITH_LEN("status"), system_charset_info);
2487
      protocol->store(STRING_WITH_LEN("OK"), system_charset_info);
2488
      break;
2489
2490
    case HA_ADMIN_FAILED:
2491
      protocol->store(STRING_WITH_LEN("status"), system_charset_info);
2492
      protocol->store(STRING_WITH_LEN("Operation failed"),
2493
                      system_charset_info);
2494
      break;
2495
2496
    case HA_ADMIN_REJECT:
2497
      protocol->store(STRING_WITH_LEN("status"), system_charset_info);
2498
      protocol->store(STRING_WITH_LEN("Operation need committed state"),
2499
                      system_charset_info);
55 by brian
Update for using real bool types.
2500
      open_for_modify= false;
1 by brian
clean slate
2501
      break;
2502
2503
    case HA_ADMIN_ALREADY_DONE:
2504
      protocol->store(STRING_WITH_LEN("status"), system_charset_info);
2505
      protocol->store(STRING_WITH_LEN("Table is already up to date"),
2506
                      system_charset_info);
2507
      break;
2508
2509
    case HA_ADMIN_CORRUPT:
2510
      protocol->store(STRING_WITH_LEN("error"), system_charset_info);
2511
      protocol->store(STRING_WITH_LEN("Corrupt"), system_charset_info);
2512
      fatal_error=1;
2513
      break;
2514
2515
    case HA_ADMIN_INVALID:
2516
      protocol->store(STRING_WITH_LEN("error"), system_charset_info);
2517
      protocol->store(STRING_WITH_LEN("Invalid argument"),
2518
                      system_charset_info);
2519
      break;
2520
2521
    case HA_ADMIN_TRY_ALTER:
2522
    {
2523
      /*
2524
        This is currently used only by InnoDB. ha_innobase::optimize() answers
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2525
        "try with alter", so here we close the table, do an ALTER Table,
1 by brian
clean slate
2526
        reopen the table and do ha_innobase::analyze() on it.
2527
      */
520.1.22 by Brian Aker
Second pass of thd cleanup
2528
      ha_autocommit_or_rollback(session, 0);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2529
      session->close_thread_tables();
327.2.4 by Brian Aker
Refactoring table.h
2530
      TableList *save_next_local= table->next_local,
1 by brian
clean slate
2531
                 *save_next_global= table->next_global;
2532
      table->next_local= table->next_global= 0;
520.1.22 by Brian Aker
Second pass of thd cleanup
2533
      result_code= mysql_recreate_table(session, table);
1 by brian
clean slate
2534
      /*
2535
        mysql_recreate_table() can push OK or ERROR.
2536
        Clear 'OK' status. If there is an error, keep it:
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
2537
        we will store the error message in a result set row
1 by brian
clean slate
2538
        and then clear.
2539
      */
520.1.22 by Brian Aker
Second pass of thd cleanup
2540
      if (session->main_da.is_ok())
2541
        session->main_da.reset_diagnostics_area();
2542
      ha_autocommit_or_rollback(session, 0);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2543
      session->close_thread_tables();
1 by brian
clean slate
2544
      if (!result_code) // recreation went ok
2545
      {
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
2546
        if ((table->table= session->open_ltable(table, lock_type)) &&
520.1.22 by Brian Aker
Second pass of thd cleanup
2547
            ((result_code= table->table->file->ha_analyze(session, check_opt)) > 0))
1 by brian
clean slate
2548
          result_code= 0; // analyze went ok
2549
      }
2550
      if (result_code) // either mysql_recreate_table or analyze failed
2551
      {
520.1.22 by Brian Aker
Second pass of thd cleanup
2552
        assert(session->is_error());
2553
        if (session->is_error())
1 by brian
clean slate
2554
        {
520.1.22 by Brian Aker
Second pass of thd cleanup
2555
          const char *err_msg= session->main_da.message();
971.3.12 by Eric Day
Started abstracting Protocol, removed init_connect, init_file.
2556
          if (!session->protocol->isConnected())
1 by brian
clean slate
2557
          {
755.2.1 by Mark Atwood
replace sql_print_error etc with errmsg_print
2558
            errmsg_printf(ERRMSG_LVL_ERROR, "%s", err_msg);
1 by brian
clean slate
2559
          }
2560
          else
2561
          {
2562
            /* Hijack the row already in-progress. */
2563
            protocol->store(STRING_WITH_LEN("error"), system_charset_info);
2564
            protocol->store(err_msg, system_charset_info);
2565
            (void)protocol->write();
2566
            /* Start off another row for HA_ADMIN_FAILED */
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
2567
            protocol->prepareForResend();
1 by brian
clean slate
2568
            protocol->store(table_name, system_charset_info);
2569
            protocol->store(operator_name, system_charset_info);
2570
          }
520.1.22 by Brian Aker
Second pass of thd cleanup
2571
          session->clear_error();
1 by brian
clean slate
2572
        }
2573
      }
2574
      result_code= result_code ? HA_ADMIN_FAILED : HA_ADMIN_OK;
2575
      table->next_local= save_next_local;
2576
      table->next_global= save_next_global;
2577
      goto send_result_message;
2578
    }
2579
    case HA_ADMIN_NEEDS_UPGRADE:
2580
    case HA_ADMIN_NEEDS_ALTER:
2581
    {
2582
      char buf[ERRMSGSIZE];
482 by Brian Aker
Remove uint.
2583
      uint32_t length;
1 by brian
clean slate
2584
2585
      protocol->store(STRING_WITH_LEN("error"), system_charset_info);
77.1.18 by Monty Taylor
Removed my_vsnprintf and my_snprintf.
2586
      length=snprintf(buf, ERRMSGSIZE, ER(ER_TABLE_NEEDS_UPGRADE), table->table_name);
1 by brian
clean slate
2587
      protocol->store(buf, length, system_charset_info);
2588
      fatal_error=1;
2589
      break;
2590
    }
2591
2592
    default:				// Probably HA_ADMIN_INTERNAL_ERROR
2593
      {
2594
        char buf[ERRMSGSIZE+20];
482 by Brian Aker
Remove uint.
2595
        uint32_t length=snprintf(buf, ERRMSGSIZE,
338 by Monty Taylor
Tagged more strings.
2596
                             _("Unknown - internal error %d during operation"),
77.1.18 by Monty Taylor
Removed my_vsnprintf and my_snprintf.
2597
                             result_code);
1 by brian
clean slate
2598
        protocol->store(STRING_WITH_LEN("error"), system_charset_info);
2599
        protocol->store(buf, length, system_charset_info);
2600
        fatal_error=1;
2601
        break;
2602
      }
2603
    }
2604
    if (table->table)
2605
    {
2606
      if (fatal_error)
2607
        table->table->s->version=0;               // Force close of table
2608
      else if (open_for_modify)
2609
      {
2610
        if (table->table->s->tmp_table)
2611
          table->table->file->info(HA_STATUS_CONST);
2612
        else
2613
        {
2614
          pthread_mutex_lock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
2615
          remove_table_from_cache(session, table->table->s->db.str,
1 by brian
clean slate
2616
                                  table->table->s->table_name.str, RTFC_NO_FLAG);
2617
          pthread_mutex_unlock(&LOCK_open);
2618
        }
2619
      }
2620
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
2621
    ha_autocommit_or_rollback(session, 0);
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
2622
    session->endTransaction(COMMIT);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2623
    session->close_thread_tables();
1 by brian
clean slate
2624
    table->table=0;				// For query cache
2625
    if (protocol->write())
2626
      goto err;
2627
  }
2628
836 by Brian Aker
Fixed session call from function to method.
2629
  session->my_eof();
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
2630
  return(false);
1 by brian
clean slate
2631
2632
err:
520.1.22 by Brian Aker
Second pass of thd cleanup
2633
  ha_autocommit_or_rollback(session, 1);
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
2634
  session->endTransaction(ROLLBACK);
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
2635
  session->close_thread_tables();			// Shouldn't be needed
1 by brian
clean slate
2636
  if (table)
2637
    table->table=0;
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
2638
  return(true);
1 by brian
clean slate
2639
}
2640
2641
520.1.22 by Brian Aker
Second pass of thd cleanup
2642
bool mysql_repair_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
1 by brian
clean slate
2643
{
520.1.22 by Brian Aker
Second pass of thd cleanup
2644
  return(mysql_admin_table(session, tables, check_opt,
792 by Brian Aker
Refactor for myisam specifics in handler
2645
                           "repair", TL_WRITE, 1,
2646
                           check_opt->use_frm,
2647
                           HA_OPEN_FOR_REPAIR,
2648
                           &prepare_for_repair,
2649
                           &handler::ha_repair));
1 by brian
clean slate
2650
}
2651
2652
520.1.22 by Brian Aker
Second pass of thd cleanup
2653
bool mysql_optimize_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
1 by brian
clean slate
2654
{
520.1.22 by Brian Aker
Second pass of thd cleanup
2655
  return(mysql_admin_table(session, tables, check_opt,
792 by Brian Aker
Refactor for myisam specifics in handler
2656
                           "optimize", TL_WRITE, 1,0,0,0,
2657
                           &handler::ha_optimize));
1 by brian
clean slate
2658
}
2659
2660
2661
/*
2662
  Assigned specified indexes for a table into key cache
2663
2664
  SYNOPSIS
2665
    mysql_assign_to_keycache()
520.1.22 by Brian Aker
Second pass of thd cleanup
2666
    session		Thread object
1 by brian
clean slate
2667
    tables	Table list (one table only)
2668
2669
  RETURN VALUES
55 by brian
Update for using real bool types.
2670
   false ok
2671
   true  error
1 by brian
clean slate
2672
*/
2673
520.1.22 by Brian Aker
Second pass of thd cleanup
2674
bool mysql_assign_to_keycache(Session* session, TableList* tables,
1 by brian
clean slate
2675
			     LEX_STRING *key_cache_name)
2676
{
2677
  HA_CHECK_OPT check_opt;
2678
  KEY_CACHE *key_cache;
2679
2680
  check_opt.init();
2681
  pthread_mutex_lock(&LOCK_global_system_variables);
2682
  if (!(key_cache= get_key_cache(key_cache_name)))
2683
  {
2684
    pthread_mutex_unlock(&LOCK_global_system_variables);
2685
    my_error(ER_UNKNOWN_KEY_CACHE, MYF(0), key_cache_name->str);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
2686
    return(true);
1 by brian
clean slate
2687
  }
2688
  pthread_mutex_unlock(&LOCK_global_system_variables);
2689
  check_opt.key_cache= key_cache;
520.1.22 by Brian Aker
Second pass of thd cleanup
2690
  return(mysql_admin_table(session, tables, &check_opt,
1 by brian
clean slate
2691
				"assign_to_keycache", TL_READ_NO_INSERT, 0, 0,
2692
				0, 0, &handler::assign_to_keycache));
2693
}
2694
2695
2696
/*
2697
  Reassign all tables assigned to a key cache to another key cache
2698
2699
  SYNOPSIS
2700
    reassign_keycache_tables()
520.1.22 by Brian Aker
Second pass of thd cleanup
2701
    session		Thread object
1 by brian
clean slate
2702
    src_cache	Reference to the key cache to clean up
2703
    dest_cache	New key cache
2704
2705
  NOTES
2706
    This is called when one sets a key cache size to zero, in which
2707
    case we have to move the tables associated to this key cache to
2708
    the "default" one.
2709
2710
    One has to ensure that one never calls this function while
2711
    some other thread is changing the key cache. This is assured by
2712
    the caller setting src_cache->in_init before calling this function.
2713
2714
    We don't delete the old key cache as there may still be pointers pointing
2715
    to it for a while after this function returns.
2716
2717
 RETURN VALUES
2718
    0	  ok
2719
*/
2720
779.1.27 by Monty Taylor
Got rid of __attribute__((unused)) and the like from the .cc files.
2721
int reassign_keycache_tables(Session *,
77.1.46 by Monty Taylor
Finished the warnings work!
2722
                             KEY_CACHE *src_cache,
2723
                             KEY_CACHE *dst_cache)
1 by brian
clean slate
2724
{
51.2.1 by Patrick Galbraith
Removed DBUG_PRINTs, DBUG_ASSERTs, DBUG_EXECUTE_IFs from
2725
  assert(src_cache != dst_cache);
2726
  assert(src_cache->in_init);
1 by brian
clean slate
2727
  src_cache->param_buff_size= 0;		// Free key cache
2728
  ha_resize_key_cache(src_cache);
2729
  ha_change_key_cache(src_cache, dst_cache);
1046.1.10 by Brian Aker
Formatting around return (style)
2730
  return 0;
1 by brian
clean slate
2731
}
2732
2733
/**
2734
  @brief          Create frm file based on I_S table
2735
520.1.22 by Brian Aker
Second pass of thd cleanup
2736
  @param[in]      session                      thread handler
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
2737
  @param[in]      schema_table             I_S table
1 by brian
clean slate
2738
  @param[in]      dst_path                 path where frm should be created
2739
  @param[in]      create_info              Create info
2740
2741
  @return         Operation status
2742
    @retval       0                        success
2743
    @retval       1                        error
2744
*/
520.1.22 by Brian Aker
Second pass of thd cleanup
2745
bool mysql_create_like_schema_frm(Session* session, TableList* schema_table,
1 by brian
clean slate
2746
                                  char *dst_path, HA_CREATE_INFO *create_info)
2747
{
2748
  HA_CREATE_INFO local_create_info;
2749
  Alter_info alter_info;
2750
  bool tmp_table= (create_info->options & HA_LEX_CREATE_TMP_TABLE);
482 by Brian Aker
Remove uint.
2751
  uint32_t keys= schema_table->table->s->keys;
2752
  uint32_t db_options= 0;
1 by brian
clean slate
2753
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
2754
  memset(&local_create_info, 0, sizeof(local_create_info));
1 by brian
clean slate
2755
  local_create_info.db_type= schema_table->table->s->db_type();
2756
  local_create_info.row_type= schema_table->table->s->row_type;
2757
  local_create_info.default_table_charset=default_charset_info;
2758
  alter_info.flags= (ALTER_CHANGE_COLUMN | ALTER_RECREATE);
2759
  schema_table->table->use_all_columns();
520.1.22 by Brian Aker
Second pass of thd cleanup
2760
  if (mysql_prepare_alter_table(session, schema_table->table,
1 by brian
clean slate
2761
                                &local_create_info, &alter_info))
757 by Brian Aker
Fix for multple execution path for FRM creation.
2762
    return true;
2763
520.1.22 by Brian Aker
Second pass of thd cleanup
2764
  if (mysql_prepare_create_table(session, &local_create_info, &alter_info,
1 by brian
clean slate
2765
                                 tmp_table, &db_options,
2766
                                 schema_table->table->file,
2767
                                 &schema_table->table->s->key_info, &keys, 0))
757 by Brian Aker
Fix for multple execution path for FRM creation.
2768
    return true;
2769
1 by brian
clean slate
2770
  local_create_info.max_rows= 0;
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
2771
  drizzled::message::Table table_proto;
2772
  table_proto.set_name("system_stupid_i_s_fix_nonsense");
1008.3.24 by Stewart Smith
fix setting table type in proto for CREATE LIKE schema table
2773
  if(tmp_table)
2774
    table_proto.set_type(drizzled::message::Table::TEMPORARY);
2775
  else
2776
    table_proto.set_type(drizzled::message::Table::STANDARD);
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
2777
757 by Brian Aker
Fix for multple execution path for FRM creation.
2778
  if (rea_create_table(session, dst_path, "system_tmp", "system_stupid_i_s_fix_nonsense",
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
2779
		       &table_proto,
1 by brian
clean slate
2780
                       &local_create_info, alter_info.create_list,
2781
                       keys, schema_table->table->s->key_info,
1008.3.23 by Stewart Smith
remove unused (originally only in ha_partition, but should really be part of ha_create there too) ha_create_handler_files
2782
		       true))
757 by Brian Aker
Fix for multple execution path for FRM creation.
2783
    return true;
2784
2785
  return false;
1 by brian
clean slate
2786
}
2787
2788
/*
2789
  Create a table identical to the specified table
2790
2791
  SYNOPSIS
2792
    mysql_create_like_table()
520.1.22 by Brian Aker
Second pass of thd cleanup
2793
    session		Thread object
1 by brian
clean slate
2794
    table       Table list element for target table
2795
    src_table   Table list element for source table
2796
    create_info Create info
2797
2798
  RETURN VALUES
55 by brian
Update for using real bool types.
2799
    false OK
2800
    true  error
1 by brian
clean slate
2801
*/
2802
520.1.22 by Brian Aker
Second pass of thd cleanup
2803
bool mysql_create_like_table(Session* session, TableList* table, TableList* src_table,
1 by brian
clean slate
2804
                             HA_CREATE_INFO *create_info)
2805
{
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
2806
  Table *name_lock= 0;
1 by brian
clean slate
2807
  char src_path[FN_REFLEN], dst_path[FN_REFLEN];
482 by Brian Aker
Remove uint.
2808
  uint32_t dst_path_length;
1 by brian
clean slate
2809
  char *db= table->db;
2810
  char *table_name= table->table_name;
2811
  int  err;
55 by brian
Update for using real bool types.
2812
  bool res= true;
482 by Brian Aker
Remove uint.
2813
  uint32_t not_used;
1 by brian
clean slate
2814
2815
  /*
2816
    By opening source table we guarantee that it exists and no concurrent
2817
    DDL operation will mess with it. Later we also take an exclusive
2818
    name-lock on target table name, which makes copying of .frm file,
2819
    call to ha_create_table() and binlogging atomic against concurrent DML
2820
    and DDL operations on target table. Thus by holding both these "locks"
2821
    we ensure that our statement is properly isolated from all concurrent
2822
    operations which matter.
2823
  */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2824
  if (session->open_tables_from_list(&src_table, &not_used, 0))
2825
    return true;
1 by brian
clean slate
2826
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
2827
  strncpy(src_path, src_table->table->s->path.str, sizeof(src_path));
1 by brian
clean slate
2828
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
2829
  /*
1 by brian
clean slate
2830
    Check that destination tables does not exist. Note that its name
2831
    was already checked when it was added to the table list.
2832
  */
2833
  if (create_info->options & HA_LEX_CREATE_TMP_TABLE)
2834
  {
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2835
    if (session->find_temporary_table(db, table_name))
1 by brian
clean slate
2836
      goto table_exists;
520.1.22 by Brian Aker
Second pass of thd cleanup
2837
    dst_path_length= build_tmptable_filename(session, dst_path, sizeof(dst_path));
1 by brian
clean slate
2838
    create_info->table_options|= HA_CREATE_DELAY_KEY_WRITE;
2839
  }
2840
  else
2841
  {
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2842
    if (session->lock_table_name_if_not_cached(db, table_name, &name_lock))
1 by brian
clean slate
2843
      goto err;
2844
    if (!name_lock)
2845
      goto table_exists;
2846
    dst_path_length= build_table_filename(dst_path, sizeof(dst_path),
1039.1.6 by Brian Aker
Refactor for build_table_filename()
2847
                                          db, table_name, false);
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
2848
    if (table_proto_exists(dst_path)==EEXIST)
1 by brian
clean slate
2849
      goto table_exists;
2850
  }
2851
2852
  /*
2853
    Create a new table by copying from source table
2854
2855
    Altough exclusive name-lock on target table protects us from concurrent
2856
    DML and DDL operations on it we still want to wrap .FRM creation and call
2857
    to ha_create_table() in critical section protected by LOCK_open in order
2858
    to provide minimal atomicity against operations which disregard name-locks,
2859
    like I_S implementation, for example. This is a temporary and should not
2860
    be copied. Instead we should fix our code to always honor name-locks.
2861
2862
    Also some engines (e.g. NDB cluster) require that LOCK_open should be held
2863
    during the call to ha_create_table(). See bug #28614 for more info.
2864
  */
1046.1.2 by Brian Aker
Comments on LOCK_open
2865
  pthread_mutex_lock(&LOCK_open); /* We lock for CREATE TABLE LIKE to copy table definition */
1 by brian
clean slate
2866
  if (src_table->schema_table)
2867
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
2868
    if (mysql_create_like_schema_frm(session, src_table, dst_path, create_info))
1 by brian
clean slate
2869
    {
398.1.10 by Monty Taylor
Actually removed VOID() this time.
2870
      pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
2871
      goto err;
2872
    }
2873
  }
820.1.6 by Stewart Smith
fix (in a bit of a hackish) way CREATE TABLE LIKE for proto file.
2874
  else
1 by brian
clean slate
2875
  {
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
2876
    int dfecopyr= copy_table_proto_file(src_path, dst_path);
2877
2878
    if(dfecopyr)
820.1.6 by Stewart Smith
fix (in a bit of a hackish) way CREATE TABLE LIKE for proto file.
2879
    {
2880
      if (my_errno == ENOENT)
2881
	my_error(ER_BAD_DB_ERROR,MYF(0),db);
2882
      else
2883
	my_error(ER_CANT_CREATE_FILE,MYF(0),dst_path,my_errno);
2884
      pthread_mutex_unlock(&LOCK_open);
2885
      goto err;
2886
    }
1 by brian
clean slate
2887
  }
2888
2889
  /*
2890
    As mysql_truncate don't work on a new table at this stage of
2891
    creation, instead create the table directly (for both normal
2892
    and temporary tables).
2893
  */
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
2894
520.1.22 by Brian Aker
Second pass of thd cleanup
2895
  if (session->variables.keep_files_on_create)
1 by brian
clean slate
2896
    create_info->options|= HA_CREATE_KEEP_FILES;
520.1.22 by Brian Aker
Second pass of thd cleanup
2897
  err= ha_create_table(session, dst_path, db, table_name, create_info, 1);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
2898
  pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
2899
2900
  if (create_info->options & HA_LEX_CREATE_TMP_TABLE)
2901
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
2902
    if (err || !open_temporary_table(session, dst_path, db, table_name, 1,
1 by brian
clean slate
2903
                                     OTM_OPEN))
2904
    {
2905
      (void) rm_temporary_table(create_info->db_type,
590.1.3 by Stewart Smith
remove frm_only create_info option
2906
				dst_path);
1 by brian
clean slate
2907
      goto err;     /* purecov: inspected */
2908
    }
2909
  }
2910
  else if (err)
2911
  {
2912
    (void) quick_rm_table(create_info->db_type, db,
1039.1.8 by Brian Aker
Style cleanup
2913
			  table_name, false); /* purecov: inspected */
1 by brian
clean slate
2914
    goto err;	    /* purecov: inspected */
2915
  }
2916
2917
  /*
2918
    We have to write the query before we unlock the tables.
2919
  */
2920
  {
2921
    /*
2922
       Since temporary tables are not replicated under row-based
2923
       replication, CREATE TABLE ... LIKE ... needs special
2924
       treatement.  We have four cases to consider, according to the
2925
       following decision table:
2926
2927
           ==== ========= ========= ==============================
2928
           Case    Target    Source Write to binary log
2929
           ==== ========= ========= ==============================
2930
           1       normal    normal Original statement
2931
           2       normal temporary Generated statement
2932
           3    temporary    normal Nothing
2933
           4    temporary temporary Nothing
2934
           ==== ========= ========= ==============================
2935
    */
2936
    if (!(create_info->options & HA_LEX_CREATE_TMP_TABLE))
2937
    {
2938
      if (src_table->table->s->tmp_table)               // Case 2
2939
      {
2940
        char buf[2048];
2941
        String query(buf, sizeof(buf), system_charset_info);
2942
        query.length(0);  // Have to zero it since constructor doesn't
2943
2944
2945
        /*
2946
          Here we open the destination table, on which we already have
2947
          name-lock. This is needed for store_create_info() to work.
2948
          The table will be closed by unlink_open_table() at the end
2949
          of this function.
2950
        */
2951
        table->table= name_lock;
1046.1.2 by Brian Aker
Comments on LOCK_open
2952
        pthread_mutex_lock(&LOCK_open); /* Open new table we have just acquired */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2953
        if (session->reopen_name_locked_table(table, false))
1 by brian
clean slate
2954
        {
398.1.10 by Monty Taylor
Actually removed VOID() this time.
2955
          pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
2956
          goto err;
2957
        }
398.1.10 by Monty Taylor
Actually removed VOID() this time.
2958
        pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
2959
1014.3.4 by Brian Aker
Remove dead session calls.
2960
        int result= store_create_info(table, &query, create_info);
1 by brian
clean slate
2961
51.2.1 by Patrick Galbraith
Removed DBUG_PRINTs, DBUG_ASSERTs, DBUG_EXECUTE_IFs from
2962
        assert(result == 0); // store_create_info() always return 0
520.1.22 by Brian Aker
Second pass of thd cleanup
2963
        write_bin_log(session, true, query.ptr(), query.length());
1 by brian
clean slate
2964
      }
2965
      else                                      // Case 1
520.1.22 by Brian Aker
Second pass of thd cleanup
2966
        write_bin_log(session, true, session->query, session->query_length);
1 by brian
clean slate
2967
    }
2968
  }
2969
55 by brian
Update for using real bool types.
2970
  res= false;
1 by brian
clean slate
2971
  goto err;
2972
2973
table_exists:
2974
  if (create_info->options & HA_LEX_CREATE_IF_NOT_EXISTS)
2975
  {
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
2976
    char warn_buff[DRIZZLE_ERRMSG_SIZE];
77.1.18 by Monty Taylor
Removed my_vsnprintf and my_snprintf.
2977
    snprintf(warn_buff, sizeof(warn_buff),
2978
             ER(ER_TABLE_EXISTS_ERROR), table_name);
520.1.22 by Brian Aker
Second pass of thd cleanup
2979
    push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
2980
		 ER_TABLE_EXISTS_ERROR,warn_buff);
55 by brian
Update for using real bool types.
2981
    res= false;
1 by brian
clean slate
2982
  }
2983
  else
2984
    my_error(ER_TABLE_EXISTS_ERROR, MYF(0), table_name);
2985
2986
err:
2987
  if (name_lock)
2988
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
2989
    pthread_mutex_lock(&LOCK_open); /* unlink open tables for create table like*/
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
2990
    session->unlink_open_table(name_lock);
1 by brian
clean slate
2991
    pthread_mutex_unlock(&LOCK_open);
2992
  }
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
2993
  return(res);
1 by brian
clean slate
2994
}
2995
2996
520.1.22 by Brian Aker
Second pass of thd cleanup
2997
bool mysql_analyze_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
1 by brian
clean slate
2998
{
2999
  thr_lock_type lock_type = TL_READ_NO_INSERT;
3000
520.1.22 by Brian Aker
Second pass of thd cleanup
3001
  return(mysql_admin_table(session, tables, check_opt,
1 by brian
clean slate
3002
				"analyze", lock_type, 1, 0, 0, 0,
3003
				&handler::ha_analyze));
3004
}
3005
3006
520.1.22 by Brian Aker
Second pass of thd cleanup
3007
bool mysql_check_table(Session* session, TableList* tables,HA_CHECK_OPT* check_opt)
1 by brian
clean slate
3008
{
3009
  thr_lock_type lock_type = TL_READ_NO_INSERT;
3010
520.1.22 by Brian Aker
Second pass of thd cleanup
3011
  return(mysql_admin_table(session, tables, check_opt,
1 by brian
clean slate
3012
				"check", lock_type,
3013
				0, 0, HA_OPEN_FOR_REPAIR, 0,
3014
				&handler::ha_check));
3015
}
3016
3017
3018
/* table_list should contain just one table */
3019
static int
520.1.22 by Brian Aker
Second pass of thd cleanup
3020
mysql_discard_or_import_tablespace(Session *session,
327.2.4 by Brian Aker
Refactoring table.h
3021
                                   TableList *table_list,
1 by brian
clean slate
3022
                                   enum tablespace_op_type tablespace_op)
3023
{
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3024
  Table *table;
199 by Brian Aker
my_bool...
3025
  bool discard;
1 by brian
clean slate
3026
  int error;
3027
3028
  /*
3029
    Note that DISCARD/IMPORT TABLESPACE always is the only operation in an
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3030
    ALTER Table
1 by brian
clean slate
3031
  */
3032
520.1.22 by Brian Aker
Second pass of thd cleanup
3033
  session->set_proc_info("discard_or_import_tablespace");
1 by brian
clean slate
3034
3035
  discard= test(tablespace_op == DISCARD_TABLESPACE);
3036
3037
 /*
3038
   We set this flag so that ha_innobase::open and ::external_lock() do
3039
   not complain when we lock the table
3040
 */
520.1.22 by Brian Aker
Second pass of thd cleanup
3041
  session->tablespace_op= true;
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
3042
  if (!(table= session->open_ltable(table_list, TL_WRITE)))
1 by brian
clean slate
3043
  {
1046.1.9 by Brian Aker
Remove caller that wasn't correctly locking, and reverted code to 5.1
3044
    session->tablespace_op= false;
1046.1.10 by Brian Aker
Formatting around return (style)
3045
    return -1;
1 by brian
clean slate
3046
  }
3047
3048
  error= table->file->ha_discard_or_import_tablespace(discard);
3049
520.1.22 by Brian Aker
Second pass of thd cleanup
3050
  session->set_proc_info("end");
1 by brian
clean slate
3051
3052
  if (error)
3053
    goto err;
3054
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3055
  /* The ALTER Table is always in its own transaction */
520.1.22 by Brian Aker
Second pass of thd cleanup
3056
  error = ha_autocommit_or_rollback(session, 0);
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
3057
  if (! session->endActiveTransaction())
1 by brian
clean slate
3058
    error=1;
3059
  if (error)
3060
    goto err;
520.1.22 by Brian Aker
Second pass of thd cleanup
3061
  write_bin_log(session, false, session->query, session->query_length);
1 by brian
clean slate
3062
3063
err:
520.1.22 by Brian Aker
Second pass of thd cleanup
3064
  ha_autocommit_or_rollback(session, error);
3065
  session->tablespace_op=false;
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
3066
1 by brian
clean slate
3067
  if (error == 0)
3068
  {
836 by Brian Aker
Fixed session call from function to method.
3069
    session->my_ok();
1046.1.10 by Brian Aker
Formatting around return (style)
3070
    return 0;
1 by brian
clean slate
3071
  }
3072
3073
  table->file->print_error(error, MYF(0));
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
3074
1046.1.10 by Brian Aker
Formatting around return (style)
3075
  return -1;
1 by brian
clean slate
3076
}
3077
3078
/**
3079
  Copy all changes detected by parser to the HA_ALTER_FLAGS
3080
*/
3081
3082
void setup_ha_alter_flags(Alter_info *alter_info, HA_ALTER_FLAGS *alter_flags)
3083
{
482 by Brian Aker
Remove uint.
3084
  uint32_t flags= alter_info->flags;
1 by brian
clean slate
3085
3086
  if (ALTER_ADD_COLUMN & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3087
    alter_flags->set(HA_ADD_COLUMN);
1 by brian
clean slate
3088
  if (ALTER_DROP_COLUMN & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3089
    alter_flags->set(HA_DROP_COLUMN);
1 by brian
clean slate
3090
  if (ALTER_RENAME & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3091
    alter_flags->set(HA_RENAME_TABLE);
1 by brian
clean slate
3092
  if (ALTER_CHANGE_COLUMN & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3093
    alter_flags->set(HA_CHANGE_COLUMN);
1 by brian
clean slate
3094
  if (ALTER_COLUMN_DEFAULT & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3095
    alter_flags->set(HA_COLUMN_DEFAULT_VALUE);
1 by brian
clean slate
3096
  if (ALTER_COLUMN_STORAGE & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3097
    alter_flags->set(HA_COLUMN_STORAGE);
1 by brian
clean slate
3098
  if (ALTER_COLUMN_FORMAT & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3099
    alter_flags->set(HA_COLUMN_FORMAT);
1 by brian
clean slate
3100
  if (ALTER_COLUMN_ORDER & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3101
    alter_flags->set(HA_ALTER_COLUMN_ORDER);
1 by brian
clean slate
3102
  if (ALTER_STORAGE & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3103
    alter_flags->set(HA_ALTER_STORAGE);
1 by brian
clean slate
3104
  if (ALTER_ROW_FORMAT & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3105
    alter_flags->set(HA_ALTER_ROW_FORMAT);
1 by brian
clean slate
3106
  if (ALTER_RECREATE & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3107
    alter_flags->set(HA_RECREATE);
1 by brian
clean slate
3108
  if (ALTER_FOREIGN_KEY & flags)
1005.2.6 by Monty Taylor
Re-added bitset<> as a replacement for Bitmap<>
3109
    alter_flags->set(HA_ALTER_FOREIGN_KEY);
1 by brian
clean slate
3110
}
3111
3112
3113
/*
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3114
  Manages enabling/disabling of indexes for ALTER Table
1 by brian
clean slate
3115
3116
  SYNOPSIS
3117
    alter_table_manage_keys()
3118
      table                  Target table
3119
      indexes_were_disabled  Whether the indexes of the from table
3120
                             were disabled
3121
      keys_onoff             ENABLE | DISABLE | LEAVE_AS_IS
3122
3123
  RETURN VALUES
55 by brian
Update for using real bool types.
3124
    false  OK
3125
    true   Error
1 by brian
clean slate
3126
*/
3127
3128
static
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3129
bool alter_table_manage_keys(Table *table, int indexes_were_disabled,
1 by brian
clean slate
3130
                             enum enum_enable_or_disable keys_onoff)
3131
{
3132
  int error= 0;
3133
  switch (keys_onoff) {
3134
  case ENABLE:
3135
    error= table->file->ha_enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
3136
    break;
3137
  case LEAVE_AS_IS:
3138
    if (!indexes_were_disabled)
3139
      break;
3140
    /* fall-through: disabled indexes */
3141
  case DISABLE:
3142
    error= table->file->ha_disable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
3143
  }
3144
3145
  if (error == HA_ERR_WRONG_COMMAND)
3146
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
3147
    push_warning_printf(current_session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
3148
                        ER_ILLEGAL_HA, ER(ER_ILLEGAL_HA),
3149
                        table->s->table_name.str);
3150
    error= 0;
3151
  } else if (error)
3152
    table->file->print_error(error, MYF(0));
3153
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
3154
  return(error);
1 by brian
clean slate
3155
}
3156
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3157
static int 
3158
create_temporary_table(Session *session,
3159
                       Table *table,
3160
                       char *new_db,
3161
                       char *tmp_name,
3162
                       HA_CREATE_INFO *create_info,
3163
                       Alter_info *alter_info,
3164
                       bool db_changed)
1 by brian
clean slate
3165
{
3166
  int error;
3167
  char index_file[FN_REFLEN], data_file[FN_REFLEN];
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
3168
  StorageEngine *old_db_type, *new_db_type;
1 by brian
clean slate
3169
  old_db_type= table->s->db_type();
3170
  new_db_type= create_info->db_type;
3171
  /*
3172
    Handling of symlinked tables:
3173
    If no rename:
3174
      Create new data file and index file on the same disk as the
3175
      old data and index files.
3176
      Copy data.
3177
      Rename new data file over old data file and new index file over
3178
      old index file.
3179
      Symlinks are not changed.
3180
3181
   If rename:
3182
      Create new data file and index file on the same disk as the
3183
      old data and index files.  Create also symlinks to point at
3184
      the new tables.
3185
      Copy data.
3186
      At end, rename intermediate tables, and symlinks to intermediate
3187
      table, to final table name.
3188
      Remove old table and old symlinks
3189
3190
    If rename is made to another database:
3191
      Create new tables in new database.
3192
      Copy data.
3193
      Remove old table and symlinks.
3194
  */
3195
  if (db_changed)		// Ignore symlink if db changed
3196
  {
3197
    if (create_info->index_file_name)
3198
    {
3199
      /* Fix index_file_name to have 'tmp_name' as basename */
641.4.1 by Toru Maesaka
First pass of replacing MySQL's my_stpcpy() with appropriate libc calls
3200
      strcpy(index_file, tmp_name);
1 by brian
clean slate
3201
      create_info->index_file_name=fn_same(index_file,
3202
                                           create_info->index_file_name,
3203
                                           1);
3204
    }
3205
    if (create_info->data_file_name)
3206
    {
3207
      /* Fix data_file_name to have 'tmp_name' as basename */
641.4.1 by Toru Maesaka
First pass of replacing MySQL's my_stpcpy() with appropriate libc calls
3208
      strcpy(data_file, tmp_name);
1 by brian
clean slate
3209
      create_info->data_file_name=fn_same(data_file,
3210
                                          create_info->data_file_name,
3211
                                          1);
3212
    }
3213
  }
3214
  else
3215
    create_info->data_file_name=create_info->index_file_name=0;
3216
3217
  /*
3218
    Create a table with a temporary name.
3219
    We don't log the statement, it will be logged later.
3220
  */
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
3221
  drizzled::message::Table table_proto;
3222
  table_proto.set_name(tmp_name);
1008.3.21 by Stewart Smith
add table type to table proto generated in parser
3223
  table_proto.set_type(drizzled::message::Table::TEMPORARY);
520.1.22 by Brian Aker
Second pass of thd cleanup
3224
  error= mysql_create_table(session, new_db, tmp_name,
1008.3.20 by Stewart Smith
Put table_name into table proto in parser, and use that when creating table proto file.
3225
                            create_info, &table_proto, alter_info, 1, 0);
1 by brian
clean slate
3226
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
3227
  return(error);
1 by brian
clean slate
3228
}
3229
3230
3231
/**
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3232
  Prepare column and key definitions for CREATE TABLE in ALTER Table.
1 by brian
clean slate
3233
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3234
  This function transforms parse output of ALTER Table - lists of
1 by brian
clean slate
3235
  columns and keys to add, drop or modify into, essentially,
3236
  CREATE TABLE definition - a list of columns and keys of the new
3237
  table. While doing so, it also performs some (bug not all)
3238
  semantic checks.
3239
3240
  This function is invoked when we know that we're going to
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3241
  perform ALTER Table via a temporary table -- i.e. fast ALTER Table
1 by brian
clean slate
3242
  is not possible, perhaps because the ALTER statement contains
3243
  instructions that require change in table data, not only in
3244
  table definition or indexes.
3245
520.1.22 by Brian Aker
Second pass of thd cleanup
3246
  @param[in,out]  session         thread handle. Used as a memory pool
1 by brian
clean slate
3247
                              and source of environment information.
3248
  @param[in]      table       the source table, open and locked
3249
                              Used as an interface to the storage engine
3250
                              to acquire additional information about
3251
                              the original table.
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3252
  @param[in,out]  create_info A blob with CREATE/ALTER Table
1 by brian
clean slate
3253
                              parameters
3254
  @param[in,out]  alter_info  Another blob with ALTER/CREATE parameters.
3255
                              Originally create_info was used only in
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3256
                              CREATE TABLE and alter_info only in ALTER Table.
1 by brian
clean slate
3257
                              But since ALTER might end-up doing CREATE,
3258
                              this distinction is gone and we just carry
3259
                              around two structures.
3260
3261
  @return
3262
    Fills various create_info members based on information retrieved
3263
    from the storage engine.
3264
    Sets create_info->varchar if the table has a VARCHAR column.
3265
    Prepares alter_info->create_list and alter_info->key_list with
3266
    columns and keys of the new table.
55 by brian
Update for using real bool types.
3267
  @retval true   error, out of memory or a semantical error in ALTER
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3268
                 Table instructions
55 by brian
Update for using real bool types.
3269
  @retval false  success
1 by brian
clean slate
3270
*/
3271
3272
static bool
520.1.22 by Brian Aker
Second pass of thd cleanup
3273
mysql_prepare_alter_table(Session *session, Table *table,
1 by brian
clean slate
3274
                          HA_CREATE_INFO *create_info,
3275
                          Alter_info *alter_info)
3276
{
3277
  /* New column definitions are added here */
3278
  List<Create_field> new_create_list;
3279
  /* New key definitions are added here */
3280
  List<Key> new_key_list;
3281
  List_iterator<Alter_drop> drop_it(alter_info->drop_list);
3282
  List_iterator<Create_field> def_it(alter_info->create_list);
3283
  List_iterator<Alter_column> alter_it(alter_info->alter_list);
3284
  List_iterator<Key> key_it(alter_info->key_list);
3285
  List_iterator<Create_field> find_it(new_create_list);
3286
  List_iterator<Create_field> field_it(new_create_list);
3287
  List<Key_part_spec> key_parts;
482 by Brian Aker
Remove uint.
3288
  uint32_t db_create_options= (table->s->db_create_options
1 by brian
clean slate
3289
                           & ~(HA_OPTION_PACK_RECORD));
482 by Brian Aker
Remove uint.
3290
  uint32_t used_fields= create_info->used_fields;
1 by brian
clean slate
3291
  KEY *key_info=table->key_info;
55 by brian
Update for using real bool types.
3292
  bool rc= true;
1 by brian
clean slate
3293
3294
55 by brian
Update for using real bool types.
3295
  create_info->varchar= false;
1 by brian
clean slate
3296
  /* Let new create options override the old ones */
3297
  if (!(used_fields & HA_CREATE_USED_MIN_ROWS))
3298
    create_info->min_rows= table->s->min_rows;
3299
  if (!(used_fields & HA_CREATE_USED_MAX_ROWS))
3300
    create_info->max_rows= table->s->max_rows;
3301
  if (!(used_fields & HA_CREATE_USED_AVG_ROW_LENGTH))
3302
    create_info->avg_row_length= table->s->avg_row_length;
244.1.1 by Harrison Fisk
Port Ebay/Google memory storage engine variable width columns.
3303
  if (!(used_fields & HA_CREATE_USED_BLOCK_SIZE))
3304
    create_info->block_size= table->s->block_size;
1 by brian
clean slate
3305
  if (!(used_fields & HA_CREATE_USED_DEFAULT_CHARSET))
3306
    create_info->default_table_charset= table->s->table_charset;
3307
  if (!(used_fields & HA_CREATE_USED_AUTO) && table->found_next_number_field)
3308
    {
3309
    /* Table has an autoincrement, copy value to new table */
3310
    table->file->info(HA_STATUS_AUTO);
3311
    create_info->auto_increment_value= table->file->stats.auto_increment_value;
3312
  }
3313
  if (!(used_fields & HA_CREATE_USED_KEY_BLOCK_SIZE))
3314
    create_info->key_block_size= table->s->key_block_size;
3315
997.5.1 by chris
Replace macros around unireg.h, store_record,restore_record,cmp_record,empty_record
3316
  table->restoreRecordAsDefault();     // Empty record for DEFAULT
1 by brian
clean slate
3317
  Create_field *def;
3318
3319
    /*
3320
    First collect all fields from table which isn't in drop_list
3321
    */
3322
  Field **f_ptr,*field;
3323
  for (f_ptr=table->field ; (field= *f_ptr) ; f_ptr++)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3324
  {
1 by brian
clean slate
3325
    /* Check if field should be dropped */
3326
    Alter_drop *drop;
3327
    drop_it.rewind();
3328
    while ((drop=drop_it++))
3329
    {
3330
      if (drop->type == Alter_drop::COLUMN &&
3331
	  !my_strcasecmp(system_charset_info,field->field_name, drop->name))
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3332
      {
1 by brian
clean slate
3333
	/* Reset auto_increment value if it was dropped */
3334
	if (MTYP_TYPENR(field->unireg_check) == Field::NEXT_NUMBER &&
3335
	    !(used_fields & HA_CREATE_USED_AUTO))
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3336
	{
1 by brian
clean slate
3337
	  create_info->auto_increment_value=0;
3338
	  create_info->used_fields|=HA_CREATE_USED_AUTO;
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3339
	}
3340
	break;
1 by brian
clean slate
3341
      }
3342
    }
3343
    if (drop)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3344
    {
1 by brian
clean slate
3345
      drop_it.remove();
3346
      continue;
3347
    }
3348
    /* Check if field is changed */
3349
    def_it.rewind();
3350
    while ((def=def_it++))
3351
    {
3352
      if (def->change &&
3353
	  !my_strcasecmp(system_charset_info,field->field_name, def->change))
3354
	break;
3355
    }
3356
    if (def)
3357
    {						// Field is changed
3358
      def->field=field;
3359
      if (!def->after)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3360
      {
1 by brian
clean slate
3361
	new_create_list.push_back(def);
3362
	def_it.remove();
3363
      }
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3364
    }
3365
    else
3366
    {
1 by brian
clean slate
3367
      /*
3368
        This field was not dropped and not changed, add it to the list
3369
        for the new table.
3370
      */
3371
      def= new Create_field(field, field);
3372
      new_create_list.push_back(def);
3373
      alter_it.rewind();			// Change default if ALTER
3374
      Alter_column *alter;
3375
      while ((alter=alter_it++))
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3376
      {
1 by brian
clean slate
3377
	if (!my_strcasecmp(system_charset_info,field->field_name, alter->name))
3378
	  break;
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3379
      }
1 by brian
clean slate
3380
      if (alter)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3381
      {
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
3382
	if (def->sql_type == DRIZZLE_TYPE_BLOB)
1 by brian
clean slate
3383
	{
3384
	  my_error(ER_BLOB_CANT_HAVE_DEFAULT, MYF(0), def->change);
3385
          goto err;
3386
	}
3387
	if ((def->def=alter->def))              // Use new default
3388
          def->flags&= ~NO_DEFAULT_VALUE_FLAG;
3389
        else
3390
          def->flags|= NO_DEFAULT_VALUE_FLAG;
3391
	alter_it.remove();
3392
      }
3393
    }
3394
  }
3395
  def_it.rewind();
3396
  while ((def=def_it++))			// Add new columns
3397
  {
3398
    if (def->change && ! def->field)
3399
    {
3400
      my_error(ER_BAD_FIELD_ERROR, MYF(0), def->change, table->s->table_name.str);
3401
      goto err;
3402
    }
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3403
    /*
1 by brian
clean slate
3404
      Check that the DATE/DATETIME not null field we are going to add is
3405
      either has a default value or the '0000-00-00' is allowed by the
3406
      set sql mode.
3407
      If the '0000-00-00' value isn't allowed then raise the error_if_not_empty
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3408
      flag to allow ALTER Table only if the table to be altered is empty.
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3409
    */
575.5.1 by David Axmark
Changed NEWDATE to DATE. One failing test but I think its somewhere else in the code
3410
    if ((def->sql_type == DRIZZLE_TYPE_DATE ||
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
3411
         def->sql_type == DRIZZLE_TYPE_DATETIME) &&
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3412
	!alter_info->datetime_field &&
3413
	!(~def->flags & (NO_DEFAULT_VALUE_FLAG | NOT_NULL_FLAG)) &&
3414
	session->variables.sql_mode & MODE_NO_ZERO_DATE)
1 by brian
clean slate
3415
    {
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3416
      alter_info->datetime_field= def;
3417
      alter_info->error_if_not_empty= true;
1 by brian
clean slate
3418
    }
3419
    if (!def->after)
3420
      new_create_list.push_back(def);
3421
    else if (def->after == first_keyword)
3422
      new_create_list.push_front(def);
3423
    else
3424
    {
3425
      Create_field *find;
3426
      find_it.rewind();
3427
      while ((find=find_it++))			// Add new columns
3428
      {
3429
	if (!my_strcasecmp(system_charset_info,def->after, find->field_name))
3430
	  break;
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3431
      }
1 by brian
clean slate
3432
      if (!find)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3433
      {
1 by brian
clean slate
3434
	my_error(ER_BAD_FIELD_ERROR, MYF(0), def->after, table->s->table_name.str);
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3435
	goto err;
3436
      }
1 by brian
clean slate
3437
      find_it.after(def);			// Put element after this
3438
      /*
3439
        XXX: hack for Bug#28427.
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3440
        If column order has changed, force OFFLINE ALTER Table
1 by brian
clean slate
3441
        without querying engine capabilities.  If we ever have an
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3442
        engine that supports online ALTER Table CHANGE COLUMN
1 by brian
clean slate
3443
        <name> AFTER <name1> (Falcon?), this fix will effectively
3444
        disable the capability.
3445
        TODO: detect the situation in compare_tables, behave based
3446
        on engine capabilities.
3447
      */
3448
      if (alter_info->build_method == HA_BUILD_ONLINE)
3449
      {
520.1.22 by Brian Aker
Second pass of thd cleanup
3450
        my_error(ER_NOT_SUPPORTED_YET, MYF(0), session->query);
1 by brian
clean slate
3451
        goto err;
3452
      }
3453
      alter_info->build_method= HA_BUILD_OFFLINE;
3454
    }
3455
  }
3456
  if (alter_info->alter_list.elements)
3457
  {
3458
    my_error(ER_BAD_FIELD_ERROR, MYF(0),
3459
             alter_info->alter_list.head()->name, table->s->table_name.str);
3460
    goto err;
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3461
  }
1 by brian
clean slate
3462
  if (!new_create_list.elements)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3463
  {
1 by brian
clean slate
3464
    my_message(ER_CANT_REMOVE_ALL_FIELDS, ER(ER_CANT_REMOVE_ALL_FIELDS),
3465
               MYF(0));
3466
    goto err;
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3467
  }
1 by brian
clean slate
3468
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3469
  /*
1 by brian
clean slate
3470
    Collect all keys which isn't in drop list. Add only those
3471
    for which some fields exists.
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3472
  */
1 by brian
clean slate
3473
482 by Brian Aker
Remove uint.
3474
  for (uint32_t i=0 ; i < table->s->keys ; i++,key_info++)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3475
  {
1 by brian
clean slate
3476
    char *key_name= key_info->name;
3477
    Alter_drop *drop;
3478
    drop_it.rewind();
3479
    while ((drop=drop_it++))
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3480
    {
1 by brian
clean slate
3481
      if (drop->type == Alter_drop::KEY &&
3482
	  !my_strcasecmp(system_charset_info,key_name, drop->name))
3483
	break;
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3484
    }
1 by brian
clean slate
3485
    if (drop)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3486
    {
1 by brian
clean slate
3487
      drop_it.remove();
3488
      continue;
3489
    }
3490
3491
    KEY_PART_INFO *key_part= key_info->key_part;
3492
    key_parts.empty();
482 by Brian Aker
Remove uint.
3493
    for (uint32_t j=0 ; j < key_info->key_parts ; j++,key_part++)
1 by brian
clean slate
3494
    {
3495
      if (!key_part->field)
3496
	continue;				// Wrong field (from UNIREG)
3497
      const char *key_part_name=key_part->field->field_name;
3498
      Create_field *cfield;
3499
      field_it.rewind();
3500
      while ((cfield=field_it++))
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3501
      {
1 by brian
clean slate
3502
	if (cfield->change)
896.3.5 by Stewart Smith
fix indentation so I can remotely read the function.
3503
	{
1 by brian
clean slate
3504
	  if (!my_strcasecmp(system_charset_info, key_part_name,
3505
			     cfield->change))
3506
	    break;
3507
	}
3508
	else if (!my_strcasecmp(system_charset_info,
3509
				key_part_name, cfield->field_name))
3510
	  break;
3511
      }
3512
      if (!cfield)
3513
	continue;				// Field is removed
482 by Brian Aker
Remove uint.
3514
      uint32_t key_part_length=key_part->length;
1 by brian
clean slate
3515
      if (cfield->field)			// Not new field
3516
      {
3517
        /*
3518
          If the field can't have only a part used in a key according to its
3519
          new type, or should not be used partially according to its
3520
          previous type, or the field length is less than the key part
3521
          length, unset the key part length.
3522
3523
          We also unset the key part length if it is the same as the
3524
          old field's length, so the whole new field will be used.
3525
3526
          BLOBs may have cfield->length == 0, which is why we test it before
3527
          checking whether cfield->length < key_part_length (in chars).
3528
         */
3529
        if (!Field::type_can_have_key_part(cfield->field->type()) ||
3530
            !Field::type_can_have_key_part(cfield->sql_type) ||
3531
            (cfield->field->field_length == key_part_length &&
3532
             !f_is_blob(key_part->key_type)) ||
3533
	    (cfield->length && (cfield->length < key_part_length /
3534
                                key_part->field->charset()->mbmaxlen)))
3535
	  key_part_length= 0;			// Use whole field
3536
      }
3537
      key_part_length /= key_part->field->charset()->mbmaxlen;
3538
      key_parts.push_back(new Key_part_spec(cfield->field_name,
3539
                                            strlen(cfield->field_name),
3540
					    key_part_length));
3541
    }
3542
    if (key_parts.elements)
3543
    {
3544
      KEY_CREATE_INFO key_create_info;
3545
      Key *key;
3546
      enum Key::Keytype key_type;
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
3547
      memset(&key_create_info, 0, sizeof(key_create_info));
1 by brian
clean slate
3548
3549
      key_create_info.algorithm= key_info->algorithm;
3550
      if (key_info->flags & HA_USES_BLOCK_SIZE)
3551
        key_create_info.block_size= key_info->block_size;
3552
      if (key_info->flags & HA_USES_COMMENT)
3553
        key_create_info.comment= key_info->comment;
3554
3555
      if (key_info->flags & HA_NOSAME)
3556
      {
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
3557
        if (is_primary_key_name(key_name))
1 by brian
clean slate
3558
          key_type= Key::PRIMARY;
3559
        else
3560
          key_type= Key::UNIQUE;
3561
      }
3562
      else
3563
        key_type= Key::MULTIPLE;
3564
3565
      key= new Key(key_type, key_name, strlen(key_name),
3566
                   &key_create_info,
3567
                   test(key_info->flags & HA_GENERATED_KEY),
3568
                   key_parts);
3569
      new_key_list.push_back(key);
3570
    }
3571
  }
3572
  {
3573
    Key *key;
3574
    while ((key=key_it++))			// Add new keys
3575
    {
383.7.1 by Andrey Zhakov
Initial submit of code and tests
3576
      if (key->type == Key::FOREIGN_KEY &&
3577
          ((Foreign_key *)key)->validate(new_create_list))
3578
        goto err;
1 by brian
clean slate
3579
      if (key->type != Key::FOREIGN_KEY)
3580
        new_key_list.push_back(key);
590.1.1 by Stewart Smith
begin moving from global const char* primary_key_name to methods is_primary_key() and is_primary_key_name()
3581
      if (key->name.str && is_primary_key_name(key->name.str))
1 by brian
clean slate
3582
      {
3583
	my_error(ER_WRONG_NAME_FOR_INDEX, MYF(0), key->name.str);
3584
        goto err;
3585
      }
3586
    }
3587
  }
3588
3589
  if (alter_info->drop_list.elements)
3590
  {
3591
    my_error(ER_CANT_DROP_FIELD_OR_KEY, MYF(0),
3592
             alter_info->drop_list.head()->name);
3593
    goto err;
3594
  }
3595
  if (alter_info->alter_list.elements)
3596
  {
3597
    my_error(ER_CANT_DROP_FIELD_OR_KEY, MYF(0),
3598
             alter_info->alter_list.head()->name);
3599
    goto err;
3600
  }
3601
3602
  if (!create_info->comment.str)
3603
  {
3604
    create_info->comment.str= table->s->comment.str;
3605
    create_info->comment.length= table->s->comment.length;
3606
  }
3607
3608
  table->file->update_create_info(create_info);
3609
  if ((create_info->table_options &
3610
       (HA_OPTION_PACK_KEYS | HA_OPTION_NO_PACK_KEYS)) ||
3611
      (used_fields & HA_CREATE_USED_PACK_KEYS))
3612
    db_create_options&= ~(HA_OPTION_PACK_KEYS | HA_OPTION_NO_PACK_KEYS);
3613
  if (create_info->table_options &
3614
      (HA_OPTION_CHECKSUM | HA_OPTION_NO_CHECKSUM))
3615
    db_create_options&= ~(HA_OPTION_CHECKSUM | HA_OPTION_NO_CHECKSUM);
3616
  if (create_info->table_options &
3617
      (HA_OPTION_DELAY_KEY_WRITE | HA_OPTION_NO_DELAY_KEY_WRITE))
3618
    db_create_options&= ~(HA_OPTION_DELAY_KEY_WRITE |
3619
			  HA_OPTION_NO_DELAY_KEY_WRITE);
3620
  create_info->table_options|= db_create_options;
3621
3622
  if (table->s->tmp_table)
3623
    create_info->options|=HA_LEX_CREATE_TMP_TABLE;
3624
55 by brian
Update for using real bool types.
3625
  rc= false;
1 by brian
clean slate
3626
  alter_info->create_list.swap(new_create_list);
3627
  alter_info->key_list.swap(new_key_list);
3628
err:
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
3629
  return(rc);
1 by brian
clean slate
3630
}
3631
3632
3633
/*
3634
  Alter table
3635
3636
  SYNOPSIS
3637
    mysql_alter_table()
520.1.22 by Brian Aker
Second pass of thd cleanup
3638
      session              Thread handle
1 by brian
clean slate
3639
      new_db           If there is a RENAME clause
3640
      new_name         If there is a RENAME clause
3641
      create_info      Information from the parsing phase about new
3642
                       table properties.
3643
      table_list       The table to change.
3644
      alter_info       Lists of fields, keys to be changed, added
3645
                       or dropped.
327.2.3 by Brian Aker
Refactoring of class Table
3646
      order_num        How many order_st BY fields has been specified.
3647
      order            List of fields to order_st BY.
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3648
      ignore           Whether we have ALTER IGNORE Table
1 by brian
clean slate
3649
3650
  DESCRIPTION
3651
    This is a veery long function and is everything but the kitchen sink :)
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3652
    It is used to alter a table and not only by ALTER Table but also
1 by brian
clean slate
3653
    CREATE|DROP INDEX are mapped on this function.
3654
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3655
    When the ALTER Table statement just does a RENAME or ENABLE|DISABLE KEYS,
1 by brian
clean slate
3656
    or both, then this function short cuts its operation by renaming
3657
    the table and/or enabling/disabling the keys. In this case, the FRM is
3658
    not changed, directly by mysql_alter_table. However, if there is a
3659
    RENAME + change of a field, or an index, the short cut is not used.
3660
    See how `create_list` is used to generate the new FRM regarding the
3661
    structure of the fields. The same is done for the indices of the table.
3662
3663
    Important is the fact, that this function tries to do as little work as
3664
    possible, by finding out whether a intermediate table is needed to copy
3665
    data into and when finishing the altering to use it as the original table.
3666
    For this reason the function compare_tables() is called, which decides
3667
    based on all kind of data how similar are the new and the original
3668
    tables.
3669
3670
  RETURN VALUES
55 by brian
Update for using real bool types.
3671
    false  OK
3672
    true   Error
1 by brian
clean slate
3673
*/
3674
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
3675
bool mysql_alter_table(Session *session, char *new_db, char *new_name,
1 by brian
clean slate
3676
                       HA_CREATE_INFO *create_info,
327.2.4 by Brian Aker
Refactoring table.h
3677
                       TableList *table_list,
1 by brian
clean slate
3678
                       Alter_info *alter_info,
482 by Brian Aker
Remove uint.
3679
                       uint32_t order_num, order_st *order, bool ignore)
1 by brian
clean slate
3680
{
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3681
  Table *table, *new_table=0, *name_lock= 0;;
670.3.3 by Toru Maesaka
Added namespacing for std to .cc files that needed it
3682
  string new_name_str;
1 by brian
clean slate
3683
  int error= 0;
3684
  char tmp_name[80],old_name[32],new_name_buff[FN_REFLEN];
1039.1.4 by Brian Aker
Modified alias to being const.
3685
  char new_alias_buff[FN_REFLEN], *table_name, *db;
3686
  const char *new_alias;
1 by brian
clean slate
3687
  char path[FN_REFLEN];
3688
  ha_rows copied= 0,deleted= 0;
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
3689
  StorageEngine *old_db_type, *new_db_type, *save_old_db_type;
1 by brian
clean slate
3690
584.2.6 by Stewart Smith
fix valgrind warning in alter table: initialise string.
3691
  new_name_buff[0]= '\0';
3692
1 by brian
clean slate
3693
  if (table_list && table_list->schema_table)
3694
  {
575.4.7 by Monty Taylor
More header cleanup.
3695
    my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), "", "", INFORMATION_SCHEMA_NAME.c_str());
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
3696
    return true;
1 by brian
clean slate
3697
  }
3698
3699
  /*
3700
    Assign variables table_name, new_name, db, new_db, path
3701
    to simplify further comparisons: we want to see if it's a RENAME
3702
    later just by comparing the pointers, avoiding the need for strcmp.
3703
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
3704
  session->set_proc_info("init");
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
3705
  table_name= table_list->table_name;
1 by brian
clean slate
3706
  db=table_list->db;
3707
  if (!new_db || !my_strcasecmp(table_alias_charset, new_db, db))
3708
    new_db= db;
1039.1.6 by Brian Aker
Refactor for build_table_filename()
3709
  build_table_filename(path, sizeof(path), db, table_name, false);
1 by brian
clean slate
3710
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3711
  /* DISCARD/IMPORT TABLESPACE is always alone in an ALTER Table */
1 by brian
clean slate
3712
  if (alter_info->tablespace_op != NO_TABLESPACE_OP)
3713
    /* Conditionally writes to binlog. */
520.1.22 by Brian Aker
Second pass of thd cleanup
3714
    return(mysql_discard_or_import_tablespace(session,table_list,
108 by Brian Aker
Removed unwanted ALTER TABLESPACE, left in valuable bits.
3715
                                              alter_info->tablespace_op));
670.3.3 by Toru Maesaka
Added namespacing for std to .cc files that needed it
3716
  ostringstream oss;
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
3717
  oss << drizzle_data_home << "/" << db << "/" << table_name;
534 by Monty Taylor
Removed stxnmov. Also deleted strstr which had already been removed.
3718
670.3.1 by Toru Maesaka
Replaced MySQL's my_stpncpy() with libc and c++ calls
3719
  (void) unpack_filename(new_name_buff, oss.str().c_str());
1 by brian
clean slate
3720
  /*
3721
    If this is just a rename of a view, short cut to the
3722
    following scenario: 1) lock LOCK_open 2) do a RENAME
3723
    2) unlock LOCK_open.
3724
    This is a copy-paste added to make sure
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3725
    ALTER (sic:) Table .. RENAME works for views. ALTER VIEW is handled
1 by brian
clean slate
3726
    as an independent branch in mysql_execute_command. The need
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3727
    for a copy-paste arose because the main code flow of ALTER Table
1 by brian
clean slate
3728
    ... RENAME tries to use open_ltable, which does not work for views
3729
    (open_ltable was never modified to merge table lists of child tables
3730
    into the main table list, like open_tables does).
3731
    This code is wrong and will be removed, please do not copy.
3732
  */
3733
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
3734
  if (!(table= session->open_ltable(table_list, TL_WRITE_ALLOW_READ)))
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3735
    return true;
1 by brian
clean slate
3736
  table->use_all_columns();
3737
3738
  /* Check that we are not trying to rename to an existing table */
3739
  if (new_name)
3740
  {
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
3741
    strcpy(new_name_buff, new_name);
1039.1.4 by Brian Aker
Modified alias to being const.
3742
    strcpy(new_alias_buff, new_name);
3743
    new_alias= new_alias_buff;
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
3744
3745
    my_casedn_str(files_charset_info, new_name_buff);
3746
    new_alias= new_name;			// Create lower case table name
3747
    my_casedn_str(files_charset_info, new_name);
3748
1 by brian
clean slate
3749
    if (new_db == db &&
3750
	!my_strcasecmp(table_alias_charset, new_name_buff, table_name))
3751
    {
3752
      /*
3753
	Source and destination table names are equal: make later check
3754
	easier.
3755
      */
3756
      new_alias= new_name= table_name;
3757
    }
3758
    else
3759
    {
3760
      if (table->s->tmp_table != NO_TMP_TABLE)
3761
      {
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
3762
	if (session->find_temporary_table(new_db, new_name_buff))
1 by brian
clean slate
3763
	{
3764
	  my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_name_buff);
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
3765
	  return true;
1 by brian
clean slate
3766
	}
3767
      }
3768
      else
3769
      {
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
3770
        if (session->lock_table_name_if_not_cached(new_db, new_name, &name_lock))
3771
          return true;
3772
1 by brian
clean slate
3773
        if (!name_lock)
3774
        {
3775
	  my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_alias);
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
3776
	  return true;
1 by brian
clean slate
3777
        }
3778
3779
        build_table_filename(new_name_buff, sizeof(new_name_buff),
1039.1.6 by Brian Aker
Refactor for build_table_filename()
3780
                             new_db, new_name_buff, false);
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
3781
        if (table_proto_exists(new_name_buff)==EEXIST)
1 by brian
clean slate
3782
	{
934.2.8 by Jay Pipes
Refactors the do_command() function out of the sql_parse.cc stuff and implements it as a member method, executeStatement() on the Session object.
3783
	  /* Table will be closed by Session::executeCommand() */
1 by brian
clean slate
3784
	  my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_alias);
3785
	  goto err;
3786
	}
3787
      }
3788
    }
3789
  }
3790
  else
3791
  {
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
3792
    new_alias= table_name;
1 by brian
clean slate
3793
    new_name= table_name;
3794
  }
3795
3796
  old_db_type= table->s->db_type();
3797
  if (!create_info->db_type)
3798
  {
3799
    create_info->db_type= old_db_type;
3800
  }
3801
520.1.22 by Brian Aker
Second pass of thd cleanup
3802
  if (check_engine(session, new_name, create_info))
1 by brian
clean slate
3803
    goto err;
3804
  new_db_type= create_info->db_type;
3805
186 by Brian Aker
Partial fix for alter table
3806
  if (new_db_type != old_db_type &&
1 by brian
clean slate
3807
      !table->file->can_switch_engines())
3808
  {
186 by Brian Aker
Partial fix for alter table
3809
    assert(0);
1 by brian
clean slate
3810
    my_error(ER_ROW_IS_REFERENCED, MYF(0));
3811
    goto err;
3812
  }
3813
3814
  if (create_info->row_type == ROW_TYPE_NOT_USED)
3815
    create_info->row_type= table->s->row_type;
3816
964.1.4 by Monty Taylor
Moved flags into private area.
3817
  if (old_db_type->check_flag(HTON_BIT_ALTER_NOT_SUPPORTED) ||
3818
      new_db_type->check_flag(HTON_BIT_ALTER_NOT_SUPPORTED))
1 by brian
clean slate
3819
  {
3820
    my_error(ER_ILLEGAL_HA, MYF(0), table_name);
3821
    goto err;
3822
  }
3823
520.1.22 by Brian Aker
Second pass of thd cleanup
3824
  session->set_proc_info("setup");
1 by brian
clean slate
3825
  if (!(alter_info->flags & ~(ALTER_RENAME | ALTER_KEYS_ONOFF)) &&
3826
      !table->s->tmp_table) // no need to touch frm
3827
  {
3828
    switch (alter_info->keys_onoff) {
3829
    case LEAVE_AS_IS:
3830
      break;
3831
    case ENABLE:
3832
      /*
3833
        wait_while_table_is_used() ensures that table being altered is
1000.1.3 by Brian Aker
Renamed TABLE_SHARE to TableShare
3834
        opened only by this thread and that Table::TableShare::version
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3835
        of Table object corresponding to this table is 0.
1 by brian
clean slate
3836
        The latter guarantees that no DML statement will open this table
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3837
        until ALTER Table finishes (i.e. until close_thread_tables())
1 by brian
clean slate
3838
        while the fact that the table is still open gives us protection
3839
        from concurrent DDL statements.
3840
      */
1046.1.2 by Brian Aker
Comments on LOCK_open
3841
      pthread_mutex_lock(&LOCK_open); /* DDL wait for/blocker */
520.1.22 by Brian Aker
Second pass of thd cleanup
3842
      wait_while_table_is_used(session, table, HA_EXTRA_FORCE_REOPEN);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
3843
      pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
3844
      error= table->file->ha_enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
3845
      /* COND_refresh will be signaled in close_thread_tables() */
3846
      break;
3847
    case DISABLE:
1046.1.2 by Brian Aker
Comments on LOCK_open
3848
      pthread_mutex_lock(&LOCK_open); /* DDL wait for/blocker */
520.1.22 by Brian Aker
Second pass of thd cleanup
3849
      wait_while_table_is_used(session, table, HA_EXTRA_FORCE_REOPEN);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
3850
      pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
3851
      error=table->file->ha_disable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
3852
      /* COND_refresh will be signaled in close_thread_tables() */
3853
      break;
3854
    default:
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
3855
      assert(false);
1 by brian
clean slate
3856
      error= 0;
3857
      break;
3858
    }
3859
    if (error == HA_ERR_WRONG_COMMAND)
3860
    {
3861
      error= 0;
520.1.22 by Brian Aker
Second pass of thd cleanup
3862
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
3863
			  ER_ILLEGAL_HA, ER(ER_ILLEGAL_HA),
3864
			  table->alias);
3865
    }
3866
1046.1.2 by Brian Aker
Comments on LOCK_open
3867
    pthread_mutex_lock(&LOCK_open); /* Lock to remove all instances of table from table cache before ALTER */
1 by brian
clean slate
3868
    /*
3869
      Unlike to the above case close_cached_table() below will remove ALL
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3870
      instances of Table from table cache (it will also remove table lock
1 by brian
clean slate
3871
      held by this thread). So to make actual table renaming and writing
3872
      to binlog atomic we have to put them into the same critical section
3873
      protected by LOCK_open mutex. This also removes gap for races between
3874
      access() and mysql_rename_table() calls.
3875
    */
3876
3877
    if (!error && (new_name != table_name || new_db != db))
3878
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
3879
      session->set_proc_info("rename");
1 by brian
clean slate
3880
      /*
3881
        Then do a 'simple' rename of the table. First we need to close all
3882
        instances of 'source' table.
3883
      */
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
3884
      session->close_cached_table(table);
1 by brian
clean slate
3885
      /*
3886
        Then, we want check once again that target table does not exist.
3887
        Actually the order of these two steps does not matter since
3888
        earlier we took name-lock on the target table, so we do them
3889
        in this particular order only to be consistent with 5.0, in which
3890
        we don't take this name-lock and where this order really matters.
3891
        TODO: Investigate if we need this access() check at all.
3892
      */
896.4.9 by Stewart Smith
No longer write the FRM. just use proto.
3893
      if (table_proto_exists(new_name)==EEXIST)
1 by brian
clean slate
3894
      {
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3895
        my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_name);
3896
        error= -1;
1 by brian
clean slate
3897
      }
3898
      else
3899
      {
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3900
        *fn_ext(new_name)=0;
3901
        if (mysql_rename_table(old_db_type, db, table_name, new_db, new_alias, 0))
3902
          error= -1;
1 by brian
clean slate
3903
        else if (0)
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3904
        {
398.1.10 by Monty Taylor
Actually removed VOID() this time.
3905
          mysql_rename_table(old_db_type, new_db, new_alias, db,
3906
                             table_name, 0);
1 by brian
clean slate
3907
          error= -1;
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3908
        }
1 by brian
clean slate
3909
      }
3910
    }
3911
3912
    if (error == HA_ERR_WRONG_COMMAND)
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3913
    {
1 by brian
clean slate
3914
      error= 0;
520.1.22 by Brian Aker
Second pass of thd cleanup
3915
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3916
                          ER_ILLEGAL_HA, ER(ER_ILLEGAL_HA),
3917
                          table->alias);
3918
    }
1 by brian
clean slate
3919
3920
    if (!error)
3921
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
3922
      write_bin_log(session, true, session->query, session->query_length);
836 by Brian Aker
Fixed session call from function to method.
3923
      session->my_ok();
1 by brian
clean slate
3924
  }
3925
    else if (error > 0)
3926
  {
3927
      table->file->print_error(error, MYF(0));
3928
      error= -1;
3929
    }
3930
    if (name_lock)
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
3931
      session->unlink_open_table(name_lock);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
3932
    pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
3933
    table_list->table= NULL;                    // For query cache
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
3934
    return(error);
1 by brian
clean slate
3935
  }
3936
3937
  /* We have to do full alter table. */
3938
3939
    /*
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
3940
    If the old table had partitions and we are doing ALTER Table ...
1 by brian
clean slate
3941
    engine= <new_engine>, the new table must preserve the original
3942
    partitioning. That means that the new engine is still the
3943
    partitioning engine, not the engine specified in the parser.
3944
    This is discovered  in prep_alter_part_table, which in such case
3945
    updates create_info->db_type.
3946
    Now we need to update the stack copy of create_info->db_type,
3947
    as otherwise we won't be able to correctly move the files of the
3948
    temporary table to the result table files.
3949
  */
3950
  new_db_type= create_info->db_type;
3951
520.1.22 by Brian Aker
Second pass of thd cleanup
3952
  if (mysql_prepare_alter_table(session, table, create_info, alter_info))
1 by brian
clean slate
3953
      goto err;
3954
1014.3.1 by Brian Aker
Simplify the calling stack for getting schema collation. We need to extend
3955
  set_table_default_charset(create_info, db);
520.1.22 by Brian Aker
Second pass of thd cleanup
3956
1003.1.16 by Brian Aker
First pass in alter table cleanup.
3957
  alter_info->build_method= HA_BUILD_OFFLINE;
1 by brian
clean slate
3958
584.2.1 by Stewart Smith
convert tmp_file_prefix to TMP_FILE_PREFIX to indicate it's a macro
3959
  snprintf(tmp_name, sizeof(tmp_name), "%s-%lx_%"PRIx64, TMP_FILE_PREFIX,
606 by Brian Aker
Remove dead lex structure and clean up use of pid_t
3960
           (unsigned long)current_pid, session->thread_id);
1 by brian
clean slate
3961
  /* Safety fix for innodb */
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
3962
  my_casedn_str(files_charset_info, tmp_name);
1 by brian
clean slate
3963
3964
3965
  /* Create a temporary table with the new format */
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
3966
  if ((error= create_temporary_table(session, table, new_db, tmp_name,
3967
                                     create_info, alter_info,
1 by brian
clean slate
3968
                                     !strcmp(db, new_db))))
3969
  {
3970
    goto err;
3971
  }
3972
3973
  /* Open the table so we need to copy the data to it. */
3974
  if (table->s->tmp_table)
3975
  {
327.2.4 by Brian Aker
Refactoring table.h
3976
    TableList tbl;
1 by brian
clean slate
3977
    tbl.db= new_db;
1039.1.4 by Brian Aker
Modified alias to being const.
3978
    tbl.alias= tmp_name;
3979
    tbl.table_name= tmp_name;
3980
520.1.22 by Brian Aker
Second pass of thd cleanup
3981
    /* Table is in session->temporary_tables */
1054.1.10 by Brian Aker
Move open_table() to session.
3982
    new_table= session->open_table(&tbl, (bool*) 0, DRIZZLE_LOCK_IGNORE_FLUSH);
1 by brian
clean slate
3983
  }
3984
  else
3985
  {
779.3.10 by Monty Taylor
Turned on -Wshadow.
3986
    char tmp_path[FN_REFLEN];
1 by brian
clean slate
3987
    /* table is a normal table: Create temporary table in same directory */
1039.1.6 by Brian Aker
Refactor for build_table_filename()
3988
    build_table_filename(tmp_path, sizeof(tmp_path), new_db, tmp_name, true);
1 by brian
clean slate
3989
    /* Open our intermediate table */
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3990
    new_table= open_temporary_table(session, tmp_path, new_db, tmp_name, 0, OTM_OPEN);
1 by brian
clean slate
3991
  }
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
3992
3993
  if (new_table == NULL)
1 by brian
clean slate
3994
    goto err1;
3995
3996
  /* Copy the data if necessary. */
520.1.22 by Brian Aker
Second pass of thd cleanup
3997
  session->count_cuted_fields= CHECK_FIELD_WARN;	// calc cuted fields
3998
  session->cuted_fields=0L;
3999
  session->set_proc_info("copy to tmp table");
1039.1.12 by Brian Aker
Removed dead code around alter table/MERGE tables. Added assert to double
4000
  copied= deleted= 0;
4001
4002
  assert(new_table);
4003
4004
  /* We don't want update TIMESTAMP fields during ALTER Table. */
4005
  new_table->timestamp_field_type= TIMESTAMP_NO_AUTO_SET;
4006
  new_table->next_number_field=new_table->found_next_number_field;
4007
  error= copy_data_between_tables(table, new_table,
4008
                                  alter_info->create_list, ignore,
4009
                                  order_num, order, &copied, &deleted,
4010
                                  alter_info->keys_onoff,
4011
                                  alter_info->error_if_not_empty);
4012
873.1.7 by Jay Pipes
Fixes Field_datetime::store(NUMBER) to throw an error when invalid
4013
  /* We must not ignore bad input! */;
4014
  session->count_cuted_fields= CHECK_FIELD_ERROR_FOR_NULL;
1 by brian
clean slate
4015
4016
  if (table->s->tmp_table != NO_TMP_TABLE)
4017
  {
4018
    /* We changed a temporary table */
4019
    if (error)
4020
      goto err1;
4021
    /* Close lock if this is a transactional table */
520.1.22 by Brian Aker
Second pass of thd cleanup
4022
    if (session->lock)
1 by brian
clean slate
4023
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
4024
      mysql_unlock_tables(session, session->lock);
4025
      session->lock=0;
1 by brian
clean slate
4026
    }
4027
    /* Remove link to old table and rename the new one */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
4028
    session->close_temporary_table(table, true, true);
1 by brian
clean slate
4029
    /* Should pass the 'new_name' as we store table name in the cache */
1019 by Brian Aker
Remove the need for TMP_TABLE_KEY_EXTRA (we don't replicate temp tables).
4030
    if (rename_temporary_table(new_table, new_db, new_name))
1 by brian
clean slate
4031
      goto err1;
4032
    goto end_temporary;
4033
  }
4034
4035
  if (new_table)
4036
  {
4037
    /*
4038
      Close the intermediate table that will be the new table.
4039
      Note that MERGE tables do not have their children attached here.
4040
    */
4041
    intern_close_table(new_table);
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.
4042
    free(new_table);
1 by brian
clean slate
4043
  }
1046.1.2 by Brian Aker
Comments on LOCK_open
4044
  pthread_mutex_lock(&LOCK_open); /* ALTER TABLE */
1 by brian
clean slate
4045
  if (error)
4046
  {
1039.1.8 by Brian Aker
Style cleanup
4047
    quick_rm_table(new_db_type, new_db, tmp_name, true);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
4048
    pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
4049
    goto err;
4050
  }
4051
4052
  /*
4053
    Data is copied. Now we:
4054
    1) Wait until all other threads close old version of table.
4055
    2) Close instances of table open by this thread and replace them
4056
       with exclusive name-locks.
4057
    3) Rename the old table to a temp name, rename the new one to the
4058
       old name.
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
4059
    4) If we are under LOCK TABLES and don't do ALTER Table ... RENAME
1 by brian
clean slate
4060
       we reopen new version of table.
4061
    5) Write statement to the binary log.
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
4062
    6) If we are under LOCK TABLES and do ALTER Table ... RENAME we
1 by brian
clean slate
4063
       remove name-locks from list of open tables and table cache.
4064
    7) If we are not not under LOCK TABLES we rely on close_thread_tables()
4065
       call to remove name-locks from table cache and list of open table.
4066
  */
4067
520.1.22 by Brian Aker
Second pass of thd cleanup
4068
  session->set_proc_info("rename result table");
584.2.1 by Stewart Smith
convert tmp_file_prefix to TMP_FILE_PREFIX to indicate it's a macro
4069
  snprintf(old_name, sizeof(old_name), "%s2-%lx-%"PRIx64, TMP_FILE_PREFIX,
606 by Brian Aker
Remove dead lex structure and clean up use of pid_t
4070
           (unsigned long)current_pid, session->thread_id);
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
4071
  my_casedn_str(files_charset_info, old_name);
1 by brian
clean slate
4072
520.1.22 by Brian Aker
Second pass of thd cleanup
4073
  wait_while_table_is_used(session, table, HA_EXTRA_PREPARE_FOR_RENAME);
1046.1.14 by Brian Aker
More redactoring of all lock issue code that is session bound, to be a
4074
  session->close_data_files_and_morph_locks(db, table_name);
1 by brian
clean slate
4075
4076
  error=0;
4077
  save_old_db_type= old_db_type;
4078
4079
  /*
4080
    This leads to the storage engine (SE) not being notified for renames in
4081
    mysql_rename_table(), because we just juggle with the FRM and nothing
4082
    more. If we have an intermediate table, then we notify the SE that
4083
    it should become the actual table. Later, we will recycle the old table.
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
4084
    However, in case of ALTER Table RENAME there might be no intermediate
1 by brian
clean slate
4085
    table. This is when the old and new tables are compatible, according to
4086
    compare_table(). Then, we need one additional call to
4087
    mysql_rename_table() with flag NO_FRM_RENAME, which does nothing else but
4088
    actual rename in the SE and the FRM is not touched. Note that, if the
4089
    table is renamed and the SE is also changed, then an intermediate table
4090
    is created and the additional call will not take place.
4091
  */
4092
  if (mysql_rename_table(old_db_type, db, table_name, db, old_name,
4093
                         FN_TO_IS_TMP))
4094
  {
4095
    error=1;
1039.1.8 by Brian Aker
Style cleanup
4096
    quick_rm_table(new_db_type, new_db, tmp_name, true);
1 by brian
clean slate
4097
  }
4098
  else if (mysql_rename_table(new_db_type, new_db, tmp_name, new_db,
4099
                              new_alias, FN_FROM_IS_TMP) || ((new_name != table_name || new_db != db) && 0))
4100
  {
4101
    /* Try to get everything back. */
1046.1.14 by Brian Aker
More redactoring of all lock issue code that is session bound, to be a
4102
    error= 1;
1039.1.8 by Brian Aker
Style cleanup
4103
    quick_rm_table(new_db_type, new_db, new_alias, false);
4104
    quick_rm_table(new_db_type, new_db, tmp_name, true);
1039.1.5 by Brian Aker
Remove lower case filename bits (aka we just lock into the most compatible
4105
    mysql_rename_table(old_db_type, db, old_name, db, table_name,
398.1.10 by Monty Taylor
Actually removed VOID() this time.
4106
                       FN_FROM_IS_TMP);
1 by brian
clean slate
4107
  }
4108
4109
  if (error)
4110
  {
4111
    /* This shouldn't happen. But let us play it safe. */
4112
    goto err_with_placeholders;
4113
  }
4114
1039.1.8 by Brian Aker
Style cleanup
4115
  quick_rm_table(old_db_type, db, old_name, true);
1 by brian
clean slate
4116
398.1.10 by Monty Taylor
Actually removed VOID() this time.
4117
  pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
4118
520.1.22 by Brian Aker
Second pass of thd cleanup
4119
  session->set_proc_info("end");
1 by brian
clean slate
4120
520.1.22 by Brian Aker
Second pass of thd cleanup
4121
  write_bin_log(session, true, session->query, session->query_length);
1 by brian
clean slate
4122
964.1.4 by Monty Taylor
Moved flags into private area.
4123
  if (old_db_type->check_flag(HTON_BIT_FLUSH_AFTER_RENAME))
1 by brian
clean slate
4124
  {
4125
    /*
4126
      For the alter table to be properly flushed to the logs, we
4127
      have to open the new table.  If not, we get a problem on server
4128
      shutdown. But we do not need to attach MERGE children.
4129
    */
779.3.10 by Monty Taylor
Turned on -Wshadow.
4130
    char table_path[FN_REFLEN];
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
4131
    Table *t_table;
1039.1.6 by Brian Aker
Refactor for build_table_filename()
4132
    build_table_filename(table_path, sizeof(table_path), new_db, table_name, false);
779.3.10 by Monty Taylor
Turned on -Wshadow.
4133
    t_table= open_temporary_table(session, table_path, new_db, tmp_name, false, OTM_OPEN);
1 by brian
clean slate
4134
    if (t_table)
4135
    {
4136
      intern_close_table(t_table);
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.
4137
      free(t_table);
1 by brian
clean slate
4138
    }
4139
    else
779.3.10 by Monty Taylor
Turned on -Wshadow.
4140
      errmsg_printf(ERRMSG_LVL_WARN,
4141
                    _("Could not open table %s.%s after rename\n"),
4142
                    new_db,table_name);
1 by brian
clean slate
4143
    ha_flush_logs(old_db_type);
4144
  }
4145
  table_list->table=0;				// For query cache
4146
4147
end_temporary:
873.1.7 by Jay Pipes
Fixes Field_datetime::store(NUMBER) to throw an error when invalid
4148
  /*
4149
   * Field::store() may have called my_error().  If this is 
4150
   * the case, we must not send an ok packet, since 
4151
   * Diagnostics_area::is_set() will fail an assert.
4152
   */
4153
  if (! session->is_error())
4154
  {
4155
    snprintf(tmp_name, sizeof(tmp_name), ER(ER_INSERT_INFO),
4156
            (ulong) (copied + deleted), (ulong) deleted,
4157
            (ulong) session->cuted_fields);
4158
    session->my_ok(copied + deleted, 0L, tmp_name);
4159
    session->some_tables_deleted=0;
4160
    return false;
4161
  }
4162
  else
4163
  {
4164
    /* my_error() was called.  Return true (which means error...) */
4165
    return true;
4166
  }
1 by brian
clean slate
4167
4168
err1:
4169
  if (new_table)
4170
  {
4171
    /* close_temporary_table() frees the new_table pointer. */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
4172
    session->close_temporary_table(new_table, true, true);
1 by brian
clean slate
4173
  }
4174
  else
1039.1.8 by Brian Aker
Style cleanup
4175
    quick_rm_table(new_db_type, new_db, tmp_name, true);
1 by brian
clean slate
4176
4177
err:
4178
  /*
4179
    No default value was provided for a DATE/DATETIME field, the
4180
    current sql_mode doesn't allow the '0000-00-00' value and
4181
    the table to be altered isn't empty.
4182
    Report error here.
4183
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
4184
  if (alter_info->error_if_not_empty && session->row_count)
1 by brian
clean slate
4185
  {
4186
    const char *f_val= 0;
236.1.25 by Monty Taylor
Fixed a few naming references.
4187
    enum enum_drizzle_timestamp_type t_type= DRIZZLE_TIMESTAMP_DATE;
1 by brian
clean slate
4188
    switch (alter_info->datetime_field->sql_type)
4189
    {
575.5.1 by David Axmark
Changed NEWDATE to DATE. One failing test but I think its somewhere else in the code
4190
      case DRIZZLE_TYPE_DATE:
1 by brian
clean slate
4191
        f_val= "0000-00-00";
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
4192
        t_type= DRIZZLE_TIMESTAMP_DATE;
1 by brian
clean slate
4193
        break;
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
4194
      case DRIZZLE_TYPE_DATETIME:
1 by brian
clean slate
4195
        f_val= "0000-00-00 00:00:00";
236.1.24 by Monty Taylor
Renamed MYSQL_TIME to DRIZZLE_TIME.
4196
        t_type= DRIZZLE_TIMESTAMP_DATETIME;
1 by brian
clean slate
4197
        break;
4198
      default:
4199
        /* Shouldn't get here. */
51.2.1 by Patrick Galbraith
Removed DBUG_PRINTs, DBUG_ASSERTs, DBUG_EXECUTE_IFs from
4200
        assert(0);
1 by brian
clean slate
4201
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
4202
    bool save_abort_on_warning= session->abort_on_warning;
4203
    session->abort_on_warning= true;
4204
    make_truncated_value_warning(session, DRIZZLE_ERROR::WARN_LEVEL_ERROR,
1 by brian
clean slate
4205
                                 f_val, strlength(f_val), t_type,
4206
                                 alter_info->datetime_field->field_name);
520.1.22 by Brian Aker
Second pass of thd cleanup
4207
    session->abort_on_warning= save_abort_on_warning;
1 by brian
clean slate
4208
  }
4209
  if (name_lock)
4210
  {
1046.1.2 by Brian Aker
Comments on LOCK_open
4211
    pthread_mutex_lock(&LOCK_open); /* ALTER TABLe */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
4212
    session->unlink_open_table(name_lock);
1 by brian
clean slate
4213
    pthread_mutex_unlock(&LOCK_open);
4214
  }
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
4215
  return true;
1 by brian
clean slate
4216
4217
err_with_placeholders:
4218
  /*
4219
    An error happened while we were holding exclusive name-lock on table
4220
    being altered. To be safe under LOCK TABLES we should remove placeholders
4221
    from list of open tables list and table cache.
4222
  */
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
4223
  session->unlink_open_table(table);
1 by brian
clean slate
4224
  if (name_lock)
1054.1.9 by Brian Aker
This is a large number of refactors against the Session class for its
4225
    session->unlink_open_table(name_lock);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
4226
  pthread_mutex_unlock(&LOCK_open);
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
4227
  return(true);
1 by brian
clean slate
4228
}
4229
/* mysql_alter_table */
4230
4231
static int
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
4232
copy_data_between_tables(Table *from,Table *to,
1 by brian
clean slate
4233
			 List<Create_field> &create,
4234
                         bool ignore,
482 by Brian Aker
Remove uint.
4235
			 uint32_t order_num, order_st *order,
1 by brian
clean slate
4236
			 ha_rows *copied,
4237
			 ha_rows *deleted,
4238
                         enum enum_enable_or_disable keys_onoff,
4239
                         bool error_if_not_empty)
4240
{
4241
  int error;
4242
  Copy_field *copy,*copy_end;
4243
  ulong found_count,delete_count;
520.1.22 by Brian Aker
Second pass of thd cleanup
4244
  Session *session= current_session;
482 by Brian Aker
Remove uint.
4245
  uint32_t length= 0;
1 by brian
clean slate
4246
  SORT_FIELD *sortorder;
4247
  READ_RECORD info;
327.2.4 by Brian Aker
Refactoring table.h
4248
  TableList   tables;
1 by brian
clean slate
4249
  List<Item>   fields;
4250
  List<Item>   all_fields;
4251
  ha_rows examined_rows;
4252
  bool auto_increment_field_copied= 0;
4253
  ulong save_sql_mode;
151 by Brian Aker
Ulonglong to uint64_t
4254
  uint64_t prev_insert_id;
1 by brian
clean slate
4255
4256
  /*
4257
    Turn off recovery logging since rollback of an alter table is to
4258
    delete the new table so there is no need to log the changes to it.
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
4259
1 by brian
clean slate
4260
    This needs to be done before external_lock
4261
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
4262
  error= ha_enable_transaction(session, false);
1 by brian
clean slate
4263
  if (error)
1046.1.10 by Brian Aker
Formatting around return (style)
4264
    return -1;
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
4265
1 by brian
clean slate
4266
  if (!(copy= new Copy_field[to->s->fields]))
1046.1.10 by Brian Aker
Formatting around return (style)
4267
    return -1;				/* purecov: inspected */
1 by brian
clean slate
4268
520.1.22 by Brian Aker
Second pass of thd cleanup
4269
  if (to->file->ha_external_lock(session, F_WRLCK))
1046.1.10 by Brian Aker
Formatting around return (style)
4270
    return -1;
1 by brian
clean slate
4271
4272
  /* We need external lock before we can disable/enable keys */
4273
  alter_table_manage_keys(to, from->file->indexes_are_disabled(), keys_onoff);
4274
4275
  /* We can abort alter table for any table type */
520.1.22 by Brian Aker
Second pass of thd cleanup
4276
  session->abort_on_warning= !ignore;
1 by brian
clean slate
4277
4278
  from->file->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
4279
  to->file->ha_start_bulk_insert(from->file->stats.records);
4280
520.1.22 by Brian Aker
Second pass of thd cleanup
4281
  save_sql_mode= session->variables.sql_mode;
1 by brian
clean slate
4282
4283
  List_iterator<Create_field> it(create);
4284
  Create_field *def;
4285
  copy_end=copy;
4286
  for (Field **ptr=to->field ; *ptr ; ptr++)
4287
  {
4288
    def=it++;
4289
    if (def->field)
4290
    {
4291
      if (*ptr == to->next_number_field)
55 by brian
Update for using real bool types.
4292
        auto_increment_field_copied= true;
359 by Brian Aker
More modes removed. 0 always becomes new number again
4293
1 by brian
clean slate
4294
      (copy_end++)->set(*ptr,def->field,0);
4295
    }
4296
4297
  }
4298
4299
  found_count=delete_count=0;
4300
4301
  if (order)
4302
  {
4303
    if (to->s->primary_key != MAX_KEY && to->file->primary_key_is_clustered())
4304
    {
261.4.1 by Felipe
- Renamed MYSQL_ERROR to DRIZZLE_ERROR.
4305
      char warn_buff[DRIZZLE_ERRMSG_SIZE];
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
4306
      snprintf(warn_buff, sizeof(warn_buff),
327.2.3 by Brian Aker
Refactoring of class Table
4307
               _("order_st BY ignored because there is a user-defined clustered "
338 by Monty Taylor
Tagged more strings.
4308
                 "index in the table '%-.192s'"),
4309
               from->s->table_name.str);
520.1.22 by Brian Aker
Second pass of thd cleanup
4310
      push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN, ER_UNKNOWN_ERROR,
1 by brian
clean slate
4311
                   warn_buff);
4312
    }
4313
    else
4314
    {
684 by Brian Aker
Mass cleanup for casting.
4315
      from->sort.io_cache= new IO_CACHE;
641.3.9 by Monty Taylor
More removal of my_malloc.
4316
      memset(from->sort.io_cache, 0, sizeof(IO_CACHE));
641.3.8 by Monty Taylor
Removed my_malloc from drizzled.
4317
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
4318
      memset(&tables, 0, sizeof(tables));
1 by brian
clean slate
4319
      tables.table= from;
4320
      tables.alias= tables.table_name= from->s->table_name.str;
4321
      tables.db= from->s->db.str;
4322
      error= 1;
4323
520.1.22 by Brian Aker
Second pass of thd cleanup
4324
      if (session->lex->select_lex.setup_ref_array(session, order_num) ||
4325
          setup_order(session, session->lex->select_lex.ref_pointer_array,
1 by brian
clean slate
4326
                      &tables, fields, all_fields, order) ||
4327
          !(sortorder= make_unireg_sortorder(order, &length, NULL)) ||
520.1.22 by Brian Aker
Second pass of thd cleanup
4328
          (from->sort.found_records= filesort(session, from, sortorder, length,
1 by brian
clean slate
4329
                                              (SQL_SELECT *) 0, HA_POS_ERROR,
4330
                                              1, &examined_rows)) ==
4331
          HA_POS_ERROR)
4332
        goto err;
4333
    }
4334
  };
4335
4336
  /* Tell handler that we have values for all columns in the to table */
4337
  to->use_all_columns();
520.1.22 by Brian Aker
Second pass of thd cleanup
4338
  init_read_record(&info, session, from, (SQL_SELECT *) 0, 1,1);
1 by brian
clean slate
4339
  if (ignore)
4340
    to->file->extra(HA_EXTRA_IGNORE_DUP_KEY);
520.1.22 by Brian Aker
Second pass of thd cleanup
4341
  session->row_count= 0;
997.5.1 by chris
Replace macros around unireg.h, store_record,restore_record,cmp_record,empty_record
4342
  to->restoreRecordAsDefault();        // Create empty record
1 by brian
clean slate
4343
  while (!(error=info.read_record(&info)))
4344
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
4345
    if (session->killed)
1 by brian
clean slate
4346
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
4347
      session->send_kill_message();
1 by brian
clean slate
4348
      error= 1;
4349
      break;
4350
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
4351
    session->row_count++;
1 by brian
clean slate
4352
    /* Return error if source table isn't empty. */
4353
    if (error_if_not_empty)
4354
    {
4355
      error= 1;
4356
      break;
4357
    }
4358
    if (to->next_number_field)
4359
    {
4360
      if (auto_increment_field_copied)
55 by brian
Update for using real bool types.
4361
        to->auto_increment_field_not_null= true;
1 by brian
clean slate
4362
      else
4363
        to->next_number_field->reset();
4364
    }
660.1.3 by Eric Herman
removed trailing whitespace with simple script:
4365
1 by brian
clean slate
4366
    for (Copy_field *copy_ptr=copy ; copy_ptr != copy_end ; copy_ptr++)
4367
    {
4368
      copy_ptr->do_copy(copy_ptr);
4369
    }
4370
    prev_insert_id= to->file->next_insert_id;
4371
    error=to->file->ha_write_row(to->record[0]);
55 by brian
Update for using real bool types.
4372
    to->auto_increment_field_not_null= false;
1 by brian
clean slate
4373
    if (error)
4374
    {
4375
      if (!ignore ||
4376
          to->file->is_fatal_error(error, HA_CHECK_DUP))
4377
      {
4378
         if (!to->file->is_fatal_error(error, HA_CHECK_DUP))
4379
         {
482 by Brian Aker
Remove uint.
4380
           uint32_t key_nr= to->file->get_dup_key(error);
1 by brian
clean slate
4381
           if ((int) key_nr >= 0)
4382
           {
4383
             const char *err_msg= ER(ER_DUP_ENTRY_WITH_KEY_NAME);
4384
             if (key_nr == 0 &&
4385
                 (to->key_info[0].key_part[0].field->flags &
4386
                  AUTO_INCREMENT_FLAG))
4387
               err_msg= ER(ER_DUP_ENTRY_AUTOINCREMENT_CASE);
4388
             to->file->print_keydup_error(key_nr, err_msg);
4389
             break;
4390
           }
4391
         }
4392
4393
	to->file->print_error(error,MYF(0));
4394
	break;
4395
      }
4396
      to->file->restore_auto_increment(prev_insert_id);
4397
      delete_count++;
4398
    }
4399
    else
4400
      found_count++;
4401
  }
4402
  end_read_record(&info);
4403
  free_io_cache(from);
4404
  delete [] copy;				// This is never 0
4405
4406
  if (to->file->ha_end_bulk_insert() && error <= 0)
4407
  {
4408
    to->file->print_error(my_errno,MYF(0));
4409
    error=1;
4410
  }
4411
  to->file->extra(HA_EXTRA_NO_IGNORE_DUP_KEY);
4412
520.1.22 by Brian Aker
Second pass of thd cleanup
4413
  if (ha_enable_transaction(session, true))
1 by brian
clean slate
4414
  {
4415
    error= 1;
4416
    goto err;
4417
  }
4418
4419
  /*
4420
    Ensure that the new table is saved properly to disk so that we
4421
    can do a rename
4422
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
4423
  if (ha_autocommit_or_rollback(session, 0))
1 by brian
clean slate
4424
    error=1;
934.2.11 by Jay Pipes
Moves end_trans(), begin_trans(), end_active_trans() out of the parser module and adds startTransaction(), endTransaction(), and endActiveTransaction() member methods of Session object.
4425
  if (! session->endActiveTransaction())
1 by brian
clean slate
4426
    error=1;
4427
4428
 err:
520.1.22 by Brian Aker
Second pass of thd cleanup
4429
  session->variables.sql_mode= save_sql_mode;
4430
  session->abort_on_warning= 0;
1 by brian
clean slate
4431
  free_io_cache(from);
4432
  *copied= found_count;
4433
  *deleted=delete_count;
4434
  to->file->ha_release_auto_increment();
520.1.22 by Brian Aker
Second pass of thd cleanup
4435
  if (to->file->ha_external_lock(session,F_UNLCK))
1 by brian
clean slate
4436
    error=1;
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
4437
  return(error > 0 ? -1 : 0);
1 by brian
clean slate
4438
}
4439
4440
4441
/*
4442
  Recreates tables by calling mysql_alter_table().
4443
4444
  SYNOPSIS
4445
    mysql_recreate_table()
520.1.22 by Brian Aker
Second pass of thd cleanup
4446
    session			Thread handler
1 by brian
clean slate
4447
    tables		Tables to recreate
4448
4449
 RETURN
4450
    Like mysql_alter_table().
4451
*/
520.1.22 by Brian Aker
Second pass of thd cleanup
4452
bool mysql_recreate_table(Session *session, TableList *table_list)
1 by brian
clean slate
4453
{
4454
  HA_CREATE_INFO create_info;
4455
  Alter_info alter_info;
4456
51.1.1 by Jay Pipes
Merged PatG's removal of various DBUG stuff with still keeping DBUG_ASSERT calls since they seem to be breaking test runs
4457
  assert(!table_list->next_global);
1 by brian
clean slate
4458
  /*
4459
    table_list->table has been closed and freed. Do not reference
4460
    uninitialized data. open_tables() could fail.
4461
  */
4462
  table_list->table= NULL;
4463
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
4464
  memset(&create_info, 0, sizeof(create_info));
1 by brian
clean slate
4465
  create_info.row_type=ROW_TYPE_NOT_USED;
4466
  create_info.default_table_charset=default_charset_info;
4467
  /* Force alter table to recreate table */
4468
  alter_info.flags= (ALTER_CHANGE_COLUMN | ALTER_RECREATE);
520.1.22 by Brian Aker
Second pass of thd cleanup
4469
  return(mysql_alter_table(session, NULL, NULL, &create_info,
1 by brian
clean slate
4470
                                table_list, &alter_info, 0,
327.2.3 by Brian Aker
Refactoring of class Table
4471
                                (order_st *) 0, 0));
1 by brian
clean slate
4472
}
4473
4474
520.1.22 by Brian Aker
Second pass of thd cleanup
4475
bool mysql_checksum_table(Session *session, TableList *tables,
1 by brian
clean slate
4476
                          HA_CHECK_OPT *check_opt)
4477
{
327.2.4 by Brian Aker
Refactoring table.h
4478
  TableList *table;
1 by brian
clean slate
4479
  List<Item> field_list;
4480
  Item *item;
520.1.22 by Brian Aker
Second pass of thd cleanup
4481
  Protocol *protocol= session->protocol;
1 by brian
clean slate
4482
4483
  field_list.push_back(item = new Item_empty_string("Table", NAME_LEN*2));
4484
  item->maybe_null= 1;
152 by Brian Aker
longlong replacement
4485
  field_list.push_back(item= new Item_int("Checksum", (int64_t) 1,
1 by brian
clean slate
4486
                                          MY_INT64_NUM_DECIMAL_DIGITS));
4487
  item->maybe_null= 1;
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
4488
  if (protocol->sendFields(&field_list,
4489
                           Protocol::SEND_NUM_ROWS | Protocol::SEND_EOF))
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
4490
    return(true);
1 by brian
clean slate
4491
4492
  /* Open one table after the other to keep lock time as short as possible. */
4493
  for (table= tables; table; table= table->next_local)
4494
  {
4495
    char table_name[NAME_LEN*2+2];
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
4496
    Table *t;
1 by brian
clean slate
4497
673.2.1 by Toru Maesaka
First pass of replacing MySQL's strxmov with libc alternatives
4498
    sprintf(table_name,"%s.%s",table->db,table->table_name);
1 by brian
clean slate
4499
1054.1.11 by Brian Aker
Remove dead lock.cc commands.
4500
    t= table->table= session->open_ltable(table, TL_READ);
520.1.22 by Brian Aker
Second pass of thd cleanup
4501
    session->clear_error();			// these errors shouldn't get client
1 by brian
clean slate
4502
971.3.19 by Eric Day
Finished first pass at Protocol cleanup, still some things to remove but they are a bit more involved.
4503
    protocol->prepareForResend();
1 by brian
clean slate
4504
    protocol->store(table_name, system_charset_info);
4505
4506
    if (!t)
4507
    {
4508
      /* Table didn't exist */
971.3.17 by Eric Day
Cleaned up int/date related store functions.
4509
      protocol->store();
520.1.22 by Brian Aker
Second pass of thd cleanup
4510
      session->clear_error();
1 by brian
clean slate
4511
    }
4512
    else
4513
    {
4514
      if (t->file->ha_table_flags() & HA_HAS_CHECKSUM &&
4515
	  !(check_opt->flags & T_EXTEND))
151 by Brian Aker
Ulonglong to uint64_t
4516
	protocol->store((uint64_t)t->file->checksum());
1 by brian
clean slate
4517
      else if (!(t->file->ha_table_flags() & HA_HAS_CHECKSUM) &&
4518
	       (check_opt->flags & T_QUICK))
971.3.17 by Eric Day
Cleaned up int/date related store functions.
4519
	protocol->store();
1 by brian
clean slate
4520
      else
4521
      {
4522
	/* calculating table's checksum */
4523
	ha_checksum crc= 0;
481 by Brian Aker
Remove all of uchar.
4524
        unsigned char null_mask=256 -  (1 << t->s->last_null_bit_pos);
1 by brian
clean slate
4525
4526
        t->use_all_columns();
4527
4528
	if (t->file->ha_rnd_init(1))
971.3.17 by Eric Day
Cleaned up int/date related store functions.
4529
	  protocol->store();
1 by brian
clean slate
4530
	else
4531
	{
4532
	  for (;;)
4533
	  {
4534
	    ha_checksum row_crc= 0;
4535
            int error= t->file->rnd_next(t->record[0]);
4536
            if (unlikely(error))
4537
            {
4538
              if (error == HA_ERR_RECORD_DELETED)
4539
                continue;
4540
              break;
4541
            }
4542
	    if (t->s->null_bytes)
4543
            {
4544
              /* fix undefined null bits */
4545
              t->record[0][t->s->null_bytes-1] |= null_mask;
4546
              if (!(t->s->db_create_options & HA_OPTION_PACK_RECORD))
4547
                t->record[0][0] |= 1;
4548
4549
	      row_crc= my_checksum(row_crc, t->record[0], t->s->null_bytes);
4550
            }
4551
482 by Brian Aker
Remove uint.
4552
	    for (uint32_t i= 0; i < t->s->fields; i++ )
1 by brian
clean slate
4553
	    {
4554
	      Field *f= t->field[i];
212.2.2 by Patrick Galbraith
Renamed FIELD_TYPE to DRIZZLE_TYPE
4555
	      if ((f->type() == DRIZZLE_TYPE_BLOB) ||
4556
                  (f->type() == DRIZZLE_TYPE_VARCHAR))
1 by brian
clean slate
4557
	      {
4558
		String tmp;
4559
		f->val_str(&tmp);
481 by Brian Aker
Remove all of uchar.
4560
		row_crc= my_checksum(row_crc, (unsigned char*) tmp.ptr(), tmp.length());
1 by brian
clean slate
4561
	      }
4562
	      else
4563
		row_crc= my_checksum(row_crc, f->ptr,
4564
				     f->pack_length());
4565
	    }
4566
4567
	    crc+= row_crc;
4568
	  }
151 by Brian Aker
Ulonglong to uint64_t
4569
	  protocol->store((uint64_t)crc);
1 by brian
clean slate
4570
          t->file->ha_rnd_end();
4571
	}
4572
      }
520.1.22 by Brian Aker
Second pass of thd cleanup
4573
      session->clear_error();
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
4574
      session->close_thread_tables();
1 by brian
clean slate
4575
      table->table=0;				// For query cache
4576
    }
4577
    if (protocol->write())
4578
      goto err;
4579
  }
4580
836 by Brian Aker
Fixed session call from function to method.
4581
  session->my_eof();
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
4582
  return(false);
1 by brian
clean slate
4583
4584
 err:
1039.1.16 by Brian Aker
A lot of little cleanups (most based off lcov)
4585
  session->close_thread_tables();			// Shouldn't be needed
1 by brian
clean slate
4586
  if (table)
4587
    table->table=0;
51.1.8 by Jay Pipes
Resolving conflicts for a few files regarding FALSE=>false
4588
  return(true);
1 by brian
clean slate
4589
}
4590
520.1.22 by Brian Aker
Second pass of thd cleanup
4591
static bool check_engine(Session *session, const char *table_name,
1 by brian
clean slate
4592
                         HA_CREATE_INFO *create_info)
4593
{
960.2.24 by Monty Taylor
Changed handlerton to StorageEngine.
4594
  StorageEngine **new_engine= &create_info->db_type;
4595
  StorageEngine *req_engine= *new_engine;
960.2.44 by Monty Taylor
Removed legacy_db_type.
4596
  if (!req_engine->is_enabled())
4597
  {
971.1.14 by Monty Taylor
Slurp around strings rather than char* for storage engine name.
4598
    string engine_name= req_engine->getName();
4599
    my_error(ER_FEATURE_DISABLED,MYF(0),
4600
             engine_name.c_str(), engine_name.c_str());
4601
             
55 by brian
Update for using real bool types.
4602
    return true;
960.2.44 by Monty Taylor
Removed legacy_db_type.
4603
  }
1 by brian
clean slate
4604
4605
  if (req_engine && req_engine != *new_engine)
4606
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
4607
    push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
4608
                       ER_WARN_USING_OTHER_HANDLER,
4609
                       ER(ER_WARN_USING_OTHER_HANDLER),
971.1.14 by Monty Taylor
Slurp around strings rather than char* for storage engine name.
4610
                       ha_resolve_storage_engine_name(*new_engine).c_str(),
1 by brian
clean slate
4611
                       table_name);
4612
  }
4613
  if (create_info->options & HA_LEX_CREATE_TMP_TABLE &&
964.1.4 by Monty Taylor
Moved flags into private area.
4614
      (*new_engine)->check_flag(HTON_BIT_TEMPORARY_NOT_SUPPORTED))
1 by brian
clean slate
4615
  {
4616
    if (create_info->used_fields & HA_CREATE_USED_ENGINE)
4617
    {
4618
      my_error(ER_ILLEGAL_HA_CREATE_OPTION, MYF(0),
971.1.14 by Monty Taylor
Slurp around strings rather than char* for storage engine name.
4619
               ha_resolve_storage_engine_name(*new_engine).c_str(),
4620
               "TEMPORARY");
1 by brian
clean slate
4621
      *new_engine= 0;
55 by brian
Update for using real bool types.
4622
      return true;
1 by brian
clean slate
4623
    }
960.2.25 by Monty Taylor
First step of hton rename.
4624
    *new_engine= myisam_engine;
1 by brian
clean slate
4625
  }
55 by brian
Update for using real bool types.
4626
  return false;
1 by brian
clean slate
4627
}