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 */
18
16
#define DRIZZLE_LEX 1
20
#include "drizzled/item/num.h"
21
#include "drizzled/abort_exception.h"
22
#include <drizzled/my_hash.h>
17
#include <drizzled/server_includes.h>
18
#include <mysys/hash.h>
19
#include <drizzled/db.h>
23
20
#include <drizzled/error.h>
24
21
#include <drizzled/nested_join.h>
25
22
#include <drizzled/query_id.h>
26
#include "drizzled/transaction_services.h"
27
23
#include <drizzled/sql_parse.h>
28
24
#include <drizzled/data_home.h>
29
25
#include <drizzled/sql_base.h>
30
26
#include <drizzled/show.h>
31
#include <drizzled/db.h>
27
#include <drizzled/plugin/info_schema_table.h>
32
28
#include <drizzled/function/time/unix_timestamp.h>
33
29
#include <drizzled/function/get_system_var.h>
34
30
#include <drizzled/item/cmpfunc.h>
35
31
#include <drizzled/item/null.h>
36
32
#include <drizzled/session.h>
37
#include <drizzled/session/cache.h>
38
33
#include <drizzled/sql_load.h>
39
34
#include <drizzled/lock.h>
40
35
#include <drizzled/select_send.h>
42
37
#include <drizzled/statement.h>
43
38
#include <drizzled/statement/alter_table.h>
44
39
#include "drizzled/probes.h"
45
#include "drizzled/global_charset_info.h"
47
41
#include "drizzled/plugin/logging.h"
48
#include "drizzled/plugin/query_rewrite.h"
49
#include "drizzled/plugin/query_cache.h"
50
#include "drizzled/plugin/authorization.h"
51
#include "drizzled/optimizer/explain_plan.h"
52
#include "drizzled/pthread_globals.h"
53
#include "drizzled/plugin/event_observer.h"
54
#include "drizzled/visibility.h"
42
#include "drizzled/plugin/info_schema_table.h"
59
45
#include <algorithm>
60
#include <boost/date_time.hpp>
61
#include "drizzled/internal/my_sys.h"
47
using namespace drizzled;
63
48
using namespace std;
65
extern int DRIZZLEparse(void *session); // from sql_yacc.cc
71
51
bool my_yyoverflow(short **a, YYSTYPE **b, ulong *yystacksize);
72
52
static bool parse_sql(Session *session, Lex_input_stream *lip);
73
void parse(Session *session, const char *inBuf, uint32_t length);
53
static void mysql_parse(Session *session, const char *inBuf, uint32_t length,
54
const char ** found_semicolon);
76
57
@defgroup Runtime_Environment Runtime Environment
80
61
extern size_t my_thread_stack_size;
81
62
extern const CHARSET_INFO *character_set_filesystem;
86
static const std::string command_name[COM_END+1]={
94
"Error" // Last command number
63
const char *any_db="*any*"; // Special symbol for check_access
65
const LEX_STRING command_name[COM_END+1]={
66
{ C_STRING_WITH_LEN("Sleep") },
67
{ C_STRING_WITH_LEN("Quit") },
68
{ C_STRING_WITH_LEN("Init DB") },
69
{ C_STRING_WITH_LEN("Query") },
70
{ C_STRING_WITH_LEN("Shutdown") },
71
{ C_STRING_WITH_LEN("Connect") },
72
{ C_STRING_WITH_LEN("Ping") },
73
{ C_STRING_WITH_LEN("Error") } // Last command number
99
76
const char *xa_state_names[]={
100
77
"NON-EXISTING", "ACTIVE", "IDLE", "PREPARED"
144
116
sql_command_flags[SQLCOM_REPLACE]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
145
117
sql_command_flags[SQLCOM_REPLACE_SELECT]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
119
sql_command_flags[SQLCOM_SHOW_STATUS]= CF_STATUS_COMMAND;
120
sql_command_flags[SQLCOM_SHOW_DATABASES]= CF_STATUS_COMMAND;
121
sql_command_flags[SQLCOM_SHOW_OPEN_TABLES]= CF_STATUS_COMMAND;
122
sql_command_flags[SQLCOM_SHOW_FIELDS]= CF_STATUS_COMMAND;
123
sql_command_flags[SQLCOM_SHOW_KEYS]= CF_STATUS_COMMAND;
124
sql_command_flags[SQLCOM_SHOW_VARIABLES]= CF_STATUS_COMMAND;
147
125
sql_command_flags[SQLCOM_SHOW_WARNS]= CF_STATUS_COMMAND;
148
126
sql_command_flags[SQLCOM_SHOW_ERRORS]= CF_STATUS_COMMAND;
127
sql_command_flags[SQLCOM_SHOW_ENGINE_STATUS]= CF_STATUS_COMMAND;
128
sql_command_flags[SQLCOM_SHOW_PROCESSLIST]= CF_STATUS_COMMAND;
149
129
sql_command_flags[SQLCOM_SHOW_CREATE_DB]= CF_STATUS_COMMAND;
150
130
sql_command_flags[SQLCOM_SHOW_CREATE]= CF_STATUS_COMMAND;
132
sql_command_flags[SQLCOM_SHOW_TABLES]= (CF_STATUS_COMMAND |
133
CF_SHOW_TABLE_COMMAND);
134
sql_command_flags[SQLCOM_SHOW_TABLE_STATUS]= (CF_STATUS_COMMAND |
135
CF_SHOW_TABLE_COMMAND);
153
137
The following admin table operations are allowed
140
sql_command_flags[SQLCOM_OPTIMIZE]= CF_WRITE_LOGS_COMMAND;
156
141
sql_command_flags[SQLCOM_ANALYZE]= CF_WRITE_LOGS_COMMAND;
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);
213
195
switch (command) {
214
196
case COM_INIT_DB:
216
if (packet_length == 0)
218
my_message(ER_NO_DB_ERROR, ER(ER_NO_DB_ERROR), MYF(0));
222
string tmp(packet, packet_length);
224
identifier::Schema identifier(tmp);
226
if (not change_db(session, identifier))
199
status_var_increment(session->status_var.com_stat[SQLCOM_CHANGE_DB]);
201
tmp.length= packet_length;
202
if (!mysql_change_db(session, &tmp, false))
228
204
session->my_ok();
234
if (not session->readAndStoreQuery(packet, packet_length))
210
if (! session->readAndStoreQuery(packet, packet_length))
235
211
break; // fatal error is set
236
DRIZZLE_QUERY_START(session->getQueryString()->c_str(),
212
DRIZZLE_QUERY_START(session->query,
237
213
session->thread_id,
238
const_cast<const char *>(session->schema()->c_str()));
214
const_cast<const char *>(session->db ? session->db : ""));
215
const char* end_of_stmt= NULL;
240
parse(session, session->getQueryString()->c_str(), session->getQueryString()->length());
217
mysql_parse(session, session->query, session->query_length, &end_of_stmt);
296
272
case Diagnostics_area::DA_ERROR:
297
273
/* The query failed, send error to log and abort bootstrap. */
298
session->getClient()->sendError(session->main_da.sql_errno(),
274
session->client->sendError(session->main_da.sql_errno(),
299
275
session->main_da.message());
302
278
case Diagnostics_area::DA_EOF:
303
session->getClient()->sendEOF();
279
session->client->sendEOF();
306
282
case Diagnostics_area::DA_OK:
307
session->getClient()->sendOK();
283
session->client->sendOK();
310
286
case Diagnostics_area::DA_DISABLED:
323
299
session->close_thread_tables();
325
301
plugin::Logging::postDo(session);
326
if (unlikely(plugin::EventObserver::afterStatement(*session)))
328
// We should do something about an error...
331
303
/* Store temp state for processlist */
332
304
session->set_proc_info("cleaning up");
333
305
session->command= COM_SLEEP;
334
session->resetQueryString();
306
memset(session->process_list_info, 0, PROCESS_LIST_WIDTH);
308
session->query_length= 0;
336
310
session->set_proc_info(NULL);
337
session->mem_root->free_root(MYF(memory::KEEP_PREALLOC));
311
free_root(session->mem_root,MYF(MY_KEEP_PREALLOC));
339
313
if (DRIZZLE_QUERY_DONE_ENABLED() || DRIZZLE_COMMAND_DONE_ENABLED())
374
348
1 out of memory or SHOW commands are not allowed
375
349
in this version of the server.
377
static bool _schema_select(Session *session, Select_Lex *sel,
378
const string& schema_table_name)
380
LEX_STRING db, table;
381
bitset<NUM_OF_TABLE_OPTIONS> table_options;
383
We have to make non const db_name & table_name
384
because of lower_case_table_names
386
session->make_lex_string(&db, "data_dictionary", sizeof("data_dictionary"), false);
387
session->make_lex_string(&table, schema_table_name, false);
389
if (! sel->add_table_to_list(session, new Table_ident(db, table),
390
NULL, table_options, TL_READ))
397
int prepare_new_schema_table(Session *session, LEX *lex,
398
const string& schema_table_name)
352
int prepare_schema_table(Session *session, LEX *lex, Table_ident *table_ident,
353
const string& schema_table_name)
400
355
Select_Lex *schema_select_lex= NULL;
358
if (schema_table_name.compare("TABLES") == 0 ||
359
schema_table_name.compare("TABLE_NAMES") == 0)
363
if (lex->select_lex.db == NULL &&
364
lex->copy_db_to(&lex->select_lex.db, &dummy))
368
schema_select_lex= new Select_Lex();
369
db.str= schema_select_lex->db= lex->select_lex.db;
370
schema_select_lex->table_list.first= NULL;
371
db.length= strlen(db.str);
373
if (check_db_name(&db))
375
my_error(ER_WRONG_DB_NAME, MYF(0), db.str);
379
else if (schema_table_name.compare("COLUMNS") == 0 ||
380
schema_table_name.compare("STATISTICS") == 0)
383
TableList **query_tables_last= lex->query_tables_last;
384
schema_select_lex= new Select_Lex();
385
/* 'parent_lex' is used in init_query() so it must be before it. */
386
schema_select_lex->parent_lex= lex;
387
schema_select_lex->init_query();
388
if (! schema_select_lex->add_table_to_list(session, table_ident, 0, 0, TL_READ))
392
lex->query_tables_last= query_tables_last;
402
395
Select_Lex *select_lex= lex->current_select;
403
396
assert(select_lex);
404
if (_schema_select(session, select_lex, schema_table_name))
397
if (make_schema_select(session, select_lex, schema_table_name))
486
486
drizzle_reset_errors(session, 0);
489
assert(session->transaction.stmt.hasModifiedNonTransData() == false);
489
status_var_increment(session->status_var.com_stat[lex->sql_command]);
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));
491
assert(session->transaction.stmt.modified_non_trans_table == false);
502
493
/* now we are ready to execute the statement */
503
494
res= lex->statement->execute();
504
496
session->set_proc_info("query end");
506
499
The return value for ROW_COUNT() is "implementation dependent" if the
507
500
statement is not DELETE, INSERT or UPDATE, but -1 is what JDBC and ODBC
641
new_select(LEX *lex, bool move_down)
615
mysql_new_select(LEX *lex, bool move_down)
643
617
Select_Lex *select_lex;
644
618
Session *session= lex->session;
646
620
if (!(select_lex= new (session->mem_root) Select_Lex()))
649
622
select_lex->select_number= ++session->select_number;
650
623
select_lex->parent_lex= lex; /* Used in init_query. */
651
624
select_lex->init_query();
652
625
select_lex->init_select();
653
626
lex->nest_level++;
655
627
if (lex->nest_level > (int) MAX_SELECT_NESTING)
657
629
my_error(ER_TOO_HIGH_LEVEL_OF_NESTING_FOR_SELECT,MYF(0),MAX_SELECT_NESTING);
661
632
select_lex->nest_level= lex->nest_level;
753
723
@param session Current thread
754
724
@param inBuf Begining of the query text
755
725
@param length Length of the query text
726
@param[out] found_semicolon For multi queries, position of the character of
727
the next query in the query text.
758
void parse(Session *session, const char *inBuf, uint32_t length)
730
static void mysql_parse(Session *session, const char *inBuf, uint32_t length,
731
const char ** found_semicolon)
760
session->lex->start(session);
735
The purpose of query_cache_send_result_to_client() is to lookup the
736
query in the query cache first, to avoid parsing and executing it.
737
So, the natural implementation would be to:
738
- first, call query_cache_send_result_to_client,
739
- second, if caching failed, initialise the lexical and syntactic parser.
740
The problem is that the query cache depends on a clean initialization
741
of (among others) lex->safe_to_cache_query and session->server_status,
742
which are reset respectively in
744
- mysql_reset_session_for_next_command()
745
So, initializing the lexical analyser *before* using the query cache
746
is required for the cache to work properly.
747
FIXME: cleanup the dependencies in the code to simplify this.
762
750
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
LEX *lex= session->lex;
777
Lex_input_stream lip(session, inBuf, length);
778
bool err= parse_sql(session, &lip);
753
LEX *lex= session->lex;
755
Lex_input_stream lip(session, inBuf, length);
757
bool err= parse_sql(session, &lip);
758
*found_semicolon= lip.found_semicolon;
782
if (not session->is_error())
784
DRIZZLE_QUERY_EXEC_START(session->getQueryString()->c_str(),
786
const_cast<const char *>(session->schema()->c_str()));
787
// Implement Views here --Brian
788
/* Actually execute the query */
791
execute_command(session);
795
// Just try to catch any random failures that could have come
799
DRIZZLE_QUERY_EXEC_DONE(0);
763
if (! session->is_error())
766
Binlog logs a string starting from session->query and having length
767
session->query_length; so we set session->query_length correctly (to not
768
log several statements in one event, when we executed only first).
769
We set it to not see the ';' (otherwise it would get into binlog
770
and Query_log_event::print() would give ';;' output).
771
This also helps display only the current query in SHOW
773
Note that we don't need LOCK_thread_count to modify query_length.
775
if (*found_semicolon &&
776
(session->query_length= (ulong)(*found_semicolon - session->query)))
777
session->query_length--;
778
DRIZZLE_QUERY_EXEC_START(session->query,
780
const_cast<const char *>(session->db ? session->db : ""));
781
/* Actually execute the query */
782
mysql_execute_command(session);
783
DRIZZLE_QUERY_EXEC_DONE(0);
805
assert(session->is_error());
808
session->set_proc_info("freeing items");
809
session->end_statement();
810
session->cleanup_after_query();
811
session->set_end_timer();
789
assert(session->is_error());
792
session->set_proc_info("freeing items");
793
session->end_statement();
794
session->cleanup_after_query();
944
938
return NULL; // End of memory
945
939
alias_str= alias ? alias->str : table->table.str;
946
if (! table_options.test(TL_OPTION_ALIAS) &&
940
if (!test(table_options & TL_OPTION_ALIAS) &&
947
941
check_table_name(table->table.str, table->table.length))
949
943
my_error(ER_WRONG_TABLE_NAME, MYF(0), table->table.str);
953
if (table->is_derived_table() == false && table->db.str)
947
if (table->is_derived_table() == false && table->db.str &&
948
check_db_name(&table->db))
955
my_casedn_str(files_charset_info, table->db.str);
957
identifier::Schema schema_identifier(string(table->db.str));
958
if (not check_db_name(session, schema_identifier))
961
my_error(ER_WRONG_DB_NAME, MYF(0), table->db.str);
950
my_error(ER_WRONG_DB_NAME, MYF(0), table->db.str);
966
954
if (!alias) /* Alias is case sensitive */
971
959
ER(ER_DERIVED_MUST_HAVE_ALIAS), MYF(0));
974
if (!(alias_str= (char*) session->getMemRoot()->duplicate(alias_str,table->table.length+1)))
962
if (!(alias_str= (char*) session->memdup(alias_str,table->table.length+1)))
977
965
if (!(ptr = (TableList *) session->calloc(sizeof(TableList))))
980
967
if (table->db.str)
982
ptr->setIsFqtn(true);
983
ptr->setSchemaName(table->db.str);
970
ptr->db= table->db.str;
984
971
ptr->db_length= table->db.length;
986
else if (lex->copy_db_to(ptr->getSchemaNamePtr(), &ptr->db_length))
973
else if (lex->copy_db_to(&ptr->db, &ptr->db_length))
989
ptr->setIsFqtn(false);
991
978
ptr->alias= alias_str;
992
ptr->setIsAlias(alias ? true : false);
993
ptr->setTableName(table->table.str);
979
ptr->is_alias= alias ? true : false;
980
if (table->table.length)
981
table->table.length= my_casedn_str(files_charset_info, table->table.str);
982
ptr->table_name=table->table.str;
994
983
ptr->table_name_length=table->table.length;
995
984
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);
985
ptr->updating= test(table_options & TL_OPTION_UPDATING);
986
ptr->force_index= test(table_options & TL_OPTION_FORCE_INDEX);
987
ptr->ignore_leaves= test(table_options & TL_OPTION_IGNORE_LEAVES);
998
988
ptr->derived= table->sel;
989
if (!ptr->derived && !my_strcasecmp(system_charset_info, ptr->db,
990
INFORMATION_SCHEMA_NAME.c_str()))
992
plugin::InfoSchemaTable *schema_table= plugin::InfoSchemaTable::getTable(ptr->table_name);
994
(schema_table->isHidden() &&
995
((sql_command_flags[lex->sql_command].test(CF_BIT_STATUS_COMMAND)) == 0 ||
997
this check is used for show columns|keys from I_S hidden table
999
lex->sql_command == SQLCOM_SHOW_FIELDS ||
1000
lex->sql_command == SQLCOM_SHOW_KEYS)))
1002
my_error(ER_UNKNOWN_TABLE, MYF(0),
1003
ptr->table_name, INFORMATION_SCHEMA_NAME.c_str());
1006
ptr->schema_table_name= ptr->table_name;
1007
ptr->schema_table= schema_table;
999
1009
ptr->select_lex= lex->current_select;
1000
1010
ptr->index_hints= index_hints_arg;
1001
1011
ptr->option= option ? option->str : 0;
1072
1082
bool Select_Lex::init_nested_join(Session *session)
1074
1084
TableList *ptr;
1075
NestedJoin *nested_join;
1085
nested_join_st *nested_join;
1077
1087
if (!(ptr= (TableList*) session->calloc(ALIGN_SIZE(sizeof(TableList))+
1078
sizeof(NestedJoin))))
1088
sizeof(nested_join_st))))
1080
ptr->setNestedJoin(((NestedJoin*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList)))));
1081
nested_join= ptr->getNestedJoin();
1090
nested_join= ptr->nested_join=
1091
((nested_join_st*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList))));
1082
1093
join_list->push_front(ptr);
1083
ptr->setEmbedding(embedding);
1084
ptr->setJoinList(join_list);
1094
ptr->embedding= embedding;
1095
ptr->join_list= join_list;
1085
1096
ptr->alias= (char*) "(nested_join)";
1086
1097
embedding= ptr;
1087
1098
join_list= &nested_join->join_list;
1107
1118
TableList *Select_Lex::end_nested_join(Session *)
1109
1120
TableList *ptr;
1110
NestedJoin *nested_join;
1121
nested_join_st *nested_join;
1112
1123
assert(embedding);
1113
1124
ptr= embedding;
1114
join_list= ptr->getJoinList();
1115
embedding= ptr->getEmbedding();
1116
nested_join= ptr->getNestedJoin();
1125
join_list= ptr->join_list;
1126
embedding= ptr->embedding;
1127
nested_join= ptr->nested_join;
1117
1128
if (nested_join->join_list.elements == 1)
1119
1130
TableList *embedded= nested_join->join_list.head();
1120
1131
join_list->pop();
1121
embedded->setJoinList(join_list);
1122
embedded->setEmbedding(embedding);
1132
embedded->join_list= join_list;
1133
embedded->embedding= embedding;
1123
1134
join_list->push_front(embedded);
1148
1159
TableList *Select_Lex::nest_last_join(Session *session)
1150
1161
TableList *ptr;
1151
NestedJoin *nested_join;
1162
nested_join_st *nested_join;
1152
1163
List<TableList> *embedded_list;
1154
1165
if (!(ptr= (TableList*) session->calloc(ALIGN_SIZE(sizeof(TableList))+
1155
sizeof(NestedJoin))))
1166
sizeof(nested_join_st))))
1157
ptr->setNestedJoin(((NestedJoin*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList)))));
1158
nested_join= ptr->getNestedJoin();
1159
ptr->setEmbedding(embedding);
1160
ptr->setJoinList(join_list);
1168
nested_join= ptr->nested_join=
1169
((nested_join_st*) ((unsigned char*) ptr + ALIGN_SIZE(sizeof(TableList))));
1171
ptr->embedding= embedding;
1172
ptr->join_list= join_list;
1161
1173
ptr->alias= (char*) "(nest_last_join)";
1162
1174
embedded_list= &nested_join->join_list;
1163
1175
embedded_list->empty();
1467
@param session Thread class
1469
@param only_kill_query Should it kill the query or the connection
1472
This is written such that we have a short lock on LOCK_thread_count
1476
kill_one_thread(Session *, ulong id, bool only_kill_query)
1479
uint32_t error=ER_NO_SUCH_THREAD;
1480
pthread_mutex_lock(&LOCK_thread_count); // For unlink from list
1482
for( vector<Session*>::iterator it= session_list.begin(); it != session_list.end(); ++it )
1484
if ((*it)->thread_id == id)
1487
pthread_mutex_lock(&tmp->LOCK_delete); // Lock from delete
1491
pthread_mutex_unlock(&LOCK_thread_count);
1494
tmp->awake(only_kill_query ? Session::KILL_QUERY : Session::KILL_CONNECTION);
1496
pthread_mutex_unlock(&tmp->LOCK_delete);
1503
kills a thread and sends response
1507
session Thread class
1509
only_kill_query Should it kill the query or the connection
1512
void sql_kill(Session *session, ulong id, bool only_kill_query)
1515
if (!(error= kill_one_thread(session, id, only_kill_query)))
1518
my_error(error, MYF(0), id);
1449
1523
Check if the select is a simple select (not an union).
1650
CREATE TABLE query pre-check.
1652
@param session Thread handler
1653
@param tables Global table list
1654
@param create_table Table which will be created
1662
bool create_table_precheck(Session *, TableList *,
1663
TableList *create_table)
1665
bool error= true; // Error message is given
1667
if (create_table && (strcmp(create_table->db, "information_schema") == 0))
1669
my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), "", "", INFORMATION_SCHEMA_NAME.c_str());
1575
1680
negate given expression.
1577
1682
@param session thread handler
1706
1812
/* Parse the query. */
1708
bool parse_status= DRIZZLEparse(session) != 0;
1814
bool mysql_parse_status= DRIZZLEparse(session) != 0;
1710
1816
/* Check that if DRIZZLEparse() failed, session->is_error() is set. */
1712
assert(!parse_status || session->is_error());
1818
assert(!mysql_parse_status || session->is_error());
1714
1820
/* Reset Lex_input_stream. */
1716
1822
session->m_lip= NULL;
1718
DRIZZLE_QUERY_PARSE_DONE(parse_status || session->is_fatal_error);
1824
DRIZZLE_QUERY_PARSE_DONE(mysql_parse_status || session->is_fatal_error);
1720
1826
/* That's it. */
1722
return parse_status || session->is_fatal_error;
1828
return mysql_parse_status || session->is_fatal_error;
1726
1832
@} (end of group Runtime_Environment)
1729
} /* namespace drizzled */