12
12
You should have received a copy of the GNU General Public License
13
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
16
16
#include "config.h"
18
18
#define DRIZZLE_LEX 1
20
#include "drizzled/item/num.h"
21
#include "drizzled/abort_exception.h"
22
20
#include <drizzled/my_hash.h>
23
21
#include <drizzled/error.h>
24
22
#include <drizzled/nested_join.h>
34
32
#include <drizzled/item/cmpfunc.h>
35
33
#include <drizzled/item/null.h>
36
34
#include <drizzled/session.h>
37
#include <drizzled/session/cache.h>
38
35
#include <drizzled/sql_load.h>
39
36
#include <drizzled/lock.h>
40
37
#include <drizzled/select_send.h>
42
39
#include <drizzled/statement.h>
43
40
#include <drizzled/statement/alter_table.h>
44
41
#include "drizzled/probes.h"
42
#include "drizzled/session_list.h"
45
43
#include "drizzled/global_charset_info.h"
47
45
#include "drizzled/plugin/logging.h"
48
46
#include "drizzled/plugin/query_rewrite.h"
49
#include "drizzled/plugin/query_cache.h"
50
47
#include "drizzled/plugin/authorization.h"
51
48
#include "drizzled/optimizer/explain_plan.h"
52
49
#include "drizzled/pthread_globals.h"
53
#include "drizzled/plugin/event_observer.h"
54
#include "drizzled/visibility.h"
56
51
#include <limits.h>
59
54
#include <algorithm>
60
#include <boost/date_time.hpp>
61
56
#include "drizzled/internal/my_sys.h"
63
58
using namespace std;
71
66
bool my_yyoverflow(short **a, YYSTYPE **b, ulong *yystacksize);
72
67
static bool parse_sql(Session *session, Lex_input_stream *lip);
73
void parse(Session *session, const char *inBuf, uint32_t length);
68
void mysql_parse(Session *session, const char *inBuf, uint32_t length);
76
71
@defgroup Runtime_Environment Runtime Environment
80
75
extern size_t my_thread_stack_size;
81
76
extern const CHARSET_INFO *character_set_filesystem;
86
static const std::string command_name[COM_END+1]={
94
"Error" // Last command number
78
const LEX_STRING command_name[COM_END+1]={
79
{ C_STRING_WITH_LEN("Sleep") },
80
{ C_STRING_WITH_LEN("Quit") },
81
{ C_STRING_WITH_LEN("Init DB") },
82
{ C_STRING_WITH_LEN("Query") },
83
{ C_STRING_WITH_LEN("Shutdown") },
84
{ C_STRING_WITH_LEN("Connect") },
85
{ C_STRING_WITH_LEN("Ping") },
86
{ C_STRING_WITH_LEN("Error") } // Last command number
99
89
const char *xa_state_names[]={
100
90
"NON-EXISTING", "ACTIVE", "IDLE", "PREPARED"
184
169
Query_id &query_id= Query_id::get_query_id();
186
DRIZZLE_COMMAND_START(session->thread_id, command);
171
DRIZZLE_COMMAND_START(session->thread_id,
188
174
session->command= command;
189
175
session->lex->sql_command= SQLCOM_END; /* to avoid confusing VIEW detectors */
197
183
/* Increase id and count all other statements. */
199
session->status_var.questions++;
185
statistic_increment(session->status_var.questions, &LOCK_status);
203
/* @todo set session->lex->sql_command to SQLCOM_END here */
189
/* TODO: set session->lex->sql_command to SQLCOM_END here */
205
191
plugin::Logging::preDo(session);
206
if (unlikely(plugin::EventObserver::beforeStatement(*session)))
208
// We should do something about an error...
211
193
session->server_status&=
212
194
~(SERVER_QUERY_NO_INDEX_USED | SERVER_QUERY_NO_GOOD_INDEX_USED);
222
204
string tmp(packet, packet_length);
224
identifier::Schema identifier(tmp);
206
SchemaIdentifier identifier(tmp);
226
if (not change_db(session, identifier))
208
if (not mysql_change_db(session, identifier))
228
210
session->my_ok();
234
if (not session->readAndStoreQuery(packet, packet_length))
216
if (! session->readAndStoreQuery(packet, packet_length))
235
217
break; // fatal error is set
236
DRIZZLE_QUERY_START(session->getQueryString()->c_str(),
218
DRIZZLE_QUERY_START(session->query.c_str(),
237
219
session->thread_id,
238
const_cast<const char *>(session->schema()->c_str()));
220
const_cast<const char *>(session->db.empty() ? "" : session->db.c_str()));
240
parse(session, session->getQueryString()->c_str(), session->getQueryString()->length());
222
plugin::QueryRewriter::rewriteQuery(session->db, session->query);
223
mysql_parse(session, session->query.c_str(), session->query.length());
270
253
/* If commit fails, we should be able to reset the OK status. */
271
254
session->main_da.can_overwrite_status= true;
272
255
TransactionServices &transaction_services= TransactionServices::singleton();
273
transaction_services.autocommitOrRollback(*session, session->is_error());
256
transaction_services.autocommitOrRollback(session, session->is_error());
274
257
session->main_da.can_overwrite_status= false;
276
259
session->transaction.stmt.reset();
282
265
if (! session->main_da.is_set())
283
266
session->send_kill_message();
285
if (session->getKilled() == Session::KILL_QUERY || session->getKilled() == Session::KILL_BAD_DATA)
268
if (session->killed == Session::KILL_QUERY || session->killed == Session::KILL_BAD_DATA)
287
session->setKilled(Session::NOT_KILLED);
288
session->setAbort(false);
270
session->killed= Session::NOT_KILLED;
271
session->mysys_var->abort= 0;
291
274
/* Can not be true, but do not take chances in production. */
296
279
case Diagnostics_area::DA_ERROR:
297
280
/* The query failed, send error to log and abort bootstrap. */
298
session->getClient()->sendError(session->main_da.sql_errno(),
281
session->client->sendError(session->main_da.sql_errno(),
299
282
session->main_da.message());
302
285
case Diagnostics_area::DA_EOF:
303
session->getClient()->sendEOF();
286
session->client->sendEOF();
306
289
case Diagnostics_area::DA_OK:
307
session->getClient()->sendOK();
290
session->client->sendOK();
310
293
case Diagnostics_area::DA_DISABLED:
323
306
session->close_thread_tables();
325
308
plugin::Logging::postDo(session);
326
if (unlikely(plugin::EventObserver::afterStatement(*session)))
328
// We should do something about an error...
331
310
/* Store temp state for processlist */
332
311
session->set_proc_info("cleaning up");
333
312
session->command= COM_SLEEP;
334
session->resetQueryString();
313
memset(session->process_list_info, 0, PROCESS_LIST_WIDTH);
314
session->query.clear();
336
316
session->set_proc_info(NULL);
337
317
session->mem_root->free_root(MYF(memory::KEEP_PREALLOC));
387
366
session->make_lex_string(&table, schema_table_name, false);
389
368
if (! sel->add_table_to_list(session, new Table_ident(db, table),
390
NULL, table_options, TL_READ))
450
430
Select_Lex *select_lex= &lex->select_lex;
451
431
/* list of all tables in query */
452
432
TableList *all_tables;
433
/* A peek into the query string */
434
size_t proc_info_len= session->query.length() > PROCESS_LIST_WIDTH ?
435
PROCESS_LIST_WIDTH : session->query.length();
437
memcpy(session->process_list_info, session->query.c_str(), proc_info_len);
438
session->process_list_info[proc_info_len]= '\0';
455
441
In many cases first table of main Select_Lex have special meaning =>
489
475
assert(session->transaction.stmt.hasModifiedNonTransData() == false);
491
if (! (session->server_status & SERVER_STATUS_AUTOCOMMIT)
492
&& ! session->inTransaction()
493
&& lex->statement->isTransactional())
495
if (session->startTransaction() == false)
497
my_error(drizzled::ER_UNKNOWN_ERROR, MYF(0));
502
477
/* now we are ready to execute the statement */
503
478
res= lex->statement->execute();
504
480
session->set_proc_info("query end");
506
483
The return value for ROW_COUNT() is "implementation dependent" if the
507
484
statement is not DELETE, INSERT or UPDATE, but -1 is what JDBC and ODBC
527
505
param->select_limit=
528
506
new Item_int((uint64_t) session->variables.select_limit);
532
&& ! (session->server_status & SERVER_STATUS_AUTOCOMMIT)
533
&& ! session->inTransaction()
534
&& ! lex->statement->isShow())
536
if (session->startTransaction() == false)
538
my_error(drizzled::ER_UNKNOWN_ERROR, MYF(0));
543
508
if (not (res= session->openTablesLock(all_tables)))
545
510
if (lex->describe)
576
541
if (!result && !(result= new select_send()))
579
/* Init the Query Cache plugin */
580
plugin::QueryCache::prepareResultset(session);
581
543
res= handle_select(session, lex, result, 0);
582
/* Send the Resultset to the cache */
583
plugin::QueryCache::setResultset(session);
585
544
if (result != lex->result)
641
new_select(LEX *lex, bool move_down)
600
mysql_new_select(LEX *lex, bool move_down)
643
602
Select_Lex *select_lex;
644
603
Session *session= lex->session;
646
605
if (!(select_lex= new (session->mem_root) Select_Lex()))
649
607
select_lex->select_number= ++session->select_number;
650
608
select_lex->parent_lex= lex; /* Used in init_query. */
651
609
select_lex->init_query();
652
610
select_lex->init_select();
653
611
lex->nest_level++;
655
612
if (lex->nest_level > (int) MAX_SELECT_NESTING)
657
614
my_error(ER_TOO_HIGH_LEVEL_OF_NESTING_FOR_SELECT,MYF(0),MAX_SELECT_NESTING);
661
617
select_lex->nest_level= lex->nest_level;
685
641
if (lex->current_select->order_list.first && !lex->current_select->braces)
687
643
my_error(ER_WRONG_USAGE, MYF(0), "UNION", "order_st BY");
691
646
select_lex->include_neighbour(lex->current_select);
692
647
Select_Lex_Unit *unit= select_lex->master_unit();
694
if (not unit->fake_select_lex && unit->add_fake_select_lex(lex->session))
648
if (!unit->fake_select_lex && unit->add_fake_select_lex(lex->session))
697
650
select_lex->context.outer_context=
698
651
unit->first_select()->context.outer_context;
720
672
@param var_name Variable name
723
void create_select_for_variable(Session *session, const char *var_name)
675
void create_select_for_variable(const char *var_name)
726
679
LEX_STRING tmp, null_lex_string;
728
681
char buff[MAX_SYS_VAR_LENGTH*2+4+8];
684
session= current_session;
731
685
lex= session->lex;
686
mysql_init_select(lex);
733
687
lex->sql_command= SQLCOM_SELECT;
734
688
tmp.str= (char*) var_name;
735
689
tmp.length=strlen(var_name);
755
710
@param length Length of the query text
758
void parse(Session *session, const char *inBuf, uint32_t length)
713
void mysql_parse(Session *session, const char *inBuf, uint32_t length)
760
session->lex->start(session);
762
716
session->reset_for_next_command();
763
/* Check if the Query is Cached if and return true if yes
764
* TODO the plugin has to make sure that the query is cacheble
765
* by setting the query_safe_cache param to TRUE
768
if (plugin::QueryCache::isCached(session))
770
res= plugin::QueryCache::sendCachedResultset(session);
776
718
LEX *lex= session->lex;
777
720
Lex_input_stream lip(session, inBuf, length);
778
722
bool err= parse_sql(session, &lip);
782
if (not session->is_error())
727
if (! session->is_error())
784
DRIZZLE_QUERY_EXEC_START(session->getQueryString()->c_str(),
729
DRIZZLE_QUERY_EXEC_START(session->query.c_str(),
785
730
session->thread_id,
786
const_cast<const char *>(session->schema()->c_str()));
731
const_cast<const char *>(session->db.empty() ? "" : session->db.c_str()));
787
732
// Implement Views here --Brian
788
734
/* Actually execute the query */
791
execute_command(session);
736
mysql_execute_command(session);
795
740
// Just try to catch any random failures that could have come
796
741
// during execution.
799
743
DRIZZLE_QUERY_EXEC_DONE(0);
805
749
assert(session->is_error());
807
752
lex->unit.cleanup();
808
753
session->set_proc_info("freeing items");
809
754
session->end_statement();
810
755
session->cleanup_after_query();
811
session->set_end_timer();
869
813
if (default_value->type() == Item::FUNC_ITEM &&
870
814
!(((Item_func*)default_value)->functype() == Item_func::NOW_FUNC &&
871
(type == DRIZZLE_TYPE_TIMESTAMP or type == DRIZZLE_TYPE_MICROTIME)))
815
type == DRIZZLE_TYPE_TIMESTAMP))
873
817
my_error(ER_INVALID_DEFAULT, MYF(0), field_name->str);
876
820
else if (default_value->type() == Item::NULL_ITEM)
878
822
default_value= 0;
879
if ((type_modifier & (NOT_NULL_FLAG | AUTO_INCREMENT_FLAG)) == NOT_NULL_FLAG)
823
if ((type_modifier & (NOT_NULL_FLAG | AUTO_INCREMENT_FLAG)) ==
881
826
my_error(ER_INVALID_DEFAULT, MYF(0), field_name->str);
892
if (on_update_value && (type != DRIZZLE_TYPE_TIMESTAMP and type != DRIZZLE_TYPE_MICROTIME))
837
if (on_update_value && type != DRIZZLE_TYPE_TIMESTAMP)
894
839
my_error(ER_INVALID_ON_UPDATE, MYF(0), field_name->str);
856
/** Store position for column in ALTER TABLE .. ADD column. */
858
void store_position_for_column(const char *name)
860
current_session->lex->last_field->after=const_cast<char*> (name);
912
864
Add a table to list of used tables.
930
882
TableList *Select_Lex::add_table_to_list(Session *session,
933
const bitset<NUM_OF_TABLE_OPTIONS>& table_options,
934
thr_lock_type lock_type,
935
List<Index_hint> *index_hints_arg,
885
uint32_t table_options,
886
thr_lock_type lock_type,
887
List<Index_hint> *index_hints_arg,
890
register TableList *ptr;
939
891
TableList *previous_table_ref; /* The table preceding the current one. */
941
893
LEX *lex= session->lex;
944
896
return NULL; // End of memory
945
897
alias_str= alias ? alias->str : table->table.str;
946
if (! table_options.test(TL_OPTION_ALIAS) &&
898
if (!test(table_options & TL_OPTION_ALIAS) &&
947
899
check_table_name(table->table.str, table->table.length))
949
901
my_error(ER_WRONG_TABLE_NAME, MYF(0), table->table.str);
955
907
my_casedn_str(files_charset_info, table->db.str);
957
identifier::Schema schema_identifier(string(table->db.str));
909
SchemaIdentifier schema_identifier(string(table->db.str));
958
910
if (not check_db_name(session, schema_identifier))
971
923
ER(ER_DERIVED_MUST_HAVE_ALIAS), MYF(0));
974
if (!(alias_str= (char*) session->getMemRoot()->duplicate(alias_str,table->table.length+1)))
926
if (!(alias_str= (char*) session->memdup(alias_str,table->table.length+1)))
977
929
if (!(ptr = (TableList *) session->calloc(sizeof(TableList))))
980
931
if (table->db.str)
982
933
ptr->setIsFqtn(true);
983
ptr->setSchemaName(table->db.str);
934
ptr->db= table->db.str;
984
935
ptr->db_length= table->db.length;
986
else if (lex->copy_db_to(ptr->getSchemaNamePtr(), &ptr->db_length))
937
else if (lex->copy_db_to(&ptr->db, &ptr->db_length))
989
940
ptr->setIsFqtn(false);
991
942
ptr->alias= alias_str;
992
943
ptr->setIsAlias(alias ? true : false);
993
ptr->setTableName(table->table.str);
944
if (table->table.length)
945
table->table.length= my_casedn_str(files_charset_info, table->table.str);
946
ptr->table_name=table->table.str;
994
947
ptr->table_name_length=table->table.length;
995
948
ptr->lock_type= lock_type;
996
ptr->force_index= table_options.test(TL_OPTION_FORCE_INDEX);
997
ptr->ignore_leaves= table_options.test(TL_OPTION_IGNORE_LEAVES);
949
ptr->force_index= test(table_options & TL_OPTION_FORCE_INDEX);
950
ptr->ignore_leaves= test(table_options & TL_OPTION_IGNORE_LEAVES);
998
951
ptr->derived= table->sel;
999
952
ptr->select_lex= lex->current_select;
1000
953
ptr->index_hints= index_hints_arg;
1008
961
tables=tables->next_local)
1010
if (not my_strcasecmp(table_alias_charset, alias_str, tables->alias) &&
1011
not my_strcasecmp(system_charset_info, ptr->getSchemaName(), tables->getSchemaName()))
963
if (!my_strcasecmp(table_alias_charset, alias_str, tables->alias) &&
964
!strcasecmp(ptr->db, tables->db))
1013
966
my_error(ER_NONUNIQ_TABLE, MYF(0), alias_str);
1072
1025
bool Select_Lex::init_nested_join(Session *session)
1074
1027
TableList *ptr;
1075
NestedJoin *nested_join;
1028
nested_join_st *nested_join;
1077
1030
if (!(ptr= (TableList*) session->calloc(ALIGN_SIZE(sizeof(TableList))+
1078
sizeof(NestedJoin))))
1031
sizeof(nested_join_st))))
1080
ptr->setNestedJoin(((NestedJoin*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList)))));
1033
ptr->setNestedJoin(((nested_join_st*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList)))));
1081
1034
nested_join= ptr->getNestedJoin();
1082
1035
join_list->push_front(ptr);
1083
1036
ptr->setEmbedding(embedding);
1148
1101
TableList *Select_Lex::nest_last_join(Session *session)
1150
1103
TableList *ptr;
1151
NestedJoin *nested_join;
1104
nested_join_st *nested_join;
1152
1105
List<TableList> *embedded_list;
1154
1107
if (!(ptr= (TableList*) session->calloc(ALIGN_SIZE(sizeof(TableList))+
1155
sizeof(NestedJoin))))
1108
sizeof(nested_join_st))))
1157
ptr->setNestedJoin(((NestedJoin*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList)))));
1110
ptr->setNestedJoin(((nested_join_st*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList)))));
1158
1111
nested_join= ptr->getNestedJoin();
1159
1112
ptr->setEmbedding(embedding);
1160
1113
ptr->setJoinList(join_list);
1404
@param session Thread class
1406
@param only_kill_query Should it kill the query or the connection
1409
This is written such that we have a short lock on LOCK_thread_count
1413
kill_one_thread(Session *, ulong id, bool only_kill_query)
1416
uint32_t error= ER_NO_SUCH_THREAD;
1417
pthread_mutex_lock(&LOCK_thread_count); // For unlink from list
1419
for (SessionList::iterator it= getSessionList().begin(); it != getSessionList().end(); ++it )
1421
if ((*it)->thread_id == id)
1424
pthread_mutex_lock(&tmp->LOCK_delete); // Lock from delete
1428
pthread_mutex_unlock(&LOCK_thread_count);
1432
if (tmp->isViewable())
1434
tmp->awake(only_kill_query ? Session::KILL_QUERY : Session::KILL_CONNECTION);
1438
pthread_mutex_unlock(&tmp->LOCK_delete);
1445
kills a thread and sends response
1449
session Thread class
1451
only_kill_query Should it kill the query or the connection
1454
void sql_kill(Session *session, ulong id, bool only_kill_query)
1457
if (!(error= kill_one_thread(session, id, only_kill_query)))
1460
my_error(error, MYF(0), id);
1449
1465
Check if the select is a simple select (not an union).
1454
1470
1 error ; In this case the error messege is sent to the client
1457
bool check_simple_select(Session::pointer session)
1473
bool check_simple_select()
1475
Session *session= current_session;
1459
1476
LEX *lex= session->lex;
1460
1477
if (lex->current_select != &lex->select_lex)
1592
CREATE TABLE query pre-check.
1594
@param session Thread handler
1595
@param tables Global table list
1596
@param create_table Table which will be created
1604
bool create_table_precheck(TableIdentifier &identifier)
1606
if (not plugin::StorageEngine::canCreateTable(identifier))
1608
my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), "", "", identifier.getSchemaName().c_str());
1612
if (not plugin::StorageEngine::doesSchemaExist(identifier))
1614
my_error(ER_BAD_DB_ERROR, MYF(0), identifier.getSchemaName().c_str());
1575
1623
negate given expression.
1577
1625
@param session thread handler
1641
bool check_identifier_name(LEX_STRING *str, error_t err_code,
1689
bool check_identifier_name(LEX_STRING *str, uint32_t err_code,
1642
1690
uint32_t max_char_length,
1643
1691
const char *param_for_err_msg)
1706
1753
/* Parse the query. */
1708
bool parse_status= DRIZZLEparse(session) != 0;
1755
bool mysql_parse_status= DRIZZLEparse(session) != 0;
1710
1757
/* Check that if DRIZZLEparse() failed, session->is_error() is set. */
1712
assert(!parse_status || session->is_error());
1759
assert(!mysql_parse_status || session->is_error());
1714
1761
/* Reset Lex_input_stream. */
1716
1763
session->m_lip= NULL;
1718
DRIZZLE_QUERY_PARSE_DONE(parse_status || session->is_fatal_error);
1765
DRIZZLE_QUERY_PARSE_DONE(mysql_parse_status || session->is_fatal_error);
1720
1767
/* That's it. */
1722
return parse_status || session->is_fatal_error;
1769
return mysql_parse_status || session->is_fatal_error;