~drizzle-trunk/drizzle/development

1 by brian
clean slate
1
/* Copyright (C) 2000 MySQL AB
2
3
   This program is free software; you can redistribute it and/or modify
4
   it under the terms of the GNU General Public License as published by
5
   the Free Software Foundation; version 2 of the License.
6
7
   This program is distributed in the hope that it will be useful,
8
   but WITHOUT ANY WARRANTY; without even the implied warranty of
9
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10
   GNU General Public License for more details.
11
12
   You should have received a copy of the GNU General Public License
13
   along with this program; if not, write to the Free Software
14
   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
15
16
/*
17
  Delete of records and truncate of tables.
18
19
  Multi-table deletes were introduced by Monty and Sinisa
20
*/
243.1.17 by Jay Pipes
FINAL PHASE removal of mysql_priv.h (Bye, bye my friend.)
21
#include <drizzled/server_includes.h>
22
#include <drizzled/sql_select.h>
549 by Monty Taylor
Took gettext.h out of header files.
23
#include <drizzled/error.h>
520.7.1 by Monty Taylor
Moved hash.c to drizzled.
24
#include <drizzled/probes.h>
575.4.7 by Monty Taylor
More header cleanup.
25
#include <drizzled/sql_parse.h>
1 by brian
clean slate
26
27
/**
28
  Implement DELETE SQL word.
29
30
  @note Like implementations of other DDL/DML in MySQL, this function
31
  relies on the caller to close the thread tables. This is done in the
32
  end of dispatch_command().
33
*/
34
520.1.22 by Brian Aker
Second pass of thd cleanup
35
bool mysql_delete(Session *session, TableList *table_list, COND *conds,
151 by Brian Aker
Ulonglong to uint64_t
36
                  SQL_LIST *order, ha_rows limit, uint64_t options,
1 by brian
clean slate
37
                  bool reset_auto_increment)
