468
494
variables, but for now this is probably good enough.
469
495
Don't reset warnings when executing a stored routine.
471
if (all_tables || ! lex->is_single_level_stmt())
497
if (all_tables || !lex->is_single_level_stmt())
473
498
drizzle_reset_errors(session, 0);
476
500
status_var_increment(session->status_var.com_stat[lex->sql_command]);
478
assert(session->transaction.stmt.hasModifiedNonTransData() == false);
480
/* now we are ready to execute the statement */
481
res= lex->statement->execute();
502
assert(session->transaction.stmt.modified_non_trans_table == false);
504
switch (lex->sql_command) {
505
case SQLCOM_SHOW_STATUS:
507
system_status_var old_status_var= session->status_var;
508
session->initial_status_var= &old_status_var;
509
res= execute_sqlcom_select(session, all_tables);
510
/* Don't log SHOW STATUS commands to slow query log */
511
session->server_status&= ~(SERVER_QUERY_NO_INDEX_USED |
512
SERVER_QUERY_NO_GOOD_INDEX_USED);
514
restore status variables, as we don't want 'show status' to cause
517
pthread_mutex_lock(&LOCK_status);
518
add_diff_to_status(&global_status_var, &session->status_var,
520
session->status_var= old_status_var;
521
pthread_mutex_unlock(&LOCK_status);
524
case SQLCOM_SHOW_DATABASES:
525
case SQLCOM_SHOW_TABLES:
526
case SQLCOM_SHOW_TABLE_STATUS:
527
case SQLCOM_SHOW_OPEN_TABLES:
528
case SQLCOM_SHOW_FIELDS:
529
case SQLCOM_SHOW_KEYS:
530
case SQLCOM_SHOW_VARIABLES:
533
session->status_var.last_query_cost= 0.0;
534
res= execute_sqlcom_select(session, all_tables);
537
case SQLCOM_EMPTY_QUERY:
541
case SQLCOM_SHOW_WARNS:
543
res= mysqld_show_warnings(session, (uint32_t)
544
((1L << (uint32_t) DRIZZLE_ERROR::WARN_LEVEL_NOTE) |
545
(1L << (uint32_t) DRIZZLE_ERROR::WARN_LEVEL_WARN) |
546
(1L << (uint32_t) DRIZZLE_ERROR::WARN_LEVEL_ERROR)
550
case SQLCOM_SHOW_ERRORS:
552
res= mysqld_show_warnings(session, (uint32_t)
553
(1L << (uint32_t) DRIZZLE_ERROR::WARN_LEVEL_ERROR));
556
case SQLCOM_ASSIGN_TO_KEYCACHE:
558
assert(first_table == all_tables && first_table != 0);
559
res= mysql_assign_to_keycache(session, first_table, &lex->ident);
562
case SQLCOM_SHOW_ENGINE_STATUS:
564
res = ha_show_status(session, lex->create_info.db_type, HA_ENGINE_STATUS);
567
case SQLCOM_CREATE_TABLE:
569
/* If CREATE TABLE of non-temporary table, do implicit commit */
570
if (!(lex->create_info.options & HA_LEX_CREATE_TMP_TABLE))
572
if (! session->endActiveTransaction())
578
assert(first_table == all_tables && first_table != 0);
580
// Skip first table, which is the table we are creating
581
TableList *create_table= lex->unlink_first_table(&link_to_local);
582
TableList *select_tables= lex->query_tables;
584
Code below (especially in mysql_create_table() and select_create
585
methods) may modify HA_CREATE_INFO structure in LEX, so we have to
586
use a copy of this structure to make execution prepared statement-
587
safe. A shallow copy is enough as this code won't modify any memory
588
referenced from this structure.
590
HA_CREATE_INFO create_info(lex->create_info);
592
We need to copy alter_info for the same reasons of re-execution
593
safety, only in case of Alter_info we have to do (almost) a deep
596
Alter_info alter_info(lex->alter_info, session->mem_root);
598
if (session->is_fatal_error)
600
/* If out of memory when creating a copy of alter_info. */
602
goto end_with_restore_list;
605
if ((res= create_table_precheck(session, select_tables, create_table)))
606
goto end_with_restore_list;
608
/* Might have been updated in create_table_precheck */
609
create_info.alias= create_table->alias;
612
/* Fix names if symlinked tables */
613
if (append_file_to_dir(session, &create_info.data_file_name,
614
create_table->table_name) ||
615
append_file_to_dir(session, &create_info.index_file_name,
616
create_table->table_name))
617
goto end_with_restore_list;
620
The create-select command will open and read-lock the select table
621
and then create, open and write-lock the new table. If a global
622
read lock steps in, we get a deadlock. The write lock waits for
623
the global read lock, while the global read lock waits for the
624
select table to be closed. So we wait until the global readlock is
625
gone before starting both steps. Note that
626
wait_if_global_read_lock() sets a protection against a new global
627
read lock when it succeeds. This needs to be released by
628
start_waiting_global_read_lock(). We protect the normal CREATE
629
TABLE in the same way. That way we avoid that a new table is
630
created during a gobal read lock.
632
if (!session->locked_tables &&
633
!(need_start_waiting= !wait_if_global_read_lock(session, 0, 1)))
636
goto end_with_restore_list;
638
if (select_lex->item_list.elements) // With select
640
select_result *result;
642
select_lex->options|= SELECT_NO_UNLOCK;
643
unit->set_limit(select_lex);
645
if (!(create_info.options & HA_LEX_CREATE_TMP_TABLE))
647
lex->link_first_table_back(create_table, link_to_local);
648
create_table->create= true;
651
if (!(res= open_and_lock_tables(session, lex->query_tables)))
654
Is table which we are changing used somewhere in other parts
657
if (!(create_info.options & HA_LEX_CREATE_TMP_TABLE))
659
TableList *duplicate;
660
create_table= lex->unlink_first_table(&link_to_local);
661
if ((duplicate= unique_table(session, create_table, select_tables, 0)))
663
update_non_unique_table_error(create_table, "CREATE", duplicate);
665
goto end_with_restore_list;
670
select_create is currently not re-execution friendly and
671
needs to be created for every execution of a PS/SP.
673
if ((result= new select_create(create_table,
676
select_lex->item_list,
682
CREATE from SELECT give its Select_Lex for SELECT,
683
and item_list belong to SELECT
685
res= handle_select(session, lex, result, 0);
689
else if (!(create_info.options & HA_LEX_CREATE_TMP_TABLE))
690
create_table= lex->unlink_first_table(&link_to_local);
695
/* So that CREATE TEMPORARY TABLE gets to binlog at commit/rollback */
696
if (create_info.options & HA_LEX_CREATE_TMP_TABLE)
697
session->options|= OPTION_KEEP_LOG;
699
if (create_info.options & HA_LEX_CREATE_TABLE_LIKE)
700
res= mysql_create_like_table(session, create_table, select_tables,
704
res= mysql_create_table(session, create_table->db,
705
create_table->table_name, &create_info,
712
/* put tables back for PS rexecuting */
713
end_with_restore_list:
714
lex->link_first_table_back(create_table, link_to_local);
717
case SQLCOM_CREATE_INDEX:
719
case SQLCOM_DROP_INDEX:
721
CREATE INDEX and DROP INDEX are implemented by calling ALTER
722
TABLE with proper arguments.
724
In the future ALTER TABLE will notice that the request is to
725
only add indexes and create these one by one for the existing
726
table without having to do a full rebuild.
729
/* Prepare stack copies to be re-execution safe */
730
HA_CREATE_INFO create_info;
731
Alter_info alter_info(lex->alter_info, session->mem_root);
733
if (session->is_fatal_error) /* out of memory creating a copy of alter_info */
736
assert(first_table == all_tables && first_table != 0);
737
if (! session->endActiveTransaction())
740
memset(&create_info, 0, sizeof(create_info));
741
create_info.db_type= 0;
742
create_info.row_type= ROW_TYPE_NOT_USED;
743
create_info.default_table_charset= session->variables.collation_database;
745
res= mysql_alter_table(session, first_table->db, first_table->table_name,
746
&create_info, first_table, &alter_info,
747
0, (order_st*) 0, 0);
750
case SQLCOM_ALTER_TABLE:
751
assert(first_table == all_tables && first_table != 0);
754
Code in mysql_alter_table() may modify its HA_CREATE_INFO argument,
755
so we have to use a copy of this structure to make execution
756
prepared statement- safe. A shallow copy is enough as no memory
757
referenced from this structure will be modified.
759
HA_CREATE_INFO create_info(lex->create_info);
760
Alter_info alter_info(lex->alter_info, session->mem_root);
762
if (session->is_fatal_error) /* out of memory creating a copy of alter_info */
767
/* Must be set in the parser */
768
assert(select_lex->db);
772
memset(&tmp_table, 0, sizeof(tmp_table));
773
tmp_table.table_name= lex->name.str;
774
tmp_table.db=select_lex->db;
777
/* Don't yet allow changing of symlinks with ALTER TABLE */
778
if (create_info.data_file_name)
779
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN, 0,
780
"DATA DIRECTORY option ignored");
781
if (create_info.index_file_name)
782
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN, 0,
783
"INDEX DIRECTORY option ignored");
784
create_info.data_file_name= create_info.index_file_name= NULL;
785
/* ALTER TABLE ends previous transaction */
786
if (! session->endActiveTransaction())
789
if (!session->locked_tables &&
790
!(need_start_waiting= !wait_if_global_read_lock(session, 0, 1)))
796
res= mysql_alter_table(session, select_lex->db, lex->name.str,
800
select_lex->order_list.elements,
801
(order_st *) select_lex->order_list.first,
805
case SQLCOM_RENAME_TABLE:
807
assert(first_table == all_tables && first_table != 0);
809
for (table= first_table; table; table= table->next_local->next_local)
811
TableList old_list, new_list;
813
we do not need initialize old_list and new_list because we will
814
come table[0] and table->next[0] there
817
new_list= table->next_local[0];
820
if (! session->endActiveTransaction() || drizzle_rename_tables(session, first_table, 0))
826
case SQLCOM_SHOW_CREATE:
827
assert(first_table == all_tables && first_table != 0);
829
res= drizzled_show_create(session, first_table);
832
case SQLCOM_CHECKSUM:
834
assert(first_table == all_tables && first_table != 0);
835
res = mysql_checksum_table(session, first_table, &lex->check_opt);
840
assert(first_table == all_tables && first_table != 0);
841
res= mysql_repair_table(session, first_table, &lex->check_opt);
842
/* ! we write after unlocking the table */
844
Presumably, REPAIR and binlog writing doesn't require synchronization
846
write_bin_log(session, true, session->query, session->query_length);
847
select_lex->table_list.first= (unsigned char*) first_table;
848
lex->query_tables=all_tables;
853
assert(first_table == all_tables && first_table != 0);
854
res = mysql_check_table(session, first_table, &lex->check_opt);
855
select_lex->table_list.first= (unsigned char*) first_table;
856
lex->query_tables=all_tables;
861
assert(first_table == all_tables && first_table != 0);
862
res= mysql_analyze_table(session, first_table, &lex->check_opt);
863
/* ! we write after unlocking the table */
864
write_bin_log(session, true, session->query, session->query_length);
865
select_lex->table_list.first= (unsigned char*) first_table;
866
lex->query_tables=all_tables;
870
case SQLCOM_OPTIMIZE:
872
assert(first_table == all_tables && first_table != 0);
873
res= mysql_optimize_table(session, first_table, &lex->check_opt);
874
/* ! we write after unlocking the table */
875
write_bin_log(session, true, session->query, session->query_length);
876
select_lex->table_list.first= (unsigned char*) first_table;
877
lex->query_tables=all_tables;
881
assert(first_table == all_tables && first_table != 0);
882
if ((res= update_precheck(session, all_tables)))
884
assert(select_lex->offset_limit == 0);
885
unit->set_limit(select_lex);
886
res= mysql_update(session, all_tables,
887
select_lex->item_list,
890
select_lex->order_list.elements,
891
(order_st *) select_lex->order_list.first,
892
unit->select_limit_cnt,
893
lex->duplicates, lex->ignore);
895
case SQLCOM_UPDATE_MULTI:
897
assert(first_table == all_tables && first_table != 0);
898
if ((res= update_precheck(session, all_tables)))
901
if ((res= mysql_multi_update_prepare(session)))
904
res= mysql_multi_update(session, all_tables,
905
&select_lex->item_list,
909
lex->duplicates, lex->ignore, unit, select_lex);
915
assert(first_table == all_tables && first_table != 0);
916
if ((res= insert_precheck(session, all_tables)))
919
if (!session->locked_tables &&
920
!(need_start_waiting= !wait_if_global_read_lock(session, 0, 1)))
926
res= mysql_insert(session, all_tables, lex->field_list, lex->many_values,
927
lex->update_list, lex->value_list,
928
lex->duplicates, lex->ignore);
932
case SQLCOM_REPLACE_SELECT:
933
case SQLCOM_INSERT_SELECT:
935
select_result *sel_result;
936
assert(first_table == all_tables && first_table != 0);
937
if ((res= insert_precheck(session, all_tables)))
940
/* Fix lock for first table */
941
if (first_table->lock_type == TL_WRITE_DELAYED)
942
first_table->lock_type= TL_WRITE;
944
/* Don't unlock tables until command is written to binary log */
945
select_lex->options|= SELECT_NO_UNLOCK;
947
unit->set_limit(select_lex);
949
if (! session->locked_tables &&
950
! (need_start_waiting= ! wait_if_global_read_lock(session, 0, 1)))
956
if (!(res= open_and_lock_tables(session, all_tables)))
958
/* Skip first table, which is the table we are inserting in */
959
TableList *second_table= first_table->next_local;
960
select_lex->table_list.first= (unsigned char*) second_table;
961
select_lex->context.table_list=
962
select_lex->context.first_name_resolution_table= second_table;
963
res= mysql_insert_select_prepare(session);
964
if (!res && (sel_result= new select_insert(first_table,
972
res= handle_select(session, lex, sel_result, OPTION_SETUP_TABLES_DONE);
974
Invalidate the table in the query cache if something changed
975
after unlocking when changes become visible.
976
TODO: this is workaround. right way will be move invalidating in
977
the unlock procedure.
979
if (first_table->lock_type == TL_WRITE_CONCURRENT_INSERT &&
982
/* INSERT ... SELECT should invalidate only the very first table */
983
TableList *save_table= first_table->next_local;
984
first_table->next_local= 0;
985
first_table->next_local= save_table;
989
/* revert changes for SP */
990
select_lex->table_list.first= (unsigned char*) first_table;
995
case SQLCOM_TRUNCATE:
996
if (! session->endActiveTransaction())
1001
assert(first_table == all_tables && first_table != 0);
1003
Don't allow this within a transaction because we want to use
1006
if (session->locked_tables || session->inTransaction())
1008
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION, ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
1012
res= mysql_truncate(session, first_table, 0);
1017
assert(first_table == all_tables && first_table != 0);
1018
assert(select_lex->offset_limit == 0);
1019
unit->set_limit(select_lex);
1021
if (!session->locked_tables &&
1022
!(need_start_waiting= !wait_if_global_read_lock(session, 0, 1)))
1028
res = mysql_delete(session, all_tables, select_lex->where,
1029
&select_lex->order_list,
1030
unit->select_limit_cnt, select_lex->options,
1034
case SQLCOM_DELETE_MULTI:
1036
assert(first_table == all_tables && first_table != 0);
1037
TableList *aux_tables=
1038
(TableList *)session->lex->auxiliary_table_list.first;
1039
multi_delete *del_result;
1041
if (!session->locked_tables &&
1042
!(need_start_waiting= !wait_if_global_read_lock(session, 0, 1)))
1048
if ((res= multi_delete_precheck(session, all_tables)))
1051
/* condition will be true on SP re-excuting */
1052
if (select_lex->item_list.elements != 0)
1053
select_lex->item_list.empty();
1054
if (session->add_item_to_list(new Item_null()))
1057
session->set_proc_info("init");
1058
if ((res= open_and_lock_tables(session, all_tables)))
1061
if ((res= mysql_multi_delete_prepare(session)))
1064
if (!session->is_fatal_error &&
1065
(del_result= new multi_delete(aux_tables, lex->table_count)))
1067
res= mysql_select(session, &select_lex->ref_pointer_array,
1068
select_lex->get_table_list(),
1069
select_lex->with_wild,
1070
select_lex->item_list,
1072
0, (order_st *)NULL, (order_st *)NULL, (Item *)NULL,
1073
select_lex->options | session->options | SELECT_NO_JOIN_CACHE | SELECT_NO_UNLOCK | OPTION_SETUP_TABLES_DONE,
1074
del_result, unit, select_lex);
1075
res|= session->is_error();
1077
del_result->abort();
1084
case SQLCOM_DROP_TABLE:
1086
assert(first_table == all_tables && first_table != 0);
1087
if (!lex->drop_temporary)
1089
if (! session->endActiveTransaction())
1094
/* So that DROP TEMPORARY TABLE gets to binlog at commit/rollback */
1095
session->options|= OPTION_KEEP_LOG;
1097
/* DDL and binlog write order protected by LOCK_open */
1098
res= mysql_rm_table(session, first_table, lex->drop_if_exists, lex->drop_temporary);
1101
case SQLCOM_SHOW_PROCESSLIST:
1102
mysqld_list_processes(session, NULL, lex->verbose);
1104
case SQLCOM_SHOW_ENGINE_LOGS:
1106
res= ha_show_status(session, lex->create_info.db_type, HA_ENGINE_LOGS);
1109
case SQLCOM_CHANGE_DB:
1111
LEX_STRING db_str= { (char *) select_lex->db, strlen(select_lex->db) };
1113
if (!mysql_change_db(session, &db_str, false))
1121
assert(first_table == all_tables && first_table != 0);
1122
res= mysql_load(session, lex->exchange, first_table, lex->field_list,
1123
lex->update_list, lex->value_list, lex->duplicates, lex->ignore);
1127
case SQLCOM_SET_OPTION:
1129
List<set_var_base> *lex_var_list= &lex->var_list;
1131
if (lex->autocommit && ! session->endActiveTransaction())
1134
if (open_and_lock_tables(session, all_tables))
1136
if (!(res= sql_set_variables(session, lex_var_list)))
1143
We encountered some sort of error, but no message was sent.
1144
Send something semi-generic here since we don't know which
1145
assignment in the list caused the error.
1147
if (!session->is_error())
1148
my_error(ER_WRONG_ARGUMENTS,MYF(0),"SET");
1155
case SQLCOM_UNLOCK_TABLES:
1157
It is critical for mysqldump --single-transaction --master-data that
1158
UNLOCK TABLES does not implicitely commit a connection which has only
1159
done FLUSH TABLES WITH READ LOCK + BEGIN. If this assumption becomes
1160
false, mysqldump will not work.
1162
unlock_locked_tables(session);
1163
if (session->options & OPTION_TABLE_LOCK)
1165
(void) session->endActiveTransaction();
1166
session->options&= ~(OPTION_TABLE_LOCK);
1168
if (session->global_read_lock)
1169
unlock_global_read_lock(session);
1172
case SQLCOM_LOCK_TABLES:
1174
We try to take transactional locks if
1175
- only transactional locks are requested (lex->lock_transactional) and
1176
- no non-transactional locks exist (!session->locked_tables).
1178
if (lex->lock_transactional && !session->locked_tables)
1182
All requested locks are transactional and no non-transactional
1185
if ((rc= try_transactional_lock(session, all_tables)) == -1)
1193
Non-transactional locking has been requested or
1194
non-transactional locks exist already or transactional locks are
1195
not supported by all storage engines. Take non-transactional
1200
One or more requested locks are non-transactional and/or
1201
non-transactional locks exist or a storage engine does not support
1202
transactional locks. Check if at least one transactional lock is
1203
requested. If yes, warn about the conversion to non-transactional
1204
locks or abort in strict mode.
1206
if (check_transactional_lock(session, all_tables))
1208
unlock_locked_tables(session);
1209
/* we must end the trasaction first, regardless of anything */
1210
if (! session->endActiveTransaction())
1212
session->in_lock_tables=1;
1213
session->options|= OPTION_TABLE_LOCK;
1215
if (!(res= simple_open_n_lock_tables(session, all_tables)))
1217
session->locked_tables=session->lock;
1219
(void) set_handler_table_locks(session, all_tables, false);
1225
Need to end the current transaction, so the storage engine (InnoDB)
1226
can free its locks if LOCK TABLES locked some tables before finding
1227
that it can't lock a table in its list
1229
ha_autocommit_or_rollback(session, 1);
1230
(void) session->endActiveTransaction();
1231
session->options&= ~(OPTION_TABLE_LOCK);
1233
session->in_lock_tables=0;
1235
case SQLCOM_CREATE_DB:
1238
As mysql_create_db() may modify HA_CREATE_INFO structure passed to
1239
it, we need to use a copy of LEX::create_info to make execution
1240
prepared statement- safe.
1242
HA_CREATE_INFO create_info(lex->create_info);
1243
if (! session->endActiveTransaction())
1249
if (!(alias=session->strmake(lex->name.str, lex->name.length)) ||
1250
check_db_name(&lex->name))
1252
my_error(ER_WRONG_DB_NAME, MYF(0), lex->name.str);
1255
res= mysql_create_db(session,(lower_case_table_names == 2 ? alias :
1256
lex->name.str), &create_info, 0);
1259
case SQLCOM_DROP_DB:
1261
if (! session->endActiveTransaction())
1266
if (check_db_name(&lex->name))
1268
my_error(ER_WRONG_DB_NAME, MYF(0), lex->name.str);
1271
if (session->locked_tables || session->inTransaction())
1273
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION, ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
1276
res= mysql_rm_db(session, lex->name.str, lex->drop_if_exists, 0);
1279
case SQLCOM_ALTER_DB:
1281
LEX_STRING *db= &lex->name;
1282
HA_CREATE_INFO create_info(lex->create_info);
1283
if (check_db_name(db))
1285
my_error(ER_WRONG_DB_NAME, MYF(0), db->str);
1288
if (session->locked_tables || session->inTransaction())
1290
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION, ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
1293
res= mysql_alter_db(session, db->str, &create_info);
1296
case SQLCOM_SHOW_CREATE_DB:
1298
if (check_db_name(&lex->name))
1300
my_error(ER_WRONG_DB_NAME, MYF(0), lex->name.str);
1303
res= mysqld_show_create_db(session, lex->name.str, &lex->create_info);
1308
bool write_to_binlog;
1311
reload_cache() will tell us if we are allowed to write to the
1314
if (!reload_cache(session, lex->type, first_table, &write_to_binlog))
1317
We WANT to write and we CAN write.
1318
! we write after unlocking the table.
1321
Presumably, RESET and binlog writing doesn't require synchronization
1323
write_bin_log(session, false, session->query, session->query_length);
1331
Item *it= (Item *)lex->value_list.head();
1333
if ((!it->fixed && it->fix_fields(lex->session, &it)) || it->check_cols(1))
1335
my_message(ER_SET_CONSTANTS_ONLY, ER(ER_SET_CONSTANTS_ONLY),
1339
sql_kill(session, (ulong)it->val_int(), lex->type & ONLY_KILL_QUERY);
1343
if (session->transaction.xid_state.xa_state != XA_NOTR)
1345
my_error(ER_XAER_RMFAIL, MYF(0),
1346
xa_state_names[session->transaction.xid_state.xa_state]);
1350
Breakpoints for backup testing.
1352
if (! session->startTransaction())
1357
if (! session->endTransaction(lex->tx_release ? COMMIT_RELEASE : lex->tx_chain ? COMMIT_AND_CHAIN : COMMIT))
1361
case SQLCOM_ROLLBACK:
1362
if (! session->endTransaction(lex->tx_release ? ROLLBACK_RELEASE : lex->tx_chain ? ROLLBACK_AND_CHAIN : ROLLBACK))
1366
case SQLCOM_RELEASE_SAVEPOINT:
1369
for (sv=session->transaction.savepoints; sv; sv=sv->prev)
1371
if (my_strnncoll(system_charset_info,
1372
(unsigned char *)lex->ident.str, lex->ident.length,
1373
(unsigned char *)sv->name, sv->length) == 0)
1378
if (ha_release_savepoint(session, sv))
1379
res= true; // cannot happen
1382
session->transaction.savepoints=sv->prev;
1385
my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "SAVEPOINT", lex->ident.str);
1388
case SQLCOM_ROLLBACK_TO_SAVEPOINT:
1391
for (sv=session->transaction.savepoints; sv; sv=sv->prev)
1393
if (my_strnncoll(system_charset_info,
1394
(unsigned char *)lex->ident.str, lex->ident.length,
1395
(unsigned char *)sv->name, sv->length) == 0)
1400
if (ha_rollback_to_savepoint(session, sv))
1401
res= true; // cannot happen
1404
if ((session->options & OPTION_KEEP_LOG) || session->transaction.all.modified_non_trans_table)
1405
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN,
1406
ER_WARNING_NOT_COMPLETE_ROLLBACK,
1407
ER(ER_WARNING_NOT_COMPLETE_ROLLBACK));
1410
session->transaction.savepoints=sv;
1413
my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "SAVEPOINT", lex->ident.str);
1416
case SQLCOM_SAVEPOINT:
1417
if (!(session->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN)))
1421
SAVEPOINT **sv, *newsv;
1422
for (sv=&session->transaction.savepoints; *sv; sv=&(*sv)->prev)
1424
if (my_strnncoll(system_charset_info,
1425
(unsigned char *)lex->ident.str, lex->ident.length,
1426
(unsigned char *)(*sv)->name, (*sv)->length) == 0)
1429
if (*sv) /* old savepoint of the same name exists */
1432
ha_release_savepoint(session, *sv); // it cannot fail
1435
else if ((newsv=(SAVEPOINT *) alloc_root(&session->transaction.mem_root,
1436
savepoint_alloc_size)) == 0)
1438
my_error(ER_OUT_OF_RESOURCES, MYF(0));
1441
newsv->name=strmake_root(&session->transaction.mem_root,
1442
lex->ident.str, lex->ident.length);
1443
newsv->length=lex->ident.length;
1445
if we'll get an error here, don't add new savepoint to the list.
1446
we'll lose a little bit of memory in transaction mem_root, but it'll
1447
be free'd when transaction ends anyway
1449
if (ha_savepoint(session, newsv))
1453
newsv->prev=session->transaction.savepoints;
1454
session->transaction.savepoints=newsv;
1460
assert(0); /* Impossible */
483
1464
session->set_proc_info("query end");