153
/* Update the table->cursor->stats.records number */
154
table->cursor->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
152
/* Update the table->file->stats.records number */
153
table->file->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
156
table->covering_keys.reset();
157
table->quick_keys.reset(); // Can't use 'only index'
158
select= optimizer::make_select(table, 0, 0, conds, 0, &error);
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);
161
if ((select && select->check_quick(session, false, limit)) || !limit)
160
if ((select && select->check_quick(session, safe_update, limit)) || !limit)
164
163
free_underlaid_joins(session, select_lex);
165
164
session->row_count_func= 0;
166
DRIZZLE_DELETE_DONE(0, 0);
168
* Resetting the Diagnostic area to prevent
171
session->main_da.reset_diagnostics_area();
172
session->my_ok((ha_rows) session->row_count_func);
165
DRIZZLE_DELETE_END();
166
my_ok(session, (ha_rows) session->row_count_func);
174
168
We don't need to call reset_auto_increment in this case, because
175
169
mysql_truncate always gives a NULL conds argument, hence we never
178
return 0; // Nothing to delete
172
return(0); // Nothing to delete
181
175
/* If running in safe sql mode, don't allow updates without keys */
182
if (table->quick_keys.none())
176
if (table->quick_keys.is_clear_all())
184
178
session->server_status|=SERVER_QUERY_NO_INDEX_USED;
179
if (safe_update && !using_limit)
182
free_underlaid_joins(session, select_lex);
183
my_message(ER_UPDATE_WITHOUT_KEY_IN_SAFE_MODE,
184
ER(ER_UPDATE_WITHOUT_KEY_IN_SAFE_MODE), MYF(0));
188
if (options & OPTION_QUICK)
189
(void) table->file->extra(HA_EXTRA_QUICK);
187
191
if (order && order->elements)
189
193
uint32_t length= 0;
190
SortField *sortorder;
194
SORT_FIELD *sortorder;
191
195
ha_rows examined_rows;
193
if ((!select || table->quick_keys.none()) && limit != HA_POS_ERROR)
194
usable_index= optimizer::get_index_for_order(table, (order_st*)(order->first), limit);
197
if ((!select || table->quick_keys.is_clear_all()) && limit != HA_POS_ERROR)
198
usable_index= get_index_for_order(table, (order_st*)(order->first), limit);
196
200
if (usable_index == MAX_KEY)
198
table->sort.io_cache= new internal::IO_CACHE;
202
table->sort.io_cache= (IO_CACHE *) my_malloc(sizeof(IO_CACHE),
203
MYF(MY_FAE | MY_ZEROFILL));
201
205
if (!(sortorder= make_unireg_sortorder((order_st*) order->first,
202
206
&length, NULL)) ||
203
207
(table->sort.found_records = filesort(session, table, sortorder, length,
378
413
/***************************************************************************
414
Delete multiple tables from join
415
***************************************************************************/
417
#define MEM_STRIP_BUF_SIZE current_session->variables.sortbuff_size
419
extern "C" int refpos_order_cmp(void* arg, const void *a,const void *b)
421
handler *file= (handler*)arg;
422
return file->cmp_ref((const unsigned char*)a, (const unsigned char*)b);
426
make delete specific preparation and checks after opening tables
429
mysql_multi_delete_prepare()
430
session thread handler
437
int mysql_multi_delete_prepare(Session *session)
439
LEX *lex= session->lex;
440
TableList *aux_tables= (TableList *)lex->auxiliary_table_list.first;
441
TableList *target_tbl;
445
setup_tables() need for VIEWs. JOIN::prepare() will not do it second
448
lex->query_tables also point on local list of DELETE SELECT_LEX
450
if (setup_tables_and_check_access(session, &session->lex->select_lex.context,
451
&session->lex->select_lex.top_join_list,
453
&lex->select_lex.leaf_tables, false))
458
Multi-delete can't be constructed over-union => we always have
459
single SELECT on top and have to check underlying SELECTs of it
461
lex->select_lex.exclude_from_table_unique_test= true;
462
/* Fix tables-to-be-deleted-from list to point at opened tables */
463
for (target_tbl= (TableList*) aux_tables;
465
target_tbl= target_tbl->next_local)
467
if (!(target_tbl->table= target_tbl->correspondent_table->table))
469
assert(target_tbl->correspondent_table->merge_underlying_list &&
470
target_tbl->correspondent_table->merge_underlying_list->
472
my_error(ER_VIEW_DELETE_MERGE_VIEW, MYF(0), "", "");
477
Check that table from which we delete is not used somewhere
478
inside subqueries/view.
481
TableList *duplicate;
482
if ((duplicate= unique_table(session, target_tbl->correspondent_table,
483
lex->query_tables, 0)))
485
update_non_unique_table_error(target_tbl->correspondent_table,
486
"DELETE", duplicate);
495
multi_delete::multi_delete(TableList *dt, uint32_t num_of_tables_arg)
496
: delete_tables(dt), deleted(0), found(0),
497
num_of_tables(num_of_tables_arg), error(0),
498
do_delete(0), transactional_tables(0), normal_tables(0), error_handled(0)
500
tempfiles= (Unique **) sql_calloc(sizeof(Unique *) * num_of_tables);
505
multi_delete::prepare(List<Item> &values __attribute__((unused)),
511
session->set_proc_info("deleting from main table");
517
multi_delete::initialize_tables(JOIN *join)
520
Unique **tempfiles_ptr;
523
if ((session->options & OPTION_SAFE_UPDATES) && error_if_full_join(join))
526
table_map tables_to_delete_from=0;
527
for (walk= delete_tables; walk; walk= walk->next_local)
528
tables_to_delete_from|= walk->table->map;
531
delete_while_scanning= 1;
532
for (JOIN_TAB *tab=join->join_tab, *end=join->join_tab+join->tables;
536
if (tab->table->map & tables_to_delete_from)
538
/* We are going to delete from this table */
539
Table *tbl=walk->table=tab->table;
540
walk= walk->next_local;
541
/* Don't use KEYREAD optimization on this table */
543
/* Don't use record cache */
545
tbl->covering_keys.clear_all();
546
if (tbl->file->has_transactions())
547
transactional_tables= 1;
550
tbl->prepare_for_position();
551
tbl->mark_columns_needed_for_delete();
553
else if ((tab->type != JT_SYSTEM && tab->type != JT_CONST) &&
554
walk == delete_tables)
557
We are not deleting from the table we are scanning. In this
558
case send_data() shouldn't delete any rows a we may touch
559
the rows in the deleted table many times
561
delete_while_scanning= 0;
565
tempfiles_ptr= tempfiles;
566
if (delete_while_scanning)
568
table_being_deleted= delete_tables;
569
walk= walk->next_local;
571
for (;walk ;walk= walk->next_local)
573
Table *table=walk->table;
574
*tempfiles_ptr++= new Unique (refpos_order_cmp,
575
(void *) table->file,
576
table->file->ref_length,
579
return(session->is_fatal_error != 0);
583
multi_delete::~multi_delete()
585
for (table_being_deleted= delete_tables;
587
table_being_deleted= table_being_deleted->next_local)
589
Table *table= table_being_deleted->table;
593
for (uint32_t counter= 0; counter < num_of_tables; counter++)
595
if (tempfiles[counter])
596
delete tempfiles[counter];
601
bool multi_delete::send_data(List<Item> &values __attribute__((unused)))
603
int secure_counter= delete_while_scanning ? -1 : 0;
604
TableList *del_table;
607
for (del_table= delete_tables;
609
del_table= del_table->next_local, secure_counter++)
611
Table *table= del_table->table;
613
/* Check if we are using outer join and we didn't find the row */
614
if (table->status & (STATUS_NULL_ROW | STATUS_DELETED))
617
table->file->position(table->record[0]);
620
if (secure_counter < 0)
622
/* We are scanning the current table */
623
assert(del_table == table_being_deleted);
624
table->status|= STATUS_DELETED;
625
if (!(error=table->file->ha_delete_row(table->record[0])))
628
if (!table->file->has_transactions())
629
session->transaction.stmt.modified_non_trans_table= true;
633
table->file->print_error(error,MYF(0));
639
error=tempfiles[secure_counter]->unique_add((char*) table->file->ref);
642
error= 1; // Fatal error
651
void multi_delete::send_error(uint32_t errcode,const char *err)
655
/* First send error what ever it is ... */
656
my_message(errcode, err, MYF(0));
662
void multi_delete::abort()
666
/* the error was handled or nothing deleted and no side effects return */
668
(!session->transaction.stmt.modified_non_trans_table && !deleted))
672
If rows from the first table only has been deleted and it is
673
transactional, just do rollback.
674
The same if all tables are transactional, regardless of where we are.
675
In all other cases do attempt deletes ...
677
if (do_delete && normal_tables &&
678
(table_being_deleted != delete_tables ||
679
!table_being_deleted->table->file->has_transactions()))
682
We have to execute the recorded do_deletes() and write info into the
687
assert(error_handled);
691
if (session->transaction.stmt.modified_non_trans_table)
694
there is only side effects; to binlog with the error
696
if (mysql_bin_log.is_open())
698
session->binlog_query(Session::ROW_QUERY_TYPE,
699
session->query, session->query_length,
700
transactional_tables, false);
702
session->transaction.all.modified_non_trans_table= true;
710
Do delete from other tables.
716
int multi_delete::do_deletes()
718
int local_error= 0, counter= 0, tmp_error;
723
do_delete= 0; // Mark called
727
table_being_deleted= (delete_while_scanning ? delete_tables->next_local :
730
for (; table_being_deleted;
731
table_being_deleted= table_being_deleted->next_local, counter++)
733
ha_rows last_deleted= deleted;
734
Table *table = table_being_deleted->table;
735
if (tempfiles[counter]->get(table))
742
init_read_record(&info,session,table,NULL,0,1);
744
Ignore any rows not found in reference tables as they may already have
745
been deleted by foreign key handling
747
info.ignore_not_found_rows= 1;
748
will_batch= !table->file->start_bulk_delete();
749
while (!(local_error=info.read_record(&info)) && !session->killed)
751
if ((local_error=table->file->ha_delete_row(table->record[0])))
753
table->file->print_error(local_error,MYF(0));
758
if (will_batch && (tmp_error= table->file->end_bulk_delete()))
762
local_error= tmp_error;
763
table->file->print_error(local_error,MYF(0));
766
if (last_deleted != deleted && !table->file->has_transactions())
767
session->transaction.stmt.modified_non_trans_table= true;
768
end_read_record(&info);
769
if (session->killed && !local_error)
771
if (local_error == -1) // End of file
779
Send ok to the client
785
bool multi_delete::send_eof()
787
Session::killed_state killed_status= Session::NOT_KILLED;
788
session->set_proc_info("deleting from reference tables");
790
/* Does deletes for the last n - 1 tables, returns 0 if ok */
791
int local_error= do_deletes(); // returns 0 if success
793
/* compute a total error to know if something failed */
794
local_error= local_error || error;
795
killed_status= (local_error == 0)? Session::NOT_KILLED : session->killed;
796
/* reset used flags */
797
session->set_proc_info("end");
799
if ((local_error == 0) || session->transaction.stmt.modified_non_trans_table)
801
if (mysql_bin_log.is_open())
803
if (local_error == 0)
804
session->clear_error();
805
if (session->binlog_query(Session::ROW_QUERY_TYPE,
806
session->query, session->query_length,
807
transactional_tables, false, killed_status) &&
810
local_error=1; // Log write failed: roll back the SQL statement
813
if (session->transaction.stmt.modified_non_trans_table)
814
session->transaction.all.modified_non_trans_table= true;
816
if (local_error != 0)
817
error_handled= true; // to force early leave from ::send_error()
821
session->row_count_func= deleted;
822
::my_ok(session, (ha_rows) session->row_count_func);
828
/***************************************************************************
380
830
****************************************************************************/
383
833
Optimize delete of all rows by doing a full generate of the table
384
834
This will work even if the .ISM and .ISD tables are destroyed
836
dont_send_ok should be set if:
837
- We should always wants to generate the table (even if the table type
838
normally can't safely do this.
839
- We don't want an ok to be sent to the end user.
840
- We don't want to log the truncate command
841
- If we want to have a name lock on the table on exit without errors.
387
bool mysql_truncate(Session& session, TableList *table_list)
844
bool mysql_truncate(Session *session, TableList *table_list, bool dont_send_ok)
846
HA_CREATE_INFO create_info;
847
char path[FN_REFLEN];
390
TransactionServices &transaction_services= TransactionServices::singleton();
392
uint64_t save_options= session.options;
850
uint32_t path_length;
853
memset(&create_info, 0, sizeof(create_info));
854
/* If it is a temporary table, close and regenerate it */
855
if (!dont_send_ok && (table= find_temporary_table(session, table_list)))
857
handlerton *table_type= table->s->db_type();
858
TABLE_SHARE *share= table->s;
859
bool frm_only= (share->tmp_table == TMP_TABLE_FRM_FILE_ONLY);
861
if (!ha_check_storage_engine_flag(table_type, HTON_CAN_RECREATE))
864
table->file->info(HA_STATUS_AUTO | HA_STATUS_NO_LOCK);
866
close_temporary_table(session, table, 0, 0); // Don't free share
867
ha_create_table(session, share->normalized_path.str,
868
share->db.str, share->table_name.str, &create_info, 1);
869
// We don't need to call invalidate() because this table is not in cache
870
if ((error= (int) !(open_temporary_table(session, share->path.str,
872
share->table_name.str, 1,
874
(void) rm_temporary_table(table_type, path, frm_only);
875
free_table_share(share);
878
If we return here we will not have logged the truncation to the bin log
879
and we will not my_ok() to the client.
884
path_length= build_table_filename(path, sizeof(path), table_list->db,
885
table_list->table_name, reg_ext, 0);
890
// Remove the .frm extension AIX 5.2 64-bit compiler bug (BUG#16155): this
891
// crashes, replacement works. *(path + path_length - reg_ext_length)=
893
path[path_length - reg_ext_length] = 0;
894
pthread_mutex_lock(&LOCK_open);
895
error= ha_create_table(session, path, table_list->db, table_list->table_name,
897
pthread_mutex_unlock(&LOCK_open);
905
TRUNCATE must always be statement-based binlogged (not row-based) so
906
we don't test current_stmt_binlog_row_based.
908
write_bin_log(session, true, session->query, session->query_length);
909
my_ok(session); // This should return record count
911
pthread_mutex_lock(&LOCK_open);
912
unlock_table_name(session, table_list);
913
pthread_mutex_unlock(&LOCK_open);
917
pthread_mutex_lock(&LOCK_open);
918
unlock_table_name(session, table_list);
919
pthread_mutex_unlock(&LOCK_open);
924
/* Probably InnoDB table */
925
uint64_t save_options= session->options;
393
926
table_list->lock_type= TL_WRITE;
394
session.options&= ~(OPTION_BEGIN | OPTION_NOT_AUTOCOMMIT);
395
mysql_init_select(session.lex);
396
error= mysql_delete(&session, table_list, (COND*) 0, (SQL_LIST*) 0,
927
session->options&= ~(OPTION_BEGIN | OPTION_NOT_AUTOCOMMIT);
928
ha_enable_transaction(session, false);
929
mysql_init_select(session->lex);
930
bool save_binlog_row_based= session->current_stmt_binlog_row_based;
931
session->clear_current_stmt_binlog_row_based();
932
error= mysql_delete(session, table_list, (COND*) 0, (SQL_LIST*) 0,
397
933
HA_POS_ERROR, 0L, true);
934
ha_enable_transaction(session, true);
399
936
Safety, in case the engine ignored ha_enable_transaction(false)
400
937
above. Also clears session->transaction.*.
402
error= transaction_services.autocommitOrRollback(&session, error);
403
session.options= save_options;
939
error= ha_autocommit_or_rollback(session, error);
941
session->options= save_options;
942
session->current_stmt_binlog_row_based= save_binlog_row_based;
408
} /* namespace drizzled */