38
{
39
  bool          will_batch;
40
  int		error, loc_error;
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
41
  Table		*table;
1 by brian
clean slate
42
  SQL_SELECT	*select=0;
43
  READ_RECORD	info;
44
  bool          using_limit=limit != HA_POS_ERROR;
45
  bool		transactional_table, safe_update, const_cond;
46
  bool          const_cond_result;
47
  ha_rows	deleted= 0;
482 by Brian Aker
Remove uint.
48
  uint32_t usable_index= MAX_KEY;
520.1.22 by Brian Aker
Second pass of thd cleanup
49
  SELECT_LEX   *select_lex= &session->lex->select_lex;
520.1.21 by Brian Aker
THD -> Session rename
50
  Session::killed_state killed_status= Session::NOT_KILLED;
51.2.2 by Patrick Galbraith
Removed DBUGs from
51
  
1 by brian
clean slate
52
520.1.22 by Brian Aker
Second pass of thd cleanup
53
  if (open_and_lock_tables(session, table_list))
163 by Brian Aker
Merge Monty's code.
54
    return(true);
1 by brian
clean slate
55
  /* TODO look at this error */
56
  if (!(table= table_list->table))
57
  {
58
    my_error(ER_VIEW_DELETE_MERGE_VIEW, MYF(0), "", "");
163 by Brian Aker
Merge Monty's code.
59
    return(true);
1 by brian
clean slate
60
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
61
  session->set_proc_info("init");
1 by brian
clean slate
62
  table->map=1;
63
520.1.22 by Brian Aker
Second pass of thd cleanup
64
  if (mysql_prepare_delete(session, table_list, &conds))
1 by brian
clean slate
65
    goto err;
66
67
  /* check ORDER BY even if it can be ignored */
68
  if (order && order->elements)
69
  {
327.2.4 by Brian Aker
Refactoring table.h
70
    TableList   tables;
1 by brian
clean slate
71
    List<Item>   fields;
72
    List<Item>   all_fields;
73
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
74
    memset(&tables, 0, sizeof(tables));
1 by brian
clean slate
75
    tables.table = table;
76
    tables.alias = table_list->alias;
77
520.1.22 by Brian Aker
Second pass of thd cleanup
78
      if (select_lex->setup_ref_array(session, order->elements) ||
79
	  setup_order(session, select_lex->ref_pointer_array, &tables,
327.2.3 by Brian Aker
Refactoring of class Table
80
                    fields, all_fields, (order_st*) order->first))
1 by brian
clean slate
81
    {
82
      delete select;
520.1.22 by Brian Aker
Second pass of thd cleanup
83
      free_underlaid_joins(session, &session->lex->select_lex);
1 by brian
clean slate
84
      goto err;
85
    }
86
  }
87
88
  const_cond= (!conds || conds->const_item());
520.1.22 by Brian Aker
Second pass of thd cleanup
89
  safe_update=test(session->options & OPTION_SAFE_UPDATES);
1 by brian
clean slate
90
  if (safe_update && const_cond)
91
  {
92
    my_message(ER_UPDATE_WITHOUT_KEY_IN_SAFE_MODE,
93
               ER(ER_UPDATE_WITHOUT_KEY_IN_SAFE_MODE), MYF(0));
94
    goto err;
95
  }
96
520.1.22 by Brian Aker
Second pass of thd cleanup
97
  select_lex->no_error= session->lex->ignore;
1 by brian
clean slate
98
99
  const_cond_result= const_cond && (!conds || conds->val_int());
520.1.22 by Brian Aker
Second pass of thd cleanup
100
  if (session->is_error())
1 by brian
clean slate
101
  {
102
    /* Error evaluating val_int(). */
163 by Brian Aker
Merge Monty's code.
103
    return(true);
1 by brian
clean slate
104
  }
105
106
  /*
107
    Test if the user wants to delete all rows and deletion doesn't have
108
    any side-effects (because of triggers), so we can use optimized
109
    handler::delete_all_rows() method.
110
111
    We implement fast TRUNCATE for InnoDB even if triggers are
112
    present.  TRUNCATE ignores triggers.
113
114
    We can use delete_all_rows() if and only if:
115
    - We allow new functions (not using option --skip-new), and are
116
      not in safe mode (not using option --safe-mode)
117
    - There is no limit clause
118
    - The condition is constant
119
    - If there is a condition, then it it produces a non-zero value
120
    - If the current command is DELETE FROM with no where clause
121
      (i.e., not TRUNCATE) then:
122
      - We should not be binlogging this statement row-based, and
123
      - there should be no delete triggers associated with the table.
124
  */
581 by Brian Aker
Second pass through on replication row patch
125
  if (!using_limit && const_cond_result)
1 by brian
clean slate
126
  {
127
    /* Update the table->file->stats.records number */
128
    table->file->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
129
    ha_rows const maybe_deleted= table->file->stats.records;
130
    if (!(error=table->file->ha_delete_all_rows()))
131
    {
132
      error= -1;				// ok
133
      deleted= maybe_deleted;
134
      goto cleanup;
135
    }
136
    if (error != HA_ERR_WRONG_COMMAND)
137
    {
138
      table->file->print_error(error,MYF(0));
139
      error=0;
140
      goto cleanup;
141
    }
142
    /* Handler didn't support fast delete; Delete rows one by one */
143
  }
144
  if (conds)
145
  {
146
    Item::cond_result result;
520.1.22 by Brian Aker
Second pass of thd cleanup
147
    conds= remove_eq_conds(session, conds, &result);
1 by brian
clean slate
148
    if (result == Item::COND_FALSE)             // Impossible where
149
      limit= 0;
150
  }
151
152
  /* Update the table->file->stats.records number */
153
  table->file->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
154
155
  table->covering_keys.clear_all();
156
  table->quick_keys.clear_all();		// Can't use 'only index'
157
  select=make_select(table, 0, 0, conds, 0, &error);
158
  if (error)
159
    goto err;
520.1.22 by Brian Aker
Second pass of thd cleanup
160
  if ((select && select->check_quick(session, safe_update, limit)) || !limit)
1 by brian
clean slate
161
  {
162
    delete select;
520.1.22 by Brian Aker
Second pass of thd cleanup
163
    free_underlaid_joins(session, select_lex);
164
    session->row_count_func= 0;
319.1.1 by Grant Limberg
renamed all instances of MYSQL_ to DRIZZLE_
165
    DRIZZLE_DELETE_END();
520.1.22 by Brian Aker
Second pass of thd cleanup
166
    my_ok(session, (ha_rows) session->row_count_func);
1 by brian
clean slate
167
    /*
168
      We don't need to call reset_auto_increment in this case, because
169
      mysql_truncate always gives a NULL conds argument, hence we never
170
      get here.
171
    */
51.2.2 by Patrick Galbraith
Removed DBUGs from
172
    return(0);				// Nothing to delete
1 by brian
clean slate
173
  }
174
175
  /* If running in safe sql mode, don't allow updates without keys */
176
  if (table->quick_keys.is_clear_all())
177
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
178
    session->server_status|=SERVER_QUERY_NO_INDEX_USED;
1 by brian
clean slate
179
    if (safe_update && !using_limit)
180
    {
181
      delete select;
520.1.22 by Brian Aker
Second pass of thd cleanup
182
      free_underlaid_joins(session, select_lex);
1 by brian
clean slate
183
      my_message(ER_UPDATE_WITHOUT_KEY_IN_SAFE_MODE,
184
                 ER(ER_UPDATE_WITHOUT_KEY_IN_SAFE_MODE), MYF(0));
185
      goto err;
186
    }
187
  }
188
  if (options & OPTION_QUICK)
189
    (void) table->file->extra(HA_EXTRA_QUICK);
190
191
  if (order && order->elements)
192
  {
482 by Brian Aker
Remove uint.
193
    uint32_t         length= 0;
1 by brian
clean slate
194
    SORT_FIELD  *sortorder;
195
    ha_rows examined_rows;
196
    
197
    if ((!select || table->quick_keys.is_clear_all()) && limit != HA_POS_ERROR)
327.2.3 by Brian Aker
Refactoring of class Table
198
      usable_index= get_index_for_order(table, (order_st*)(order->first), limit);
1 by brian
clean slate
199
200
    if (usable_index == MAX_KEY)
201
    {
202
      table->sort.io_cache= (IO_CACHE *) my_malloc(sizeof(IO_CACHE),
203
                                                   MYF(MY_FAE | MY_ZEROFILL));
204
    
327.2.3 by Brian Aker
Refactoring of class Table
205
      if (!(sortorder= make_unireg_sortorder((order_st*) order->first,
1 by brian
clean slate
206
                                             &length, NULL)) ||
520.1.22 by Brian Aker
Second pass of thd cleanup
207
	  (table->sort.found_records = filesort(session, table, sortorder, length,
1 by brian
clean slate
208
                                                select, HA_POS_ERROR, 1,
209
                                                &examined_rows))
210
	  == HA_POS_ERROR)
211
      {
212
        delete select;
520.1.22 by Brian Aker
Second pass of thd cleanup
213
        free_underlaid_joins(session, &session->lex->select_lex);
1 by brian
clean slate
214
        goto err;
215
      }
216
      /*
217
        Filesort has already found and selected the rows we want to delete,
218
        so we don't need the where clause
219
      */
220
      delete select;
520.1.22 by Brian Aker
Second pass of thd cleanup
221
      free_underlaid_joins(session, select_lex);
1 by brian
clean slate
222
      select= 0;
223
    }
224
  }
225
226
  /* If quick select is used, initialize it before retrieving rows. */
227
  if (select && select->quick && select->quick->reset())
228
  {
229
    delete select;
520.1.22 by Brian Aker
Second pass of thd cleanup
230
    free_underlaid_joins(session, select_lex);
1 by brian
clean slate
231
    goto err;
232
  }
233
  if (usable_index==MAX_KEY)
520.1.22 by Brian Aker
Second pass of thd cleanup
234
    init_read_record(&info,session,table,select,1,1);
1 by brian
clean slate
235
  else
520.1.22 by Brian Aker
Second pass of thd cleanup
236
    init_read_record_idx(&info, session, table, 1, usable_index);
1 by brian
clean slate
237
520.1.22 by Brian Aker
Second pass of thd cleanup
238
  session->set_proc_info("updating");
1 by brian
clean slate
239
240
  will_batch= !table->file->start_bulk_delete();
241
242
243
  table->mark_columns_needed_for_delete();
244
520.1.22 by Brian Aker
Second pass of thd cleanup
245
  while (!(error=info.read_record(&info)) && !session->killed &&
246
	 ! session->is_error())
1 by brian
clean slate
247
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
248
    // session->is_error() is tested to disallow delete row on error
249
    if (!(select && select->skip_record())&& ! session->is_error() )
1 by brian
clean slate
250
    {
251
      if (!(error= table->file->ha_delete_row(table->record[0])))
252
      {
253
	deleted++;
254
	if (!--limit && using_limit)
255
	{
256
	  error= -1;
257
	  break;
258
	}
259
      }
260
      else
261
      {
262
	table->file->print_error(error,MYF(0));
263
	/*
264
	  In < 4.0.14 we set the error number to 0 here, but that
265
	  was not sensible, because then MySQL would not roll back the
266
	  failed DELETE, and also wrote it to the binlog. For MyISAM
267
	  tables a DELETE probably never should fail (?), but for
268
	  InnoDB it can fail in a FOREIGN KEY error or an
269
	  out-of-tablespace error.
270
	*/
271
 	error= 1;
272
	break;
273
      }
274
    }
275
    else
276
      table->file->unlock_row();  // Row failed selection, release lock on it
277
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
278
  killed_status= session->killed;
279
  if (killed_status != Session::NOT_KILLED || session->is_error())
1 by brian
clean slate
280
    error= 1;					// Aborted
281
  if (will_batch && (loc_error= table->file->end_bulk_delete()))
282
  {
283
    if (error != 1)
284
      table->file->print_error(loc_error,MYF(0));
285
    error=1;
286
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
287
  session->set_proc_info("end");
1 by brian
clean slate
288
  end_read_record(&info);
289
  if (options & OPTION_QUICK)
290
    (void) table->file->extra(HA_EXTRA_NORMAL);
291
292
  if (reset_auto_increment && (error < 0))
293
  {
294
    /*
295
      We're really doing a truncate and need to reset the table's
296
      auto-increment counter.
297
    */
298
    int error2= table->file->ha_reset_auto_increment(0);
299
300
    if (error2 && (error2 != HA_ERR_WRONG_COMMAND))
301
    {
302
      table->file->print_error(error2, MYF(0));
303
      error= 1;
304
    }
305
  }
306
307
cleanup:
308
309
  delete select;
310
  transactional_table= table->file->has_transactions();
311
312
  if (!transactional_table && deleted > 0)
520.1.22 by Brian Aker
Second pass of thd cleanup
313
    session->transaction.stmt.modified_non_trans_table= true;
1 by brian
clean slate
314
  
315
  /* See similar binlogging code in sql_update.cc, for comments */
520.1.22 by Brian Aker
Second pass of thd cleanup
316
  if ((error < 0) || session->transaction.stmt.modified_non_trans_table)
1 by brian
clean slate
317
  {
318
    if (mysql_bin_log.is_open())
319
    {
320
      if (error < 0)
520.1.22 by Brian Aker
Second pass of thd cleanup
321
        session->clear_error();
1 by brian
clean slate
322
      /*
323
        [binlog]: If 'handler::delete_all_rows()' was called and the
324
        storage engine does not inject the rows itself, we replicate
325
        statement-based; otherwise, 'ha_delete_row()' was used to
326
        delete specific rows which we might log row-based.
327
      */
520.1.22 by Brian Aker
Second pass of thd cleanup
328
      int log_result= session->binlog_query(Session::ROW_QUERY_TYPE,
329
                                        session->query, session->query_length,
163 by Brian Aker
Merge Monty's code.
330
                                        transactional_table, false, killed_status);
1 by brian
clean slate
331
332
      if (log_result && transactional_table)
333
      {
334
	error=1;
335
      }
336
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
337
    if (session->transaction.stmt.modified_non_trans_table)
338
      session->transaction.all.modified_non_trans_table= true;
1 by brian
clean slate
339
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
340
  assert(transactional_table || !deleted || session->transaction.stmt.modified_non_trans_table);
341
  free_underlaid_joins(session, select_lex);
1 by brian
clean slate
342
319.1.1 by Grant Limberg
renamed all instances of MYSQL_ to DRIZZLE_
343
  DRIZZLE_DELETE_END();
520.1.22 by Brian Aker
Second pass of thd cleanup
344
  if (error < 0 || (session->lex->ignore && !session->is_fatal_error))
1 by brian
clean slate
345
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
346
    session->row_count_func= deleted;
347
    my_ok(session, (ha_rows) session->row_count_func);
1 by brian
clean slate
348
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
349
  return(error >= 0 || session->is_error());
1 by brian
clean slate
350
351
err:
319.1.1 by Grant Limberg
renamed all instances of MYSQL_ to DRIZZLE_
352
  DRIZZLE_DELETE_END();
163 by Brian Aker
Merge Monty's code.
353
  return(true);
1 by brian
clean slate
354
}
355
356
357
/*
358
  Prepare items in DELETE statement
359
360
  SYNOPSIS
361
    mysql_prepare_delete()
520.1.22 by Brian Aker
Second pass of thd cleanup
362
    session			- thread handler
1 by brian
clean slate
363
    table_list		- global/local table list
364
    conds		- conditions
365
366
  RETURN VALUE
163 by Brian Aker
Merge Monty's code.
367
    false OK
368
    true  error
1 by brian
clean slate
369
*/
520.1.22 by Brian Aker
Second pass of thd cleanup
370
int mysql_prepare_delete(Session *session, TableList *table_list, Item **conds)
1 by brian
clean slate
371
{
520.1.22 by Brian Aker
Second pass of thd cleanup
372
  SELECT_LEX *select_lex= &session->lex->select_lex;
51.2.2 by Patrick Galbraith
Removed DBUGs from
373
  
1 by brian
clean slate
374
  List<Item> all_fields;
375
520.1.22 by Brian Aker
Second pass of thd cleanup
376
  session->lex->allow_sum_func= 0;
377
  if (setup_tables_and_check_access(session, &session->lex->select_lex.context,
378
                                    &session->lex->select_lex.top_join_list,
1 by brian
clean slate
379
                                    table_list, 
380
                                    &select_lex->leaf_tables, false) ||
520.1.22 by Brian Aker
Second pass of thd cleanup
381
      setup_conds(session, table_list, select_lex->leaf_tables, conds))
163 by Brian Aker
Merge Monty's code.
382
    return(true);
1 by brian
clean slate
383
  {
327.2.4 by Brian Aker
Refactoring table.h
384
    TableList *duplicate;
520.1.22 by Brian Aker
Second pass of thd cleanup
385
    if ((duplicate= unique_table(session, table_list, table_list->next_global, 0)))
1 by brian
clean slate
386
    {
387
      update_non_unique_table_error(table_list, "DELETE", duplicate);
163 by Brian Aker
Merge Monty's code.
388
      return(true);
1 by brian
clean slate
389
    }
390
  }
391
392
  if (select_lex->inner_refs_list.elements &&
520.1.22 by Brian Aker
Second pass of thd cleanup
393
    fix_inner_refs(session, all_fields, select_lex, select_lex->ref_pointer_array))
51.2.2 by Patrick Galbraith
Removed DBUGs from
394
    return(-1);
1 by brian
clean slate
395
163 by Brian Aker
Merge Monty's code.
396
  return(false);
1 by brian
clean slate
397
}
398
399
400
/***************************************************************************
401
  Delete multiple tables from join 
402
***************************************************************************/
403
520.1.22 by Brian Aker
Second pass of thd cleanup
404
#define MEM_STRIP_BUF_SIZE current_session->variables.sortbuff_size
1 by brian
clean slate
405
406
extern "C" int refpos_order_cmp(void* arg, const void *a,const void *b)
407
{
408
  handler *file= (handler*)arg;
481 by Brian Aker
Remove all of uchar.
409
  return file->cmp_ref((const unsigned char*)a, (const unsigned char*)b);
1 by brian
clean slate
410
}
411
412
/*
413
  make delete specific preparation and checks after opening tables
414
415
  SYNOPSIS
416
    mysql_multi_delete_prepare()
520.1.22 by Brian Aker
Second pass of thd cleanup
417
    session         thread handler
1 by brian
clean slate
418
419
  RETURN
163 by Brian Aker
Merge Monty's code.
420
    false OK
421
    true  Error
1 by brian
clean slate
422
*/
423
520.1.22 by Brian Aker
Second pass of thd cleanup
424
int mysql_multi_delete_prepare(Session *session)
1 by brian
clean slate
425
{
520.1.22 by Brian Aker
Second pass of thd cleanup
426
  LEX *lex= session->lex;
327.2.4 by Brian Aker
Refactoring table.h
427
  TableList *aux_tables= (TableList *)lex->auxiliary_table_list.first;
428
  TableList *target_tbl;
51.2.2 by Patrick Galbraith
Removed DBUGs from
429
  
1 by brian
clean slate
430
431
  /*
432
    setup_tables() need for VIEWs. JOIN::prepare() will not do it second
433
    time.
434
435
    lex->query_tables also point on local list of DELETE SELECT_LEX
436
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
437
  if (setup_tables_and_check_access(session, &session->lex->select_lex.context,
438
                                    &session->lex->select_lex.top_join_list,
1 by brian
clean slate
439
                                    lex->query_tables,
440
                                    &lex->select_lex.leaf_tables, false))
163 by Brian Aker
Merge Monty's code.
441
    return(true);
1 by brian
clean slate
442
443
444
  /*
445
    Multi-delete can't be constructed over-union => we always have
446
    single SELECT on top and have to check underlying SELECTs of it
447
  */
163 by Brian Aker
Merge Monty's code.
448
  lex->select_lex.exclude_from_table_unique_test= true;
1 by brian
clean slate
449
  /* Fix tables-to-be-deleted-from list to point at opened tables */
327.2.4 by Brian Aker
Refactoring table.h
450
  for (target_tbl= (TableList*) aux_tables;
1 by brian
clean slate
451
       target_tbl;
452
       target_tbl= target_tbl->next_local)
453
  {
454
    if (!(target_tbl->table= target_tbl->correspondent_table->table))
455
    {
51.2.2 by Patrick Galbraith
Removed DBUGs from
456
      assert(target_tbl->correspondent_table->merge_underlying_list &&
1 by brian
clean slate
457
                  target_tbl->correspondent_table->merge_underlying_list->
458
                  next_local);
459
      my_error(ER_VIEW_DELETE_MERGE_VIEW, MYF(0), "", "");
163 by Brian Aker
Merge Monty's code.
460
      return(true);
1 by brian
clean slate
461
    }
462
463
    /*
464
      Check that table from which we delete is not used somewhere
465
      inside subqueries/view.
466
    */
467
    {
327.2.4 by Brian Aker
Refactoring table.h
468
      TableList *duplicate;
520.1.22 by Brian Aker
Second pass of thd cleanup
469
      if ((duplicate= unique_table(session, target_tbl->correspondent_table,
1 by brian
clean slate
470
                                   lex->query_tables, 0)))
471
      {
472
        update_non_unique_table_error(target_tbl->correspondent_table,
473
                                      "DELETE", duplicate);
163 by Brian Aker
Merge Monty's code.
474
        return(true);
1 by brian
clean slate
475
      }
476
    }
477
  }
163 by Brian Aker
Merge Monty's code.
478
  return(false);
1 by brian
clean slate
479
}
480
481
482 by Brian Aker
Remove uint.
482
multi_delete::multi_delete(TableList *dt, uint32_t num_of_tables_arg)
1 by brian
clean slate
483
  : delete_tables(dt), deleted(0), found(0),
484
    num_of_tables(num_of_tables_arg), error(0),
485
    do_delete(0), transactional_tables(0), normal_tables(0), error_handled(0)
486
{
487
  tempfiles= (Unique **) sql_calloc(sizeof(Unique *) * num_of_tables);
488
}
489
490
491
int
575.1.2 by Monty Taylor
Changed a bunch of __attribute__((unused)) to removing the parameter name instead.
492
multi_delete::prepare(List<Item> &, SELECT_LEX_UNIT *u)
1 by brian
clean slate
493
{
51.2.2 by Patrick Galbraith
Removed DBUGs from
494
  
1 by brian
clean slate
495
  unit= u;
496
  do_delete= 1;
520.1.22 by Brian Aker
Second pass of thd cleanup
497
  session->set_proc_info("deleting from main table");
51.2.2 by Patrick Galbraith
Removed DBUGs from
498
  return(0);
1 by brian
clean slate
499
}
500
501
502
bool
503
multi_delete::initialize_tables(JOIN *join)
504
{
327.2.4 by Brian Aker
Refactoring table.h
505
  TableList *walk;
1 by brian
clean slate
506
  Unique **tempfiles_ptr;
51.2.2 by Patrick Galbraith
Removed DBUGs from
507
  
1 by brian
clean slate
508
520.1.22 by Brian Aker
Second pass of thd cleanup
509
  if ((session->options & OPTION_SAFE_UPDATES) && error_if_full_join(join))
51.2.2 by Patrick Galbraith
Removed DBUGs from
510
    return(1);
1 by brian
clean slate
511
512
  table_map tables_to_delete_from=0;
513
  for (walk= delete_tables; walk; walk= walk->next_local)
514
    tables_to_delete_from|= walk->table->map;
515
516
  walk= delete_tables;
517
  delete_while_scanning= 1;
518
  for (JOIN_TAB *tab=join->join_tab, *end=join->join_tab+join->tables;
519
       tab < end;
520
       tab++)
521
  {
522
    if (tab->table->map & tables_to_delete_from)
523
    {
524
      /* We are going to delete from this table */
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
525
      Table *tbl=walk->table=tab->table;
1 by brian
clean slate
526
      walk= walk->next_local;
527
      /* Don't use KEYREAD optimization on this table */
528
      tbl->no_keyread=1;
529
      /* Don't use record cache */
530
      tbl->no_cache= 1;
531
      tbl->covering_keys.clear_all();
532
      if (tbl->file->has_transactions())
533
	transactional_tables= 1;
534
      else
535
	normal_tables= 1;
536
      tbl->prepare_for_position();
537
      tbl->mark_columns_needed_for_delete();
538
    }
539
    else if ((tab->type != JT_SYSTEM && tab->type != JT_CONST) &&
540
             walk == delete_tables)
541
    {
542
      /*
543
        We are not deleting from the table we are scanning. In this
544
        case send_data() shouldn't delete any rows a we may touch
545
        the rows in the deleted table many times
546
      */
547
      delete_while_scanning= 0;
548
    }
549
  }
550
  walk= delete_tables;
551
  tempfiles_ptr= tempfiles;
552
  if (delete_while_scanning)
553
  {
554
    table_being_deleted= delete_tables;
555
    walk= walk->next_local;
556
  }
557
  for (;walk ;walk= walk->next_local)
558
  {
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
559
    Table *table=walk->table;
1 by brian
clean slate
560
    *tempfiles_ptr++= new Unique (refpos_order_cmp,
561
				  (void *) table->file,
562
				  table->file->ref_length,
563
				  MEM_STRIP_BUF_SIZE);
564
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
565
  return(session->is_fatal_error != 0);
1 by brian
clean slate
566
}
567
568
569
multi_delete::~multi_delete()
570
{
571
  for (table_being_deleted= delete_tables;
572
       table_being_deleted;
573
       table_being_deleted= table_being_deleted->next_local)
574
  {
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
575
    Table *table= table_being_deleted->table;
1 by brian
clean slate
576
    table->no_keyread=0;
577
  }
578
482 by Brian Aker
Remove uint.
579
  for (uint32_t counter= 0; counter < num_of_tables; counter++)
1 by brian
clean slate
580
  {
581
    if (tempfiles[counter])
582
      delete tempfiles[counter];
583
  }
584
}
585
586
575.1.2 by Monty Taylor
Changed a bunch of __attribute__((unused)) to removing the parameter name instead.
587
bool multi_delete::send_data(List<Item> &)
1 by brian
clean slate
588
{
589
  int secure_counter= delete_while_scanning ? -1 : 0;
327.2.4 by Brian Aker
Refactoring table.h
590
  TableList *del_table;
575.1.2 by Monty Taylor
Changed a bunch of __attribute__((unused)) to removing the parameter name instead.
591
1 by brian
clean slate
592
593
  for (del_table= delete_tables;
594
       del_table;
595
       del_table= del_table->next_local, secure_counter++)
596
  {
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
597
    Table *table= del_table->table;
1 by brian
clean slate
598
599
    /* Check if we are using outer join and we didn't find the row */
600
    if (table->status & (STATUS_NULL_ROW | STATUS_DELETED))
601
      continue;
602
603
    table->file->position(table->record[0]);
604
    found++;
605
606
    if (secure_counter < 0)
607
    {
608
      /* We are scanning the current table */
51.2.2 by Patrick Galbraith
Removed DBUGs from
609
      assert(del_table == table_being_deleted);
1 by brian
clean slate
610
      table->status|= STATUS_DELETED;
611
      if (!(error=table->file->ha_delete_row(table->record[0])))
612
      {
613
        deleted++;
614
        if (!table->file->has_transactions())
520.1.22 by Brian Aker
Second pass of thd cleanup
615
          session->transaction.stmt.modified_non_trans_table= true;
1 by brian
clean slate
616
      }
617
      else
618
      {
619
        table->file->print_error(error,MYF(0));
51.2.2 by Patrick Galbraith
Removed DBUGs from
620
        return(1);
1 by brian
clean slate
621
      }
622
    }
623
    else
624
    {
625
      error=tempfiles[secure_counter]->unique_add((char*) table->file->ref);
626
      if (error)
627
      {
628
	error= 1;                               // Fatal error
51.2.2 by Patrick Galbraith
Removed DBUGs from
629
	return(1);
1 by brian
clean slate
630
      }
631
    }
632
  }
51.2.2 by Patrick Galbraith
Removed DBUGs from
633
  return(0);
1 by brian
clean slate
634
}
635
636
482 by Brian Aker
Remove uint.
637
void multi_delete::send_error(uint32_t errcode,const char *err)
1 by brian
clean slate
638
{
51.2.2 by Patrick Galbraith
Removed DBUGs from
639
  
1 by brian
clean slate
640
641
  /* First send error what ever it is ... */
642
  my_message(errcode, err, MYF(0));
643
51.2.2 by Patrick Galbraith
Removed DBUGs from
644
  return;
1 by brian
clean slate
645
}
646
647
648
void multi_delete::abort()
649
{
51.2.2 by Patrick Galbraith
Removed DBUGs from
650
  
1 by brian
clean slate
651
652
  /* the error was handled or nothing deleted and no side effects return */
653
  if (error_handled ||
520.1.22 by Brian Aker
Second pass of thd cleanup
654
      (!session->transaction.stmt.modified_non_trans_table && !deleted))
51.2.2 by Patrick Galbraith
Removed DBUGs from
655
    return;
1 by brian
clean slate
656
657
  /*
658
    If rows from the first table only has been deleted and it is
659
    transactional, just do rollback.
660
    The same if all tables are transactional, regardless of where we are.
661
    In all other cases do attempt deletes ...
662
  */
663
  if (do_delete && normal_tables &&
664
      (table_being_deleted != delete_tables ||
665
       !table_being_deleted->table->file->has_transactions()))
666
  {
667
    /*
668
      We have to execute the recorded do_deletes() and write info into the
669
      error log
670
    */
671
    error= 1;
672
    send_eof();
51.2.2 by Patrick Galbraith
Removed DBUGs from
673
    assert(error_handled);
674
    return;
1 by brian
clean slate
675
  }
676
  
520.1.22 by Brian Aker
Second pass of thd cleanup
677
  if (session->transaction.stmt.modified_non_trans_table)
1 by brian
clean slate
678
  {
679
    /* 
680
       there is only side effects; to binlog with the error
681
    */
682
    if (mysql_bin_log.is_open())
683
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
684
      session->binlog_query(Session::ROW_QUERY_TYPE,
685
                        session->query, session->query_length,
163 by Brian Aker
Merge Monty's code.
686
                        transactional_tables, false);
1 by brian
clean slate
687
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
688
    session->transaction.all.modified_non_trans_table= true;
1 by brian
clean slate
689
  }
51.2.2 by Patrick Galbraith
Removed DBUGs from
690
  return;
1 by brian
clean slate
691
}
692
693
694
695
/*
696
  Do delete from other tables.
697
  Returns values:
698
	0 ok
699
	1 error
700
*/
701
702
int multi_delete::do_deletes()
703
{
704
  int local_error= 0, counter= 0, tmp_error;
705
  bool will_batch;
51.2.2 by Patrick Galbraith
Removed DBUGs from
706
  
707
  assert(do_delete);
1 by brian
clean slate
708
709
  do_delete= 0;                                 // Mark called
710
  if (!found)
51.2.2 by Patrick Galbraith
Removed DBUGs from
711
    return(0);
1 by brian
clean slate
712
713
  table_being_deleted= (delete_while_scanning ? delete_tables->next_local :
714
                        delete_tables);
715
 
716
  for (; table_being_deleted;
717
       table_being_deleted= table_being_deleted->next_local, counter++)
718
  { 
719
    ha_rows last_deleted= deleted;
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
720
    Table *table = table_being_deleted->table;
1 by brian
clean slate
721
    if (tempfiles[counter]->get(table))
722
    {
723
      local_error=1;
724
      break;
725
    }
726
727
    READ_RECORD	info;
520.1.22 by Brian Aker
Second pass of thd cleanup
728
    init_read_record(&info,session,table,NULL,0,1);
1 by brian
clean slate
729
    /*
730
      Ignore any rows not found in reference tables as they may already have
731
      been deleted by foreign key handling
732
    */
733
    info.ignore_not_found_rows= 1;
734
    will_batch= !table->file->start_bulk_delete();
520.1.22 by Brian Aker
Second pass of thd cleanup
735
    while (!(local_error=info.read_record(&info)) && !session->killed)
1 by brian
clean slate
736
    {
737
      if ((local_error=table->file->ha_delete_row(table->record[0])))
738
      {
739
	table->file->print_error(local_error,MYF(0));
740
	break;
741
      }
742
      deleted++;
743
    }
744
    if (will_batch && (tmp_error= table->file->end_bulk_delete()))
745
    {
746
      if (!local_error)
747
      {
748
        local_error= tmp_error;
749
        table->file->print_error(local_error,MYF(0));
750
      }
751
    }
752
    if (last_deleted != deleted && !table->file->has_transactions())
520.1.22 by Brian Aker
Second pass of thd cleanup
753
      session->transaction.stmt.modified_non_trans_table= true;
1 by brian
clean slate
754
    end_read_record(&info);
520.1.22 by Brian Aker
Second pass of thd cleanup
755
    if (session->killed && !local_error)
1 by brian
clean slate
756
      local_error= 1;
757
    if (local_error == -1)				// End of file
758
      local_error = 0;
759
  }
51.2.2 by Patrick Galbraith
Removed DBUGs from
760
  return(local_error);
1 by brian
clean slate
761
}
762
763
764
/*
765
  Send ok to the client
766
767
  return:  0 sucess
768
	   1 error
769
*/
770
771
bool multi_delete::send_eof()
772
{
520.1.21 by Brian Aker
THD -> Session rename
773
  Session::killed_state killed_status= Session::NOT_KILLED;
520.1.22 by Brian Aker
Second pass of thd cleanup
774
  session->set_proc_info("deleting from reference tables");
1 by brian
clean slate
775
776
  /* Does deletes for the last n - 1 tables, returns 0 if ok */
777
  int local_error= do_deletes();		// returns 0 if success
778
779
  /* compute a total error to know if something failed */
780
  local_error= local_error || error;
520.1.22 by Brian Aker
Second pass of thd cleanup
781
  killed_status= (local_error == 0)? Session::NOT_KILLED : session->killed;
1 by brian
clean slate
782
  /* reset used flags */
520.1.22 by Brian Aker
Second pass of thd cleanup
783
  session->set_proc_info("end");
1 by brian
clean slate
784
520.1.22 by Brian Aker
Second pass of thd cleanup
785
  if ((local_error == 0) || session->transaction.stmt.modified_non_trans_table)
1 by brian
clean slate
786
  {
787
    if (mysql_bin_log.is_open())
788
    {
789
      if (local_error == 0)
520.1.22 by Brian Aker
Second pass of thd cleanup
790
        session->clear_error();
791
      if (session->binlog_query(Session::ROW_QUERY_TYPE,
792
                            session->query, session->query_length,
163 by Brian Aker
Merge Monty's code.
793
                            transactional_tables, false, killed_status) &&
1 by brian
clean slate
794
          !normal_tables)
795
      {
796
	local_error=1;  // Log write failed: roll back the SQL statement
797
      }
798
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
799
    if (session->transaction.stmt.modified_non_trans_table)
800
      session->transaction.all.modified_non_trans_table= true;
1 by brian
clean slate
801
  }
802
  if (local_error != 0)
163 by Brian Aker
Merge Monty's code.
803
    error_handled= true; // to force early leave from ::send_error()
1 by brian
clean slate
804
805
  if (!local_error)
806
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
807
    session->row_count_func= deleted;
808
    ::my_ok(session, (ha_rows) session->row_count_func);
1 by brian
clean slate
809
  }
810
  return 0;
811
}
812
813
814
/***************************************************************************
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
815
  TRUNCATE Table
1 by brian
clean slate
816
****************************************************************************/
817
818
/*
819
  Optimize delete of all rows by doing a full generate of the table
820
  This will work even if the .ISM and .ISD tables are destroyed
821
822
  dont_send_ok should be set if:
823
  - We should always wants to generate the table (even if the table type
824
    normally can't safely do this.
825
  - We don't want an ok to be sent to the end user.
826
  - We don't want to log the truncate command
827
  - If we want to have a name lock on the table on exit without errors.
828
*/
829
520.1.22 by Brian Aker
Second pass of thd cleanup
830
bool mysql_truncate(Session *session, TableList *table_list, bool dont_send_ok)
1 by brian
clean slate
831
{
832
  HA_CREATE_INFO create_info;
833
  char path[FN_REFLEN];
327.1.5 by Brian Aker
Refactor around classes. TABLE_LIST has been factored out of table.h
834
  Table *table;
1 by brian
clean slate
835
  bool error;
482 by Brian Aker
Remove uint.
836
  uint32_t path_length;
51.2.2 by Patrick Galbraith
Removed DBUGs from
837
  
1 by brian
clean slate
838
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
839
  memset(&create_info, 0, sizeof(create_info));
1 by brian
clean slate
840
  /* If it is a temporary table, close and regenerate it */
520.1.22 by Brian Aker
Second pass of thd cleanup
841
  if (!dont_send_ok && (table= find_temporary_table(session, table_list)))
1 by brian
clean slate
842
  {
843
    handlerton *table_type= table->s->db_type();
844
    TABLE_SHARE *share= table->s;
845
    bool frm_only= (share->tmp_table == TMP_TABLE_FRM_FILE_ONLY);
846
847
    if (!ha_check_storage_engine_flag(table_type, HTON_CAN_RECREATE))
848
      goto trunc_by_del;
849
850
    table->file->info(HA_STATUS_AUTO | HA_STATUS_NO_LOCK);
851
    
520.1.22 by Brian Aker
Second pass of thd cleanup
852
    close_temporary_table(session, table, 0, 0);    // Don't free share
853
    ha_create_table(session, share->normalized_path.str,
1 by brian
clean slate
854
                    share->db.str, share->table_name.str, &create_info, 1);
855
    // We don't need to call invalidate() because this table is not in cache
520.1.22 by Brian Aker
Second pass of thd cleanup
856
    if ((error= (int) !(open_temporary_table(session, share->path.str,
1 by brian
clean slate
857
                                             share->db.str,
858
					     share->table_name.str, 1,
859
                                             OTM_OPEN))))
860
      (void) rm_temporary_table(table_type, path, frm_only);
861
    free_table_share(share);
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.
862
    free((char*) table);
1 by brian
clean slate
863
    /*
864
      If we return here we will not have logged the truncation to the bin log
865
      and we will not my_ok() to the client.
866
    */
867
    goto end;
868
  }
869
870
  path_length= build_table_filename(path, sizeof(path), table_list->db,
871
                                    table_list->table_name, reg_ext, 0);
872
873
  if (!dont_send_ok)
410 by Brian Aker
Removed legacy type, but truncate is no longer doing recreate. This should
874
    goto trunc_by_del;
1 by brian
clean slate
875
876
  // Remove the .frm extension AIX 5.2 64-bit compiler bug (BUG#16155): this
877
  // crashes, replacement works.  *(path + path_length - reg_ext_length)=
878
  // '\0';
879
  path[path_length - reg_ext_length] = 0;
398.1.10 by Monty Taylor
Actually removed VOID() this time.
880
  pthread_mutex_lock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
881
  error= ha_create_table(session, path, table_list->db, table_list->table_name,
1 by brian
clean slate
882
                         &create_info, 1);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
883
  pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
884
885
end:
886
  if (!dont_send_ok)
887
  {
888
    if (!error)
889
    {
890
      /*
891
        TRUNCATE must always be statement-based binlogged (not row-based) so
892
        we don't test current_stmt_binlog_row_based.
893
      */
520.1.22 by Brian Aker
Second pass of thd cleanup
894
      write_bin_log(session, true, session->query, session->query_length);
895
      my_ok(session);		// This should return record count
1 by brian
clean slate
896
    }
398.1.10 by Monty Taylor
Actually removed VOID() this time.
897
    pthread_mutex_lock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
898
    unlock_table_name(session, table_list);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
899
    pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
900
  }
901
  else if (error)
902
  {
398.1.10 by Monty Taylor
Actually removed VOID() this time.
903
    pthread_mutex_lock(&LOCK_open);
520.1.22 by Brian Aker
Second pass of thd cleanup
904
    unlock_table_name(session, table_list);
398.1.10 by Monty Taylor
Actually removed VOID() this time.
905
    pthread_mutex_unlock(&LOCK_open);
1 by brian
clean slate
906
  }
51.2.2 by Patrick Galbraith
Removed DBUGs from
907
  return(error);
1 by brian
clean slate
908
909
trunc_by_del:
910
  /* Probably InnoDB table */
520.1.22 by Brian Aker
Second pass of thd cleanup
911
  uint64_t save_options= session->options;
1 by brian
clean slate
912
  table_list->lock_type= TL_WRITE;
520.1.22 by Brian Aker
Second pass of thd cleanup
913
  session->options&= ~(OPTION_BEGIN | OPTION_NOT_AUTOCOMMIT);
914
  ha_enable_transaction(session, false);
915
  mysql_init_select(session->lex);
916
  error= mysql_delete(session, table_list, (COND*) 0, (SQL_LIST*) 0,
398.1.8 by Monty Taylor
Enabled -Wlong-long.
917
                      HA_POS_ERROR, 0L, true);
520.1.22 by Brian Aker
Second pass of thd cleanup
918
  ha_enable_transaction(session, true);
1 by brian
clean slate
919
  /*
163 by Brian Aker
Merge Monty's code.
920
    Safety, in case the engine ignored ha_enable_transaction(false)
520.1.22 by Brian Aker
Second pass of thd cleanup
921
    above. Also clears session->transaction.*.
1 by brian
clean slate
922
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
923
  error= ha_autocommit_or_rollback(session, error);
924
  ha_commit(session);
925
  session->options= save_options;
51.2.2 by Patrick Galbraith
Removed DBUGs from
926
  return(error);
1 by brian
clean slate
927
}