mariadb/sql/sql_yacc.yy

6808 lines
166 KiB
Text
Raw Normal View History

/* Copyright (C) 2000-2003 MySQL AB
2000-07-31 21:29:14 +02:00
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
2000-07-31 21:29:14 +02:00
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
2000-07-31 21:29:14 +02:00
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
2000-12-15 13:18:52 +02:00
/* sql_yacc.yy */
2000-07-31 21:29:14 +02:00
%{
2002-11-30 16:43:53 +01:00
/* thd is passed as an arg to yyparse(), and subsequently to yylex().
** The type will be void*, so it must be cast to (THD*) when used.
** Use the YYTHD macro for this.
*/
#define YYPARSE_PARAM yythd
2002-11-30 16:43:53 +01:00
#define YYLEX_PARAM yythd
#define YYTHD ((THD *)yythd)
2000-07-31 21:29:14 +02:00
#define MYSQL_YACC
#define YYINITDEPTH 100
#define YYMAXDEPTH 3200 /* Because of 64K stack */
#define Lex (YYTHD->lex)
#define Select Lex->current_select
2000-07-31 21:29:14 +02:00
#include "mysql_priv.h"
2001-10-09 14:53:54 +02:00
#include "slave.h"
2000-07-31 21:29:14 +02:00
#include "sql_acl.h"
#include "lex_symbol.h"
#include "item_create.h"
#include "sp_head.h"
#include "sp_pcontext.h"
#include "sp_rcontext.h"
#include "sp.h"
2000-07-31 21:29:14 +02:00
#include <myisam.h>
#include <myisammrg.h>
2002-11-30 16:43:53 +01:00
int yylex(void *yylval, void *yythd);
2000-07-31 21:29:14 +02:00
#define yyoverflow(A,B,C,D,E,F) if (my_yyoverflow((B),(D),(int*) (F))) { yyerror((char*) (A)); return 2; }
2000-07-31 21:29:14 +02:00
#define WARN_DEPRECATED(A,B) \
push_warning_printf(((THD *)yythd), MYSQL_ERROR::WARN_LEVEL_WARN, \
ER_WARN_DEPRECATED_SYNTAX, \
ER(ER_WARN_DEPRECATED_SYNTAX), (A), (B));
inline Item *or_or_concat(THD *thd, Item* A, Item* B)
2000-07-31 21:29:14 +02:00
{
return (thd->variables.sql_mode & MODE_PIPES_AS_CONCAT ?
2000-07-31 21:29:14 +02:00
(Item*) new Item_func_concat(A,B) : (Item*) new Item_cond_or(A,B));
}
%}
%union {
int num;
ulong ulong_num;
ulonglong ulonglong_number;
2000-07-31 21:29:14 +02:00
LEX_STRING lex_str;
LEX_STRING *lex_str_ptr;
LEX_SYMBOL symbol;
Table_ident *table;
char *simple_string;
Item *item;
List<Item> *item_list;
List<String> *string_list;
String *string;
key_part_spec *key_part;
TABLE_LIST *table_list;
udf_func *udf;
LEX_USER *lex_user;
struct sys_var_with_base variable;
2000-07-31 21:29:14 +02:00
Key::Keytype key_type;
enum ha_key_alg key_alg;
2000-07-31 21:29:14 +02:00
enum db_type db_type;
enum row_type row_type;
2001-11-06 00:05:45 +02:00
enum ha_rkey_function ha_rkey_mode;
enum enum_tx_isolation tx_isolation;
enum Cast_target cast_type;
2000-07-31 21:29:14 +02:00
enum Item_udftype udf_type;
CHARSET_INFO *charset;
thr_lock_type lock_type;
interval_type interval, interval_time_st;
timestamp_type date_time_type;
2002-10-27 23:27:00 +02:00
st_select_lex *select_lex;
chooser_compare_func_creator boolfunc2creator;
struct sp_cond_type *spcondtype;
struct { int vars, conds, hndlrs, curs; } spblock;
struct st_lex *lex;
2000-07-31 21:29:14 +02:00
}
%{
bool my_yyoverflow(short **a, YYSTYPE **b,int *yystacksize);
%}
%pure_parser /* We have threads */
%token END_OF_INPUT
%token CLOSE_SYM
%token HANDLER_SYM
%token LAST_SYM
%token NEXT_SYM
%token PREV_SYM
%token DIV_SYM
2000-07-31 21:29:14 +02:00
%token EQ
%token EQUAL_SYM
2002-12-14 02:18:14 +04:00
%token SOUNDS_SYM
2000-07-31 21:29:14 +02:00
%token GE
%token GT_SYM
%token LE
%token LT
%token NE
%token IS
%token MOD_SYM
2000-07-31 21:29:14 +02:00
%token SHIFT_LEFT
%token SHIFT_RIGHT
%token SET_VAR
%token ABORT_SYM
2000-07-31 21:29:14 +02:00
%token ADD
%token AFTER_SYM
%token ALTER
%token ANALYZE_SYM
%token ANY_SYM
%token AVG_SYM
%token BEGIN_SYM
%token BINLOG_SYM
%token CALL_SYM
2000-07-31 21:29:14 +02:00
%token CHANGE
%token CLIENT_SYM
%token COMMENT_SYM
%token COMMIT_SYM
%token COUNT_SYM
2000-07-31 21:29:14 +02:00
%token CREATE
%token CROSS
%token CUBE_SYM
%token DEFINER_SYM
2000-07-31 21:29:14 +02:00
%token DELETE_SYM
%token DETERMINISTIC_SYM
%token DUAL_SYM
2001-12-17 19:59:20 +02:00
%token DO_SYM
2000-07-31 21:29:14 +02:00
%token DROP
%token EVENTS_SYM
%token EXECUTE_SYM
%token EXPANSION_SYM
2000-07-31 21:29:14 +02:00
%token FLUSH_SYM
%token HELP_SYM
%token INSERT
%token RELAY_THREAD
2000-07-31 21:29:14 +02:00
%token KILL_SYM
%token LOAD
%token LOCKS_SYM
%token LOCK_SYM
%token MASTER_SYM
%token MAX_SYM
%token MIN_SYM
%token NONE_SYM
%token OPTIMIZE
%token PURGE
%token REPAIR
%token REPLICATION
%token RESET_SYM
%token ROLLBACK_SYM
%token ROLLUP_SYM
%token SAVEPOINT_SYM
%token SELECT_SYM
%token SHOW
%token SLAVE
%token SQL_SYM
%token SQL_THREAD
%token START_SYM
%token STD_SYM
2002-12-14 03:36:59 +04:00
%token VARIANCE_SYM
%token STOP_SYM
%token SUM_SYM
2003-06-23 12:56:44 +05:00
%token ADDDATE_SYM
%token SUPER_SYM
%token TRUNCATE_SYM
2000-07-31 21:29:14 +02:00
%token UNLOCK_SYM
%token UNTIL_SYM
%token UPDATE_SYM
2000-07-31 21:29:14 +02:00
%token ACTION
%token AGGREGATE_SYM
%token ALL
%token AND
%token AS
%token ASC
%token AUTO_INC
2000-07-31 21:29:14 +02:00
%token AVG_ROW_LENGTH
%token BACKUP_SYM
2000-07-31 21:29:14 +02:00
%token BERKELEY_DB_SYM
%token BINARY
%token BIT_SYM
%token BOOL_SYM
2001-10-09 14:53:54 +02:00
%token BOOLEAN_SYM
2000-07-31 21:29:14 +02:00
%token BOTH
2002-02-22 15:24:42 +04:00
%token BTREE_SYM
2000-07-31 21:29:14 +02:00
%token BY
%token BYTE_SYM
2001-12-02 14:34:01 +02:00
%token CACHE_SYM
2000-07-31 21:29:14 +02:00
%token CASCADE
%token CAST_SYM
%token CHARSET
2000-07-31 21:29:14 +02:00
%token CHECKSUM_SYM
%token CHECK_SYM
%token COMMITTED_SYM
%token COLLATE_SYM
%token COLLATION_SYM
2000-07-31 21:29:14 +02:00
%token COLUMNS
%token COLUMN_SYM
2001-05-05 09:41:47 +03:00
%token CONCURRENT
%token CONDITION_SYM
%token CONNECTION_SYM
2000-07-31 21:29:14 +02:00
%token CONSTRAINT
%token CONTINUE_SYM
%token CONVERT_SYM
2000-07-31 21:29:14 +02:00
%token DATABASES
%token DATA_SYM
%token DECLARE_SYM
2000-07-31 21:29:14 +02:00
%token DEFAULT
%token DELAYED_SYM
%token DELAY_KEY_WRITE_SYM
%token DESC
%token DESCRIBE
%token DES_KEY_FILE
%token DISABLE_SYM
%token DISCARD
%token DISTINCT
%token DUPLICATE_SYM
2000-07-31 21:29:14 +02:00
%token DYNAMIC_SYM
%token ENABLE_SYM
2000-07-31 21:29:14 +02:00
%token ENCLOSED
%token ESCAPED
%token DIRECTORY_SYM
2000-07-31 21:29:14 +02:00
%token ESCAPE_SYM
%token EXISTS
%token EXIT_SYM
2000-07-31 21:29:14 +02:00
%token EXTENDED_SYM
%token FALSE_SYM
%token FETCH_SYM
2000-07-31 21:29:14 +02:00
%token FILE_SYM
%token FIRST_SYM
%token FIXED_SYM
%token FLOAT_NUM
%token FORCE_SYM
2000-07-31 21:29:14 +02:00
%token FOREIGN
%token FOUND_SYM
2000-07-31 21:29:14 +02:00
%token FROM
%token FULL
%token FULLTEXT_SYM
%token GLOBAL_SYM
2000-07-31 21:29:14 +02:00
%token GRANT
%token GRANTS
%token GREATEST_SYM
%token GROUP
%token HAVING
2002-02-22 15:24:42 +04:00
%token HASH_SYM
2000-07-31 21:29:14 +02:00
%token HEX_NUM
%token HIGH_PRIORITY
%token HOSTS_SYM
%token IDENT
%token IDENT_QUOTED
2000-07-31 21:29:14 +02:00
%token IGNORE_SYM
%token IMPORT
2000-07-31 21:29:14 +02:00
%token INDEX
%token INDEXES
2000-07-31 21:29:14 +02:00
%token INFILE
%token INNER_SYM
2000-10-20 17:39:23 +03:00
%token INNOBASE_SYM
%token INOUT_SYM
2000-07-31 21:29:14 +02:00
%token INTO
%token IN_SYM
%token INVOKER_SYM
%token ISOLATION
2000-07-31 21:29:14 +02:00
%token JOIN_SYM
%token KEYS
%token KEY_SYM
%token LEADING
%token LEAST_SYM
%token LEAVES
%token LEVEL_SYM
2000-07-31 21:29:14 +02:00
%token LEX_HOSTNAME
%token LANGUAGE_SYM
2000-07-31 21:29:14 +02:00
%token LIKE
%token LINES
%token LOCAL_SYM
%token LOCATOR_SYM
%token LOG_SYM
2000-07-31 21:29:14 +02:00
%token LOGS_SYM
%token LONG_NUM
%token LONG_SYM
%token LOW_PRIORITY
%token MASTER_HOST_SYM
%token MASTER_USER_SYM
%token MASTER_LOG_FILE_SYM
%token MASTER_LOG_POS_SYM
%token MASTER_PASSWORD_SYM
%token MASTER_PORT_SYM
%token MASTER_CONNECT_RETRY_SYM
%token MASTER_SERVER_ID_SYM
%token MASTER_SSL_SYM
%token MASTER_SSL_CA_SYM
%token MASTER_SSL_CAPATH_SYM
%token MASTER_SSL_CERT_SYM
%token MASTER_SSL_CIPHER_SYM
%token MASTER_SSL_KEY_SYM
%token RELAY_LOG_FILE_SYM
%token RELAY_LOG_POS_SYM
2000-07-31 21:29:14 +02:00
%token MATCH
%token MAX_ROWS
%token MAX_CONNECTIONS_PER_HOUR
%token MAX_QUERIES_PER_HOUR
%token MAX_UPDATES_PER_HOUR
%token MEDIUM_SYM
2000-07-31 21:29:14 +02:00
%token MIN_ROWS
%token NAMES_SYM
%token NAME_SYM
2000-07-31 21:29:14 +02:00
%token NATIONAL_SYM
%token NATURAL
%token NEW_SYM
2000-07-31 21:29:14 +02:00
%token NCHAR_SYM
2003-03-20 22:01:03 +04:00
%token NCHAR_STRING
%token NVARCHAR_SYM
2000-07-31 21:29:14 +02:00
%token NOT
%token NO_SYM
%token NULL_SYM
%token NUM
%token OFFSET_SYM
2000-07-31 21:29:14 +02:00
%token ON
%token OPEN_SYM
2000-07-31 21:29:14 +02:00
%token OPTION
%token OPTIONALLY
%token OR
%token OR_OR_CONCAT
2000-07-31 21:29:14 +02:00
%token ORDER_SYM
%token OUT_SYM
2000-07-31 21:29:14 +02:00
%token OUTER
%token OUTFILE
%token DUMPFILE
2000-07-31 21:29:14 +02:00
%token PACK_KEYS_SYM
%token PARTIAL
%token PRIMARY_SYM
%token PRIVILEGES
%token PROCESS
%token PROCESSLIST_SYM
2001-12-02 14:34:01 +02:00
%token QUERY_SYM
2000-07-31 21:29:14 +02:00
%token RAID_0_SYM
%token RAID_STRIPED_SYM
%token RAID_TYPE
2000-07-31 21:29:14 +02:00
%token RAID_CHUNKS
%token RAID_CHUNKSIZE
%token READ_SYM
%token REAL_NUM
%token REFERENCES
%token REGEXP
%token RELOAD
%token RENAME
%token REPEATABLE_SYM
2001-09-01 16:29:37 +08:00
%token REQUIRE_SYM
%token RESOURCES
%token RESTORE_SYM
2000-07-31 21:29:14 +02:00
%token RESTRICT
%token REVOKE
%token ROWS_SYM
%token ROW_FORMAT_SYM
%token ROW_SYM
2002-02-22 15:24:42 +04:00
%token RTREE_SYM
%token SECURITY_SYM
2000-07-31 21:29:14 +02:00
%token SET
%token SEPARATOR_SYM
%token SERIAL_SYM
%token SERIALIZABLE_SYM
%token SESSION_SYM
%token SIMPLE_SYM
2000-07-31 21:29:14 +02:00
%token SHUTDOWN
2002-02-22 15:24:42 +04:00
%token SPATIAL_SYM
%token SPECIFIC_SYM
%token SQLEXCEPTION_SYM
%token SQLSTATE_SYM
%token SQLWARNING_SYM
%token SSL_SYM
2000-07-31 21:29:14 +02:00
%token STARTING
%token STATUS_SYM
%token STORAGE_SYM
2000-07-31 21:29:14 +02:00
%token STRAIGHT_JOIN
2001-09-01 16:29:37 +08:00
%token SUBJECT_SYM
2000-07-31 21:29:14 +02:00
%token TABLES
%token TABLE_SYM
%token TABLESPACE
2000-07-31 21:29:14 +02:00
%token TEMPORARY
%token TERMINATED
%token TEXT_STRING
%token TO_SYM
%token TRAILING
%token TRANSACTION_SYM
%token TRUE_SYM
%token TYPE_SYM
%token TYPES_SYM
2000-07-31 21:29:14 +02:00
%token FUNC_ARG0
%token FUNC_ARG1
%token FUNC_ARG2
%token FUNC_ARG3
%token RETURN_SYM
%token RETURNS_SYM
%token UDF_SONAME_SYM
%token FUNCTION_SYM
%token UNCOMMITTED_SYM
%token UNDERSCORE_CHARSET
%token UNDO_SYM
%token UNICODE_SYM
%token UNION_SYM
%token UNIQUE_SYM
%token USAGE
%token USE_FRM
%token USE_SYM
%token USING
%token VALUE_SYM
%token VALUES
2000-07-31 21:29:14 +02:00
%token VARIABLES
%token WHERE
%token WITH
%token WRITE_SYM
%token NO_WRITE_TO_BINLOG
%token X509_SYM
%token XOR
%token COMPRESSED_SYM
2000-07-31 21:29:14 +02:00
%token ERRORS
%token WARNINGS
%token ASCII_SYM
2000-07-31 21:29:14 +02:00
%token BIGINT
%token BLOB_SYM
%token CHAR_SYM
%token CHANGED
2000-07-31 21:29:14 +02:00
%token COALESCE
%token DATETIME
%token DATE_SYM
%token DECIMAL_SYM
%token DOUBLE_SYM
%token ENUM
%token FAST_SYM
2000-07-31 21:29:14 +02:00
%token FLOAT_SYM
%token GEOMETRY_SYM
2000-07-31 21:29:14 +02:00
%token INT_SYM
%token LIMIT
%token LONGBLOB
%token LONGTEXT
%token MEDIUMBLOB
%token MEDIUMINT
%token MEDIUMTEXT
%token NUMERIC_SYM
%token PRECISION
%token QUICK
2000-07-31 21:29:14 +02:00
%token REAL
%token SIGNED_SYM
2000-07-31 21:29:14 +02:00
%token SMALLINT
%token STRING_SYM
%token TEXT_SYM
%token TIMESTAMP
%token TIMESTAMP_ADD
%token TIMESTAMP_DIFF
2000-07-31 21:29:14 +02:00
%token TIME_SYM
%token TINYBLOB
%token TINYINT
%token TINYTEXT
%token ULONGLONG_NUM
2000-07-31 21:29:14 +02:00
%token UNSIGNED
%token VARBINARY
%token VARCHAR
%token VARYING
2000-07-31 21:29:14 +02:00
%token ZEROFILL
2003-06-23 12:56:44 +05:00
%token ADDDATE_SYM
%token AGAINST
2000-07-31 21:29:14 +02:00
%token ATAN
%token BETWEEN_SYM
%token BIT_AND
%token BIT_OR
%token BIT_XOR
2000-07-31 21:29:14 +02:00
%token CASE_SYM
%token CONCAT
%token CONCAT_WS
2000-07-31 21:29:14 +02:00
%token CURDATE
%token CURTIME
%token DATABASE
%token DATE_ADD_INTERVAL
%token DATE_SUB_INTERVAL
%token DAY_HOUR_SYM
2003-06-23 12:56:44 +05:00
%token DAY_MICROSECOND_SYM
2000-07-31 21:29:14 +02:00
%token DAY_MINUTE_SYM
%token DAY_SECOND_SYM
%token DAY_SYM
%token DECODE_SYM
2001-12-13 03:36:36 +02:00
%token DES_ENCRYPT_SYM
%token DES_DECRYPT_SYM
2000-07-31 21:29:14 +02:00
%token ELSE
%token ELT_FUNC
%token ENCODE_SYM
%token ENGINE_SYM
%token ENGINES_SYM
2000-07-31 21:29:14 +02:00
%token ENCRYPT
%token EXPORT_SET
%token EXTRACT_SYM
%token FIELD_FUNC
%token FORMAT_SYM
%token FOR_SYM
%token FRAC_SECOND_SYM
2000-07-31 21:29:14 +02:00
%token FROM_UNIXTIME
%token GEOMCOLLFROMTEXT
%token GEOMFROMTEXT
%token GEOMFROMWKB
2002-02-22 15:24:42 +04:00
%token GEOMETRYCOLLECTION
%token GROUP_CONCAT_SYM
2000-07-31 21:29:14 +02:00
%token GROUP_UNIQUE_USERS
%token GET_FORMAT
2003-06-23 12:56:44 +05:00
%token HOUR_MICROSECOND_SYM
2000-07-31 21:29:14 +02:00
%token HOUR_MINUTE_SYM
%token HOUR_SECOND_SYM
%token HOUR_SYM
%token IDENTIFIED_SYM
%token IF
%token INSERT_METHOD
2000-07-31 21:29:14 +02:00
%token INTERVAL_SYM
%token LAST_INSERT_ID
%token LEFT
%token LINEFROMTEXT
2002-02-22 15:24:42 +04:00
%token LINESTRING
2000-07-31 21:29:14 +02:00
%token LOCATE
%token MAKE_SET_SYM
%token MASTER_POS_WAIT
2003-06-23 12:56:44 +05:00
%token MICROSECOND_SYM
%token MINUTE_MICROSECOND_SYM
2000-07-31 21:29:14 +02:00
%token MINUTE_SECOND_SYM
%token MINUTE_SYM
%token MODE_SYM
2000-07-31 21:29:14 +02:00
%token MODIFY_SYM
%token MONTH_SYM
%token MLINEFROMTEXT
%token MPOINTFROMTEXT
%token MPOLYFROMTEXT
2002-02-22 15:24:42 +04:00
%token MULTILINESTRING
%token MULTIPOINT
%token MULTIPOLYGON
2000-07-31 21:29:14 +02:00
%token NOW_SYM
%token OLD_PASSWORD
2000-07-31 21:29:14 +02:00
%token PASSWORD
%token POINTFROMTEXT
%token POINT_SYM
%token POLYFROMTEXT
2002-02-22 15:24:42 +04:00
%token POLYGON
2000-07-31 21:29:14 +02:00
%token POSITION_SYM
%token PROCEDURE
%token QUARTER_SYM
2000-07-31 21:29:14 +02:00
%token RAND
%token REPLACE
%token RIGHT
%token ROUND
%token SECOND_SYM
2003-06-23 12:56:44 +05:00
%token SECOND_MICROSECOND_SYM
%token SHARE_SYM
%token SP_FUNC
2003-06-23 12:56:44 +05:00
%token SUBDATE_SYM
2000-07-31 21:29:14 +02:00
%token SUBSTRING
%token SUBSTRING_INDEX
%token TRIM
%token UDA_CHAR_SUM
%token UDA_FLOAT_SUM
%token UDA_INT_SUM
%token UDF_CHAR_FUNC
%token UDF_FLOAT_FUNC
%token UDF_INT_FUNC
%token UNIQUE_USERS
%token UNIX_TIMESTAMP
%token USER
%token UTC_DATE_SYM
%token UTC_TIME_SYM
%token UTC_TIMESTAMP_SYM
2000-07-31 21:29:14 +02:00
%token WEEK_SYM
%token WHEN_SYM
%token WORK_SYM
2000-07-31 21:29:14 +02:00
%token YEAR_MONTH_SYM
%token YEAR_SYM
%token YEARWEEK
%token BENCHMARK_SYM
%token END
%token THEN_SYM
2000-07-31 21:29:14 +02:00
%token SQL_BIG_RESULT
%token SQL_CACHE_SYM
%token SQL_CALC_FOUND_ROWS
%token SQL_NO_CACHE_SYM
2000-07-31 21:29:14 +02:00
%token SQL_SMALL_RESULT
%token SQL_BUFFER_RESULT
2000-07-31 21:29:14 +02:00
%token CURSOR_SYM
%token ELSEIF_SYM
%token ITERATE_SYM
%token LEAVE_SYM
%token LOOP_SYM
%token REPEAT_SYM
%token UNTIL_SYM
%token WHILE_SYM
%token ASENSITIVE_SYM
%token INSENSITIVE_SYM
%token SENSITIVE_SYM
2001-09-19 19:45:13 -06:00
%token ISSUER_SYM
%token SUBJECT_SYM
%token CIPHER_SYM
2001-09-19 19:45:13 -06:00
%token BEFORE_SYM
2000-07-31 21:29:14 +02:00
%left SET_VAR
%left OR_OR_CONCAT OR
%left AND
%left BETWEEN_SYM CASE_SYM WHEN_SYM THEN_SYM ELSE
%left EQ EQUAL_SYM GE GT_SYM LE LT NE IS LIKE REGEXP IN_SYM
%left '|'
%left '&'
%left SHIFT_LEFT SHIFT_RIGHT
%left '-' '+'
%left '*' '/' '%' DIV_SYM MOD_SYM
%left XOR
%left '^'
%left NEG '~'
2000-07-31 21:29:14 +02:00
%right NOT
%right BINARY COLLATE_SYM
2000-07-31 21:29:14 +02:00
%type <lex_str>
IDENT IDENT_QUOTED TEXT_STRING REAL_NUM FLOAT_NUM NUM LONG_NUM HEX_NUM
LEX_HOSTNAME ULONGLONG_NUM field_ident select_alias ident ident_or_text
UNDERSCORE_CHARSET IDENT_sys TEXT_STRING_sys TEXT_STRING_literal
NCHAR_STRING opt_component key_cache_name
SP_FUNC ident_or_spfunc sp_opt_label
2000-07-31 21:29:14 +02:00
%type <lex_str_ptr>
opt_table_alias
%type <table>
table_ident references
2000-07-31 21:29:14 +02:00
%type <simple_string>
remember_name remember_end opt_ident opt_db text_or_password
opt_escape opt_constraint
2000-07-31 21:29:14 +02:00
%type <string>
text_string opt_gconcat_separator
2000-07-31 21:29:14 +02:00
%type <num>
type int_type real_type order_dir opt_field_spec lock_option
2000-07-31 21:29:14 +02:00
udf_type if_exists opt_local opt_table_options table_options
table_option opt_if_not_exists opt_no_write_to_binlog opt_var_type
opt_var_ident_type delete_option opt_temporary all_or_any opt_distinct
opt_ignore_leaves fulltext_options
2000-07-31 21:29:14 +02:00
%type <ulong_num>
ULONG_NUM raid_types merge_insert_types
2000-07-31 21:29:14 +02:00
%type <ulonglong_number>
ulonglong_num
2000-07-31 21:29:14 +02:00
%type <lock_type>
replace_lock_option opt_low_priority insert_lock_option load_data_lock
2000-07-31 21:29:14 +02:00
%type <item>
2001-12-10 17:51:07 +02:00
literal text_literal insert_ident order_ident
2000-07-31 21:29:14 +02:00
simple_ident select_item2 expr opt_expr opt_else sum_expr in_sum_expr
table_wild no_in_expr expr_expr simple_expr no_and_expr udf_expr
using_list expr_or_default set_expr_or_default interval_expr
2002-12-19 21:15:09 +02:00
param_marker singlerow_subselect singlerow_subselect_init
2003-12-11 16:05:51 +00:00
signed_literal NUM_literal
exists_subselect exists_subselect_init sp_opt_default
2000-07-31 21:29:14 +02:00
%type <item_list>
expr_list sp_expr_list udf_expr_list udf_expr_list2 when_list
ident_list ident_list_arg
2000-07-31 21:29:14 +02:00
%type <key_type>
key_type opt_unique_or_fulltext constraint_key_type
2000-07-31 21:29:14 +02:00
2002-02-22 15:24:42 +04:00
%type <key_alg>
key_alg opt_btree_or_rtree
2000-07-31 21:29:14 +02:00
%type <string_list>
key_usage_list
%type <key_part>
key_part
%type <table_list>
join_table_list join_table
2000-07-31 21:29:14 +02:00
%type <udf>
UDF_CHAR_FUNC UDF_FLOAT_FUNC UDF_INT_FUNC
UDA_CHAR_SUM UDA_FLOAT_SUM UDA_INT_SUM
%type <date_time_type> date_time_type;
2000-07-31 21:29:14 +02:00
%type <interval> interval
%type <interval_time_st> interval_time_st
%type <db_type> storage_engines
2000-07-31 21:29:14 +02:00
%type <row_type> row_types
%type <tx_isolation> isolation_types
2001-11-06 00:05:45 +02:00
%type <ha_rkey_mode> handler_rkey_mode
2003-08-21 14:15:25 +05:00
%type <cast_type> cast_type cast_type_finalize
2000-07-31 21:29:14 +02:00
%type <udf_type> udf_func_type
%type <symbol> FUNC_ARG0 FUNC_ARG1 FUNC_ARG2 FUNC_ARG3 keyword
%type <lex_user> user grant_user
%type <charset>
opt_collate
charset_name
charset_name_or_default
2003-04-05 18:56:15 +05:00
old_or_new_charset_name
old_or_new_charset_name_or_default
2003-01-09 15:37:59 +04:00
collation_name
collation_name_or_default
%type <variable> internal_variable_name
2002-10-27 23:27:00 +02:00
%type <select_lex> in_subselect in_subselect_init
%type <boolfunc2creator> comp_op
2000-07-31 21:29:14 +02:00
%type <NONE>
2001-12-17 19:59:20 +02:00
query verb_clause create change select do drop insert replace insert2
insert_values update delete truncate rename
2003-08-26 00:15:49 -07:00
show describe load alter optimize keycache preload flush
reset purge begin commit rollback savepoint
slave master_def master_defs master_file_def slave_until_opts
repair restore backup analyze check start checksum
field_list field_list_item field_spec kill column_def key_def
2003-08-26 00:15:49 -07:00
keycache_list assign_to_keycache preload_list preload_keys
2000-07-31 21:29:14 +02:00
select_item_list select_item values_list no_braces
opt_limit_clause delete_limit_clause fields opt_values values
2000-07-31 21:29:14 +02:00
procedure_list procedure_list2 procedure_item
when_list2 expr_list2 udf_expr_list3 handler
2000-07-31 21:29:14 +02:00
opt_precision opt_ignore opt_column opt_restrict
grant revoke set lock unlock string_list field_options field_option
field_opt_list opt_binary table_lock_list table_lock
ref_list opt_on_delete opt_on_delete_list opt_on_delete_item use
opt_delete_options opt_delete_option varchar nchar nvarchar
opt_outer table_list table_name opt_option opt_place
2000-07-31 21:29:14 +02:00
opt_attribute opt_attribute_list attribute column_list column_list_id
opt_column_list grant_privileges opt_table user_list grant_option
grant_privilege grant_privilege_list
flush_options flush_option
2000-07-31 21:29:14 +02:00
equal optional_braces opt_key_definition key_usage_list2
opt_mi_check_type opt_to mi_check_types normal_join
table_to_table_list table_to_table opt_table_list opt_as
2001-11-06 00:05:45 +02:00
handler_rkey_function handler_read_or_scan
2002-11-21 15:56:48 +02:00
single_multi table_wild_list table_wild_one opt_wild
union_clause union_list union_option
precision subselect_start opt_and charset
subselect_end select_var_list select_var_list_init help opt_len
opt_extended_describe
statement sp_suid
sp_c_chistics sp_a_chistics sp_chistic sp_c_chistic sp_a_chistic
END_OF_INPUT
2000-07-31 21:29:14 +02:00
%type <NONE> call sp_proc_stmts sp_proc_stmt
%type <num> sp_decl_idents sp_opt_inout sp_handler_type sp_hcond_list
%type <spcondtype> sp_cond sp_hcond
%type <spblock> sp_decls sp_decl
%type <lex> sp_cursor_stmt
2000-07-31 21:29:14 +02:00
%type <NONE>
'-' '+' '*' '/' '%' '(' ')'
',' '!' '{' '}' '&' '|' AND OR OR_OR_CONCAT BETWEEN_SYM CASE_SYM
THEN_SYM WHEN_SYM DIV_SYM MOD_SYM
2000-07-31 21:29:14 +02:00
%%
query:
END_OF_INPUT
{
THD *thd= YYTHD;
if (!thd->bootstrap &&
2003-05-05 14:54:37 -04:00
(!(thd->lex->select_lex.options & OPTION_FOUND_COMMENT)))
{
send_error(thd,ER_EMPTY_QUERY);
YYABORT;
}
else
{
thd->lex->sql_command= SQLCOM_EMPTY_QUERY;
}
2000-07-31 21:29:14 +02:00
}
| verb_clause END_OF_INPUT {};
2000-07-31 21:29:14 +02:00
verb_clause:
statement
| begin
;
/* Verb clauses, except begin */
statement:
2000-07-31 21:29:14 +02:00
alter
| analyze
| backup
| call
2000-07-31 21:29:14 +02:00
| change
| check
| checksum
2000-07-31 21:29:14 +02:00
| commit
| create
| delete
| describe
2001-12-17 19:59:20 +02:00
| do
2000-07-31 21:29:14 +02:00
| drop
| flush
2000-07-31 21:29:14 +02:00
| grant
| handler
| help
2000-07-31 21:29:14 +02:00
| insert
| kill
2000-07-31 21:29:14 +02:00
| load
| lock
| optimize
2003-08-26 00:15:49 -07:00
| keycache
| preload
| purge
| rename
| repair
2000-07-31 21:29:14 +02:00
| replace
2000-10-14 11:16:17 +03:00
| reset
| restore
2000-07-31 21:29:14 +02:00
| revoke
| rollback
| savepoint
2000-07-31 21:29:14 +02:00
| select
| set
| show
2000-08-22 00:39:08 +03:00
| slave
| start
| truncate
2000-07-31 21:29:14 +02:00
| unlock
| update
2002-10-28 17:44:19 +04:00
| use
;
2002-10-28 17:44:19 +04:00
/* help */
help:
HELP_SYM ident_or_text
2002-10-28 17:44:19 +04:00
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_HELP;
lex->help_arg= $2.str;
2002-10-30 17:38:07 +04:00
};
2000-07-31 21:29:14 +02:00
/* change master */
change:
2000-08-22 00:39:08 +03:00
CHANGE MASTER_SYM TO_SYM
2000-07-31 21:29:14 +02:00
{
LEX *lex = Lex;
lex->sql_command = SQLCOM_CHANGE_MASTER;
bzero((char*) &lex->mi, sizeof(lex->mi));
}
master_defs
{}
;
2000-07-31 21:29:14 +02:00
master_defs:
master_def
| master_defs ',' master_def;
2000-08-22 00:39:08 +03:00
2000-07-31 21:29:14 +02:00
master_def:
2003-03-17 21:56:34 +04:00
MASTER_HOST_SYM EQ TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
Lex->mi.host = $3.str;
}
|
2003-03-17 21:56:34 +04:00
MASTER_USER_SYM EQ TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
Lex->mi.user = $3.str;
}
|
2003-03-17 21:56:34 +04:00
MASTER_PASSWORD_SYM EQ TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
Lex->mi.password = $3.str;
}
|
MASTER_PORT_SYM EQ ULONG_NUM
2000-07-31 21:29:14 +02:00
{
Lex->mi.port = $3;
}
|
MASTER_CONNECT_RETRY_SYM EQ ULONG_NUM
2000-07-31 21:29:14 +02:00
{
Lex->mi.connect_retry = $3;
}
| MASTER_SSL_SYM EQ ULONG_NUM
{
Lex->mi.ssl= $3 ?
LEX_MASTER_INFO::SSL_ENABLE : LEX_MASTER_INFO::SSL_DISABLE;
}
| MASTER_SSL_CA_SYM EQ TEXT_STRING_sys
{
Lex->mi.ssl_ca= $3.str;
}
| MASTER_SSL_CAPATH_SYM EQ TEXT_STRING_sys
{
Lex->mi.ssl_capath= $3.str;
}
| MASTER_SSL_CERT_SYM EQ TEXT_STRING_sys
{
Lex->mi.ssl_cert= $3.str;
}
| MASTER_SSL_CIPHER_SYM EQ TEXT_STRING_sys
{
Lex->mi.ssl_cipher= $3.str;
}
| MASTER_SSL_KEY_SYM EQ TEXT_STRING_sys
{
Lex->mi.ssl_key= $3.str;
}
|
master_file_def
2002-10-16 16:55:08 +03:00
;
2000-07-31 21:29:14 +02:00
master_file_def:
MASTER_LOG_FILE_SYM EQ TEXT_STRING_sys
{
Lex->mi.log_file_name = $3.str;
2002-10-16 16:55:08 +03:00
}
| MASTER_LOG_POS_SYM EQ ulonglong_num
{
Lex->mi.pos = $3;
/*
If the user specified a value < BIN_LOG_HEADER_SIZE, adjust it
instead of causing subsequent errors.
We need to do it in this file, because only there we know that
MASTER_LOG_POS has been explicitely specified. On the contrary
in change_master() (sql_repl.cc) we cannot distinguish between 0
(MASTER_LOG_POS explicitely specified as 0) and 0 (unspecified),
whereas we want to distinguish (specified 0 means "read the binlog
from 0" (4 in fact), unspecified means "don't change the position
(keep the preceding value)").
*/
Lex->mi.pos = max(BIN_LOG_HEADER_SIZE, Lex->mi.pos);
}
| RELAY_LOG_FILE_SYM EQ TEXT_STRING_sys
{
Lex->mi.relay_log_name = $3.str;
}
| RELAY_LOG_POS_SYM EQ ULONG_NUM
{
Lex->mi.relay_log_pos = $3;
/* Adjust if < BIN_LOG_HEADER_SIZE (same comment as Lex->mi.pos) */
Lex->mi.relay_log_pos = max(BIN_LOG_HEADER_SIZE, Lex->mi.relay_log_pos);
}
2002-10-16 16:55:08 +03:00
;
2000-07-31 21:29:14 +02:00
/* create a table */
create:
CREATE opt_table_options TABLE_SYM opt_if_not_exists table_ident
{
THD *thd= YYTHD;
2000-07-31 21:29:14 +02:00
LEX *lex=Lex;
lex->sql_command= SQLCOM_CREATE_TABLE;
if (!lex->select_lex.add_table_to_list(thd,$5,
($2 &
HA_LEX_CREATE_TMP_TABLE ?
&tmp_table_alias :
(LEX_STRING*) 0),
TL_OPTION_UPDATING,
2002-11-23 18:54:15 +02:00
((using_update_log)?
TL_READ_NO_INSERT:
TL_READ)))
2000-07-31 21:29:14 +02:00
YYABORT;
lex->create_list.empty();
lex->key_list.empty();
lex->col_list.empty();
lex->change=NullS;
bzero((char*) &lex->create_info,sizeof(lex->create_info));
lex->create_info.options=$2 | $4;
lex->create_info.db_type= (enum db_type) lex->thd->variables.table_type;
lex->create_info.default_table_charset= thd->variables.collation_database;
lex->name=0;
2000-07-31 21:29:14 +02:00
}
create2
{ Lex->current_select= &Lex->select_lex; }
2002-02-22 15:24:42 +04:00
| CREATE opt_unique_or_fulltext INDEX ident key_alg ON table_ident
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_CREATE_INDEX;
if (!lex->current_select->add_table_to_list(lex->thd, $7, NULL,
TL_OPTION_UPDATING))
2000-07-31 21:29:14 +02:00
YYABORT;
lex->create_list.empty();
lex->key_list.empty();
lex->col_list.empty();
lex->change=NullS;
2000-07-31 21:29:14 +02:00
}
'(' key_list ')'
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
2002-02-22 15:24:42 +04:00
lex->key_list.push_back(new Key($2,$4.str, $5, lex->col_list));
lex->col_list.empty();
2000-07-31 21:29:14 +02:00
}
| CREATE DATABASE opt_if_not_exists ident
{ Lex->create_info.default_table_charset=NULL; }
2003-01-09 15:37:59 +04:00
opt_create_database_options
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command=SQLCOM_CREATE_DB;
lex->name=$4.str;
lex->create_info.options=$3;
2000-07-31 21:29:14 +02:00
}
| CREATE udf_func_type FUNCTION_SYM ident_or_spfunc
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->udf.name = $4;
lex->udf.type= $2;
2000-07-31 21:29:14 +02:00
}
create_function_tail
{}
| CREATE PROCEDURE ident
{
LEX *lex= Lex;
sp_head *sp;
if (lex->sphead)
{
net_printf(YYTHD, ER_SP_NO_RECURSIVE_CREATE, "PROCEDURE");
YYABORT;
}
/* Order is important here: new - reset - init */
sp= new sp_head();
sp->reset_thd_mem_root(YYTHD);
sp->init(lex);
sp->m_type= TYPE_ENUM_PROCEDURE;
lex->sphead= sp;
/*
* We have to turn of CLIENT_MULTI_QUERIES while parsing a
* stored procedure, otherwise yylex will chop it into pieces
* at each ';'.
*/
sp->m_old_cmq= YYTHD->client_capabilities & CLIENT_MULTI_QUERIES;
YYTHD->client_capabilities &= (~CLIENT_MULTI_QUERIES);
}
'('
{
LEX *lex= Lex;
lex->sphead->m_param_begin= lex->tok_start+1;
}
sp_pdparam_list
')'
{
LEX *lex= Lex;
lex->sphead->m_param_end= lex->tok_start;
lex->spcont->set_params();
bzero((char *)&lex->sp_chistics, sizeof(st_sp_chistics));
}
sp_c_chistics
{
LEX *lex= Lex;
lex->sphead->m_chistics= &lex->sp_chistics;
lex->sphead->m_body_begin= lex->tok_start;
}
sp_proc_stmt
{
LEX *lex= Lex;
2003-12-15 13:24:16 +01:00
lex->sphead->init_strings(YYTHD, lex, &$3);
lex->sql_command= SQLCOM_CREATE_PROCEDURE;
/* Restore flag if it was cleared above */
if (lex->sphead->m_old_cmq)
YYTHD->client_capabilities |= CLIENT_MULTI_QUERIES;
lex->sphead->restore_thd_mem_root(YYTHD);
}
;
ident_or_spfunc:
2003-03-18 16:12:50 +01:00
IDENT_sys { $$= $1; }
| SP_FUNC { $$= $1; }
;
create_function_tail:
2003-03-18 16:12:50 +01:00
RETURNS_SYM udf_type UDF_SONAME_SYM TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_CREATE_FUNCTION;
lex->udf.returns=(Item_result) $2;
lex->udf.dl=$4.str;
2002-10-16 16:55:08 +03:00
}
| '('
{
LEX *lex= Lex;
sp_head *sp;
if (lex->sphead)
{
net_printf(YYTHD, ER_SP_NO_RECURSIVE_CREATE, "FUNCTION");
YYABORT;
}
/* Order is important here: new - reset - init */
sp= new sp_head();
sp->reset_thd_mem_root(YYTHD);
sp->init(lex);
sp->m_type= TYPE_ENUM_FUNCTION;
lex->sphead= sp;
/*
* We have to turn of CLIENT_MULTI_QUERIES while parsing a
* stored procedure, otherwise yylex will chop it into pieces
* at each ';'.
*/
sp->m_old_cmq= YYTHD->client_capabilities & CLIENT_MULTI_QUERIES;
YYTHD->client_capabilities &= ~CLIENT_MULTI_QUERIES;
lex->sphead->m_param_begin= lex->tok_start+1;
}
sp_fdparam_list ')'
{
LEX *lex= Lex;
lex->spcont->set_params();
lex->sphead->m_param_end= lex->tok_start;
}
RETURNS_SYM
{
Lex->sphead->m_returns_begin= Lex->tok_start;
}
type
{
LEX *lex= Lex;
lex->sphead->m_returns_end= lex->tok_start;
lex->sphead->m_returns= (enum enum_field_types)$8;
bzero((char *)&lex->sp_chistics, sizeof(st_sp_chistics));
}
sp_c_chistics
{
LEX *lex= Lex;
lex->sphead->m_chistics= &lex->sp_chistics;
lex->sphead->m_body_begin= lex->tok_start;
}
sp_proc_stmt
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
lex->sql_command= SQLCOM_CREATE_SPFUNCTION;
2003-12-15 13:24:16 +01:00
sp->init_strings(YYTHD, lex, &lex->udf.name);
/* Restore flag if it was cleared above */
if (sp->m_old_cmq)
YYTHD->client_capabilities |= CLIENT_MULTI_QUERIES;
sp->restore_thd_mem_root(YYTHD);
}
;
sp_a_chistics:
/* Empty */ {}
| sp_a_chistics sp_a_chistic {}
;
sp_c_chistics:
/* Empty */ {}
| sp_c_chistics sp_c_chistic {}
;
/* Characteristics for both create and alter */
sp_chistic:
COMMENT_SYM TEXT_STRING_sys { Lex->sp_chistics.comment= $2; }
| sp_suid { }
;
/* Alter characteristics */
sp_a_chistic:
sp_chistic { }
| NAME_SYM ident { Lex->name= $2.str; }
;
/* Create characteristics */
sp_c_chistic:
sp_chistic { }
| LANGUAGE_SYM SQL_SYM { }
| DETERMINISTIC_SYM { Lex->sp_chistics.detistic= TRUE; }
| NOT DETERMINISTIC_SYM { Lex->sp_chistics.detistic= FALSE; }
;
sp_suid:
SQL_SYM SECURITY_SYM DEFINER_SYM
{
Lex->sp_chistics.suid= IS_SUID;
}
| SQL_SYM SECURITY_SYM INVOKER_SYM
{
Lex->sp_chistics.suid= IS_NOT_SUID;
}
;
call:
CALL_SYM ident_or_spfunc
{
LEX *lex = Lex;
lex->sql_command= SQLCOM_CALL;
lex->udf.name= $2;
lex->value_list.empty();
}
'(' sp_cparam_list ')' {}
;
/* CALL parameters */
sp_cparam_list:
/* Empty */
| sp_cparams
;
sp_cparams:
sp_cparams ',' expr
{
Lex->value_list.push_back($3);
}
| expr
{
Lex->value_list.push_back($1);
}
;
/* Stored FUNCTION parameter declaration list */
sp_fdparam_list:
/* Empty */
| sp_fdparams
;
sp_fdparams:
sp_fdparams ',' sp_fdparam
| sp_fdparam
;
sp_fdparam:
ident type sp_opt_locator
{
LEX *lex= Lex;
sp_pcontext *spc= lex->spcont;
if (spc->find_pvar(&$1, TRUE))
{
net_printf(YYTHD, ER_SP_DUP_PARAM, $1.str);
YYABORT;
}
spc->push_pvar(&$1, (enum enum_field_types)$2, sp_param_in);
}
;
/* Stored PROCEDURE parameter declaration list */
sp_pdparam_list:
/* Empty */
| sp_pdparams
;
sp_pdparams:
sp_pdparams ',' sp_pdparam
| sp_pdparam
;
sp_pdparam:
sp_opt_inout ident type sp_opt_locator
{
LEX *lex= Lex;
sp_pcontext *spc= lex->spcont;
if (spc->find_pvar(&$2, TRUE))
{
net_printf(YYTHD, ER_SP_DUP_PARAM, $2.str);
YYABORT;
}
spc->push_pvar(&$2, (enum enum_field_types)$3,
(sp_param_mode_t)$1);
}
;
sp_opt_inout:
/* Empty */ { $$= sp_param_in; }
| IN_SYM { $$= sp_param_in; }
| OUT_SYM { $$= sp_param_out; }
| INOUT_SYM { $$= sp_param_inout; }
;
sp_opt_locator:
/* Empty */
| AS LOCATOR_SYM
;
sp_proc_stmts:
/* Empty */ {}
| sp_proc_stmts sp_proc_stmt ';'
;
sp_decls:
/* Empty */
{
$$.vars= $$.conds= $$.hndlrs= $$.curs= 0;
}
| sp_decls sp_decl ';'
{
$$.vars= $1.vars + $2.vars;
$$.conds= $1.conds + $2.conds;
$$.hndlrs= $1.hndlrs + $2.hndlrs;
$$.curs= $1.curs + $2.curs;
}
;
sp_decl:
DECLARE_SYM sp_decl_idents type sp_opt_default
{
LEX *lex= Lex;
uint max= lex->spcont->current_framesize();
enum enum_field_types type= (enum enum_field_types)$3;
Item *it= $4;
for (uint i = max-$2 ; i < max ; i++)
{
lex->spcont->set_type(i, type);
if (! it)
lex->spcont->set_isset(i, FALSE);
else
{
sp_instr_set *in= new sp_instr_set(lex->sphead->instructions(),
i, it, type);
lex->sphead->add_instr(in);
lex->spcont->set_isset(i, TRUE);
}
}
$$.vars= $2;
$$.conds= $$.hndlrs= $$.curs= 0;
}
| DECLARE_SYM ident CONDITION_SYM FOR_SYM sp_cond
{
LEX *lex= Lex;
sp_pcontext *spc= lex->spcont;
if (spc->find_cond(&$2, TRUE))
{
net_printf(YYTHD, ER_SP_DUP_COND, $2.str);
YYABORT;
}
YYTHD->lex->spcont->push_cond(&$2, $5);
$$.vars= $$.hndlrs= $$.curs= 0;
$$.conds= 1;
}
| DECLARE_SYM sp_handler_type HANDLER_SYM FOR_SYM
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_pcontext *ctx= lex->spcont;
sp_instr_hpush_jump *i=
new sp_instr_hpush_jump(sp->instructions(), $2,
ctx->current_framesize());
sp->add_instr(i);
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
ctx->add_handler();
}
sp_hcond_list sp_proc_stmt
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_label_t *hlab= lex->spcont->pop_label(); /* After this hdlr */
if ($2 == SP_HANDLER_CONTINUE)
sp->add_instr(new sp_instr_hreturn(sp->instructions(),
lex->spcont->current_framesize()));
else
{ /* EXIT or UNDO handler, just jump to the end of the block */
sp_instr_jump *i= new sp_instr_jump(sp->instructions());
sp->add_instr(i);
sp->push_backpatch(i, lex->spcont->last_label()); /* Block end */
}
lex->sphead->backpatch(hlab);
$$.vars= $$.conds= $$.curs= 0;
$$.hndlrs= $6;
}
| DECLARE_SYM ident CURSOR_SYM FOR_SYM sp_cursor_stmt
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_pcontext *spc= lex->spcont;
uint offp;
sp_instr_cpush *i;
if (spc->find_cursor(&$2, &offp, TRUE))
{
net_printf(YYTHD, ER_SP_DUP_CURS, $2.str);
delete $5;
YYABORT;
}
i= new sp_instr_cpush(sp->instructions(), $5);
sp->add_instr(i);
lex->spcont->push_cursor(&$2);
$$.vars= $$.conds= $$.hndlrs= 0;
$$.curs= 1;
}
;
sp_cursor_stmt:
{
Lex->sphead->reset_lex(YYTHD);
/* We use statement here just be able to get a better
error message. Using 'select' works too, but will then
result in a generic "syntax error" if a non-select
statement is given. */
}
statement
{
LEX *lex= Lex;
if (lex->sql_command != SQLCOM_SELECT)
{
send_error(YYTHD, ER_SP_BAD_CURSOR_QUERY);
YYABORT;
}
if (lex->result)
{
send_error(YYTHD, ER_SP_BAD_CURSOR_SELECT);
YYABORT;
}
lex->sp_lex_in_use= TRUE;
$$= lex;
lex->sphead->restore_lex(YYTHD);
}
;
sp_handler_type:
EXIT_SYM { $$= SP_HANDLER_EXIT; }
| CONTINUE_SYM { $$= SP_HANDLER_CONTINUE; }
/* | UNDO_SYM { QQ No yet } */
;
sp_hcond_list:
sp_hcond
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_instr_hpush_jump *i= (sp_instr_hpush_jump *)sp->last_instruction();
i->add_condition($1);
$$= 1;
}
| sp_hcond_list ',' sp_hcond
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_instr_hpush_jump *i= (sp_instr_hpush_jump *)sp->last_instruction();
i->add_condition($3);
$$= $1 + 1;
}
;
sp_cond:
ULONG_NUM
{ /* mysql errno */
$$= (sp_cond_type_t *)YYTHD->alloc(sizeof(sp_cond_type_t));
$$->type= sp_cond_type_t::number;
$$->mysqlerr= $1;
}
| SQLSTATE_SYM opt_value TEXT_STRING_literal
{ /* SQLSTATE */
uint len= ($3.length < sizeof($$->sqlstate)-1 ?
$3.length : sizeof($$->sqlstate)-1);
$$= (sp_cond_type_t *)YYTHD->alloc(sizeof(sp_cond_type_t));
$$->type= sp_cond_type_t::state;
memcpy($$->sqlstate, $3.str, len);
$$->sqlstate[len]= '\0';
}
;
opt_value:
/* Empty */ {}
| VALUE_SYM {}
;
sp_hcond:
sp_cond
{
$$= $1;
}
| ident /* CONDITION name */
{
$$= Lex->spcont->find_cond(&$1);
if ($$ == NULL)
{
net_printf(YYTHD, ER_SP_COND_MISMATCH, $1.str);
YYABORT;
}
}
| SQLWARNING_SYM /* SQLSTATEs 01??? */
{
$$= (sp_cond_type_t *)YYTHD->alloc(sizeof(sp_cond_type_t));
$$->type= sp_cond_type_t::warning;
}
| NOT FOUND_SYM /* SQLSTATEs 02??? */
{
$$= (sp_cond_type_t *)YYTHD->alloc(sizeof(sp_cond_type_t));
$$->type= sp_cond_type_t::notfound;
}
| SQLEXCEPTION_SYM /* All other SQLSTATEs */
{
$$= (sp_cond_type_t *)YYTHD->alloc(sizeof(sp_cond_type_t));
$$->type= sp_cond_type_t::exception;
}
;
sp_decl_idents:
ident
{
LEX *lex= Lex;
sp_pcontext *spc= lex->spcont;
if (spc->find_pvar(&$1, TRUE))
{
net_printf(YYTHD, ER_SP_DUP_VAR, $1.str);
YYABORT;
}
spc->push_pvar(&$1, (enum_field_types)0, sp_param_in);
$$= 1;
}
| sp_decl_idents ',' ident
{
LEX *lex= Lex;
sp_pcontext *spc= lex->spcont;
if (spc->find_pvar(&$3, TRUE))
{
net_printf(YYTHD, ER_SP_DUP_VAR, $3.str);
YYABORT;
}
spc->push_pvar(&$3, (enum_field_types)0, sp_param_in);
$$= $1 + 1;
}
;
sp_opt_default:
/* Empty */ { $$ = NULL; }
| DEFAULT expr { $$ = $2; }
;
sp_proc_stmt:
{
Lex->sphead->reset_lex(YYTHD);
}
statement
{
LEX *lex= Lex;
if (lex->sql_command == SQLCOM_SELECT && !lex->result)
{
/* We maybe have one or more SELECT without INTO */
lex->sphead->m_multi_results= TRUE;
}
/* Don't add an instruction for empty SET statements.
** (This happens if the SET only contained local variables,
** which get their set instructions generated separately.)
*/
if (lex->sql_command != SQLCOM_SET_OPTION ||
! lex->var_list.is_empty())
{
/* Currently we can't handle queries inside a FUNCTION,
** because of the way table locking works.
** This is unfortunate, and limits the usefulness of functions
** a great deal, but it's nothing we can do about this at the
** moment.
*/
if (lex->sphead->m_type == TYPE_ENUM_FUNCTION &&
lex->sql_command != SQLCOM_SET_OPTION)
{
send_error(YYTHD, ER_SP_BADSTATEMENT);
YYABORT;
}
else
{
sp_instr_stmt *i=new sp_instr_stmt(lex->sphead->instructions());
i->set_lex(lex);
lex->sphead->add_instr(i);
lex->sp_lex_in_use= TRUE;
}
}
lex->sphead->restore_lex(YYTHD);
}
| RETURN_SYM expr
{
LEX *lex= Lex;
if (lex->sphead->m_type == TYPE_ENUM_PROCEDURE)
{
send_error(YYTHD, ER_SP_BADRETURN);
YYABORT;
}
else
{
sp_instr_freturn *i;
if ($2->type() == Item::SUBSELECT_ITEM)
{ /* QQ For now, just disallow subselects as values */
send_error(lex->thd, ER_SP_BADSTATEMENT);
YYABORT;
}
i= new sp_instr_freturn(lex->sphead->instructions(),
$2, lex->sphead->m_returns);
lex->sphead->add_instr(i);
lex->sphead->m_has_return= TRUE;
}
}
| IF sp_if END IF {}
| CASE_SYM WHEN_SYM
{
Lex->sphead->m_simple_case= FALSE;
}
sp_case END CASE_SYM {}
| CASE_SYM expr WHEN_SYM
{
/* We "fake" this by using an anonymous variable which we
set to the expression. Note that all WHENs are evaluate
at the same frame level, so we then know that it's the
top-most variable in the frame. */
LEX *lex= Lex;
uint offset= lex->spcont->current_framesize();
sp_instr_set *i = new sp_instr_set(lex->sphead->instructions(),
offset, $2, MYSQL_TYPE_STRING);
LEX_STRING dummy;
dummy.str= (char *)"";
dummy.length= 0;
lex->spcont->push_pvar(&dummy, MYSQL_TYPE_STRING, sp_param_in);
lex->sphead->add_instr(i);
lex->sphead->m_simple_case= TRUE;
}
sp_case END CASE_SYM
{
Lex->spcont->pop_pvar();
}
| sp_labeled_control
{}
| { /* Unlabeled controls get a secret label. */
LEX *lex= Lex;
lex->spcont->push_label((char *)"", lex->sphead->instructions());
}
sp_unlabeled_control
{
LEX *lex= Lex;
lex->sphead->backpatch(lex->spcont->pop_label());
}
| LEAVE_SYM IDENT
{
LEX *lex= Lex;
sp_head *sp = lex->sphead;
sp_label_t *lab= lex->spcont->find_label($2.str);
if (! lab)
{
net_printf(YYTHD, ER_SP_LILABEL_MISMATCH, "LEAVE", $2.str);
YYABORT;
}
else
{
sp_instr_jump *i= new sp_instr_jump(sp->instructions());
sp->push_backpatch(i, lab); /* Jumping forward */
sp->add_instr(i);
}
}
| ITERATE_SYM IDENT
{
LEX *lex= Lex;
sp_label_t *lab= lex->spcont->find_label($2.str);
if (! lab || lab->isbegin)
{
net_printf(YYTHD, ER_SP_LILABEL_MISMATCH, "ITERATE", $2.str);
YYABORT;
}
else
{
uint ip= lex->sphead->instructions();
sp_instr_jump *i= new sp_instr_jump(ip, lab->ip); /* Jump back */
lex->sphead->add_instr(i);
}
}
| OPEN_SYM ident
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
uint offset;
sp_instr_copen *i;
if (! lex->spcont->find_cursor(&$2, &offset))
{
net_printf(YYTHD, ER_SP_CURSOR_MISMATCH, $2.str);
YYABORT;
}
i= new sp_instr_copen(sp->instructions(), offset);
sp->add_instr(i);
}
| FETCH_SYM ident INTO
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
uint offset;
sp_instr_cfetch *i;
if (! lex->spcont->find_cursor(&$2, &offset))
{
net_printf(YYTHD, ER_SP_CURSOR_MISMATCH, $2.str);
YYABORT;
}
i= new sp_instr_cfetch(sp->instructions(), offset);
sp->add_instr(i);
}
sp_fetch_list
{ }
| CLOSE_SYM ident
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
uint offset;
sp_instr_cclose *i;
if (! lex->spcont->find_cursor(&$2, &offset))
{
net_printf(YYTHD, ER_SP_CURSOR_MISMATCH, $2.str);
YYABORT;
}
i= new sp_instr_cclose(sp->instructions(), offset);
sp->add_instr(i);
}
;
sp_fetch_list:
ident
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_pcontext *spc= lex->spcont;
sp_pvar_t *spv;
if (!spc || !(spv = spc->find_pvar(&$1)))
{
net_printf(YYTHD, ER_SP_UNDECLARED_VAR, $1.str);
YYABORT;
}
else
{ /* An SP local variable */
sp_instr_cfetch *i= (sp_instr_cfetch *)sp->last_instruction();
i->add_to_varlist(spv);
spv->isset= TRUE;
}
}
|
sp_fetch_list ',' ident
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_pcontext *spc= lex->spcont;
sp_pvar_t *spv;
if (!spc || !(spv = spc->find_pvar(&$3)))
{
net_printf(YYTHD, ER_SP_UNDECLARED_VAR, $3.str);
YYABORT;
}
else
{ /* An SP local variable */
sp_instr_cfetch *i= (sp_instr_cfetch *)sp->last_instruction();
i->add_to_varlist(spv);
spv->isset= TRUE;
}
}
;
2000-07-31 21:29:14 +02:00
sp_if:
expr THEN_SYM
{
sp_head *sp= Lex->sphead;
sp_pcontext *ctx= Lex->spcont;
uint ip= sp->instructions();
sp_instr_jump_if_not *i = new sp_instr_jump_if_not(ip, $1);
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
sp->add_instr(i);
}
sp_proc_stmts
{
sp_head *sp= Lex->sphead;
sp_pcontext *ctx= Lex->spcont;
uint ip= sp->instructions();
sp_instr_jump *i = new sp_instr_jump(ip);
sp->add_instr(i);
sp->backpatch(ctx->pop_label());
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
}
sp_elseifs
{
LEX *lex= Lex;
lex->sphead->backpatch(lex->spcont->pop_label());
}
;
sp_elseifs:
/* Empty */
| ELSEIF_SYM sp_if
| ELSE sp_proc_stmts
;
sp_case:
expr THEN_SYM
{
sp_head *sp= Lex->sphead;
sp_pcontext *ctx= Lex->spcont;
uint ip= sp->instructions();
sp_instr_jump_if_not *i;
if (! sp->m_simple_case)
i= new sp_instr_jump_if_not(ip, $1);
else
{ /* Simple case: <caseval> = <whenval> */
2003-11-21 14:00:40 +01:00
LEX_STRING ivar;
2003-12-01 18:24:09 +01:00
ivar.str= (char *)"_tmp_";
2003-11-21 14:00:40 +01:00
ivar.length= 5;
Item *var= (Item*) new Item_splocal(ivar,
ctx->current_framesize()-1);
2003-11-19 16:59:35 +01:00
Item *expr= new Item_func_eq(var, $1);
i= new sp_instr_jump_if_not(ip, expr);
}
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
sp->add_instr(i);
}
sp_proc_stmts
{
sp_head *sp= Lex->sphead;
sp_pcontext *ctx= Lex->spcont;
uint ip= sp->instructions();
sp_instr_jump *i = new sp_instr_jump(ip);
sp->add_instr(i);
sp->backpatch(ctx->pop_label());
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
}
sp_whens
{
LEX *lex= Lex;
lex->sphead->backpatch(lex->spcont->pop_label());
}
;
sp_whens:
/* Empty */ {}
| WHEN_SYM sp_case {}
| ELSE sp_proc_stmts {}
;
sp_labeled_control:
IDENT ':'
{
LEX *lex= Lex;
sp_label_t *lab= lex->spcont->find_label($1.str);
if (lab)
{
net_printf(YYTHD, ER_SP_LABEL_REDEFINE, $1.str);
YYABORT;
}
else
{
lex->spcont->push_label($1.str,
lex->sphead->instructions());
}
}
sp_unlabeled_control sp_opt_label
{
LEX *lex= Lex;
if ($5.str)
{
sp_label_t *lab= lex->spcont->find_label($5.str);
if (!lab ||
my_strcasecmp(system_charset_info, $5.str, lab->name) != 0)
{
net_printf(YYTHD, ER_SP_LABEL_MISMATCH, $5.str);
YYABORT;
}
}
lex->sphead->backpatch(lex->spcont->pop_label());
}
;
sp_opt_label:
/* Empty */
{ $$.str= NULL; $$.length= 0; }
| IDENT
{ $$= $1; }
;
sp_unlabeled_control:
BEGIN_SYM
{ /* QQ This is just a dummy for grouping declarations and statements
together. No [[NOT] ATOMIC] yet, and we need to figure out how
make it coexist with the existing BEGIN COMMIT/ROLLBACK. */
LEX *lex= Lex;
sp_label_t *lab= lex->spcont->last_label();
lab->isbegin= TRUE;
/* Scope duplicate checking */
lex->spcont->push_scope();
}
sp_decls
sp_proc_stmts
END
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
sp_pcontext *ctx= lex->spcont;
sp->backpatch(ctx->last_label()); /* We always has a label */
ctx->pop_pvar($3.vars);
ctx->pop_cond($3.conds);
ctx->pop_cursor($3.curs);
if ($3.hndlrs)
sp->add_instr(new sp_instr_hpop(sp->instructions(),$3.hndlrs));
if ($3.curs)
sp->add_instr(new sp_instr_cpop(sp->instructions(), $3.curs));
ctx->pop_scope();
}
| LOOP_SYM
sp_proc_stmts END LOOP_SYM
{
LEX *lex= Lex;
uint ip= lex->sphead->instructions();
sp_label_t *lab= lex->spcont->last_label(); /* Jumping back */
sp_instr_jump *i = new sp_instr_jump(ip, lab->ip);
lex->sphead->add_instr(i);
}
| WHILE_SYM expr DO_SYM
{
LEX *lex= Lex;
sp_head *sp= lex->sphead;
uint ip= sp->instructions();
sp_instr_jump_if_not *i = new sp_instr_jump_if_not(ip, $2);
/* Jumping forward */
sp->push_backpatch(i, lex->spcont->last_label());
sp->add_instr(i);
}
sp_proc_stmts END WHILE_SYM
{
LEX *lex= Lex;
uint ip= lex->sphead->instructions();
sp_label_t *lab= lex->spcont->last_label(); /* Jumping back */
sp_instr_jump *i = new sp_instr_jump(ip, lab->ip);
lex->sphead->add_instr(i);
}
| REPEAT_SYM sp_proc_stmts UNTIL_SYM expr END REPEAT_SYM
{
LEX *lex= Lex;
uint ip= lex->sphead->instructions();
sp_label_t *lab= lex->spcont->last_label(); /* Jumping back */
sp_instr_jump_if_not *i = new sp_instr_jump_if_not(ip, $4, lab->ip);
lex->sphead->add_instr(i);
}
;
2000-07-31 21:29:14 +02:00
create2:
2003-08-11 22:44:43 +03:00
'(' create2a {}
2002-10-16 16:55:08 +03:00
| opt_create_table_options create3 {}
| LIKE table_ident
{
LEX *lex=Lex;
if (!(lex->name= (char *)$2))
YYABORT;
}
| '(' LIKE table_ident ')'
{
LEX *lex=Lex;
if (!(lex->name= (char *)$3))
YYABORT;
}
2003-08-11 22:44:43 +03:00
;
2000-07-31 21:29:14 +02:00
create2a:
field_list ')' opt_create_table_options create3 {}
| create_select ')' { Select->set_braces(1);} union_opt {}
;
2000-07-31 21:29:14 +02:00
create3:
/* empty */ {}
| opt_duplicate opt_as create_select
{ Select->set_braces(0);} union_clause {}
| opt_duplicate opt_as '(' create_select ')'
{ Select->set_braces(1);} union_opt {}
;
create_select:
SELECT_SYM
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->lock_option= (using_update_log) ? TL_READ_NO_INSERT : TL_READ;
if (lex->sql_command == SQLCOM_INSERT)
lex->sql_command= SQLCOM_INSERT_SELECT;
else if (lex->sql_command == SQLCOM_REPLACE)
lex->sql_command= SQLCOM_REPLACE_SELECT;
lex->current_select->table_list.save_and_clear(&lex->save_list);
mysql_init_select(lex);
2003-05-17 10:05:07 +03:00
lex->current_select->parsing_place= SELECT_LEX_NODE::SELECT_LIST;
2000-07-31 21:29:14 +02:00
}
2003-05-17 10:05:07 +03:00
select_options select_item_list
{
Select->parsing_place= SELECT_LEX_NODE::NO_MATTER;
}
2003-08-11 22:44:43 +03:00
opt_select_from
{ Lex->current_select->table_list.push_front(&Lex->save_list); }
2003-08-11 22:44:43 +03:00
;
2000-07-31 21:29:14 +02:00
opt_as:
/* empty */ {}
| AS {};
2003-01-09 15:37:59 +04:00
opt_create_database_options:
/* empty */ {}
| create_database_options {};
create_database_options:
create_database_option {}
| create_database_options create_database_option {};
create_database_option:
opt_default COLLATE_SYM collation_name_or_default
{ Lex->create_info.default_table_charset=$3; }
| opt_default charset charset_name_or_default
{ Lex->create_info.default_table_charset=$3; }
;
2003-01-09 15:37:59 +04:00
2000-07-31 21:29:14 +02:00
opt_table_options:
2000-08-22 00:39:08 +03:00
/* empty */ { $$= 0; }
| table_options { $$= $1;};
2000-07-31 21:29:14 +02:00
table_options:
table_option { $$=$1; }
| table_option table_options { $$= $1 | $2; };
2000-08-22 00:39:08 +03:00
2000-07-31 21:29:14 +02:00
table_option:
TEMPORARY { $$=HA_LEX_CREATE_TMP_TABLE; };
2000-07-31 21:29:14 +02:00
opt_if_not_exists:
2000-08-22 00:39:08 +03:00
/* empty */ { $$= 0; }
| IF NOT EXISTS { $$=HA_LEX_CREATE_IF_NOT_EXISTS; };
2000-07-31 21:29:14 +02:00
opt_create_table_options:
/* empty */
| create_table_options;
2000-07-31 21:29:14 +02:00
create_table_options_space_separated:
create_table_option
| create_table_option create_table_options_space_separated;
2000-07-31 21:29:14 +02:00
create_table_options:
create_table_option
2002-12-07 12:35:57 +01:00
| create_table_option create_table_options
| create_table_option ',' create_table_options;
2000-07-31 21:29:14 +02:00
create_table_option:
ENGINE_SYM opt_equal storage_engines { Lex->create_info.db_type= $3; }
| TYPE_SYM opt_equal storage_engines { Lex->create_info.db_type= $3; WARN_DEPRECATED("TYPE=database_engine","ENGINE=database_engine"); }
| MAX_ROWS opt_equal ulonglong_num { Lex->create_info.max_rows= $3; Lex->create_info.used_fields|= HA_CREATE_USED_MAX_ROWS;}
| MIN_ROWS opt_equal ulonglong_num { Lex->create_info.min_rows= $3; Lex->create_info.used_fields|= HA_CREATE_USED_MIN_ROWS;}
| AVG_ROW_LENGTH opt_equal ULONG_NUM { Lex->create_info.avg_row_length=$3; Lex->create_info.used_fields|= HA_CREATE_USED_AVG_ROW_LENGTH;}
2003-03-17 21:56:34 +04:00
| PASSWORD opt_equal TEXT_STRING_sys { Lex->create_info.password=$3.str; }
| COMMENT_SYM opt_equal TEXT_STRING_sys { Lex->create_info.comment=$3.str; }
| AUTO_INC opt_equal ulonglong_num { Lex->create_info.auto_increment_value=$3; Lex->create_info.used_fields|= HA_CREATE_USED_AUTO;}
| PACK_KEYS_SYM opt_equal ULONG_NUM { Lex->create_info.table_options|= $3 ? HA_OPTION_PACK_KEYS : HA_OPTION_NO_PACK_KEYS; Lex->create_info.used_fields|= HA_CREATE_USED_PACK_KEYS;}
| PACK_KEYS_SYM opt_equal DEFAULT { Lex->create_info.table_options&= ~(HA_OPTION_PACK_KEYS | HA_OPTION_NO_PACK_KEYS); Lex->create_info.used_fields|= HA_CREATE_USED_PACK_KEYS;}
| CHECKSUM_SYM opt_equal ULONG_NUM { Lex->create_info.table_options|= $3 ? HA_OPTION_CHECKSUM : HA_OPTION_NO_CHECKSUM; }
| DELAY_KEY_WRITE_SYM opt_equal ULONG_NUM { Lex->create_info.table_options|= $3 ? HA_OPTION_DELAY_KEY_WRITE : HA_OPTION_NO_DELAY_KEY_WRITE; }
| ROW_FORMAT_SYM opt_equal row_types { Lex->create_info.row_type= $3; }
| RAID_TYPE opt_equal raid_types { Lex->create_info.raid_type= $3; Lex->create_info.used_fields|= HA_CREATE_USED_RAID;}
| RAID_CHUNKS opt_equal ULONG_NUM { Lex->create_info.raid_chunks= $3; Lex->create_info.used_fields|= HA_CREATE_USED_RAID;}
| RAID_CHUNKSIZE opt_equal ULONG_NUM { Lex->create_info.raid_chunksize= $3*RAID_BLOCK_SIZE; Lex->create_info.used_fields|= HA_CREATE_USED_RAID;}
| UNION_SYM opt_equal '(' table_list ')'
2000-09-14 02:39:07 +03:00
{
/* Move the union list to the merge_list */
LEX *lex=Lex;
TABLE_LIST *table_list= lex->select_lex.get_table_list();
lex->create_info.merge_list= lex->select_lex.table_list;
2000-09-14 02:39:07 +03:00
lex->create_info.merge_list.elements--;
lex->create_info.merge_list.first= (byte*) (table_list->next);
lex->select_lex.table_list.elements=1;
lex->select_lex.table_list.next= (byte**) &(table_list->next);
2000-09-14 02:39:07 +03:00
table_list->next=0;
lex->create_info.used_fields|= HA_CREATE_USED_UNION;
2000-09-14 02:39:07 +03:00
}
| DEFAULT charset opt_equal charset_name_or_default
{
Lex->create_info.default_table_charset= $4;
Lex->create_info.used_fields|= HA_CREATE_USED_DEFAULT_CHARSET;
}
| charset opt_equal charset_name_or_default
{
Lex->create_info.table_charset= $3;
Lex->create_info.used_fields|= HA_CREATE_USED_CHARSET;
}
| DEFAULT COLLATE_SYM opt_equal collation_name_or_default
{
Lex->create_info.table_charset= $4;
Lex->create_info.used_fields|= HA_CREATE_USED_DEFAULT_CHARSET;
}
2003-01-09 15:37:59 +04:00
| COLLATE_SYM opt_equal collation_name_or_default
{
Lex->create_info.table_charset= $3;
Lex->create_info.used_fields|= HA_CREATE_USED_CHARSET;
}
| INSERT_METHOD opt_equal merge_insert_types { Lex->create_info.merge_insert_method= $3; Lex->create_info.used_fields|= HA_CREATE_USED_INSERT_METHOD;}
2003-03-17 21:56:34 +04:00
| DATA_SYM DIRECTORY_SYM opt_equal TEXT_STRING_sys
{ Lex->create_info.data_file_name= $4.str; }
| INDEX DIRECTORY_SYM opt_equal TEXT_STRING_sys { Lex->create_info.index_file_name= $4.str; };
2000-07-31 21:29:14 +02:00
storage_engines:
ident_or_text
{
$$ = ha_resolve_by_name($1.str,$1.length);
if ($$ == DB_TYPE_UNKNOWN) {
net_printf(YYTHD, ER_UNKNOWN_STORAGE_ENGINE, $1.str);
YYABORT;
}
};
2000-07-31 21:29:14 +02:00
row_types:
DEFAULT { $$= ROW_TYPE_DEFAULT; }
| FIXED_SYM { $$= ROW_TYPE_FIXED; }
| DYNAMIC_SYM { $$= ROW_TYPE_DYNAMIC; }
| COMPRESSED_SYM { $$= ROW_TYPE_COMPRESSED; };
2000-07-31 21:29:14 +02:00
raid_types:
RAID_STRIPED_SYM { $$= RAID_TYPE_0; }
| RAID_0_SYM { $$= RAID_TYPE_0; }
| ULONG_NUM { $$=$1;};
2000-07-31 21:29:14 +02:00
merge_insert_types:
NO_SYM { $$= MERGE_INSERT_DISABLED; }
| FIRST_SYM { $$= MERGE_INSERT_TO_FIRST; }
| LAST_SYM { $$= MERGE_INSERT_TO_LAST; };
2000-07-31 21:29:14 +02:00
opt_select_from:
opt_limit_clause {}
2002-12-26 17:48:19 +02:00
| FROM DUAL_SYM {}
| select_from select_lock_type;
2000-07-31 21:29:14 +02:00
udf_func_type:
/* empty */ { $$ = UDFTYPE_FUNCTION; }
| AGGREGATE_SYM { $$ = UDFTYPE_AGGREGATE; };
2000-07-31 21:29:14 +02:00
udf_type:
STRING_SYM {$$ = (int) STRING_RESULT; }
| REAL {$$ = (int) REAL_RESULT; }
| INT_SYM {$$ = (int) INT_RESULT; };
2000-07-31 21:29:14 +02:00
field_list:
field_list_item
| field_list ',' field_list_item;
2000-07-31 21:29:14 +02:00
field_list_item:
2003-02-26 01:03:47 +02:00
column_def
| key_def
;
column_def:
field_spec check_constraint
2000-07-31 21:29:14 +02:00
| field_spec references
{
Lex->col_list.empty(); /* Alloced by sql_alloc */
}
2003-02-26 01:03:47 +02:00
;
key_def:
2003-02-26 01:03:47 +02:00
key_type opt_ident key_alg '(' key_list ')'
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->key_list.push_back(new Key($1,$2, $3, lex->col_list));
lex->col_list.empty(); /* Alloced by sql_alloc */
2000-07-31 21:29:14 +02:00
}
| opt_constraint constraint_key_type opt_ident key_alg '(' key_list ')'
{
LEX *lex=Lex;
const char *key_name= $3 ? $3:$1;
lex->key_list.push_back(new Key($2, key_name, $4, lex->col_list));
lex->col_list.empty(); /* Alloced by sql_alloc */
}
| opt_constraint FOREIGN KEY_SYM opt_ident '(' key_list ')' references
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->key_list.push_back(new foreign_key($4, lex->col_list,
$8,
lex->ref_list,
lex->fk_delete_opt,
lex->fk_update_opt,
lex->fk_match_option));
lex->col_list.empty(); /* Alloced by sql_alloc */
2000-07-31 21:29:14 +02:00
}
| opt_constraint check_constraint
2000-07-31 21:29:14 +02:00
{
Lex->col_list.empty(); /* Alloced by sql_alloc */
}
;
check_constraint:
/* empty */
| CHECK_SYM expr
;
2000-07-31 21:29:14 +02:00
opt_constraint:
/* empty */ { $$=(char*) 0; }
| CONSTRAINT opt_ident { $$=$2; };
2000-07-31 21:29:14 +02:00
field_spec:
field_ident
{
LEX *lex=Lex;
lex->length=lex->dec=0; lex->type=0; lex->interval=0;
lex->default_value=0;
lex->comment=0;
lex->charset=NULL;
2000-07-31 21:29:14 +02:00
}
type opt_attribute
{
LEX *lex=Lex;
if (add_field_to_list(lex->thd, $1.str,
2000-07-31 21:29:14 +02:00
(enum enum_field_types) $3,
lex->length,lex->dec,lex->type,
lex->default_value, lex->comment,
lex->change,lex->interval,lex->charset,
lex->uint_geom_type))
2000-07-31 21:29:14 +02:00
YYABORT;
};
2000-07-31 21:29:14 +02:00
type:
int_type opt_len field_options { $$=$1; }
2000-07-31 21:29:14 +02:00
| real_type opt_precision field_options { $$=$1; }
| FLOAT_SYM float_options field_options { $$=FIELD_TYPE_FLOAT; }
| BIT_SYM opt_len { Lex->length=(char*) "1";
$$=FIELD_TYPE_TINY; }
| BOOL_SYM { Lex->length=(char*) "1";
$$=FIELD_TYPE_TINY; }
| BOOLEAN_SYM { Lex->length=(char*) "1";
$$=FIELD_TYPE_TINY; }
| char '(' NUM ')' opt_binary { Lex->length=$3.str;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_STRING; }
| char opt_binary { Lex->length=(char*) "1";
$$=FIELD_TYPE_STRING; }
| nchar '(' NUM ')' { Lex->length=$3.str;
2003-08-11 22:44:43 +03:00
$$=FIELD_TYPE_STRING;
Lex->charset=national_charset_info; }
| nchar { Lex->length=(char*) "1";
2003-08-11 22:44:43 +03:00
$$=FIELD_TYPE_STRING;
Lex->charset=national_charset_info; }
| BINARY '(' NUM ')' { Lex->length=$3.str;
2003-01-29 17:31:20 +04:00
Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_STRING; }
| varchar '(' NUM ')' opt_binary { Lex->length=$3.str;
$$=FIELD_TYPE_VAR_STRING; }
| nvarchar '(' NUM ')' { Lex->length=$3.str;
$$=FIELD_TYPE_VAR_STRING;
Lex->charset=national_charset_info; }
2000-07-31 21:29:14 +02:00
| VARBINARY '(' NUM ')' { Lex->length=$3.str;
2003-01-29 17:31:20 +04:00
Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_VAR_STRING; }
| YEAR_SYM opt_len field_options { $$=FIELD_TYPE_YEAR; }
2000-07-31 21:29:14 +02:00
| DATE_SYM { $$=FIELD_TYPE_DATE; }
| TIME_SYM { $$=FIELD_TYPE_TIME; }
| TIMESTAMP
{
if (YYTHD->variables.sql_mode & MODE_MAXDB)
$$=FIELD_TYPE_DATETIME;
else
$$=FIELD_TYPE_TIMESTAMP;
}
2000-07-31 21:29:14 +02:00
| TIMESTAMP '(' NUM ')' { Lex->length=$3.str;
$$=FIELD_TYPE_TIMESTAMP; }
| DATETIME { $$=FIELD_TYPE_DATETIME; }
2003-01-29 17:31:20 +04:00
| TINYBLOB { Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_TINY_BLOB; }
2003-01-29 17:31:20 +04:00
| BLOB_SYM opt_len { Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_BLOB; }
2003-01-29 17:31:20 +04:00
| GEOMETRY_SYM { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_GEOMETRY;
2002-02-22 15:24:42 +04:00
$$=FIELD_TYPE_GEOMETRY; }
| GEOMETRYCOLLECTION { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_GEOMETRYCOLLECTION;
$$=FIELD_TYPE_GEOMETRY; }
| POINT_SYM { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_POINT;
$$=FIELD_TYPE_GEOMETRY; }
| MULTIPOINT { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_MULTIPOINT;
$$=FIELD_TYPE_GEOMETRY; }
| LINESTRING { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_LINESTRING;
$$=FIELD_TYPE_GEOMETRY; }
| MULTILINESTRING { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_MULTILINESTRING;
$$=FIELD_TYPE_GEOMETRY; }
| POLYGON { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_POLYGON;
$$=FIELD_TYPE_GEOMETRY; }
| MULTIPOLYGON { Lex->charset=&my_charset_bin;
Lex->uint_geom_type= (uint) Field::GEOM_MULTIPOLYGON;
$$=FIELD_TYPE_GEOMETRY; }
2003-01-29 17:31:20 +04:00
| MEDIUMBLOB { Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_MEDIUM_BLOB; }
2003-01-29 17:31:20 +04:00
| LONGBLOB { Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_LONG_BLOB; }
2003-01-29 17:31:20 +04:00
| LONG_SYM VARBINARY { Lex->charset=&my_charset_bin;
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_MEDIUM_BLOB; }
| LONG_SYM varchar opt_binary { $$=FIELD_TYPE_MEDIUM_BLOB; }
| TINYTEXT opt_binary { $$=FIELD_TYPE_TINY_BLOB; }
| TEXT_SYM opt_len opt_binary { $$=FIELD_TYPE_BLOB; }
| MEDIUMTEXT opt_binary { $$=FIELD_TYPE_MEDIUM_BLOB; }
| LONGTEXT opt_binary { $$=FIELD_TYPE_LONG_BLOB; }
2000-07-31 21:29:14 +02:00
| DECIMAL_SYM float_options field_options
{ $$=FIELD_TYPE_DECIMAL;}
| NUMERIC_SYM float_options field_options
{ $$=FIELD_TYPE_DECIMAL;}
| FIXED_SYM float_options field_options
{ $$=FIELD_TYPE_DECIMAL;}
2002-10-25 15:08:47 +05:00
| ENUM {Lex->interval_list.empty();} '(' string_list ')' opt_binary
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->interval=typelib(lex->interval_list);
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_ENUM;
}
2002-10-25 15:08:47 +05:00
| SET { Lex->interval_list.empty();} '(' string_list ')' opt_binary
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->interval=typelib(lex->interval_list);
2000-07-31 21:29:14 +02:00
$$=FIELD_TYPE_SET;
}
| LONG_SYM opt_binary { $$=FIELD_TYPE_MEDIUM_BLOB; }
| SERIAL_SYM
{
$$=FIELD_TYPE_LONGLONG;
Lex->type|= (AUTO_INCREMENT_FLAG | NOT_NULL_FLAG | UNSIGNED_FLAG |
UNIQUE_FLAG);
}
;
2000-07-31 21:29:14 +02:00
char:
CHAR_SYM {}
;
nchar:
NCHAR_SYM {}
| NATIONAL_SYM CHAR_SYM {}
;
2000-08-22 00:39:08 +03:00
2000-07-31 21:29:14 +02:00
varchar:
char VARYING {}
| VARCHAR {}
;
nvarchar:
NATIONAL_SYM VARCHAR {}
| NVARCHAR_SYM {}
| NCHAR_SYM VARCHAR {}
| NATIONAL_SYM CHAR_SYM VARYING {}
| NCHAR_SYM VARYING {}
;
2000-07-31 21:29:14 +02:00
int_type:
INT_SYM { $$=FIELD_TYPE_LONG; }
| TINYINT { $$=FIELD_TYPE_TINY; }
| SMALLINT { $$=FIELD_TYPE_SHORT; }
| MEDIUMINT { $$=FIELD_TYPE_INT24; }
| BIGINT { $$=FIELD_TYPE_LONGLONG; };
2000-07-31 21:29:14 +02:00
real_type:
REAL { $$= YYTHD->variables.sql_mode & MODE_REAL_AS_FLOAT ?
2000-07-31 21:29:14 +02:00
FIELD_TYPE_FLOAT : FIELD_TYPE_DOUBLE; }
| DOUBLE_SYM { $$=FIELD_TYPE_DOUBLE; }
| DOUBLE_SYM PRECISION { $$=FIELD_TYPE_DOUBLE; };
2000-07-31 21:29:14 +02:00
float_options:
/* empty */ {}
| '(' NUM ')' { Lex->length=$2.str; }
| precision {};
precision:
'(' NUM ',' NUM ')'
{
LEX *lex=Lex;
lex->length=$2.str; lex->dec=$4.str;
};
2000-07-31 21:29:14 +02:00
field_options:
/* empty */ {}
| field_opt_list {};
2000-07-31 21:29:14 +02:00
field_opt_list:
field_opt_list field_option {}
| field_option {};
2000-07-31 21:29:14 +02:00
field_option:
SIGNED_SYM {}
| UNSIGNED { Lex->type|= UNSIGNED_FLAG;}
| ZEROFILL { Lex->type|= UNSIGNED_FLAG | ZEROFILL_FLAG; };
2000-07-31 21:29:14 +02:00
opt_len:
/* empty */ { Lex->length=(char*) 0; } /* use default length */
| '(' NUM ')' { Lex->length= $2.str; };
2000-07-31 21:29:14 +02:00
opt_precision:
/* empty */ {}
| precision {};
2000-07-31 21:29:14 +02:00
opt_attribute:
/* empty */ {}
| opt_attribute_list {};
2000-07-31 21:29:14 +02:00
opt_attribute_list:
opt_attribute_list attribute {}
| attribute;
2000-07-31 21:29:14 +02:00
attribute:
NULL_SYM { Lex->type&= ~ NOT_NULL_FLAG; }
| NOT NULL_SYM { Lex->type|= NOT_NULL_FLAG; }
2003-12-11 16:05:51 +00:00
| DEFAULT signed_literal { Lex->default_value=$2; }
2000-07-31 21:29:14 +02:00
| AUTO_INC { Lex->type|= AUTO_INCREMENT_FLAG | NOT_NULL_FLAG; }
| SERIAL_SYM DEFAULT VALUE_SYM
{ Lex->type|= AUTO_INCREMENT_FLAG | NOT_NULL_FLAG | UNIQUE_FLAG; }
| opt_primary KEY_SYM { Lex->type|= PRI_KEY_FLAG | NOT_NULL_FLAG; }
2000-07-31 21:29:14 +02:00
| UNIQUE_SYM { Lex->type|= UNIQUE_FLAG; }
| UNIQUE_SYM KEY_SYM { Lex->type|= UNIQUE_KEY_FLAG; }
| COMMENT_SYM TEXT_STRING_sys { Lex->comment= &$2; }
2003-08-11 22:44:43 +03:00
| COLLATE_SYM collation_name
{
2003-03-16 17:19:24 +04:00
if (Lex->charset && !my_charset_same(Lex->charset,$2))
{
net_printf(YYTHD,ER_COLLATION_CHARSET_MISMATCH,
$2->name,Lex->charset->csname);
YYABORT;
}
else
{
Lex->charset=$2;
}
}
;
2000-07-31 21:29:14 +02:00
charset:
CHAR_SYM SET {}
| CHARSET {}
;
charset_name:
ident_or_text
{
if (!($$=get_charset_by_csname($1.str,MY_CS_PRIMARY,MYF(0))))
{
net_printf(YYTHD,ER_UNKNOWN_CHARACTER_SET,$1.str);
YYABORT;
}
}
| BINARY { $$= &my_charset_bin; }
;
charset_name_or_default:
charset_name { $$=$1; }
| DEFAULT { $$=NULL; } ;
2000-07-31 21:29:14 +02:00
2003-04-05 18:56:15 +05:00
old_or_new_charset_name:
ident_or_text
{
if (!($$=get_charset_by_csname($1.str,MY_CS_PRIMARY,MYF(0))) &&
!($$=get_old_charset_by_name($1.str)))
{
net_printf(YYTHD,ER_UNKNOWN_CHARACTER_SET,$1.str);
YYABORT;
}
}
| BINARY { $$= &my_charset_bin; }
;
old_or_new_charset_name_or_default:
old_or_new_charset_name { $$=$1; }
| DEFAULT { $$=NULL; } ;
2003-01-09 15:37:59 +04:00
collation_name:
ident_or_text
2003-01-09 15:37:59 +04:00
{
if (!($$=get_charset_by_name($1.str,MYF(0))))
{
net_printf(YYTHD,ER_UNKNOWN_COLLATION,$1.str);
2003-01-09 15:37:59 +04:00
YYABORT;
}
};
opt_collate:
/* empty */ { $$=NULL; }
2003-04-05 18:56:15 +05:00
| COLLATE_SYM collation_name_or_default { $$=$2; }
;
2003-01-09 15:37:59 +04:00
collation_name_or_default:
collation_name { $$=$1; }
| DEFAULT { $$=NULL; } ;
opt_default:
/* empty */ {}
| DEFAULT {};
2000-07-31 21:29:14 +02:00
opt_binary:
/* empty */ { Lex->charset=NULL; }
2003-01-29 17:31:20 +04:00
| ASCII_SYM { Lex->charset=&my_charset_latin1; }
| BYTE_SYM { Lex->charset=&my_charset_bin; }
| BINARY { Lex->charset=&my_charset_bin; }
| UNICODE_SYM
{
if (!(Lex->charset=get_charset_by_csname("ucs2",MY_CS_PRIMARY,MYF(0))))
{
net_printf(YYTHD,ER_UNKNOWN_CHARACTER_SET,"ucs2");
YYABORT;
}
}
| charset charset_name { Lex->charset=$2; } ;
2000-07-31 21:29:14 +02:00
opt_primary:
/* empty */
| PRIMARY_SYM
;
2000-07-31 21:29:14 +02:00
references:
REFERENCES table_ident
{
LEX *lex=Lex;
lex->fk_delete_opt= lex->fk_update_opt= lex->fk_match_option= 0;
lex->ref_list.empty();
}
opt_ref_list
{
$$=$2;
2002-06-03 12:59:31 +03:00
};
opt_ref_list:
/* empty */ opt_on_delete {}
2002-06-03 12:59:31 +03:00
| '(' ref_list ')' opt_on_delete {};
ref_list:
ref_list ',' ident { Lex->ref_list.push_back(new key_part_spec($3.str)); }
2002-06-03 12:59:31 +03:00
| ident { Lex->ref_list.push_back(new key_part_spec($1.str)); };
2000-07-31 21:29:14 +02:00
opt_on_delete:
/* empty */ {}
| opt_on_delete_list {};
2000-07-31 21:29:14 +02:00
opt_on_delete_list:
opt_on_delete_list opt_on_delete_item {}
| opt_on_delete_item {};
2000-07-31 21:29:14 +02:00
opt_on_delete_item:
ON DELETE_SYM delete_option { Lex->fk_delete_opt= $3; }
| ON UPDATE_SYM delete_option { Lex->fk_update_opt= $3; }
| MATCH FULL { Lex->fk_match_option= foreign_key::FK_MATCH_FULL; }
| MATCH PARTIAL { Lex->fk_match_option= foreign_key::FK_MATCH_PARTIAL; }
2002-06-03 12:59:31 +03:00
| MATCH SIMPLE_SYM { Lex->fk_match_option= foreign_key::FK_MATCH_SIMPLE; };
2000-07-31 21:29:14 +02:00
delete_option:
RESTRICT { $$= (int) foreign_key::FK_OPTION_RESTRICT; }
| CASCADE { $$= (int) foreign_key::FK_OPTION_CASCADE; }
| SET NULL_SYM { $$= (int) foreign_key::FK_OPTION_SET_NULL; }
| NO_SYM ACTION { $$= (int) foreign_key::FK_OPTION_NO_ACTION; }
2002-06-03 12:59:31 +03:00
| SET DEFAULT { $$= (int) foreign_key::FK_OPTION_DEFAULT; };
2000-07-31 21:29:14 +02:00
key_type:
key_or_index { $$= Key::MULTIPLE; }
| FULLTEXT_SYM { $$= Key::FULLTEXT; }
| FULLTEXT_SYM key_or_index { $$= Key::FULLTEXT; }
2002-02-22 15:24:42 +04:00
| SPATIAL_SYM { $$= Key::SPATIAL; }
| SPATIAL_SYM key_or_index { $$= Key::SPATIAL; };
constraint_key_type:
PRIMARY_SYM KEY_SYM { $$= Key::PRIMARY; }
| UNIQUE_SYM { $$= Key::UNIQUE; }
| UNIQUE_SYM key_or_index { $$= Key::UNIQUE; };
2000-07-31 21:29:14 +02:00
key_or_index:
KEY_SYM {}
| INDEX {};
2000-07-31 21:29:14 +02:00
opt_keys_or_index:
/* empty */ {}
| keys_or_index
;
2000-07-31 21:29:14 +02:00
keys_or_index:
KEYS {}
| INDEX {}
| INDEXES {};
2000-07-31 21:29:14 +02:00
opt_unique_or_fulltext:
2000-07-31 21:29:14 +02:00
/* empty */ { $$= Key::MULTIPLE; }
| UNIQUE_SYM { $$= Key::UNIQUE; }
2003-06-16 00:13:23 +02:00
| FULLTEXT_SYM { $$= Key::FULLTEXT;}
| SPATIAL_SYM { $$= Key::SPATIAL; }
;
2002-02-22 15:24:42 +04:00
key_alg:
/* empty */ { $$= HA_KEY_ALG_UNDEF; }
| USING opt_btree_or_rtree { $$= $2; }
| TYPE_SYM opt_btree_or_rtree { $$= $2; };
2002-02-22 15:24:42 +04:00
opt_btree_or_rtree:
BTREE_SYM { $$= HA_KEY_ALG_BTREE; }
| RTREE_SYM { $$= HA_KEY_ALG_RTREE; }
2002-06-03 12:59:31 +03:00
| HASH_SYM { $$= HA_KEY_ALG_HASH; };
2000-07-31 21:29:14 +02:00
key_list:
key_list ',' key_part order_dir { Lex->col_list.push_back($3); }
| key_part order_dir { Lex->col_list.push_back($1); };
2000-07-31 21:29:14 +02:00
key_part:
ident { $$=new key_part_spec($1.str); }
| ident '(' NUM ')' { $$=new key_part_spec($1.str,(uint) atoi($3.str)); };
2000-07-31 21:29:14 +02:00
opt_ident:
/* empty */ { $$=(char*) 0; } /* Defaultlength */
| field_ident { $$=$1.str; };
2000-07-31 21:29:14 +02:00
opt_component:
/* empty */ { $$.str= 0; $$.length= 0; }
| '.' ident { $$=$2; };
2000-07-31 21:29:14 +02:00
string_list:
text_string { Lex->interval_list.push_back($1); }
| string_list ',' text_string { Lex->interval_list.push_back($3); };
2000-07-31 21:29:14 +02:00
/*
** Alter table
*/
alter:
ALTER opt_ignore TABLE_SYM table_ident
{
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
LEX *lex= thd->lex;
2000-07-31 21:29:14 +02:00
lex->sql_command = SQLCOM_ALTER_TABLE;
lex->name=0;
if (!lex->select_lex.add_table_to_list(thd, $4, NULL,
TL_OPTION_UPDATING))
2000-07-31 21:29:14 +02:00
YYABORT;
lex->drop_primary=0;
lex->create_list.empty();
lex->key_list.empty();
lex->col_list.empty();
lex->drop_list.empty();
lex->alter_list.empty();
2002-11-05 01:10:05 +02:00
lex->select_lex.init_order();
lex->select_lex.db=lex->name=0;
bzero((char*) &lex->create_info,sizeof(lex->create_info));
2000-07-31 21:29:14 +02:00
lex->create_info.db_type= DB_TYPE_DEFAULT;
lex->create_info.default_table_charset= thd->variables.collation_database;
lex->create_info.row_type= ROW_TYPE_NOT_USED;
lex->alter_keys_onoff=LEAVE_AS_IS;
lex->tablespace_op=NO_TABLESPACE_OP;
lex->simple_alter=1;
2000-07-31 21:29:14 +02:00
}
alter_list
{}
2003-01-09 15:37:59 +04:00
| ALTER DATABASE ident opt_create_database_options
{
LEX *lex=Lex;
lex->sql_command=SQLCOM_ALTER_DB;
lex->name=$3.str;
}
| ALTER PROCEDURE ident
{
LEX *lex= Lex;
bzero((char *)&lex->sp_chistics, sizeof(st_sp_chistics));
lex->name= 0;
}
sp_a_chistics
{
THD *thd= YYTHD;
LEX *lex=Lex;
lex->sql_command= SQLCOM_ALTER_PROCEDURE;
lex->udf.name= $3;
}
| ALTER FUNCTION_SYM ident
{
LEX *lex= Lex;
bzero((char *)&lex->sp_chistics, sizeof(st_sp_chistics));
lex->name= 0;
}
sp_a_chistics
{
THD *thd= YYTHD;
LEX *lex=Lex;
lex->sql_command= SQLCOM_ALTER_FUNCTION;
lex->udf.name= $3;
}
;
2000-07-31 21:29:14 +02:00
alter_list:
| DISCARD TABLESPACE { Lex->tablespace_op=DISCARD_TABLESPACE; }
| IMPORT TABLESPACE { Lex->tablespace_op=IMPORT_TABLESPACE; }
2000-11-11 12:27:34 -06:00
| alter_list_item
| alter_list ',' alter_list_item;
2000-07-31 21:29:14 +02:00
add_column:
ADD opt_column { Lex->change=0; };
2000-07-31 21:29:14 +02:00
alter_list_item:
add_column column_def opt_place { Lex->simple_alter=0; }
| ADD key_def { Lex->simple_alter=0; }
| add_column '(' field_list ')' { Lex->simple_alter=0; }
| CHANGE opt_column field_ident
{
LEX *lex=Lex;
lex->change= $3.str; lex->simple_alter=0;
}
field_spec opt_place
| MODIFY_SYM opt_column field_ident
{
LEX *lex=Lex;
lex->length=lex->dec=0; lex->type=0; lex->interval=0;
lex->default_value=0;
lex->comment=0;
lex->charset= NULL;
lex->simple_alter=0;
}
type opt_attribute
{
LEX *lex=Lex;
if (add_field_to_list(lex->thd,$3.str,
(enum enum_field_types) $5,
lex->length,lex->dec,lex->type,
lex->default_value, lex->comment,
$3.str, lex->interval, lex->charset,
lex->uint_geom_type))
YYABORT;
}
opt_place
2000-07-31 21:29:14 +02:00
| DROP opt_column field_ident opt_restrict
{
LEX *lex=Lex;
lex->drop_list.push_back(new Alter_drop(Alter_drop::COLUMN,
$3.str)); lex->simple_alter=0;
}
| DROP PRIMARY_SYM KEY_SYM
{
LEX *lex=Lex;
lex->drop_primary=1; lex->simple_alter=0;
}
| DROP FOREIGN KEY_SYM opt_ident { Lex->simple_alter=0; }
2000-07-31 21:29:14 +02:00
| DROP key_or_index field_ident
{
LEX *lex=Lex;
lex->drop_list.push_back(new Alter_drop(Alter_drop::KEY,
$3.str));
lex->simple_alter=0;
}
| DISABLE_SYM KEYS { Lex->alter_keys_onoff=DISABLE; }
| ENABLE_SYM KEYS { Lex->alter_keys_onoff=ENABLE; }
2003-12-11 16:05:51 +00:00
| ALTER opt_column field_ident SET DEFAULT signed_literal
{
LEX *lex=Lex;
lex->alter_list.push_back(new Alter_column($3.str,$6));
lex->simple_alter=0;
}
2000-07-31 21:29:14 +02:00
| ALTER opt_column field_ident DROP DEFAULT
{
LEX *lex=Lex;
lex->alter_list.push_back(new Alter_column($3.str,(Item*) 0));
lex->simple_alter=0;
}
| RENAME opt_to table_ident
2002-12-02 20:38:00 +01:00
{
LEX *lex=Lex;
lex->select_lex.db=$3->db.str;
lex->name= $3->table.str;
}
| create_table_options_space_separated { Lex->simple_alter=0; }
| order_clause { Lex->simple_alter=0; };
2000-07-31 21:29:14 +02:00
opt_column:
/* empty */ {}
| COLUMN_SYM {};
2000-07-31 21:29:14 +02:00
opt_ignore:
/* empty */ { Lex->duplicates=DUP_ERROR; }
| IGNORE_SYM { Lex->duplicates=DUP_IGNORE; };
2000-07-31 21:29:14 +02:00
opt_restrict:
/* empty */ {}
| RESTRICT {}
| CASCADE {};
2000-07-31 21:29:14 +02:00
opt_place:
/* empty */ {}
| AFTER_SYM ident { store_position_for_column($2.str); }
| FIRST_SYM { store_position_for_column(first_keyword); };
2000-07-31 21:29:14 +02:00
opt_to:
/* empty */ {}
| TO_SYM {}
| EQ {}
| AS {};
2000-07-31 21:29:14 +02:00
/*
SLAVE START and SLAVE STOP are deprecated. We keep them for compatibility.
To use UNTIL, one must use START SLAVE, not SLAVE START.
2002-11-21 15:56:48 +02:00
*/
2000-07-31 21:29:14 +02:00
slave:
START_SYM SLAVE slave_thread_opts
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_SLAVE_START;
lex->type = 0;
/* We'll use mi structure for UNTIL options */
bzero((char*) &lex->mi, sizeof(lex->mi));
}
slave_until
{}
2002-11-21 15:56:48 +02:00
| STOP_SYM SLAVE slave_thread_opts
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_SLAVE_STOP;
lex->type = 0;
}
| SLAVE START_SYM slave_thread_opts
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_SLAVE_START;
lex->type = 0;
}
| SLAVE STOP_SYM slave_thread_opts
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_SLAVE_STOP;
lex->type = 0;
}
;
2000-08-22 00:39:08 +03:00
start:
START_SYM TRANSACTION_SYM { Lex->sql_command = SQLCOM_BEGIN;}
{}
;
slave_thread_opts:
{ Lex->slave_thd_opt= 0; }
slave_thread_opt_list
{}
2003-02-13 10:14:35 +04:00
;
slave_thread_opt_list:
slave_thread_opt
| slave_thread_opt_list ',' slave_thread_opt
;
slave_thread_opt:
/*empty*/ {}
| SQL_THREAD { Lex->slave_thd_opt|=SLAVE_SQL; }
| RELAY_THREAD { Lex->slave_thd_opt|=SLAVE_IO; }
;
slave_until:
/*empty*/ {}
| UNTIL_SYM slave_until_opts
{
LEX *lex=Lex;
if ((lex->mi.log_file_name || lex->mi.pos) &&
(lex->mi.relay_log_name || lex->mi.relay_log_pos) ||
!((lex->mi.log_file_name && lex->mi.pos) ||
(lex->mi.relay_log_name && lex->mi.relay_log_pos)))
{
send_error(lex->thd, ER_BAD_SLAVE_UNTIL_COND);
YYABORT;
}
}
;
slave_until_opts:
master_file_def
| slave_until_opts ',' master_file_def ;
restore:
RESTORE_SYM table_or_tables
{
Lex->sql_command = SQLCOM_RESTORE_TABLE;
}
2003-03-17 21:56:34 +04:00
table_list FROM TEXT_STRING_sys
{
Lex->backup_dir = $6.str;
};
backup:
BACKUP_SYM table_or_tables
{
Lex->sql_command = SQLCOM_BACKUP_TABLE;
}
2003-03-17 21:56:34 +04:00
table_list TO_SYM TEXT_STRING_sys
{
Lex->backup_dir = $6.str;
};
checksum:
CHECKSUM_SYM table_or_tables
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_CHECKSUM;
}
table_list opt_checksum_type
{}
;
opt_checksum_type:
/* nothing */ { Lex->check_opt.flags= 0; }
| QUICK { Lex->check_opt.flags= T_QUICK; }
| EXTENDED_SYM { Lex->check_opt.flags= T_EXTEND; }
;
2000-07-31 21:29:14 +02:00
repair:
REPAIR opt_no_write_to_binlog table_or_tables
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_REPAIR;
lex->no_write_to_binlog= $2;
lex->check_opt.init();
2000-07-31 21:29:14 +02:00
}
table_list opt_mi_repair_type
{}
;
2000-07-31 21:29:14 +02:00
opt_mi_repair_type:
2000-07-31 21:29:14 +02:00
/* empty */ { Lex->check_opt.flags = T_MEDIUM; }
| mi_repair_types {};
2000-07-31 21:29:14 +02:00
mi_repair_types:
mi_repair_type {}
| mi_repair_type mi_repair_types {};
mi_repair_type:
2002-03-21 17:32:37 +00:00
QUICK { Lex->check_opt.flags|= T_QUICK; }
| EXTENDED_SYM { Lex->check_opt.flags|= T_EXTEND; }
| USE_FRM { Lex->check_opt.sql_flags|= TT_USEFRM; };
2000-07-31 21:29:14 +02:00
analyze:
ANALYZE_SYM opt_no_write_to_binlog table_or_tables
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_ANALYZE;
lex->no_write_to_binlog= $2;
lex->check_opt.init();
2000-07-31 21:29:14 +02:00
}
table_list opt_mi_check_type
{}
;
2000-07-31 21:29:14 +02:00
check:
2000-08-22 00:39:08 +03:00
CHECK_SYM table_or_tables
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_CHECK;
lex->check_opt.init();
2000-07-31 21:29:14 +02:00
}
table_list opt_mi_check_type
{}
;
2000-07-31 21:29:14 +02:00
opt_mi_check_type:
/* empty */ { Lex->check_opt.flags = T_MEDIUM; }
| mi_check_types {};
mi_check_types:
mi_check_type {}
| mi_check_type mi_check_types {};
mi_check_type:
QUICK { Lex->check_opt.flags|= T_QUICK; }
| FAST_SYM { Lex->check_opt.flags|= T_FAST; }
| MEDIUM_SYM { Lex->check_opt.flags|= T_MEDIUM; }
| EXTENDED_SYM { Lex->check_opt.flags|= T_EXTEND; }
| CHANGED { Lex->check_opt.flags|= T_CHECK_ONLY_CHANGED; };
2000-07-31 21:29:14 +02:00
optimize:
OPTIMIZE opt_no_write_to_binlog table_or_tables
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_OPTIMIZE;
2003-08-11 22:44:43 +03:00
lex->no_write_to_binlog= $2;
lex->check_opt.init();
2000-07-31 21:29:14 +02:00
}
table_list opt_mi_check_type
{}
;
2000-07-31 21:29:14 +02:00
opt_no_write_to_binlog:
/* empty */ { $$= 0; }
| NO_WRITE_TO_BINLOG { $$= 1; }
| LOCAL_SYM { $$= 1; }
;
rename:
RENAME table_or_tables
{
Lex->sql_command=SQLCOM_RENAME_TABLE;
}
table_to_table_list
{}
;
table_to_table_list:
table_to_table
| table_to_table_list ',' table_to_table;
table_to_table:
table_ident TO_SYM table_ident
{
2003-08-11 22:44:43 +03:00
LEX *lex=Lex;
SELECT_LEX *sl= lex->current_select;
if (!sl->add_table_to_list(lex->thd, $1,NULL,TL_OPTION_UPDATING,
TL_IGNORE) ||
!sl->add_table_to_list(lex->thd, $3,NULL,TL_OPTION_UPDATING,
TL_IGNORE))
YYABORT;
};
2003-08-26 00:15:49 -07:00
keycache:
CACHE_SYM INDEX keycache_list IN_SYM key_cache_name
2003-08-26 00:15:49 -07:00
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_ASSIGN_TO_KEYCACHE;
lex->name_and_length= $5;
2003-08-26 00:15:49 -07:00
}
;
keycache_list:
assign_to_keycache
| keycache_list ',' assign_to_keycache;
assign_to_keycache:
table_ident cache_keys_spec
2003-08-26 00:15:49 -07:00
{
LEX *lex=Lex;
SELECT_LEX *sel= &lex->select_lex;
if (!sel->add_table_to_list(lex->thd, $1, NULL, 0,
TL_READ,
sel->get_use_index(),
(List<String> *)0))
2003-08-26 00:15:49 -07:00
YYABORT;
}
;
key_cache_name:
ident { $$= $1; }
| DEFAULT { $$ = default_key_cache_base; }
;
preload:
LOAD INDEX INTO CACHE_SYM
{
LEX *lex=Lex;
lex->sql_command=SQLCOM_PRELOAD_KEYS;
}
preload_list
{}
;
preload_list:
preload_keys
| preload_list ',' preload_keys;
preload_keys:
2003-08-26 00:15:49 -07:00
table_ident cache_keys_spec opt_ignore_leaves
{
LEX *lex=Lex;
SELECT_LEX *sel= &lex->select_lex;
2003-08-11 22:44:43 +03:00
if (!sel->add_table_to_list(lex->thd, $1, NULL, $3,
TL_READ,
sel->get_use_index(),
(List<String> *)0))
YYABORT;
}
;
cache_keys_spec:
{ Select->interval_list.empty(); }
cache_key_list_or_empty
{
LEX *lex=Lex;
SELECT_LEX *sel= &lex->select_lex;
sel->use_index= sel->interval_list;
}
;
2003-08-26 00:15:49 -07:00
cache_key_list_or_empty:
/* empty */ { Lex->select_lex.use_index_ptr= 0; }
| opt_keys_or_index '(' key_usage_list2 ')'
{
SELECT_LEX *sel= &Lex->select_lex;
sel->use_index_ptr= &sel->use_index;
}
;
opt_ignore_leaves:
/* empty */
{ $$= 0; }
| IGNORE_SYM LEAVES { $$= TL_OPTION_IGNORE_LEAVES; }
;
2000-07-31 21:29:14 +02:00
/*
2001-12-17 19:59:20 +02:00
Select : retrieve data from table
2000-07-31 21:29:14 +02:00
*/
select:
select_init
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_SELECT;
lex->select_lex.resolve_mode= SELECT_LEX::SELECT_MODE;
2003-08-12 17:40:11 +03:00
}
;
/* Need select_init2 for subselects. */
select_init:
SELECT_SYM select_init2
|
'(' SELECT_SYM select_part2 ')'
{
LEX *lex= Lex;
SELECT_LEX * sel= lex->current_select;
2002-12-05 19:38:42 +02:00
if (sel->set_braces(1))
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
if (sel->linkage == UNION_TYPE &&
!sel->master_unit()->first_select()->braces)
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
/* select in braces, can't contain global parameters */
if (sel->master_unit()->fake_select_lex)
sel->master_unit()->global_parameters=
sel->master_unit()->fake_select_lex;
2002-06-05 21:32:22 +03:00
} union_opt;
select_init2:
select_part2
{
LEX *lex= Lex;
SELECT_LEX * sel= lex->current_select;
2002-12-05 19:38:42 +02:00
if (lex->current_select->set_braces(0))
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
if (sel->linkage == UNION_TYPE &&
sel->master_unit()->first_select()->braces)
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
}
union_clause
;
select_part2:
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
SELECT_LEX * sel= lex->current_select;
2002-11-23 20:40:16 +02:00
if (lex->current_select == &lex->select_lex)
lex->lock_option= TL_READ; /* Only for global SELECT */
if (sel->linkage != UNION_TYPE)
mysql_init_select(lex);
2003-05-17 10:05:07 +03:00
lex->current_select->parsing_place= SELECT_LEX_NODE::SELECT_LIST;
}
select_options select_item_list
{
Select->parsing_place= SELECT_LEX_NODE::NO_MATTER;
2000-07-31 21:29:14 +02:00
}
2003-05-17 10:05:07 +03:00
select_into select_lock_type;
2000-07-31 21:29:14 +02:00
select_into:
opt_limit_clause {}
| FROM DUAL_SYM /* oracle compatibility: oracle always requires FROM
clause, and DUAL is system table without fields.
Is "SELECT 1 FROM DUAL" any better than
"SELECT 1" ? Hmmm :) */
| into
2000-07-31 21:29:14 +02:00
| select_from
| into select_from
| select_from into;
2000-07-31 21:29:14 +02:00
select_from:
FROM join_table_list where_clause group_clause having_clause opt_order_clause opt_limit_clause procedure_clause;
2000-07-31 21:29:14 +02:00
select_options:
/* empty*/
| select_option_list;
2000-07-31 21:29:14 +02:00
select_option_list:
select_option_list select_option
| select_option;
2000-07-31 21:29:14 +02:00
select_option:
STRAIGHT_JOIN { Select->options|= SELECT_STRAIGHT_JOIN; }
| HIGH_PRIORITY
{
if (check_simple_select())
YYABORT;
Lex->lock_option= TL_READ_HIGH_PRIORITY;
}
| DISTINCT { Select->options|= SELECT_DISTINCT; }
| SQL_SMALL_RESULT { Select->options|= SELECT_SMALL_RESULT; }
| SQL_BIG_RESULT { Select->options|= SELECT_BIG_RESULT; }
| SQL_BUFFER_RESULT
{
if (check_simple_select())
YYABORT;
Select->options|= OPTION_BUFFER_RESULT;
}
| SQL_CALC_FOUND_ROWS
{
if (check_simple_select())
YYABORT;
Select->options|= OPTION_FOUND_ROWS;
}
| SQL_NO_CACHE_SYM { Lex->safe_to_cache_query=0; }
2003-02-27 22:26:09 +02:00
| SQL_CACHE_SYM
{
Lex->select_lex.options|= OPTION_TO_QUERY_CACHE;
}
| ALL {}
;
2000-07-31 21:29:14 +02:00
select_lock_type:
/* empty */
| FOR_SYM UPDATE_SYM
{
LEX *lex=Lex;
lex->current_select->set_lock_for_tables(TL_WRITE);
lex->safe_to_cache_query=0;
}
2001-12-11 20:45:48 +02:00
| LOCK_SYM IN_SYM SHARE_SYM MODE_SYM
{
LEX *lex=Lex;
lex->current_select->
set_lock_for_tables(TL_READ_WITH_SHARED_LOCKS);
lex->safe_to_cache_query=0;
}
;
2000-07-31 21:29:14 +02:00
select_item_list:
select_item_list ',' select_item
| select_item
| '*'
{
THD *thd= YYTHD;
if (add_item_to_list(thd, new Item_field(NULL, NULL, "*")))
2000-07-31 21:29:14 +02:00
YYABORT;
2003-08-26 11:51:09 +02:00
(thd->lex->current_select->with_wild)++;
};
2000-07-31 21:29:14 +02:00
select_item:
remember_name select_item2 remember_end select_alias
{
if (add_item_to_list(YYTHD, $2))
2000-07-31 21:29:14 +02:00
YYABORT;
if ($4.str)
2003-03-17 21:56:34 +04:00
$2->set_name($4.str,$4.length,system_charset_info);
2000-07-31 21:29:14 +02:00
else if (!$2->name)
2003-03-17 21:56:34 +04:00
$2->set_name($1,(uint) ($3 - $1), YYTHD->charset());
};
2000-07-31 21:29:14 +02:00
remember_name:
{ $$=(char*) Lex->tok_start; };
2000-07-31 21:29:14 +02:00
remember_end:
{ $$=(char*) Lex->tok_end; };
2000-07-31 21:29:14 +02:00
select_item2:
table_wild { $$=$1; } /* table.* */
| expr { $$=$1; };
2000-07-31 21:29:14 +02:00
select_alias:
2003-03-17 21:56:34 +04:00
/* empty */ { $$.str=0;}
| AS ident { $$=$2; }
| AS TEXT_STRING_sys { $$=$2; }
| ident { $$=$1; }
| TEXT_STRING_sys { $$=$1; }
;
2000-07-31 21:29:14 +02:00
optional_braces:
/* empty */ {}
| '(' ')' {};
2000-08-22 00:39:08 +03:00
2000-07-31 21:29:14 +02:00
/* all possible expressions */
2003-01-21 20:20:46 +04:00
expr:
expr_expr { $$= $1; }
| simple_expr { $$= $1; }
;
2000-07-31 21:29:14 +02:00
comp_op: EQ { $$ = &comp_eq_creator; }
| GE { $$ = &comp_ge_creator; }
| GT_SYM { $$ = &comp_gt_creator; }
| LE { $$ = &comp_le_creator; }
| LT { $$ = &comp_lt_creator; }
| NE { $$ = &comp_ne_creator; }
;
all_or_any: ALL { $$ = 1; }
| ANY_SYM { $$ = 0; }
;
2000-07-31 21:29:14 +02:00
/* expressions that begin with 'expr' */
expr_expr:
2003-01-21 20:20:46 +04:00
expr IN_SYM '(' expr_list ')'
{ $4->push_front($1); $$= new Item_func_in(*$4); }
2000-07-31 21:29:14 +02:00
| expr NOT IN_SYM '(' expr_list ')'
{ $5->push_front($1); $$= new Item_func_not(new Item_func_in(*$5)); }
2002-10-27 23:27:00 +02:00
| expr IN_SYM in_subselect
{ $$= new Item_in_subselect($1, $3); }
2002-10-27 23:27:00 +02:00
| expr NOT IN_SYM in_subselect
{
$$= new Item_func_not(new Item_in_subselect($1, $4));
2002-10-27 23:27:00 +02:00
}
2000-07-31 21:29:14 +02:00
| expr BETWEEN_SYM no_and_expr AND expr
{ $$= new Item_func_between($1,$3,$5); }
| expr NOT BETWEEN_SYM no_and_expr AND expr
{ $$= new Item_func_not(new Item_func_between($1,$4,$6)); }
| expr OR_OR_CONCAT expr { $$= or_or_concat(YYTHD, $1,$3); }
2000-07-31 21:29:14 +02:00
| expr OR expr { $$= new Item_cond_or($1,$3); }
| expr XOR expr { $$= new Item_cond_xor($1,$3); }
2000-07-31 21:29:14 +02:00
| expr AND expr { $$= new Item_cond_and($1,$3); }
2003-11-03 12:28:36 +02:00
| expr SOUNDS_SYM LIKE expr
{
$$= new Item_func_eq(new Item_func_soundex($1),
new Item_func_soundex($4));
}
| expr LIKE simple_expr opt_escape
{ $$= new Item_func_like($1,$3,$4); }
| expr NOT LIKE simple_expr opt_escape
{ $$= new Item_func_not(new Item_func_like($1,$4,$5));}
2000-07-31 21:29:14 +02:00
| expr REGEXP expr { $$= new Item_func_regex($1,$3); }
2003-11-03 12:28:36 +02:00
| expr NOT REGEXP expr
{ $$= new Item_func_not(new Item_func_regex($1,$4)); }
2000-07-31 21:29:14 +02:00
| expr IS NULL_SYM { $$= new Item_func_isnull($1); }
| expr IS NOT NULL_SYM { $$= new Item_func_isnotnull($1); }
| expr EQUAL_SYM expr { $$= new Item_func_equal($1,$3); }
2003-11-03 12:28:36 +02:00
| expr comp_op expr %prec EQ { $$= (*$2)(0)->create($1,$3); }
| expr comp_op all_or_any in_subselect %prec EQ
{
$$= all_any_subquery_creator($1, $2, $3, $4);
}
2000-07-31 21:29:14 +02:00
| expr SHIFT_LEFT expr { $$= new Item_func_shift_left($1,$3); }
| expr SHIFT_RIGHT expr { $$= new Item_func_shift_right($1,$3); }
| expr '+' expr { $$= new Item_func_plus($1,$3); }
| expr '-' expr { $$= new Item_func_minus($1,$3); }
| expr '*' expr { $$= new Item_func_mul($1,$3); }
| expr '/' expr { $$= new Item_func_div($1,$3); }
| expr DIV_SYM expr { $$= new Item_func_int_div($1,$3); }
| expr MOD_SYM expr { $$= new Item_func_mod($1,$3); }
2000-07-31 21:29:14 +02:00
| expr '|' expr { $$= new Item_func_bit_or($1,$3); }
| expr '^' expr { $$= new Item_func_bit_xor($1,$3); }
2000-07-31 21:29:14 +02:00
| expr '&' expr { $$= new Item_func_bit_and($1,$3); }
| expr '%' expr { $$= new Item_func_mod($1,$3); }
| expr '+' interval_expr interval
{ $$= new Item_date_add_interval($1,$3,$4,0); }
| expr '-' interval_expr interval
{ $$= new Item_date_add_interval($1,$3,$4,1); }
;
2000-07-31 21:29:14 +02:00
/* expressions that begin with 'expr' that do NOT follow IN_SYM */
no_in_expr:
no_in_expr BETWEEN_SYM no_and_expr AND expr
{ $$= new Item_func_between($1,$3,$5); }
| no_in_expr NOT BETWEEN_SYM no_and_expr AND expr
{ $$= new Item_func_not(new Item_func_between($1,$4,$6)); }
| no_in_expr OR_OR_CONCAT expr { $$= or_or_concat(YYTHD, $1,$3); }
2000-07-31 21:29:14 +02:00
| no_in_expr OR expr { $$= new Item_cond_or($1,$3); }
| no_in_expr XOR expr { $$= new Item_cond_xor($1,$3); }
2000-07-31 21:29:14 +02:00
| no_in_expr AND expr { $$= new Item_cond_and($1,$3); }
2003-11-03 12:28:36 +02:00
| no_in_expr SOUNDS_SYM LIKE expr
{
$$= new Item_func_eq(new Item_func_soundex($1),
new Item_func_soundex($4));
}
| no_in_expr LIKE simple_expr opt_escape
{ $$= new Item_func_like($1,$3,$4); }
| no_in_expr NOT LIKE simple_expr opt_escape
{ $$= new Item_func_not(new Item_func_like($1,$4,$5)); }
2000-07-31 21:29:14 +02:00
| no_in_expr REGEXP expr { $$= new Item_func_regex($1,$3); }
2003-11-03 12:28:36 +02:00
| no_in_expr NOT REGEXP expr
{ $$= new Item_func_not(new Item_func_regex($1,$4)); }
2000-07-31 21:29:14 +02:00
| no_in_expr IS NULL_SYM { $$= new Item_func_isnull($1); }
| no_in_expr IS NOT NULL_SYM { $$= new Item_func_isnotnull($1); }
| no_in_expr EQUAL_SYM expr { $$= new Item_func_equal($1,$3); }
2003-11-03 12:28:36 +02:00
| no_in_expr comp_op expr %prec EQ { $$= (*$2)(0)->create($1,$3); }
| no_in_expr comp_op all_or_any in_subselect %prec EQ
{
all_any_subquery_creator($1, $2, $3, $4);
}
2000-07-31 21:29:14 +02:00
| no_in_expr SHIFT_LEFT expr { $$= new Item_func_shift_left($1,$3); }
| no_in_expr SHIFT_RIGHT expr { $$= new Item_func_shift_right($1,$3); }
| no_in_expr '+' expr { $$= new Item_func_plus($1,$3); }
| no_in_expr '-' expr { $$= new Item_func_minus($1,$3); }
| no_in_expr '*' expr { $$= new Item_func_mul($1,$3); }
| no_in_expr '/' expr { $$= new Item_func_div($1,$3); }
| no_in_expr DIV_SYM expr { $$= new Item_func_int_div($1,$3); }
2000-07-31 21:29:14 +02:00
| no_in_expr '|' expr { $$= new Item_func_bit_or($1,$3); }
| no_in_expr '^' expr { $$= new Item_func_bit_xor($1,$3); }
2000-07-31 21:29:14 +02:00
| no_in_expr '&' expr { $$= new Item_func_bit_and($1,$3); }
| no_in_expr '%' expr { $$= new Item_func_mod($1,$3); }
| no_in_expr MOD_SYM expr { $$= new Item_func_mod($1,$3); }
| no_in_expr '+' interval_expr interval
{ $$= new Item_date_add_interval($1,$3,$4,0); }
| no_in_expr '-' interval_expr interval
{ $$= new Item_date_add_interval($1,$3,$4,1); }
| simple_expr;
2000-07-31 21:29:14 +02:00
/* expressions that begin with 'expr' that does NOT follow AND */
no_and_expr:
2002-10-27 23:27:00 +02:00
no_and_expr IN_SYM '(' expr_list ')'
{ $4->push_front($1); $$= new Item_func_in(*$4); }
2000-07-31 21:29:14 +02:00
| no_and_expr NOT IN_SYM '(' expr_list ')'
{ $5->push_front($1); $$= new Item_func_not(new Item_func_in(*$5)); }
2002-10-27 23:27:00 +02:00
| no_and_expr IN_SYM in_subselect
{ $$= new Item_in_subselect($1, $3); }
2002-10-27 23:27:00 +02:00
| no_and_expr NOT IN_SYM in_subselect
{
$$= new Item_func_not(new Item_in_subselect($1, $4));
2002-10-27 23:27:00 +02:00
}
2000-07-31 21:29:14 +02:00
| no_and_expr BETWEEN_SYM no_and_expr AND expr
{ $$= new Item_func_between($1,$3,$5); }
| no_and_expr NOT BETWEEN_SYM no_and_expr AND expr
{ $$= new Item_func_not(new Item_func_between($1,$4,$6)); }
| no_and_expr OR_OR_CONCAT expr { $$= or_or_concat(YYTHD, $1,$3); }
2000-07-31 21:29:14 +02:00
| no_and_expr OR expr { $$= new Item_cond_or($1,$3); }
| no_and_expr XOR expr { $$= new Item_cond_xor($1,$3); }
2003-11-03 12:28:36 +02:00
| no_and_expr SOUNDS_SYM LIKE expr
{
$$= new Item_func_eq(new Item_func_soundex($1),
new Item_func_soundex($4));
}
| no_and_expr LIKE simple_expr opt_escape
{ $$= new Item_func_like($1,$3,$4); }
| no_and_expr NOT LIKE simple_expr opt_escape
{ $$= new Item_func_not(new Item_func_like($1,$4,$5)); }
2000-07-31 21:29:14 +02:00
| no_and_expr REGEXP expr { $$= new Item_func_regex($1,$3); }
2003-11-03 12:28:36 +02:00
| no_and_expr NOT REGEXP expr
{ $$= new Item_func_not(new Item_func_regex($1,$4)); }
2000-07-31 21:29:14 +02:00
| no_and_expr IS NULL_SYM { $$= new Item_func_isnull($1); }
| no_and_expr IS NOT NULL_SYM { $$= new Item_func_isnotnull($1); }
| no_and_expr EQUAL_SYM expr { $$= new Item_func_equal($1,$3); }
2003-11-03 12:28:36 +02:00
| no_and_expr comp_op expr %prec EQ { $$= (*$2)(0)->create($1,$3); }
| no_and_expr comp_op all_or_any in_subselect %prec EQ
{
all_any_subquery_creator($1, $2, $3, $4);
}
2000-07-31 21:29:14 +02:00
| no_and_expr SHIFT_LEFT expr { $$= new Item_func_shift_left($1,$3); }
| no_and_expr SHIFT_RIGHT expr { $$= new Item_func_shift_right($1,$3); }
| no_and_expr '+' expr { $$= new Item_func_plus($1,$3); }
| no_and_expr '-' expr { $$= new Item_func_minus($1,$3); }
| no_and_expr '*' expr { $$= new Item_func_mul($1,$3); }
| no_and_expr '/' expr { $$= new Item_func_div($1,$3); }
| no_and_expr DIV_SYM expr { $$= new Item_func_int_div($1,$3); }
2000-07-31 21:29:14 +02:00
| no_and_expr '|' expr { $$= new Item_func_bit_or($1,$3); }
| no_and_expr '^' expr { $$= new Item_func_bit_xor($1,$3); }
2000-07-31 21:29:14 +02:00
| no_and_expr '&' expr { $$= new Item_func_bit_and($1,$3); }
| no_and_expr '%' expr { $$= new Item_func_mod($1,$3); }
| no_and_expr MOD_SYM expr { $$= new Item_func_mod($1,$3); }
| no_and_expr '+' interval_expr interval
{ $$= new Item_date_add_interval($1,$3,$4,0); }
| no_and_expr '-' interval_expr interval
{ $$= new Item_date_add_interval($1,$3,$4,1); }
| simple_expr;
2000-07-31 21:29:14 +02:00
interval_expr:
2003-01-27 13:46:33 +04:00
INTERVAL_SYM expr { $$=$2; }
;
2000-07-31 21:29:14 +02:00
simple_expr:
simple_ident
| simple_expr COLLATE_SYM ident_or_text %prec NEG
2003-08-11 22:44:43 +03:00
{
$$= new Item_func_set_collation($1,
new Item_string($3.str,
$3.length,
YYTHD->charset()));
}
2000-07-31 21:29:14 +02:00
| literal
| param_marker
2001-12-02 14:34:01 +02:00
| '@' ident_or_text SET_VAR expr
{
$$= new Item_func_set_user_var($2,$4);
Lex->uncacheable(UNCACHEABLE_RAND);
2001-12-02 14:34:01 +02:00
}
| '@' ident_or_text
{
$$= new Item_func_get_user_var($2);
Lex->uncacheable(UNCACHEABLE_RAND);
2001-12-02 14:34:01 +02:00
}
| '@' '@' opt_var_ident_type ident_or_text opt_component
{
if (!($$= get_system_var(YYTHD, (enum_var_type) $3, $4, $5)))
YYABORT;
2001-12-02 14:34:01 +02:00
}
2000-07-31 21:29:14 +02:00
| sum_expr
| '+' expr %prec NEG { $$= $2; }
| '-' expr %prec NEG { $$= new Item_func_neg($2); }
2000-07-31 21:29:14 +02:00
| '~' expr %prec NEG { $$= new Item_func_bit_neg($2); }
| NOT expr %prec NEG { $$= new Item_func_not($2); }
| '!' expr %prec NEG { $$= new Item_func_not($2); }
| '(' expr ')' { $$= $2; }
| '(' expr ',' expr_list ')'
{
$4->push_front($2);
$$= new Item_row(*$4);
}
| ROW_SYM '(' expr ',' expr_list ')'
2002-11-15 20:32:09 +02:00
{
$5->push_front($3);
$$= new Item_row(*$5);
2002-11-15 20:32:09 +02:00
}
| EXISTS exists_subselect { $$= $2; }
2002-12-19 21:15:09 +02:00
| singlerow_subselect { $$= $1; }
2000-07-31 21:29:14 +02:00
| '{' ident expr '}' { $$= $3; }
| MATCH ident_list_arg AGAINST '(' expr fulltext_options ')'
{ $2->push_front($5);
Select->add_ftfunc_to_list((Item_func_match*)
($$=new Item_func_match(*$2,$6))); }
| ASCII_SYM '(' expr ')' { $$= new Item_func_ascii($3); }
2003-08-11 22:44:43 +03:00
| BINARY expr %prec NEG
{
$$= new Item_func_set_collation($2,new Item_string(binary_keyword,
6, &my_charset_latin1));
}
| CAST_SYM '(' expr AS cast_type ')'
{
$$= create_func_cast($3, $5,
2003-08-21 14:15:25 +05:00
Lex->length ? atoi(Lex->length) : -1,
Lex->charset);
2003-08-21 14:15:25 +05:00
}
2000-07-31 21:29:14 +02:00
| CASE_SYM opt_expr WHEN_SYM when_list opt_else END
{ $$= new Item_func_case(* $4, $2, $5 ); }
| CONVERT_SYM '(' expr ',' cast_type ')'
2003-08-21 14:15:25 +05:00
{
$$= create_func_cast($3, $5,
Lex->length ? atoi(Lex->length) : -1,
Lex->charset);
}
| CONVERT_SYM '(' expr USING charset_name ')'
{ $$= new Item_func_conv_charset($3,$5); }
| CONVERT_SYM '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_conv_charset3($3,$7,$5); }
2003-01-05 14:07:24 +04:00
| DEFAULT '(' simple_ident ')'
{ $$= new Item_default_value($3); }
2003-05-03 01:16:56 +02:00
| VALUES '(' simple_ident ')'
{ $$= new Item_insert_value($3); }
2000-07-31 21:29:14 +02:00
| FUNC_ARG0 '(' ')'
{ $$= ((Item*(*)(void))($1.symbol->create_func))();}
| FUNC_ARG1 '(' expr ')'
{ $$= ((Item*(*)(Item*))($1.symbol->create_func))($3);}
| FUNC_ARG2 '(' expr ',' expr ')'
{ $$= ((Item*(*)(Item*,Item*))($1.symbol->create_func))($3,$5);}
| FUNC_ARG3 '(' expr ',' expr ',' expr ')'
{ $$= ((Item*(*)(Item*,Item*,Item*))($1.symbol->create_func))($3,$5,$7);}
2003-06-23 12:56:44 +05:00
| ADDDATE_SYM '(' expr ',' expr ')'
{ $$= new Item_date_add_interval($3, $5, INTERVAL_DAY, 0);}
| ADDDATE_SYM '(' expr ',' INTERVAL_SYM expr interval ')'
{ $$= new Item_date_add_interval($3, $6, $7, 0); }
| REPEAT_SYM '(' expr ',' expr ')'
{ $$= new Item_func_repeat($3,$5); }
2000-07-31 21:29:14 +02:00
| ATAN '(' expr ')'
{ $$= new Item_func_atan($3); }
| ATAN '(' expr ',' expr ')'
{ $$= new Item_func_atan($3,$5); }
| CHAR_SYM '(' expr_list ')'
{ $$= new Item_func_char(*$3); }
2002-06-20 23:26:04 +05:00
| CHARSET '(' expr ')'
{ $$= new Item_func_charset($3); }
2000-07-31 21:29:14 +02:00
| COALESCE '(' expr_list ')'
{ $$= new Item_func_coalesce(* $3); }
| COLLATION_SYM '(' expr ')'
{ $$= new Item_func_collation($3); }
2000-07-31 21:29:14 +02:00
| CONCAT '(' expr_list ')'
{ $$= new Item_func_concat(* $3); }
| CONCAT_WS '(' expr ',' expr_list ')'
{ $$= new Item_func_concat_ws($3, *$5); }
| CURDATE optional_braces
{ $$= new Item_func_curdate_local(); Lex->safe_to_cache_query=0; }
2000-07-31 21:29:14 +02:00
| CURTIME optional_braces
{ $$= new Item_func_curtime_local(); Lex->safe_to_cache_query=0; }
2000-07-31 21:29:14 +02:00
| CURTIME '(' expr ')'
{
$$= new Item_func_curtime_local($3);
Lex->safe_to_cache_query=0;
2001-12-02 14:34:01 +02:00
}
| DATE_ADD_INTERVAL '(' expr ',' interval_expr interval ')'
{ $$= new Item_date_add_interval($3,$5,$6,0); }
| DATE_SUB_INTERVAL '(' expr ',' interval_expr interval ')'
{ $$= new Item_date_add_interval($3,$5,$6,1); }
2000-07-31 21:29:14 +02:00
| DATABASE '(' ')'
{
2001-12-02 14:34:01 +02:00
$$= new Item_func_database();
Lex->safe_to_cache_query=0;
2001-12-02 14:34:01 +02:00
}
2003-06-23 12:56:44 +05:00
| DATE_SYM '(' expr ')'
2003-07-08 15:06:05 +05:00
{ $$= new Item_date_typecast($3); }
2003-06-23 12:56:44 +05:00
| DAY_SYM '(' expr ')'
{ $$= new Item_func_dayofmonth($3); }
2000-07-31 21:29:14 +02:00
| ELT_FUNC '(' expr ',' expr_list ')'
{ $5->push_front($3); $$= new Item_func_elt(*$5); }
2000-07-31 21:29:14 +02:00
| MAKE_SET_SYM '(' expr ',' expr_list ')'
{ $$= new Item_func_make_set($3, *$5); }
2001-12-02 14:34:01 +02:00
| ENCRYPT '(' expr ')'
{
$$= new Item_func_encrypt($3);
Lex->uncacheable(UNCACHEABLE_RAND);
2001-12-02 14:34:01 +02:00
}
2000-07-31 21:29:14 +02:00
| ENCRYPT '(' expr ',' expr ')' { $$= new Item_func_encrypt($3,$5); }
| DECODE_SYM '(' expr ',' TEXT_STRING_literal ')'
2000-07-31 21:29:14 +02:00
{ $$= new Item_func_decode($3,$5.str); }
| ENCODE_SYM '(' expr ',' TEXT_STRING_literal ')'
2000-07-31 21:29:14 +02:00
{ $$= new Item_func_encode($3,$5.str); }
2001-12-13 03:36:36 +02:00
| DES_DECRYPT_SYM '(' expr ')'
{ $$= new Item_func_des_decrypt($3); }
2001-12-13 03:36:36 +02:00
| DES_DECRYPT_SYM '(' expr ',' expr ')'
{ $$= new Item_func_des_decrypt($3,$5); }
2001-12-13 03:36:36 +02:00
| DES_ENCRYPT_SYM '(' expr ')'
{ $$= new Item_func_des_encrypt($3); }
2001-12-13 03:36:36 +02:00
| DES_ENCRYPT_SYM '(' expr ',' expr ')'
{ $$= new Item_func_des_encrypt($3,$5); }
2000-07-31 21:29:14 +02:00
| EXPORT_SET '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_export_set($3, $5, $7); }
| EXPORT_SET '(' expr ',' expr ',' expr ',' expr ')'
{ $$= new Item_func_export_set($3, $5, $7, $9); }
| EXPORT_SET '(' expr ',' expr ',' expr ',' expr ',' expr ')'
{ $$= new Item_func_export_set($3, $5, $7, $9, $11); }
| FALSE_SYM
{ $$= new Item_int((char*) "FALSE",0,1); }
2000-07-31 21:29:14 +02:00
| FORMAT_SYM '(' expr ',' NUM ')'
{ $$= new Item_func_format($3,atoi($5.str)); }
| FROM_UNIXTIME '(' expr ')'
{ $$= new Item_func_from_unixtime($3); }
| FROM_UNIXTIME '(' expr ',' expr ')'
{
2001-12-02 14:34:01 +02:00
$$= new Item_func_date_format (new Item_func_from_unixtime($3),$5,0);
2000-07-31 21:29:14 +02:00
}
| FIELD_FUNC '(' expr ',' expr_list ')'
{ $5->push_front($3); $$= new Item_func_field(*$5); }
| GEOMFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| GEOMFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| GEOMFROMWKB '(' expr ')'
{ $$= new Item_func_geometry_from_wkb($3); }
| GEOMFROMWKB '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_wkb($3, $5); }
2002-02-22 15:24:42 +04:00
| GEOMETRYCOLLECTION '(' expr_list ')'
{ $$= new Item_func_spatial_collection(* $3,
Geometry::wkbGeometryCollection,
2002-02-22 15:24:42 +04:00
Geometry::wkbPoint); }
| GET_FORMAT '(' date_time_type ',' expr ')'
{ $$= new Item_func_get_format($3, $5); }
2000-07-31 21:29:14 +02:00
| HOUR_SYM '(' expr ')'
{ $$= new Item_func_hour($3); }
| IF '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_if($3,$5,$7); }
| INSERT '(' expr ',' expr ',' expr ',' expr ')'
{ $$= new Item_func_insert($3,$5,$7,$9); }
| interval_expr interval '+' expr
2000-07-31 21:29:14 +02:00
/* we cannot put interval before - */
{ $$= new Item_date_add_interval($4,$1,$2,0); }
| interval_expr
{
if ($1->type() != Item::ROW_ITEM)
{
send_error(Lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
$$= new Item_func_interval((Item_row *)$1);
}
2000-07-31 21:29:14 +02:00
| LAST_INSERT_ID '(' ')'
{
$$= get_system_var(YYTHD, OPT_SESSION, "last_insert_id", 14,
"last_insert_id()");
Lex->safe_to_cache_query= 0;
2000-07-31 21:29:14 +02:00
}
| LAST_INSERT_ID '(' expr ')'
{
$$= new Item_func_set_last_insert_id($3);
Lex->safe_to_cache_query= 0;
2000-07-31 21:29:14 +02:00
}
| LEFT '(' expr ',' expr ')'
{ $$= new Item_func_left($3,$5); }
2002-02-22 15:24:42 +04:00
| LINESTRING '(' expr_list ')'
{ $$= new Item_func_spatial_collection(* $3,
2002-02-22 15:24:42 +04:00
Geometry::wkbLineString, Geometry::wkbPoint); }
2000-07-31 21:29:14 +02:00
| LOCATE '(' expr ',' expr ')'
{ $$= new Item_func_locate($5,$3); }
| LOCATE '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_locate($5,$3,$7); }
| GEOMCOLLFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| GEOMCOLLFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| GREATEST_SYM '(' expr ',' expr_list ')'
2000-07-31 21:29:14 +02:00
{ $5->push_front($3); $$= new Item_func_max(*$5); }
| LEAST_SYM '(' expr ',' expr_list ')'
{ $5->push_front($3); $$= new Item_func_min(*$5); }
| LOG_SYM '(' expr ')'
{ $$= new Item_func_log($3); }
| LOG_SYM '(' expr ',' expr ')'
{ $$= new Item_func_log($3, $5); }
| LINEFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| LINEFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| MASTER_POS_WAIT '(' expr ',' expr ')'
2003-08-11 22:44:43 +03:00
{
$$= new Item_master_pos_wait($3, $5);
2003-08-11 22:44:43 +03:00
Lex->safe_to_cache_query=0;
2003-02-04 21:52:14 +02:00
}
| MASTER_POS_WAIT '(' expr ',' expr ',' expr ')'
2003-08-11 22:44:43 +03:00
{
$$= new Item_master_pos_wait($3, $5, $7);
2003-08-11 22:44:43 +03:00
Lex->safe_to_cache_query=0;
}
2003-06-23 12:56:44 +05:00
| MICROSECOND_SYM '(' expr ')'
{ $$= new Item_func_microsecond($3); }
2000-07-31 21:29:14 +02:00
| MINUTE_SYM '(' expr ')'
{ $$= new Item_func_minute($3); }
| MOD_SYM '(' expr ',' expr ')'
{ $$ = new Item_func_mod( $3, $5); }
2000-07-31 21:29:14 +02:00
| MONTH_SYM '(' expr ')'
{ $$= new Item_func_month($3); }
2002-02-22 15:24:42 +04:00
| MULTILINESTRING '(' expr_list ')'
{ $$= new Item_func_spatial_collection(* $3,
2002-02-22 15:24:42 +04:00
Geometry::wkbMultiLineString, Geometry::wkbLineString); }
| MLINEFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| MLINEFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| MPOINTFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| MPOINTFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| MPOLYFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| MPOLYFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| MULTIPOINT '(' expr_list ')'
{ $$= new Item_func_spatial_collection(* $3,
2002-02-22 15:24:42 +04:00
Geometry::wkbMultiPoint, Geometry::wkbPoint); }
| MULTIPOLYGON '(' expr_list ')'
{ $$= new Item_func_spatial_collection(* $3,
2002-02-22 15:24:42 +04:00
Geometry::wkbMultiPolygon, Geometry::wkbPolygon ); }
2000-07-31 21:29:14 +02:00
| NOW_SYM optional_braces
{ $$= new Item_func_now_local(); Lex->safe_to_cache_query=0;}
2000-07-31 21:29:14 +02:00
| NOW_SYM '(' expr ')'
{ $$= new Item_func_now_local($3); Lex->safe_to_cache_query=0;}
2001-12-02 14:34:01 +02:00
| PASSWORD '(' expr ')'
{
$$= YYTHD->variables.old_passwords ?
(Item *) new Item_func_old_password($3) :
(Item *) new Item_func_password($3);
}
| OLD_PASSWORD '(' expr ')'
{ $$= new Item_func_old_password($3); }
| POINT_SYM '(' expr ',' expr ')'
{ $$= new Item_func_point($3,$5); }
| POINTFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| POINTFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| POLYFROMTEXT '(' expr ')'
{ $$= new Item_func_geometry_from_text($3); }
| POLYFROMTEXT '(' expr ',' expr ')'
{ $$= new Item_func_geometry_from_text($3, $5); }
| POLYGON '(' expr_list ')'
{ $$= new Item_func_spatial_collection(* $3,
2002-02-22 15:24:42 +04:00
Geometry::wkbPolygon, Geometry::wkbLineString); }
2000-07-31 21:29:14 +02:00
| POSITION_SYM '(' no_in_expr IN_SYM expr ')'
{ $$ = new Item_func_locate($5,$3); }
| QUARTER_SYM '(' expr ')'
{ $$ = new Item_func_quarter($3); }
2001-12-02 14:34:01 +02:00
| RAND '(' expr ')'
{ $$= new Item_func_rand($3); Lex->uncacheable(UNCACHEABLE_RAND);}
2001-12-02 14:34:01 +02:00
| RAND '(' ')'
{ $$= new Item_func_rand(); Lex->uncacheable(UNCACHEABLE_RAND);}
2000-07-31 21:29:14 +02:00
| REPLACE '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_replace($3,$5,$7); }
| RIGHT '(' expr ',' expr ')'
{ $$= new Item_func_right($3,$5); }
| ROUND '(' expr ')'
{ $$= new Item_func_round($3, new Item_int((char*)"0",0,1),0); }
| ROUND '(' expr ',' expr ')' { $$= new Item_func_round($3,$5,0); }
2003-06-23 12:56:44 +05:00
| SUBDATE_SYM '(' expr ',' expr ')'
{ $$= new Item_date_add_interval($3, $5, INTERVAL_DAY, 1);}
| SUBDATE_SYM '(' expr ',' INTERVAL_SYM expr interval ')'
{ $$= new Item_date_add_interval($3, $6, $7, 1); }
2000-07-31 21:29:14 +02:00
| SECOND_SYM '(' expr ')'
{ $$= new Item_func_second($3); }
| SUBSTRING '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_substr($3,$5,$7); }
| SUBSTRING '(' expr ',' expr ')'
{ $$= new Item_func_substr($3,$5); }
| SUBSTRING '(' expr FROM expr FOR_SYM expr ')'
{ $$= new Item_func_substr($3,$5,$7); }
| SUBSTRING '(' expr FROM expr ')'
{ $$= new Item_func_substr($3,$5); }
| SUBSTRING_INDEX '(' expr ',' expr ',' expr ')'
{ $$= new Item_func_substr_index($3,$5,$7); }
2003-06-23 12:56:44 +05:00
| TIME_SYM '(' expr ')'
2003-07-08 15:06:05 +05:00
{ $$= new Item_time_typecast($3); }
| TIMESTAMP '(' expr ')'
{ $$= new Item_datetime_typecast($3); }
2003-06-23 12:56:44 +05:00
| TIMESTAMP '(' expr ',' expr ')'
2003-07-08 15:06:05 +05:00
{ $$= new Item_func_add_time($3, $5, 1, 0); }
| TIMESTAMP_ADD '(' interval_time_st ',' expr ',' expr ')'
{ $$= new Item_date_add_interval($7,$5,$3,0); }
| TIMESTAMP_DIFF '(' interval_time_st ',' expr ',' expr ')'
{ $$= new Item_func_timestamp_diff($5,$7,$3); }
2000-07-31 21:29:14 +02:00
| TRIM '(' expr ')'
{ $$= new Item_func_trim($3); }
| TRIM '(' LEADING expr FROM expr ')'
2000-07-31 21:29:14 +02:00
{ $$= new Item_func_ltrim($6,$4); }
| TRIM '(' TRAILING expr FROM expr ')'
2000-07-31 21:29:14 +02:00
{ $$= new Item_func_rtrim($6,$4); }
| TRIM '(' BOTH expr FROM expr ')'
2000-07-31 21:29:14 +02:00
{ $$= new Item_func_trim($6,$4); }
| TRIM '(' LEADING FROM expr ')'
{ $$= new Item_func_ltrim($5); }
| TRIM '(' TRAILING FROM expr ')'
{ $$= new Item_func_rtrim($5); }
| TRIM '(' BOTH FROM expr ')'
{ $$= new Item_func_trim($5); }
2000-07-31 21:29:14 +02:00
| TRIM '(' expr FROM expr ')'
{ $$= new Item_func_trim($5,$3); }
| TRUNCATE_SYM '(' expr ',' expr ')'
{ $$= new Item_func_round($3,$5,1); }
| TRUE_SYM
{ $$= new Item_int((char*) "TRUE",1,1); }
| SP_FUNC '(' sp_expr_list ')'
{
sp_add_fun_to_lex(Lex, $1);
if ($3)
$$= new Item_func_sp($1, *$3);
else
$$= new Item_func_sp($1);
}
2000-07-31 21:29:14 +02:00
| UDA_CHAR_SUM '(' udf_expr_list ')'
{
if ($3 != NULL)
$$ = new Item_sum_udf_str($1, *$3);
else
$$ = new Item_sum_udf_str($1);
}
| UDA_FLOAT_SUM '(' udf_expr_list ')'
{
if ($3 != NULL)
$$ = new Item_sum_udf_float($1, *$3);
else
$$ = new Item_sum_udf_float($1);
}
| UDA_INT_SUM '(' udf_expr_list ')'
{
if ($3 != NULL)
$$ = new Item_sum_udf_int($1, *$3);
else
$$ = new Item_sum_udf_int($1);
}
| UDF_CHAR_FUNC '(' udf_expr_list ')'
{
if ($3 != NULL)
$$ = new Item_func_udf_str($1, *$3);
else
$$ = new Item_func_udf_str($1);
}
| UDF_FLOAT_FUNC '(' udf_expr_list ')'
{
if ($3 != NULL)
$$ = new Item_func_udf_float($1, *$3);
else
$$ = new Item_func_udf_float($1);
}
| UDF_INT_FUNC '(' udf_expr_list ')'
{
if ($3 != NULL)
$$ = new Item_func_udf_int($1, *$3);
else
$$ = new Item_func_udf_int($1);
}
| UNIQUE_USERS '(' text_literal ',' NUM ',' NUM ',' expr_list ')'
{
2001-12-02 14:34:01 +02:00
$$= new Item_func_unique_users($3,atoi($5.str),atoi($7.str), * $9);
}
2000-07-31 21:29:14 +02:00
| UNIX_TIMESTAMP '(' ')'
2001-12-02 14:34:01 +02:00
{
$$= new Item_func_unix_timestamp();
Lex->safe_to_cache_query=0;
2001-12-02 14:34:01 +02:00
}
2000-07-31 21:29:14 +02:00
| UNIX_TIMESTAMP '(' expr ')'
{ $$= new Item_func_unix_timestamp($3); }
| USER '(' ')'
{ $$= new Item_func_user(); Lex->safe_to_cache_query=0; }
| UTC_DATE_SYM optional_braces
{ $$= new Item_func_curdate_utc(); Lex->safe_to_cache_query=0;}
| UTC_TIME_SYM optional_braces
{ $$= new Item_func_curtime_utc(); Lex->safe_to_cache_query=0;}
| UTC_TIMESTAMP_SYM optional_braces
{ $$= new Item_func_now_utc(); Lex->safe_to_cache_query=0;}
2000-07-31 21:29:14 +02:00
| WEEK_SYM '(' expr ')'
2003-08-11 22:44:43 +03:00
{
$$= new Item_func_week($3,new Item_int((char*) "0",
2003-08-11 22:44:43 +03:00
YYTHD->variables.default_week_format,1));
}
2000-07-31 21:29:14 +02:00
| WEEK_SYM '(' expr ',' expr ')'
{ $$= new Item_func_week($3,$5); }
| YEAR_SYM '(' expr ')'
{ $$= new Item_func_year($3); }
| YEARWEEK '(' expr ')'
{ $$= new Item_func_yearweek($3,new Item_int((char*) "0",0,1)); }
| YEARWEEK '(' expr ',' expr ')'
{ $$= new Item_func_yearweek($3, $5); }
| BENCHMARK_SYM '(' ULONG_NUM ',' expr ')'
{
2001-12-02 14:34:01 +02:00
$$=new Item_func_benchmark($3,$5);
Lex->uncacheable(UNCACHEABLE_SIDEEFFECT);
2001-12-02 14:34:01 +02:00
}
2000-08-22 00:39:08 +03:00
| EXTRACT_SYM '(' interval FROM expr ')'
{ $$=new Item_extract( $3, $5); };
2000-07-31 21:29:14 +02:00
fulltext_options:
/* nothing */ { $$= FT_NL; }
| WITH QUERY_SYM EXPANSION_SYM { $$= FT_NL | FT_EXPAND; }
| IN_SYM BOOLEAN_SYM MODE_SYM { $$= FT_BOOL; }
;
sp_expr_list:
2000-07-31 21:29:14 +02:00
/* empty */ { $$= NULL; }
| expr_list { $$= $1;};
2000-07-31 21:29:14 +02:00
udf_expr_list:
/* empty */ { $$= NULL; }
| udf_expr_list2 { $$= $1;}
;
udf_expr_list2:
{ Select->expr_list.push_front(new List<Item>); }
udf_expr_list3
{ $$= Select->expr_list.pop(); }
;
udf_expr_list3:
udf_expr
{
Select->expr_list.head()->push_back($1);
}
| udf_expr_list3 ',' udf_expr
{
Select->expr_list.head()->push_back($3);
}
;
udf_expr:
remember_name expr remember_end select_alias
{
if ($4.str)
$2->set_name($4.str,$4.length,system_charset_info);
else
$2->set_name($1,(uint) ($3 - $1), YYTHD->charset());
$$= $2;
}
;
2000-07-31 21:29:14 +02:00
sum_expr:
AVG_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_avg($3); }
| BIT_AND '(' in_sum_expr ')'
{ $$=new Item_sum_and($3); }
| BIT_OR '(' in_sum_expr ')'
{ $$=new Item_sum_or($3); }
| BIT_XOR '(' in_sum_expr ')'
{ $$=new Item_sum_xor($3); }
| COUNT_SYM '(' opt_all '*' ')'
2000-07-31 21:29:14 +02:00
{ $$=new Item_sum_count(new Item_int((int32) 0L,1)); }
| COUNT_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_count($3); }
| COUNT_SYM '(' DISTINCT
{ Select->in_sum_expr++; }
expr_list
{ Select->in_sum_expr--; }
')'
{ $$=new Item_sum_count_distinct(* $5); }
2000-07-31 21:29:14 +02:00
| GROUP_UNIQUE_USERS '(' text_literal ',' NUM ',' NUM ',' in_sum_expr ')'
{ $$= new Item_sum_unique_users($3,atoi($5.str),atoi($7.str),$9); }
| MIN_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_min($3); }
/*
According to ANSI SQL, DISTINCT is allowed and has
no sence inside MIN and MAX grouping functions; so MIN|MAX(DISTINCT ...)
is processed like an ordinary MIN | MAX()
*/
| MIN_SYM '(' DISTINCT in_sum_expr ')'
{ $$=new Item_sum_min($4); }
2000-07-31 21:29:14 +02:00
| MAX_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_max($3); }
| MAX_SYM '(' DISTINCT in_sum_expr ')'
{ $$=new Item_sum_max($4); }
2000-07-31 21:29:14 +02:00
| STD_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_std($3); }
2002-12-14 03:36:59 +04:00
| VARIANCE_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_variance($3); }
2000-07-31 21:29:14 +02:00
| SUM_SYM '(' in_sum_expr ')'
{ $$=new Item_sum_sum($3); }
| SUM_SYM '(' DISTINCT in_sum_expr ')'
{ $$=new Item_sum_sum_distinct($4); }
2003-08-11 22:44:43 +03:00
| GROUP_CONCAT_SYM '(' opt_distinct expr_list opt_gorder_clause
opt_gconcat_separator ')'
{
$$=new Item_func_group_concat($3,$4,Lex->gorder_list,$6);
$4->empty();
};
opt_distinct:
/* empty */ { $$ = 0; }
|DISTINCT { $$ = 1; };
opt_gconcat_separator:
/* empty */ { $$ = new String(",",1,default_charset_info); }
|SEPARATOR_SYM text_string { $$ = $2; };
2003-08-11 22:44:43 +03:00
opt_gorder_clause:
/* empty */
{
LEX *lex=Lex;
lex->gorder_list = NULL;
}
| order_clause
{
LEX *lex=Lex;
lex->gorder_list=
(SQL_LIST*) sql_memdup((char*) &lex->current_select->order_list,
sizeof(st_sql_list));
lex->current_select->order_list.empty();
};
2003-08-11 22:44:43 +03:00
2000-07-31 21:29:14 +02:00
in_sum_expr:
opt_all
{
LEX *lex= Lex;
if (lex->current_select->inc_in_sum_expr())
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
}
2000-07-31 21:29:14 +02:00
expr
{
Select->in_sum_expr--;
2003-02-07 16:38:37 +02:00
$$= $3;
};
2000-07-31 21:29:14 +02:00
2003-08-21 14:15:25 +05:00
cast_type_init:
{ Lex->charset= NULL; Lex->length= (char*)0; }
;
cast_type_finalize:
BINARY { $$=ITEM_CAST_BINARY; }
| CHAR_SYM opt_len opt_binary { $$=ITEM_CAST_CHAR; }
| NCHAR_SYM opt_len { $$=ITEM_CAST_CHAR; Lex->charset= national_charset_info; }
| SIGNED_SYM { $$=ITEM_CAST_SIGNED_INT; }
| SIGNED_SYM INT_SYM { $$=ITEM_CAST_SIGNED_INT; }
| UNSIGNED { $$=ITEM_CAST_UNSIGNED_INT; }
| UNSIGNED INT_SYM { $$=ITEM_CAST_UNSIGNED_INT; }
| DATE_SYM { $$=ITEM_CAST_DATE; }
| TIME_SYM { $$=ITEM_CAST_TIME; }
| DATETIME { $$=ITEM_CAST_DATETIME; }
;
cast_type:
2003-08-21 14:15:25 +05:00
cast_type_init cast_type_finalize { $$= $2; }
;
2000-07-31 21:29:14 +02:00
expr_list:
{ Select->expr_list.push_front(new List<Item>); }
2000-07-31 21:29:14 +02:00
expr_list2
{ $$= Select->expr_list.pop(); };
2000-07-31 21:29:14 +02:00
expr_list2:
expr { Select->expr_list.head()->push_back($1); }
| expr_list2 ',' expr { Select->expr_list.head()->push_back($3); };
2000-07-31 21:29:14 +02:00
2001-10-09 14:53:54 +02:00
ident_list_arg:
ident_list { $$= $1; }
| '(' ident_list ')' { $$= $2; };
2001-10-09 14:53:54 +02:00
2000-07-31 21:29:14 +02:00
ident_list:
{ Select->expr_list.push_front(new List<Item>); }
2000-07-31 21:29:14 +02:00
ident_list2
{ $$= Select->expr_list.pop(); };
2000-07-31 21:29:14 +02:00
ident_list2:
simple_ident { Select->expr_list.head()->push_back($1); }
| ident_list2 ',' simple_ident { Select->expr_list.head()->push_back($3); };
2000-07-31 21:29:14 +02:00
opt_expr:
/* empty */ { $$= NULL; }
| expr { $$= $1; };
2000-07-31 21:29:14 +02:00
opt_else:
/* empty */ { $$= NULL; }
| ELSE expr { $$= $2; };
2000-07-31 21:29:14 +02:00
when_list:
2002-02-28 21:58:32 +02:00
{ Select->when_list.push_front(new List<Item>); }
2000-07-31 21:29:14 +02:00
when_list2
{ $$= Select->when_list.pop(); };
2000-07-31 21:29:14 +02:00
when_list2:
2000-08-22 00:39:08 +03:00
expr THEN_SYM expr
2000-07-31 21:29:14 +02:00
{
SELECT_LEX *sel=Select;
sel->when_list.head()->push_back($1);
sel->when_list.head()->push_back($3);
2000-07-31 21:29:14 +02:00
}
2000-08-22 00:39:08 +03:00
| when_list2 WHEN_SYM expr THEN_SYM expr
2000-07-31 21:29:14 +02:00
{
SELECT_LEX *sel=Select;
sel->when_list.head()->push_back($3);
sel->when_list.head()->push_back($5);
};
2000-07-31 21:29:14 +02:00
join_table_list:
'(' join_table_list ')' { $$=$2; }
| join_table { $$=$1; }
| join_table_list ',' join_table_list { $$=$3; }
| join_table_list normal_join join_table_list { $$=$3; }
| join_table_list STRAIGHT_JOIN join_table_list
{ $$=$3 ; $1->next->straight=1; }
| join_table_list normal_join join_table_list ON expr
{ add_join_on($3,$5); $$=$3; }
| join_table_list normal_join join_table_list
2003-08-11 22:44:43 +03:00
USING
{
SELECT_LEX *sel= Select;
sel->db1=$1->db; sel->table1=$1->alias;
sel->db2=$3->db; sel->table2=$3->alias;
}
'(' using_list ')'
{ add_join_on($3,$7); $$=$3; }
| join_table_list LEFT opt_outer JOIN_SYM join_table_list ON expr
{ add_join_on($5,$7); $5->outer_join|=JOIN_TYPE_LEFT; $$=$5; }
| join_table_list LEFT opt_outer JOIN_SYM join_table_list
{
SELECT_LEX *sel= Select;
sel->db1=$1->db; sel->table1=$1->alias;
sel->db2=$5->db; sel->table2=$5->alias;
}
2000-07-31 21:29:14 +02:00
USING '(' using_list ')'
{ add_join_on($5,$9); $5->outer_join|=JOIN_TYPE_LEFT; $$=$5; }
| join_table_list NATURAL LEFT opt_outer JOIN_SYM join_table_list
{
add_join_natural($1,$1->next);
$1->next->outer_join|=JOIN_TYPE_LEFT;
$$=$6;
}
| join_table_list RIGHT opt_outer JOIN_SYM join_table_list ON expr
{ add_join_on($1,$7); $1->outer_join|=JOIN_TYPE_RIGHT; $$=$5; }
| join_table_list RIGHT opt_outer JOIN_SYM join_table_list
{
SELECT_LEX *sel= Select;
sel->db1=$1->db; sel->table1=$1->alias;
sel->db2=$5->db; sel->table2=$5->alias;
}
USING '(' using_list ')'
{ add_join_on($1,$9); $1->outer_join|=JOIN_TYPE_RIGHT; $$=$5; }
| join_table_list NATURAL RIGHT opt_outer JOIN_SYM join_table_list
{
add_join_natural($1->next,$1);
$1->outer_join|=JOIN_TYPE_RIGHT;
$$=$6;
}
| join_table_list NATURAL JOIN_SYM join_table_list
{ add_join_natural($1,$1->next); $$=$4; };
2000-07-31 21:29:14 +02:00
normal_join:
JOIN_SYM {}
| INNER_SYM JOIN_SYM {}
| CROSS JOIN_SYM {}
;
2000-07-31 21:29:14 +02:00
join_table:
{
SELECT_LEX *sel= Select;
sel->use_index_ptr=sel->ignore_index_ptr=0;
sel->table_join_options= 0;
}
2000-07-31 21:29:14 +02:00
table_ident opt_table_alias opt_key_definition
{
2002-11-23 18:54:15 +02:00
LEX *lex= Lex;
SELECT_LEX *sel= lex->current_select;
if (!($$= sel->add_table_to_list(lex->thd, $2, $3,
sel->get_table_join_options(),
lex->lock_option,
sel->get_use_index(),
sel->get_ignore_index())))
YYABORT;
}
2000-07-31 21:29:14 +02:00
| '{' ident join_table LEFT OUTER JOIN_SYM join_table ON expr '}'
{ add_join_on($7,$9); $7->outer_join|=JOIN_TYPE_LEFT; $$=$7; }
| '(' SELECT_SYM select_derived ')' opt_table_alias
2002-03-26 15:06:05 +02:00
{
LEX *lex=Lex;
if (lex->sql_command == SQLCOM_UPDATE &&
&lex->select_lex == lex->current_select->outer_select())
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
SELECT_LEX_UNIT *unit= lex->current_select->master_unit();
lex->current_select= unit->outer_select();
if (!($$= lex->current_select->
add_table_to_list(lex->thd, new Table_ident(unit), $5, 0,
TL_READ,(List<String> *)0,
(List<String> *)0)))
2002-03-26 15:06:05 +02:00
YYABORT;
2002-06-05 21:32:22 +03:00
};
2002-03-26 15:06:05 +02:00
select_derived:
2002-03-26 15:06:05 +02:00
{
2002-05-07 00:04:16 +03:00
LEX *lex= Lex;
2002-12-05 19:38:42 +02:00
lex->derived_tables= 1;
2003-08-11 22:44:43 +03:00
if (((int)lex->sql_command >= (int)SQLCOM_HA_OPEN &&
lex->sql_command <= (int)SQLCOM_HA_READ) ||
lex->sql_command == (int)SQLCOM_KILL)
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
if (lex->current_select->linkage == GLOBAL_OPTIONS_TYPE ||
2002-05-07 00:04:16 +03:00
mysql_new_select(lex, 1))
2002-03-26 15:06:05 +02:00
YYABORT;
mysql_init_select(lex);
lex->current_select->linkage= DERIVED_TABLE_TYPE;
2002-03-26 15:06:05 +02:00
}
select_options select_item_list opt_select_from union_opt
;
2000-07-31 21:29:14 +02:00
opt_outer:
/* empty */ {}
| OUTER {};
2000-07-31 21:29:14 +02:00
opt_key_definition:
/* empty */ {}
| USE_SYM key_usage_list
{
SELECT_LEX *sel= Select;
sel->use_index= *$2;
sel->use_index_ptr= &sel->use_index;
}
| FORCE_SYM key_usage_list
{
SELECT_LEX *sel= Select;
sel->use_index= *$2;
sel->use_index_ptr= &sel->use_index;
sel->table_join_options|= TL_OPTION_FORCE_INDEX;
}
2000-07-31 21:29:14 +02:00
| IGNORE_SYM key_usage_list
{
SELECT_LEX *sel= Select;
sel->ignore_index= *$2;
sel->ignore_index_ptr= &sel->ignore_index;
};
2000-07-31 21:29:14 +02:00
key_usage_list:
key_or_index { Select->interval_list.empty(); }
'(' key_list_or_empty ')'
{ $$= &Select->interval_list; }
;
key_list_or_empty:
/* empty */ {}
| key_usage_list2 {}
;
2000-07-31 21:29:14 +02:00
key_usage_list2:
key_usage_list2 ',' ident
{ Select->
interval_list.push_back(new String((const char*) $3.str, $3.length,
system_charset_info)); }
2000-07-31 21:29:14 +02:00
| ident
{ Select->
interval_list.push_back(new String((const char*) $1.str, $1.length,
system_charset_info)); }
2000-07-31 21:29:14 +02:00
| PRIMARY_SYM
{ Select->
interval_list.push_back(new String("PRIMARY", 7,
system_charset_info)); };
2000-07-31 21:29:14 +02:00
using_list:
ident
{
SELECT_LEX *sel= Select;
if (!($$= new Item_func_eq(new Item_field(sel->db1, sel->table1,
$1.str),
new Item_field(sel->db2, sel->table2,
$1.str))))
2000-07-31 21:29:14 +02:00
YYABORT;
}
| using_list ',' ident
{
SELECT_LEX *sel= Select;
if (!($$= new Item_cond_and(new Item_func_eq(new Item_field(sel->db1,sel->table1,$3.str), new Item_field(sel->db2,sel->table2,$3.str)), $1)))
2000-07-31 21:29:14 +02:00
YYABORT;
};
2000-07-31 21:29:14 +02:00
interval:
interval_time_st {}
| DAY_HOUR_SYM { $$=INTERVAL_DAY_HOUR; }
2003-06-23 12:56:44 +05:00
| DAY_MICROSECOND_SYM { $$=INTERVAL_DAY_MICROSECOND; }
2000-07-31 21:29:14 +02:00
| DAY_MINUTE_SYM { $$=INTERVAL_DAY_MINUTE; }
| DAY_SECOND_SYM { $$=INTERVAL_DAY_SECOND; }
2003-06-23 12:56:44 +05:00
| HOUR_MICROSECOND_SYM { $$=INTERVAL_HOUR_MICROSECOND; }
2000-07-31 21:29:14 +02:00
| HOUR_MINUTE_SYM { $$=INTERVAL_HOUR_MINUTE; }
| HOUR_SECOND_SYM { $$=INTERVAL_HOUR_SECOND; }
2003-06-23 12:56:44 +05:00
| MICROSECOND_SYM { $$=INTERVAL_MICROSECOND; }
| MINUTE_MICROSECOND_SYM { $$=INTERVAL_MINUTE_MICROSECOND; }
2000-07-31 21:29:14 +02:00
| MINUTE_SECOND_SYM { $$=INTERVAL_MINUTE_SECOND; }
| SECOND_MICROSECOND_SYM { $$=INTERVAL_SECOND_MICROSECOND; }
| YEAR_MONTH_SYM { $$=INTERVAL_YEAR_MONTH; };
interval_time_st:
DAY_SYM { $$=INTERVAL_DAY; }
| WEEK_SYM { $$=INTERVAL_WEEK; }
| HOUR_SYM { $$=INTERVAL_HOUR; }
| FRAC_SECOND_SYM { $$=INTERVAL_MICROSECOND; }
2000-07-31 21:29:14 +02:00
| MINUTE_SYM { $$=INTERVAL_MINUTE; }
| MONTH_SYM { $$=INTERVAL_MONTH; }
| QUARTER_SYM { $$=INTERVAL_QUARTER; }
2000-07-31 21:29:14 +02:00
| SECOND_SYM { $$=INTERVAL_SECOND; }
| YEAR_SYM { $$=INTERVAL_YEAR; };
2000-07-31 21:29:14 +02:00
date_time_type:
DATE_SYM {$$=TIMESTAMP_DATE;}
| TIME_SYM {$$=TIMESTAMP_TIME;}
| DATETIME {$$=TIMESTAMP_DATETIME;};
2000-07-31 21:29:14 +02:00
table_alias:
/* empty */
| AS
| EQ;
2000-07-31 21:29:14 +02:00
opt_table_alias:
/* empty */ { $$=0; }
| table_alias ident
{ $$= (LEX_STRING*) sql_memdup(&$2,sizeof(LEX_STRING)); };
2000-07-31 21:29:14 +02:00
opt_all:
/* empty */
| ALL
;
2000-07-31 21:29:14 +02:00
where_clause:
/* empty */ { Select->where= 0; }
| WHERE expr
{
Select->where= $2;
if ($2)
$2->top_level_item();
}
2002-11-21 15:56:48 +02:00
;
2000-07-31 21:29:14 +02:00
having_clause:
/* empty */
2003-05-17 10:05:07 +03:00
| HAVING
{
Select->parsing_place= SELECT_LEX_NODE::IN_HAVING;
2003-05-17 10:05:07 +03:00
}
expr
2002-11-21 15:56:48 +02:00
{
SELECT_LEX *sel= Select;
2003-05-17 10:05:07 +03:00
sel->having= $3;
sel->parsing_place= SELECT_LEX_NODE::NO_MATTER;
2002-11-21 15:56:48 +02:00
if ($3)
$3->top_level_item();
}
;
2000-07-31 21:29:14 +02:00
opt_escape:
ESCAPE_SYM TEXT_STRING_literal { $$= $2.str; }
2003-03-17 21:56:34 +04:00
| /* empty */ { $$= (char*) "\\"; };
2000-07-31 21:29:14 +02:00
/*
2001-12-17 19:59:20 +02:00
group by statement in select
2000-07-31 21:29:14 +02:00
*/
group_clause:
/* empty */
| GROUP BY group_list olap_opt;
2000-07-31 21:29:14 +02:00
group_list:
2001-12-10 17:51:07 +02:00
group_list ',' order_ident order_dir
{ if (add_group_to_list(YYTHD, $3,(bool) $4)) YYABORT; }
2001-12-10 17:51:07 +02:00
| order_ident order_dir
{ if (add_group_to_list(YYTHD, $1,(bool) $2)) YYABORT; };
2000-07-31 21:29:14 +02:00
olap_opt:
/* empty */ {}
| WITH CUBE_SYM
{
LEX *lex=Lex;
if (lex->current_select->linkage == GLOBAL_OPTIONS_TYPE)
{
net_printf(lex->thd, ER_WRONG_USAGE, "WITH CUBE",
"global union parameters");
YYABORT;
}
lex->current_select->olap= CUBE_TYPE;
net_printf(lex->thd, ER_NOT_SUPPORTED_YET, "CUBE");
YYABORT; /* To be deleted in 5.1 */
}
| WITH ROLLUP_SYM
{
LEX *lex= Lex;
if (lex->current_select->linkage == GLOBAL_OPTIONS_TYPE)
{
net_printf(lex->thd, ER_WRONG_USAGE, "WITH ROLLUP",
"global union parameters");
YYABORT;
}
lex->current_select->olap= ROLLUP_TYPE;
}
;
2000-07-31 21:29:14 +02:00
/*
2001-12-17 19:59:20 +02:00
Order by statement in select
2000-07-31 21:29:14 +02:00
*/
opt_order_clause:
2000-07-31 21:29:14 +02:00
/* empty */
| order_clause;
order_clause:
ORDER_SYM BY
{
LEX *lex=Lex;
if (lex->current_select->linkage != GLOBAL_OPTIONS_TYPE &&
lex->current_select->olap !=
UNSPECIFIED_OLAP_TYPE)
{
net_printf(lex->thd, ER_WRONG_USAGE,
"CUBE/ROLLUP",
"ORDER BY");
YYABORT;
}
} order_list;
2000-07-31 21:29:14 +02:00
order_list:
order_list ',' order_ident order_dir
{ if (add_order_to_list(YYTHD, $3,(bool) $4)) YYABORT; }
2000-07-31 21:29:14 +02:00
| order_ident order_dir
{ if (add_order_to_list(YYTHD, $1,(bool) $2)) YYABORT; };
2000-07-31 21:29:14 +02:00
order_dir:
/* empty */ { $$ = 1; }
2001-12-10 17:51:07 +02:00
| ASC { $$ =1; }
| DESC { $$ =0; };
2000-07-31 21:29:14 +02:00
opt_limit_clause_init:
/* empty */
{
SELECT_LEX *sel= Select;
sel->offset_limit= 0L;
sel->select_limit= Lex->thd->variables.select_limit;
}
| limit_clause {}
;
opt_limit_clause:
/* empty */ {}
| limit_clause {}
;
limit_clause:
LIMIT limit_options {}
2002-11-21 15:56:48 +02:00
;
limit_options:
ULONG_NUM
{
SELECT_LEX *sel= Select;
sel->select_limit= $1;
sel->offset_limit= 0L;
}
| ULONG_NUM ',' ULONG_NUM
{
SELECT_LEX *sel= Select;
sel->select_limit= $3;
sel->offset_limit= $1;
}
| ULONG_NUM OFFSET_SYM ULONG_NUM
{
SELECT_LEX *sel= Select;
sel->select_limit= $1;
sel->offset_limit= $3;
}
;
2000-07-31 21:29:14 +02:00
delete_limit_clause:
/* empty */
{
LEX *lex=Lex;
lex->current_select->select_limit= HA_POS_ERROR;
2000-07-31 21:29:14 +02:00
}
| LIMIT ulonglong_num
{ Select->select_limit= (ha_rows) $2; };
2000-07-31 21:29:14 +02:00
ULONG_NUM:
NUM { $$= strtoul($1.str,NULL,10); }
| LONG_NUM { $$= (ulong) strtoll($1.str,NULL,10); }
2001-09-15 16:22:34 +03:00
| ULONGLONG_NUM { $$= (ulong) strtoull($1.str,NULL,10); }
| REAL_NUM { $$= strtoul($1.str,NULL,10); }
| FLOAT_NUM { $$= strtoul($1.str,NULL,10); };
2000-07-31 21:29:14 +02:00
ulonglong_num:
NUM { $$= (ulonglong) strtoul($1.str,NULL,10); }
2001-09-15 16:22:34 +03:00
| ULONGLONG_NUM { $$= strtoull($1.str,NULL,10); }
| LONG_NUM { $$= (ulonglong) strtoll($1.str,NULL,10); }
| REAL_NUM { $$= strtoull($1.str,NULL,10); }
| FLOAT_NUM { $$= strtoull($1.str,NULL,10); };
2000-07-31 21:29:14 +02:00
procedure_clause:
/* empty */
| PROCEDURE ident /* Procedure name */
{
LEX *lex=Lex;
if (&lex->select_lex != lex->current_select)
{
net_printf(lex->thd, ER_WRONG_USAGE,
"PROCEDURE",
"subquery");
YYABORT;
}
2000-07-31 21:29:14 +02:00
lex->proc_list.elements=0;
lex->proc_list.first=0;
lex->proc_list.next= (byte**) &lex->proc_list.first;
if (add_proc_to_list(lex->thd, new Item_field(NULL,NULL,$2.str)))
2000-07-31 21:29:14 +02:00
YYABORT;
Lex->uncacheable(UNCACHEABLE_SIDEEFFECT);
2000-07-31 21:29:14 +02:00
}
'(' procedure_list ')';
2000-07-31 21:29:14 +02:00
procedure_list:
/* empty */ {}
| procedure_list2 {};
2000-07-31 21:29:14 +02:00
procedure_list2:
procedure_list2 ',' procedure_item
| procedure_item;
2000-07-31 21:29:14 +02:00
procedure_item:
remember_name expr
{
LEX *lex= Lex;
if (add_proc_to_list(lex->thd, $2))
2000-07-31 21:29:14 +02:00
YYABORT;
if (!$2->name)
2003-03-17 21:56:34 +04:00
$2->set_name($1,(uint) ((char*) lex->tok_end - $1), YYTHD->charset());
2002-10-16 16:55:08 +03:00
}
;
select_var_list_init:
{
LEX *lex=Lex;
if (!lex->describe && (!(lex->result= new select_dumpvar())))
2002-10-16 16:55:08 +03:00
YYABORT;
}
select_var_list
2002-12-07 12:35:57 +01:00
{}
2002-10-16 16:55:08 +03:00
;
2000-07-31 21:29:14 +02:00
2002-10-11 21:49:10 +03:00
select_var_list:
2002-10-16 16:55:08 +03:00
select_var_list ',' select_var_ident
| select_var_ident {}
;
select_var_ident:
'@' ident_or_text
2002-10-11 21:49:10 +03:00
{
2002-10-16 16:55:08 +03:00
LEX *lex=Lex;
if (lex->result)
((select_dumpvar *)lex->result)->var_list.push_back( new my_var($2,0,0,(enum_field_types)0));
else
2002-10-11 21:49:10 +03:00
YYABORT;
}
| ident_or_text
2002-10-11 21:49:10 +03:00
{
2002-10-16 16:55:08 +03:00
LEX *lex=Lex;
sp_pvar_t *t;
if (!lex->spcont || !(t=lex->spcont->find_pvar(&$1)))
{
net_printf(YYTHD, ER_SP_UNDECLARED_VAR, $1.str);
2002-10-11 21:49:10 +03:00
YYABORT;
}
if (! lex->result)
2002-10-11 21:49:10 +03:00
YYABORT;
else
{
((select_dumpvar *)lex->result)->var_list.push_back( new my_var($1,1,t->offset,t->type));
t->isset= TRUE;
}
2002-10-11 21:49:10 +03:00
}
;
2000-07-31 21:29:14 +02:00
into:
2003-03-17 21:56:34 +04:00
INTO OUTFILE TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
2002-10-16 16:55:08 +03:00
LEX *lex=Lex;
if (!lex->describe)
{
lex->uncacheable(UNCACHEABLE_SIDEEFFECT);
if (!(lex->exchange= new sql_exchange($3.str,0)))
YYABORT;
if (!(lex->result= new select_export(lex->exchange)))
YYABORT;
}
2000-07-31 21:29:14 +02:00
}
opt_field_term opt_line_term
2003-03-17 21:56:34 +04:00
| INTO DUMPFILE TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
2002-10-16 16:55:08 +03:00
LEX *lex=Lex;
if (!lex->describe)
{
lex->uncacheable(UNCACHEABLE_SIDEEFFECT);
if (!(lex->exchange= new sql_exchange($3.str,1)))
YYABORT;
if (!(lex->result= new select_dump(lex->exchange)))
YYABORT;
}
2002-10-11 21:49:10 +03:00
}
2002-10-16 16:55:08 +03:00
| INTO select_var_list_init
2002-10-11 21:49:10 +03:00
{
Lex->uncacheable(UNCACHEABLE_SIDEEFFECT);
2002-10-16 16:55:08 +03:00
}
;
2000-07-31 21:29:14 +02:00
2001-12-17 19:59:20 +02:00
/*
DO statement
*/
2000-07-31 21:29:14 +02:00
do: DO_SYM
2001-12-17 19:59:20 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_DO;
if (!(lex->insert_list = new List_item))
YYABORT;
}
values
{}
;
2000-07-31 21:29:14 +02:00
/*
Drop : delete tables or index or user
2000-07-31 21:29:14 +02:00
*/
drop:
DROP opt_temporary table_or_tables if_exists table_list opt_restrict
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_DROP_TABLE;
lex->drop_temporary= $2;
lex->drop_if_exists= $4;
2000-07-31 21:29:14 +02:00
}
| DROP INDEX ident ON table_ident {}
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_DROP_INDEX;
lex->drop_list.empty();
lex->drop_list.push_back(new Alter_drop(Alter_drop::KEY,
2000-07-31 21:29:14 +02:00
$3.str));
if (!lex->current_select->add_table_to_list(lex->thd, $5, NULL,
TL_OPTION_UPDATING))
2000-07-31 21:29:14 +02:00
YYABORT;
}
| DROP DATABASE if_exists ident
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_DROP_DB;
lex->drop_if_exists=$3;
lex->name=$4.str;
2000-07-31 21:29:14 +02:00
}
| DROP FUNCTION_SYM if_exists IDENT_sys opt_restrict
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_DROP_FUNCTION;
lex->drop_if_exists= $3;
lex->udf.name= $4;
}
| DROP PROCEDURE if_exists IDENT_sys opt_restrict
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_DROP_PROCEDURE;
lex->drop_if_exists= $3;
lex->udf.name= $4;
}
| DROP USER
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_DROP_USER;
lex->users_list.empty();
2003-08-11 22:44:43 +03:00
}
user_list
{}
;
2000-07-31 21:29:14 +02:00
table_list:
table_name
| table_list ',' table_name;
2000-07-31 21:29:14 +02:00
table_name:
2000-07-31 21:29:14 +02:00
table_ident
{
if (!Select->add_table_to_list(YYTHD, $1, NULL, TL_OPTION_UPDATING))
YYABORT;
}
;
2000-07-31 21:29:14 +02:00
if_exists:
/* empty */ { $$= 0; }
| IF EXISTS { $$= 1; }
;
2000-07-31 21:29:14 +02:00
opt_temporary:
/* empty */ { $$= 0; }
| TEMPORARY { $$= 1; }
;
2000-07-31 21:29:14 +02:00
/*
** Insert : add new data to table
*/
insert:
2002-11-26 01:00:05 +02:00
INSERT
{
LEX *lex= Lex;
lex->sql_command = SQLCOM_INSERT;
/* for subselects */
lex->lock_option= (using_update_log) ? TL_READ_NO_INSERT : TL_READ;
lex->select_lex.resolve_mode= SELECT_LEX::INSERT_MODE;
2002-11-26 01:00:05 +02:00
} insert_lock_option
opt_ignore insert2
{
2002-11-21 22:25:53 +02:00
Select->set_lock_for_tables($3);
Lex->current_select= &Lex->select_lex;
}
insert_field_spec opt_insert_update
2002-12-07 12:35:57 +01:00
{}
;
2000-07-31 21:29:14 +02:00
replace:
REPLACE
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_REPLACE;
lex->duplicates= DUP_REPLACE;
lex->select_lex.resolve_mode= SELECT_LEX::INSERT_MODE;
}
replace_lock_option insert2
{
2002-11-21 22:25:53 +02:00
Select->set_lock_for_tables($3);
Lex->current_select= &Lex->select_lex;
}
insert_field_spec
{}
;
2000-07-31 21:29:14 +02:00
insert_lock_option:
/* empty */ { $$= TL_WRITE_CONCURRENT_INSERT; }
| LOW_PRIORITY { $$= TL_WRITE_LOW_PRIORITY; }
| DELAYED_SYM { $$= TL_WRITE_DELAYED; }
| HIGH_PRIORITY { $$= TL_WRITE; }
;
2000-07-31 21:29:14 +02:00
replace_lock_option:
opt_low_priority { $$= $1; }
| DELAYED_SYM { $$= TL_WRITE_DELAYED; };
2000-07-31 21:29:14 +02:00
insert2:
INTO insert_table {}
| insert_table {};
2000-07-31 21:29:14 +02:00
insert_table:
table_name
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->field_list.empty();
lex->many_values.empty();
lex->insert_list=0;
};
2000-07-31 21:29:14 +02:00
insert_field_spec:
insert_values {}
| '(' ')' insert_values {}
| '(' fields ')' insert_values {}
2000-07-31 21:29:14 +02:00
| SET
{
LEX *lex=Lex;
if (!(lex->insert_list = new List_item) ||
lex->many_values.push_back(lex->insert_list))
2000-07-31 21:29:14 +02:00
YYABORT;
}
ident_eq_list;
2000-07-31 21:29:14 +02:00
opt_field_spec:
/* empty */ { }
| '(' fields ')' { }
| '(' ')' { };
2000-07-31 21:29:14 +02:00
fields:
fields ',' insert_ident { Lex->field_list.push_back($3); }
| insert_ident { Lex->field_list.push_back($1); };
2000-07-31 21:29:14 +02:00
insert_values:
VALUES values_list {}
| VALUE_SYM values_list {}
| create_select { Select->set_braces(0);} union_clause {}
| '(' create_select ')' { Select->set_braces(1);} union_opt {}
;
2000-07-31 21:29:14 +02:00
values_list:
values_list ',' no_braces
| no_braces;
2000-07-31 21:29:14 +02:00
ident_eq_list:
ident_eq_list ',' ident_eq_value
|
ident_eq_value;
2000-07-31 21:29:14 +02:00
ident_eq_value:
2002-07-25 01:00:56 +03:00
simple_ident equal expr_or_default
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
if (lex->field_list.push_back($1) ||
lex->insert_list->push_back($3))
2000-07-31 21:29:14 +02:00
YYABORT;
};
2000-07-31 21:29:14 +02:00
equal: EQ {}
| SET_VAR {}
;
opt_equal:
/* empty */ {}
| equal {}
;
2000-07-31 21:29:14 +02:00
no_braces:
'('
{
if (!(Lex->insert_list = new List_item))
YYABORT;
}
opt_values ')'
{
LEX *lex=Lex;
if (lex->many_values.push_back(lex->insert_list))
2000-07-31 21:29:14 +02:00
YYABORT;
};
2000-07-31 21:29:14 +02:00
opt_values:
/* empty */ {}
| values;
2000-07-31 21:29:14 +02:00
values:
2002-07-25 01:00:56 +03:00
values ',' expr_or_default
2000-07-31 21:29:14 +02:00
{
if (Lex->insert_list->push_back($3))
YYABORT;
}
2002-07-25 01:00:56 +03:00
| expr_or_default
{
if (Lex->insert_list->push_back($1))
YYABORT;
}
;
expr_or_default:
expr { $$= $1;}
2003-01-21 20:20:46 +04:00
| DEFAULT {$$= new Item_default_value(); }
2002-07-25 01:00:56 +03:00
;
2000-07-31 21:29:14 +02:00
opt_insert_update:
/* empty */
| ON DUPLICATE_SYM
{ /* for simplisity, let's forget about
INSERT ... SELECT ... UPDATE
for a moment */
if (Lex->sql_command != SQLCOM_INSERT)
{
send_error(Lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
}
KEY_SYM UPDATE_SYM update_list
;
2000-07-31 21:29:14 +02:00
/* Update rows in a table */
update:
UPDATE_SYM
{
LEX *lex= Lex;
mysql_init_select(lex);
lex->sql_command= SQLCOM_UPDATE;
}
opt_low_priority opt_ignore join_table_list
SET update_list where_clause opt_order_clause delete_limit_clause
{
LEX *lex= Lex;
2002-11-21 22:25:53 +02:00
Select->set_lock_for_tables($3);
if (lex->select_lex.table_list.elements > 1)
lex->sql_command= SQLCOM_UPDATE_MULTI;
}
;
2000-07-31 21:29:14 +02:00
update_list:
2003-01-05 14:07:24 +04:00
update_list ',' simple_ident equal expr_or_default
2000-07-31 21:29:14 +02:00
{
if (add_item_to_list(YYTHD, $3) || add_value_to_list(YYTHD, $5))
2000-07-31 21:29:14 +02:00
YYABORT;
}
2003-01-05 14:07:24 +04:00
| simple_ident equal expr_or_default
2000-07-31 21:29:14 +02:00
{
if (add_item_to_list(YYTHD, $1) || add_value_to_list(YYTHD, $3))
2000-07-31 21:29:14 +02:00
YYABORT;
};
2000-07-31 21:29:14 +02:00
opt_low_priority:
/* empty */ { $$= YYTHD->update_lock_default; }
| LOW_PRIORITY { $$= TL_WRITE_LOW_PRIORITY; };
2000-07-31 21:29:14 +02:00
/* Delete rows from a table */
delete:
DELETE_SYM
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_DELETE;
lex->lock_option= lex->thd->update_lock_default;
2002-11-05 01:10:05 +02:00
lex->select_lex.init_order();
}
opt_delete_options single_multi {}
;
single_multi:
FROM table_ident
{
if (!Select->add_table_to_list(YYTHD, $2, NULL, TL_OPTION_UPDATING,
Lex->lock_option))
YYABORT;
}
where_clause opt_order_clause
delete_limit_clause {}
| table_wild_list
{ mysql_init_multi_delete(Lex); }
FROM join_table_list where_clause
| FROM table_wild_list
{ mysql_init_multi_delete(Lex); }
USING join_table_list where_clause
{}
;
table_wild_list:
table_wild_one {}
| table_wild_list ',' table_wild_one {};
table_wild_one:
ident opt_wild opt_table_alias
2002-11-21 15:56:48 +02:00
{
if (!Select->add_table_to_list(YYTHD, new Table_ident($1), $3,
TL_OPTION_UPDATING, Lex->lock_option))
2002-11-21 15:56:48 +02:00
YYABORT;
}
| ident '.' ident opt_wild opt_table_alias
2002-11-21 15:56:48 +02:00
{
if (!Select->add_table_to_list(YYTHD,
new Table_ident(YYTHD, $1, $3, 0),
$5, TL_OPTION_UPDATING,
Lex->lock_option))
YYABORT;
2002-11-21 15:56:48 +02:00
}
;
opt_wild:
/* empty */ {}
| '.' '*' {};
2000-07-31 21:29:14 +02:00
opt_delete_options:
/* empty */ {}
| opt_delete_option opt_delete_options {};
opt_delete_option:
QUICK { Select->options|= OPTION_QUICK; }
| LOW_PRIORITY { Lex->lock_option= TL_WRITE_LOW_PRIORITY; }
| IGNORE_SYM { Lex->duplicates= DUP_IGNORE; };
truncate:
TRUNCATE_SYM opt_table_sym table_name
{
LEX* lex= Lex;
lex->sql_command= SQLCOM_TRUNCATE;
lex->select_lex.options= 0;
2002-11-05 01:10:05 +02:00
lex->select_lex.init_order();
}
;
opt_table_sym:
/* empty */
| TABLE_SYM;
2000-07-31 21:29:14 +02:00
/* Show things */
show: SHOW
{
LEX *lex=Lex;
lex->wild=0;
bzero((char*) &lex->create_info,sizeof(lex->create_info));
}
2002-12-05 19:38:42 +02:00
show_param
{}
;
2000-07-31 21:29:14 +02:00
show_param:
DATABASES wild
{ Lex->sql_command= SQLCOM_SHOW_DATABASES; }
| TABLES opt_db wild
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_SHOW_TABLES;
lex->select_lex.db= $2;
lex->select_lex.options= 0;
}
2000-07-31 21:29:14 +02:00
| TABLE_SYM STATUS_SYM opt_db wild
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_SHOW_TABLES;
lex->select_lex.options|= SELECT_DESCRIBE;
lex->select_lex.db= $3;
2000-07-31 21:29:14 +02:00
}
| OPEN_SYM TABLES opt_db wild
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_SHOW_OPEN_TABLES;
lex->select_lex.db= $3;
lex->select_lex.options= 0;
}
| ENGINE_SYM storage_engines
{ Lex->create_info.db_type= $2; }
show_engine_param
| opt_full COLUMNS from_or_in table_ident opt_db wild
2000-07-31 21:29:14 +02:00
{
Lex->sql_command= SQLCOM_SHOW_FIELDS;
if ($5)
$4->change_db($5);
if (!Select->add_table_to_list(YYTHD, $4, NULL, 0))
2000-07-31 21:29:14 +02:00
YYABORT;
}
| NEW_SYM MASTER_SYM FOR_SYM SLAVE WITH MASTER_LOG_FILE_SYM EQ
2003-03-17 21:56:34 +04:00
TEXT_STRING_sys AND MASTER_LOG_POS_SYM EQ ulonglong_num
AND MASTER_SERVER_ID_SYM EQ
2001-07-04 17:14:31 -06:00
ULONG_NUM
{
Lex->sql_command = SQLCOM_SHOW_NEW_MASTER;
Lex->mi.log_file_name = $8.str;
Lex->mi.pos = $12;
Lex->mi.server_id = $16;
2001-07-04 17:14:31 -06:00
}
| master_or_binary LOGS_SYM
{
Lex->sql_command = SQLCOM_SHOW_BINLOGS;
}
| SLAVE HOSTS_SYM
{
Lex->sql_command = SQLCOM_SHOW_SLAVE_HOSTS;
}
| BINLOG_SYM EVENTS_SYM binlog_in binlog_from
{
LEX *lex= Lex;
lex->sql_command= SQLCOM_SHOW_BINLOG_EVENTS;
} opt_limit_clause_init
2000-07-31 21:29:14 +02:00
| keys_or_index FROM table_ident opt_db
{
Lex->sql_command= SQLCOM_SHOW_KEYS;
if ($4)
$3->change_db($4);
if (!Select->add_table_to_list(YYTHD, $3, NULL, 0))
2000-07-31 21:29:14 +02:00
YYABORT;
}
| COLUMN_SYM TYPES_SYM
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_SHOW_COLUMN_TYPES;
}
| TABLE_SYM TYPES_SYM
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_SHOW_STORAGE_ENGINES;
WARN_DEPRECATED("SHOW TABLE TYPES", "SHOW [STORAGE] ENGINES");
}
| opt_storage ENGINES_SYM
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_SHOW_STORAGE_ENGINES;
}
| PRIVILEGES
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_SHOW_PRIVILEGES;
}
| COUNT_SYM '(' '*' ')' WARNINGS
{ (void) create_select_for_variable("warning_count"); }
| COUNT_SYM '(' '*' ')' ERRORS
{ (void) create_select_for_variable("error_count"); }
| WARNINGS opt_limit_clause_init
{ Lex->sql_command = SQLCOM_SHOW_WARNS;}
| ERRORS opt_limit_clause_init
{ Lex->sql_command = SQLCOM_SHOW_ERRORS;}
2000-07-31 21:29:14 +02:00
| STATUS_SYM wild
{ Lex->sql_command= SQLCOM_SHOW_STATUS; }
2002-07-08 19:34:49 +03:00
| INNOBASE_SYM STATUS_SYM
{ Lex->sql_command = SQLCOM_SHOW_INNODB_STATUS; WARN_DEPRECATED("SHOW INNODB STATUS", "SHOW ENGINE INNODB STATUS"); }
| opt_full PROCESSLIST_SYM
{ Lex->sql_command= SQLCOM_SHOW_PROCESSLIST;}
| opt_var_type VARIABLES wild
{
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
thd->lex->sql_command= SQLCOM_SHOW_VARIABLES;
thd->lex->option_type= (enum_var_type) $1;
}
| charset wild
{ Lex->sql_command= SQLCOM_SHOW_CHARSETS; }
| COLLATION_SYM wild
{ Lex->sql_command= SQLCOM_SHOW_COLLATIONS; }
| BERKELEY_DB_SYM LOGS_SYM
{ Lex->sql_command= SQLCOM_SHOW_LOGS; WARN_DEPRECATED("SHOW BDB LOGS", "SHOW ENGINE BDB LOGS"); }
2000-12-15 13:18:52 +02:00
| LOGS_SYM
{ Lex->sql_command= SQLCOM_SHOW_LOGS; WARN_DEPRECATED("SHOW LOGS", "SHOW ENGINE BDB LOGS"); }
2000-07-31 21:29:14 +02:00
| GRANTS FOR_SYM user
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_SHOW_GRANTS;
lex->grant_user=$3;
lex->grant_user->password.str=NullS;
}
| CREATE DATABASE opt_if_not_exists ident
{
Lex->sql_command=SQLCOM_SHOW_CREATE_DB;
Lex->create_info.options=$3;
Lex->name=$4.str;
}
2000-07-31 21:29:14 +02:00
| CREATE TABLE_SYM table_ident
{
Lex->sql_command = SQLCOM_SHOW_CREATE;
if (!Select->add_table_to_list(YYTHD, $3, NULL,0))
2000-07-31 21:29:14 +02:00
YYABORT;
}
| MASTER_SYM STATUS_SYM
{
Lex->sql_command = SQLCOM_SHOW_MASTER_STAT;
2000-08-22 00:39:08 +03:00
}
2000-07-31 21:29:14 +02:00
| SLAVE STATUS_SYM
{
Lex->sql_command = SQLCOM_SHOW_SLAVE_STAT;
}
| CREATE PROCEDURE ident
{
Lex->sql_command = SQLCOM_SHOW_CREATE_PROC;
Lex->udf.name= $3;
}
| CREATE FUNCTION_SYM ident
{
Lex->sql_command = SQLCOM_SHOW_CREATE_FUNC;
Lex->udf.name= $3;
}
| PROCEDURE STATUS_SYM wild
{
Lex->sql_command = SQLCOM_SHOW_STATUS_PROC;
}
| FUNCTION_SYM STATUS_SYM wild
{
Lex->sql_command = SQLCOM_SHOW_STATUS_FUNC;
};
show_engine_param:
STATUS_SYM
{
switch (Lex->create_info.db_type) {
case DB_TYPE_INNODB:
Lex->sql_command = SQLCOM_SHOW_INNODB_STATUS;
break;
default:
net_printf(YYTHD, ER_NOT_SUPPORTED_YET, "STATUS");
YYABORT;
}
}
| LOGS_SYM
{
switch (Lex->create_info.db_type) {
case DB_TYPE_BERKELEY_DB:
Lex->sql_command = SQLCOM_SHOW_LOGS;
break;
default:
net_printf(YYTHD, ER_NOT_SUPPORTED_YET, "LOGS");
YYABORT;
}
};
2000-07-31 21:29:14 +02:00
master_or_binary:
MASTER_SYM
| BINARY;
opt_storage:
/* empty */
| STORAGE_SYM;
2000-07-31 21:29:14 +02:00
opt_db:
/* empty */ { $$= 0; }
| from_or_in ident { $$= $2.str; };
2000-07-31 21:29:14 +02:00
wild:
/* empty */
| LIKE text_string { Lex->wild= $2; };
2000-07-31 21:29:14 +02:00
opt_full:
/* empty */ { Lex->verbose=0; }
| FULL { Lex->verbose=1; };
from_or_in:
FROM
| IN_SYM;
binlog_in:
/* empty */ { Lex->mi.log_file_name = 0; }
2003-03-17 21:56:34 +04:00
| IN_SYM TEXT_STRING_sys { Lex->mi.log_file_name = $2.str; };
binlog_from:
/* empty */ { Lex->mi.pos = 4; /* skip magic number */ }
| FROM ulonglong_num { Lex->mi.pos = $2; };
2000-07-31 21:29:14 +02:00
/* A Oracle compatible synonym for show */
describe:
describe_command table_ident
{
LEX *lex=Lex;
lex->wild=0;
lex->verbose=0;
lex->sql_command=SQLCOM_SHOW_FIELDS;
if (!Select->add_table_to_list(lex->thd, $2, NULL,0))
2000-07-31 21:29:14 +02:00
YYABORT;
}
opt_describe_column {}
| describe_command opt_extended_describe
{ Lex->describe|= DESCRIBE_NORMAL; }
select
{
LEX *lex=Lex;
lex->select_lex.options|= SELECT_DESCRIBE;
2002-12-05 19:38:42 +02:00
}
;
2000-07-31 21:29:14 +02:00
describe_command:
DESC
| DESCRIBE;
2000-07-31 21:29:14 +02:00
opt_extended_describe:
/* empty */ {}
| EXTENDED_SYM { Lex->describe|= DESCRIBE_EXTENDED; }
;
2000-07-31 21:29:14 +02:00
opt_describe_column:
/* empty */ {}
| text_string { Lex->wild= $1; }
| ident
{ Lex->wild= new String((const char*) $1.str,$1.length,system_charset_info); };
2000-07-31 21:29:14 +02:00
/* flush things */
flush:
FLUSH_SYM opt_no_write_to_binlog
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_FLUSH; lex->type=0;
2003-08-11 22:44:43 +03:00
lex->no_write_to_binlog= $2;
}
flush_options
{}
;
2000-07-31 21:29:14 +02:00
flush_options:
flush_options ',' flush_option
| flush_option;
2000-07-31 21:29:14 +02:00
flush_option:
table_or_tables { Lex->type|= REFRESH_TABLES; } opt_table_list {}
2000-07-31 21:29:14 +02:00
| TABLES WITH READ_SYM LOCK_SYM { Lex->type|= REFRESH_TABLES | REFRESH_READ_LOCK; }
2001-12-02 14:34:01 +02:00
| QUERY_SYM CACHE_SYM { Lex->type|= REFRESH_QUERY_CACHE_FREE; }
2000-07-31 21:29:14 +02:00
| HOSTS_SYM { Lex->type|= REFRESH_HOSTS; }
| PRIVILEGES { Lex->type|= REFRESH_GRANT; }
| LOGS_SYM { Lex->type|= REFRESH_LOG; }
| STATUS_SYM { Lex->type|= REFRESH_STATUS; }
2000-08-22 00:39:08 +03:00
| SLAVE { Lex->type|= REFRESH_SLAVE; }
| MASTER_SYM { Lex->type|= REFRESH_MASTER; }
| DES_KEY_FILE { Lex->type|= REFRESH_DES_KEY_FILE; }
| RESOURCES { Lex->type|= REFRESH_USER_RESOURCES; };
2000-07-31 21:29:14 +02:00
opt_table_list:
/* empty */ {;}
| table_list {;};
2000-07-31 21:29:14 +02:00
2000-10-14 11:16:17 +03:00
reset:
RESET_SYM
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_RESET; lex->type=0;
} reset_options
{}
;
2000-10-14 11:16:17 +03:00
reset_options:
reset_options ',' reset_option
| reset_option;
2000-10-14 11:16:17 +03:00
reset_option:
2001-12-02 14:34:01 +02:00
SLAVE { Lex->type|= REFRESH_SLAVE; }
| MASTER_SYM { Lex->type|= REFRESH_MASTER; }
| QUERY_SYM CACHE_SYM { Lex->type|= REFRESH_QUERY_CACHE;};
2000-10-14 11:16:17 +03:00
purge:
PURGE
{
LEX *lex=Lex;
lex->type=0;
} purge_options
{}
;
purge_options:
master_or_binary LOGS_SYM purge_option
;
purge_option:
2003-03-17 21:56:34 +04:00
TO_SYM TEXT_STRING_sys
{
Lex->sql_command = SQLCOM_PURGE;
Lex->to_log = $2.str;
}
| BEFORE_SYM expr
{
if ($2->check_cols(1) || $2->fix_fields(Lex->thd, 0, &$2))
{
net_printf(Lex->thd, ER_WRONG_ARGUMENTS, "PURGE LOGS BEFORE");
2003-08-11 22:44:43 +03:00
YYABORT;
}
Item *tmp= new Item_func_unix_timestamp($2);
Lex->sql_command = SQLCOM_PURGE_BEFORE;
Lex->purge_time= (ulong) tmp->val_int();
}
;
2000-07-31 21:29:14 +02:00
/* kill threads */
kill:
KILL_SYM kill_option expr
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
if ($3->fix_fields(lex->thd, 0, &$3) || $3->check_cols(1))
{
send_error(lex->thd, ER_SET_CONSTANTS_ONLY);
YYABORT;
}
lex->sql_command=SQLCOM_KILL;
lex->thread_id= (ulong) $3->val_int();
};
2000-07-31 21:29:14 +02:00
kill_option:
/* empty */ { Lex->type= 0; }
| CONNECTION_SYM { Lex->type= 0; }
| QUERY_SYM { Lex->type= ONLY_KILL_QUERY; };
2000-07-31 21:29:14 +02:00
/* change database */
use: USE_SYM ident
{
LEX *lex=Lex;
lex->sql_command=SQLCOM_CHANGE_DB;
lex->select_lex.db= $2.str;
};
2000-07-31 21:29:14 +02:00
/* import, export of files */
2003-03-17 21:56:34 +04:00
load: LOAD DATA_SYM load_data_lock opt_local INFILE TEXT_STRING_sys
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_LOAD;
lex->lock_option= $3;
lex->local_file= $4;
if (!(lex->exchange= new sql_exchange($6.str,0)))
2000-07-31 21:29:14 +02:00
YYABORT;
lex->field_list.empty();
2000-07-31 21:29:14 +02:00
}
opt_duplicate INTO TABLE_SYM table_ident opt_field_term opt_line_term
opt_ignore_lines opt_field_spec
{
if (!Select->add_table_to_list(YYTHD, $11, NULL, TL_OPTION_UPDATING))
2000-07-31 21:29:14 +02:00
YYABORT;
}
|
2000-08-22 00:39:08 +03:00
LOAD TABLE_SYM table_ident FROM MASTER_SYM
2000-07-31 21:29:14 +02:00
{
Lex->sql_command = SQLCOM_LOAD_MASTER_TABLE;
if (!Select->add_table_to_list(YYTHD, $3, NULL, TL_OPTION_UPDATING))
2000-07-31 21:29:14 +02:00
YYABORT;
2000-08-22 00:39:08 +03:00
}
|
LOAD DATA_SYM FROM MASTER_SYM
{
Lex->sql_command = SQLCOM_LOAD_MASTER_DATA;
};
2000-07-31 21:29:14 +02:00
opt_local:
/* empty */ { $$=0;}
| LOCAL_SYM { $$=1;};
2000-07-31 21:29:14 +02:00
2001-05-05 09:41:47 +03:00
load_data_lock:
/* empty */ { $$= YYTHD->update_lock_default; }
| CONCURRENT { $$= TL_WRITE_CONCURRENT_INSERT ; }
| LOW_PRIORITY { $$= TL_WRITE_LOW_PRIORITY; };
2001-05-05 09:41:47 +03:00
2000-07-31 21:29:14 +02:00
opt_duplicate:
/* empty */ { Lex->duplicates=DUP_ERROR; }
| REPLACE { Lex->duplicates=DUP_REPLACE; }
| IGNORE_SYM { Lex->duplicates=DUP_IGNORE; };
2000-07-31 21:29:14 +02:00
opt_field_term:
/* empty */
| COLUMNS field_term_list;
2000-07-31 21:29:14 +02:00
field_term_list:
field_term_list field_term
| field_term;
2000-07-31 21:29:14 +02:00
field_term:
TERMINATED BY text_string { Lex->exchange->field_term= $3;}
| OPTIONALLY ENCLOSED BY text_string
{
LEX *lex=Lex;
lex->exchange->enclosed= $4;
lex->exchange->opt_enclosed=1;
}
2000-07-31 21:29:14 +02:00
| ENCLOSED BY text_string { Lex->exchange->enclosed= $3;}
| ESCAPED BY text_string { Lex->exchange->escaped= $3;};
2000-07-31 21:29:14 +02:00
opt_line_term:
/* empty */
| LINES line_term_list;
2000-07-31 21:29:14 +02:00
line_term_list:
line_term_list line_term
| line_term;
2000-07-31 21:29:14 +02:00
line_term:
TERMINATED BY text_string { Lex->exchange->line_term= $3;}
| STARTING BY text_string { Lex->exchange->line_start= $3;};
2000-07-31 21:29:14 +02:00
opt_ignore_lines:
/* empty */
| IGNORE_SYM NUM LINES
{ Lex->exchange->skip_lines=atol($2.str); };
2000-07-31 21:29:14 +02:00
/* Common definitions */
text_literal:
TEXT_STRING_literal
2003-03-17 21:56:34 +04:00
{
THD *thd= YYTHD;
$$ = new Item_string($1.str,$1.length,thd->variables.collation_connection);
2003-03-17 21:56:34 +04:00
}
2003-03-20 22:01:03 +04:00
| NCHAR_STRING
{ $$= new Item_string($1.str,$1.length,national_charset_info); }
| UNDERSCORE_CHARSET TEXT_STRING
{ $$ = new Item_string($2.str,$2.length,Lex->charset); }
| text_literal TEXT_STRING_literal
2003-03-20 22:01:03 +04:00
{ ((Item_string*) $1)->append($2.str,$2.length); }
;
2000-07-31 21:29:14 +02:00
text_string:
TEXT_STRING_literal
{ $$= new String($1.str,$1.length,YYTHD->variables.collation_connection); }
2000-07-31 21:29:14 +02:00
| HEX_NUM
{
Item *tmp = new Item_varbinary($1.str,$1.length);
2000-07-31 21:29:14 +02:00
$$= tmp ? tmp->val_str((String*) 0) : (String*) 0;
}
;
param_marker:
'?'
{
LEX *lex=Lex;
if (YYTHD->command == COM_PREPARE)
{
lex->param_list.push_back($$=new Item_param((uint)(lex->tok_start-(uchar *)YYTHD->query)));
lex->param_count++;
}
else
{
yyerror("You have an error in your SQL syntax");
YYABORT;
}
}
;
2003-12-11 16:05:51 +00:00
signed_literal:
literal { $$ = $1; }
| '+' NUM_literal { $$ = $2; }
| '-' NUM_literal { $$ = new Item_func_neg($2); }
;
2000-07-31 21:29:14 +02:00
literal:
text_literal { $$ = $1; }
2003-12-11 16:05:51 +00:00
| NUM_literal { $$ = $1; }
2000-07-31 21:29:14 +02:00
| NULL_SYM { $$ = new Item_null();
Lex->next_state=MY_LEX_OPERATOR_OR_IDENT;}
| HEX_NUM { $$ = new Item_varbinary($1.str,$1.length);}
| UNDERSCORE_CHARSET HEX_NUM
{
Item *tmp= new Item_varbinary($2.str,$2.length);
String *str= tmp ? tmp->val_str((String*) 0) : (String*) 0;
$$= new Item_string(str ? str->ptr() : "",
str ? str->length() : 0,
Lex->charset);
}
2000-07-31 21:29:14 +02:00
| DATE_SYM text_literal { $$ = $2; }
| TIME_SYM text_literal { $$ = $2; }
| TIMESTAMP text_literal { $$ = $2; };
2000-07-31 21:29:14 +02:00
NUM_literal:
NUM { $$ = new Item_int($1.str, (longlong) strtol($1.str, NULL, 10),$1.length); }
| LONG_NUM { $$ = new Item_int($1.str, (longlong) strtoll($1.str,NULL,10), $1.length); }
| ULONGLONG_NUM { $$ = new Item_uint($1.str, $1.length); }
| REAL_NUM { $$ = new Item_real($1.str, $1.length); }
| FLOAT_NUM { $$ = new Item_float($1.str, $1.length); }
;
2000-07-31 21:29:14 +02:00
/**********************************************************************
** Createing different items.
**********************************************************************/
insert_ident:
simple_ident { $$=$1; }
| table_wild { $$=$1; };
2000-07-31 21:29:14 +02:00
table_wild:
2003-08-11 22:44:43 +03:00
ident '.' '*'
{
$$ = new Item_field(NullS,$1.str,"*");
Lex->current_select->with_wild++;
}
2000-07-31 21:29:14 +02:00
| ident '.' ident '.' '*'
{
$$ = new Item_field((YYTHD->client_capabilities &
CLIENT_NO_SCHEMA ? NullS : $1.str),
$3.str,"*");
Lex->current_select->with_wild++;
}
;
2000-07-31 21:29:14 +02:00
order_ident:
expr { $$=$1; };
2000-07-31 21:29:14 +02:00
simple_ident:
ident
{
sp_pvar_t *spv;
LEX *lex = Lex;
sp_pcontext *spc = lex->spcont;
if (spc && (spv = spc->find_pvar(&$1)))
{ /* We're compiling a stored procedure and found a variable */
if (lex->sql_command != SQLCOM_CALL && ! spv->isset)
{
push_warning_printf(YYTHD, MYSQL_ERROR::WARN_LEVEL_WARN,
ER_SP_UNINIT_VAR, ER(ER_SP_UNINIT_VAR),
$1.str);
}
$$ = (Item*) new Item_splocal($1, spv->offset);
}
else
{
2003-08-26 11:51:09 +02:00
SELECT_LEX *sel=Select;
2003-07-08 17:50:23 +02:00
$$= (sel->parsing_place != SELECT_LEX_NODE::IN_HAVING ||
sel->get_in_sum_expr() > 0) ?
(Item*) new Item_field(NullS,NullS,$1.str) :
(Item*) new Item_ref(NullS,NullS,$1.str);
}
}
2000-07-31 21:29:14 +02:00
| ident '.' ident
{
2003-01-14 18:00:34 +02:00
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
LEX *lex= thd->lex;
SELECT_LEX *sel= lex->current_select;
2003-01-14 18:00:34 +02:00
if (sel->no_table_names_allowed)
{
2003-08-11 22:44:43 +03:00
my_printf_error(ER_TABLENAME_NOT_ALLOWED_HERE,
2003-01-14 18:00:34 +02:00
ER(ER_TABLENAME_NOT_ALLOWED_HERE),
MYF(0), $1.str, thd->where);
}
2003-05-17 10:05:07 +03:00
$$= (sel->parsing_place != SELECT_LEX_NODE::IN_HAVING ||
sel->get_in_sum_expr() > 0) ?
(Item*) new Item_field(NullS,$1.str,$3.str) :
(Item*) new Item_ref(NullS,$1.str,$3.str);
}
2000-07-31 21:29:14 +02:00
| '.' ident '.' ident
{
2003-01-14 18:00:34 +02:00
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
LEX *lex= thd->lex;
SELECT_LEX *sel= lex->current_select;
2003-01-14 18:00:34 +02:00
if (sel->no_table_names_allowed)
{
2003-08-11 22:44:43 +03:00
my_printf_error(ER_TABLENAME_NOT_ALLOWED_HERE,
2003-01-14 18:00:34 +02:00
ER(ER_TABLENAME_NOT_ALLOWED_HERE),
MYF(0), $2.str, thd->where);
}
2003-05-17 10:05:07 +03:00
$$= (sel->parsing_place != SELECT_LEX_NODE::IN_HAVING ||
sel->get_in_sum_expr() > 0) ?
(Item*) new Item_field(NullS,$2.str,$4.str) :
(Item*) new Item_ref(NullS,$2.str,$4.str);
}
2000-07-31 21:29:14 +02:00
| ident '.' ident '.' ident
{
2003-01-14 18:00:34 +02:00
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
LEX *lex= thd->lex;
SELECT_LEX *sel= lex->current_select;
2003-01-14 18:00:34 +02:00
if (sel->no_table_names_allowed)
{
2003-08-11 22:44:43 +03:00
my_printf_error(ER_TABLENAME_NOT_ALLOWED_HERE,
2003-01-14 18:00:34 +02:00
ER(ER_TABLENAME_NOT_ALLOWED_HERE),
MYF(0), $3.str, thd->where);
}
2003-05-17 10:05:07 +03:00
$$= (sel->parsing_place != SELECT_LEX_NODE::IN_HAVING ||
sel->get_in_sum_expr() > 0) ?
(Item*) new Item_field((YYTHD->client_capabilities &
CLIENT_NO_SCHEMA ? NullS : $1.str),
$3.str, $5.str) :
(Item*) new Item_ref((YYTHD->client_capabilities &
CLIENT_NO_SCHEMA ? NullS : $1.str),
$3.str, $5.str);
};
2000-07-31 21:29:14 +02:00
field_ident:
ident { $$=$1;}
| ident '.' ident { $$=$3;} /* Skipp schema name in create*/
| '.' ident { $$=$2;} /* For Delphi */;
2000-07-31 21:29:14 +02:00
table_ident:
ident { $$=new Table_ident($1); }
| ident '.' ident { $$=new Table_ident(YYTHD, $1,$3,0);}
| '.' ident { $$=new Table_ident($2);}
/* For Delphi */;
2000-07-31 21:29:14 +02:00
2003-03-17 21:56:34 +04:00
IDENT_sys:
IDENT { $$= $1; }
| IDENT_QUOTED
{
THD *thd= YYTHD;
if (thd->charset_is_system_charset)
$$= $1;
else
thd->convert_string(&$$, system_charset_info,
$1.str, $1.length, thd->charset());
}
2003-03-17 21:56:34 +04:00
;
TEXT_STRING_sys:
TEXT_STRING
{
THD *thd= YYTHD;
if (thd->charset_is_system_charset)
$$= $1;
2003-03-17 21:56:34 +04:00
else
thd->convert_string(&$$, system_charset_info,
$1.str, $1.length, thd->charset());
2003-03-17 21:56:34 +04:00
}
;
TEXT_STRING_literal:
2003-03-17 21:56:34 +04:00
TEXT_STRING
{
THD *thd= YYTHD;
if (thd->charset_is_collation_connection)
$$= $1;
2003-03-17 21:56:34 +04:00
else
thd->convert_string(&$$, thd->variables.collation_connection,
$1.str, $1.length, thd->charset());
2003-03-17 21:56:34 +04:00
}
;
2000-07-31 21:29:14 +02:00
ident:
2003-03-17 21:56:34 +04:00
IDENT_sys { $$=$1; }
2000-07-31 21:29:14 +02:00
| keyword
{
THD *thd= YYTHD;
$$.str= thd->strmake($1.str, $1.length);
$$.length= $1.length;
}
;
2000-07-31 21:29:14 +02:00
ident_or_text:
2003-03-17 21:56:34 +04:00
ident { $$=$1;}
| TEXT_STRING_sys { $$=$1;}
| LEX_HOSTNAME { $$=$1;};
2000-07-31 21:29:14 +02:00
user:
ident_or_text
{
THD *thd= YYTHD;
if (!($$=(LEX_USER*) thd->alloc(sizeof(st_lex_user))))
2000-07-31 21:29:14 +02:00
YYABORT;
$$->user = $1;
$$->host.str= (char *) "%";
$$->host.length= 1;
}
2000-07-31 21:29:14 +02:00
| ident_or_text '@' ident_or_text
{
THD *thd= YYTHD;
if (!($$=(LEX_USER*) thd->alloc(sizeof(st_lex_user))))
2000-07-31 21:29:14 +02:00
YYABORT;
$$->user = $1; $$->host=$3;
};
2000-07-31 21:29:14 +02:00
/* Keyword that we allow for identifiers */
keyword:
ACTION {}
2003-06-23 12:56:44 +05:00
| ADDDATE_SYM {}
2000-07-31 21:29:14 +02:00
| AFTER_SYM {}
| AGAINST {}
2000-07-31 21:29:14 +02:00
| AGGREGATE_SYM {}
| ANY_SYM {}
| ASCII_SYM {}
| AUTO_INC {}
2000-07-31 21:29:14 +02:00
| AVG_ROW_LENGTH {}
| AVG_SYM {}
| BACKUP_SYM {}
2000-07-31 21:29:14 +02:00
| BEGIN_SYM {}
2000-10-20 17:39:23 +03:00
| BERKELEY_DB_SYM {}
2001-10-24 11:52:19 -06:00
| BINLOG_SYM {}
2000-07-31 21:29:14 +02:00
| BIT_SYM {}
| BOOL_SYM {}
2001-10-09 14:53:54 +02:00
| BOOLEAN_SYM {}
| BYTE_SYM {}
| BTREE_SYM {}
2001-12-02 14:34:01 +02:00
| CACHE_SYM {}
| CHANGED {}
| CHARSET {}
2000-07-31 21:29:14 +02:00
| CHECKSUM_SYM {}
| CIPHER_SYM {}
| CLIENT_SYM {}
| CLOSE_SYM {}
| COLLATION_SYM {}
2000-07-31 21:29:14 +02:00
| COMMENT_SYM {}
| COMMITTED_SYM {}
| COMMIT_SYM {}
2000-07-31 21:29:14 +02:00
| COMPRESSED_SYM {}
2001-05-05 09:41:47 +03:00
| CONCURRENT {}
| CUBE_SYM {}
2000-07-31 21:29:14 +02:00
| DATA_SYM {}
| DATETIME {}
| DATE_SYM {}
| DAY_SYM {}
| DEFINER_SYM {}
2000-07-31 21:29:14 +02:00
| DELAY_KEY_WRITE_SYM {}
| DES_KEY_FILE {}
| DIRECTORY_SYM {}
| DISCARD {}
2001-12-17 19:59:20 +02:00
| DO_SYM {}
| DUAL_SYM {}
| DUMPFILE {}
| DUPLICATE_SYM {}
2000-07-31 21:29:14 +02:00
| DYNAMIC_SYM {}
| END {}
| ENUM {}
| ENGINE_SYM {}
| ENGINES_SYM {}
| ERRORS {}
2000-07-31 21:29:14 +02:00
| ESCAPE_SYM {}
2001-10-24 11:52:19 -06:00
| EVENTS_SYM {}
| EXECUTE_SYM {}
2003-10-23 15:21:06 +02:00
| EXPANSION_SYM {}
2000-07-31 21:29:14 +02:00
| EXTENDED_SYM {}
| FAST_SYM {}
| DISABLE_SYM {}
| ENABLE_SYM {}
2000-09-07 04:55:17 +03:00
| FULL {}
2000-07-31 21:29:14 +02:00
| FILE_SYM {}
| FIRST_SYM {}
| FIXED_SYM {}
| FLUSH_SYM {}
| GEOMETRY_SYM {}
| GEOMETRYCOLLECTION {}
| GET_FORMAT {}
| GRANTS {}
| GLOBAL_SYM {}
| HANDLER_SYM {}
| HASH_SYM {}
| HELP_SYM {}
2000-07-31 21:29:14 +02:00
| HOSTS_SYM {}
| HOUR_SYM {}
| IDENTIFIED_SYM {}
| INVOKER_SYM {}
| IMPORT {}
| INDEXES {}
| ISOLATION {}
2001-09-19 19:45:13 -06:00
| ISSUER_SYM {}
2000-10-20 17:39:23 +03:00
| INNOBASE_SYM {}
| INSERT_METHOD {}
| RELAY_THREAD {}
| LANGUAGE_SYM {}
| LAST_SYM {}
| LEAVES {}
| LEVEL_SYM {}
| LINESTRING {}
2000-07-31 21:29:14 +02:00
| LOCAL_SYM {}
| LOCKS_SYM {}
2000-07-31 21:29:14 +02:00
| LOGS_SYM {}
| MAX_ROWS {}
| MASTER_SYM {}
| MASTER_HOST_SYM {}
| MASTER_PORT_SYM {}
| MASTER_LOG_FILE_SYM {}
| MASTER_LOG_POS_SYM {}
| MASTER_USER_SYM {}
| MASTER_PASSWORD_SYM {}
| MASTER_CONNECT_RETRY_SYM {}
| MASTER_SSL_SYM {}
| MASTER_SSL_CA_SYM {}
| MASTER_SSL_CAPATH_SYM {}
| MASTER_SSL_CERT_SYM {}
| MASTER_SSL_CIPHER_SYM {}
| MASTER_SSL_KEY_SYM {}
| MAX_CONNECTIONS_PER_HOUR {}
| MAX_QUERIES_PER_HOUR {}
| MAX_UPDATES_PER_HOUR {}
| MEDIUM_SYM {}
2003-07-08 15:06:05 +05:00
| MICROSECOND_SYM {}
2000-07-31 21:29:14 +02:00
| MINUTE_SYM {}
| MIN_ROWS {}
| MODIFY_SYM {}
| MODE_SYM {}
2000-07-31 21:29:14 +02:00
| MONTH_SYM {}
| MULTILINESTRING {}
| MULTIPOINT {}
| MULTIPOLYGON {}
| NAME_SYM {}
| NAMES_SYM {}
2000-07-31 21:29:14 +02:00
| NATIONAL_SYM {}
| NCHAR_SYM {}
| NEXT_SYM {}
2001-09-19 19:45:13 -06:00
| NEW_SYM {}
2000-07-31 21:29:14 +02:00
| NO_SYM {}
| NONE_SYM {}
| NVARCHAR_SYM {}
| OFFSET_SYM {}
| OLD_PASSWORD {}
| OPEN_SYM {}
2000-07-31 21:29:14 +02:00
| PACK_KEYS_SYM {}
| PARTIAL {}
2000-07-31 21:29:14 +02:00
| PASSWORD {}
| POINT_SYM {}
| POLYGON {}
| PREV_SYM {}
2000-07-31 21:29:14 +02:00
| PROCESS {}
| PROCESSLIST_SYM {}
| QUARTER_SYM {}
2001-12-02 14:34:01 +02:00
| QUERY_SYM {}
| QUICK {}
| RAID_0_SYM {}
2000-08-22 00:39:08 +03:00
| RAID_CHUNKS {}
| RAID_CHUNKSIZE {}
| RAID_STRIPED_SYM {}
2000-08-22 00:39:08 +03:00
| RAID_TYPE {}
| RELAY_LOG_FILE_SYM {}
| RELAY_LOG_POS_SYM {}
2000-07-31 21:29:14 +02:00
| RELOAD {}
| REPAIR {}
| REPEATABLE_SYM {}
| REPLICATION {}
2000-10-14 11:16:17 +03:00
| RESET_SYM {}
| RESOURCES {}
| RESTORE_SYM {}
| RETURNS_SYM {}
2000-07-31 21:29:14 +02:00
| ROLLBACK_SYM {}
| ROLLUP_SYM {}
2000-07-31 21:29:14 +02:00
| ROWS_SYM {}
| ROW_FORMAT_SYM {}
| ROW_SYM {}
| RTREE_SYM {}
| SAVEPOINT_SYM {}
2000-07-31 21:29:14 +02:00
| SECOND_SYM {}
| SECURITY_SYM {}
| SERIAL_SYM {}
| SERIALIZABLE_SYM {}
| SESSION_SYM {}
| SIGNED_SYM {}
| SIMPLE_SYM {}
| SHARE_SYM {}
2000-07-31 21:29:14 +02:00
| SHUTDOWN {}
| SLAVE {}
2003-06-04 19:21:51 +03:00
| SOUNDS_SYM {}
2001-12-02 14:34:01 +02:00
| SQL_CACHE_SYM {}
| SQL_BUFFER_RESULT {}
2001-12-02 14:34:01 +02:00
| SQL_NO_CACHE_SYM {}
| SQL_THREAD {}
2000-07-31 21:29:14 +02:00
| START_SYM {}
| STATUS_SYM {}
| STOP_SYM {}
| STORAGE_SYM {}
2000-07-31 21:29:14 +02:00
| STRING_SYM {}
2003-06-23 12:56:44 +05:00
| SUBDATE_SYM {}
2001-09-19 19:45:13 -06:00
| SUBJECT_SYM {}
| SUPER_SYM {}
| TABLESPACE {}
2000-07-31 21:29:14 +02:00
| TEMPORARY {}
| TEXT_SYM {}
| TRANSACTION_SYM {}
| TRUNCATE_SYM {}
2000-07-31 21:29:14 +02:00
| TIMESTAMP {}
| TIMESTAMP_ADD {}
| TIMESTAMP_DIFF {}
2000-07-31 21:29:14 +02:00
| TIME_SYM {}
| TYPE_SYM {}
| TYPES_SYM {}
| FUNCTION_SYM {}
| UNCOMMITTED_SYM {}
| UNICODE_SYM {}
| UNTIL_SYM {}
| USER {}
| USE_FRM {}
2000-07-31 21:29:14 +02:00
| VARIABLES {}
| VALUE_SYM {}
| WARNINGS {}
| WEEK_SYM {}
2000-07-31 21:29:14 +02:00
| WORK_SYM {}
| X509_SYM {}
| YEAR_SYM {}
;
2000-07-31 21:29:14 +02:00
/* Option functions */
set:
SET opt_option
{
LEX *lex=Lex;
lex->sql_command= SQLCOM_SET_OPTION;
lex->option_type=OPT_DEFAULT;
lex->var_list.empty();
2000-07-31 21:29:14 +02:00
}
option_value_list
{}
;
2000-07-31 21:29:14 +02:00
opt_option:
/* empty */ {}
| OPTION {};
2000-07-31 21:29:14 +02:00
option_value_list:
option_type option_value
| option_value_list ',' option_type option_value;
option_type:
/* empty */ {}
| GLOBAL_SYM { Lex->option_type= OPT_GLOBAL; }
| LOCAL_SYM { Lex->option_type= OPT_SESSION; }
| SESSION_SYM { Lex->option_type= OPT_SESSION; }
;
opt_var_type:
/* empty */ { $$=OPT_SESSION; }
| GLOBAL_SYM { $$=OPT_GLOBAL; }
| LOCAL_SYM { $$=OPT_SESSION; }
| SESSION_SYM { $$=OPT_SESSION; }
;
opt_var_ident_type:
/* empty */ { $$=OPT_DEFAULT; }
| GLOBAL_SYM '.' { $$=OPT_GLOBAL; }
| LOCAL_SYM '.' { $$=OPT_SESSION; }
| SESSION_SYM '.' { $$=OPT_SESSION; }
;
2000-07-31 21:29:14 +02:00
option_value:
'@' ident_or_text equal expr
2000-07-31 21:29:14 +02:00
{
2003-08-26 17:41:40 +02:00
Lex->var_list.push_back(new set_var_user(new Item_func_set_user_var($2,$4)));
}
2002-07-25 01:00:56 +03:00
| internal_variable_name equal set_expr_or_default
2000-07-31 21:29:14 +02:00
{
LEX *lex=Lex;
2003-08-26 17:41:40 +02:00
if ($1.var)
{ /* System variable */
lex->var_list.push_back(new set_var(lex->option_type, $1.var,
&$1.base_name, $3));
}
else
{ /* An SP local variable */
if ($3 && $3->type() == Item::SUBSELECT_ITEM)
{ /* QQ For now, just disallow subselects as values */
send_error(lex->thd, ER_SP_SUBSELECT_NYI);
YYABORT;
}
2003-08-26 17:41:40 +02:00
sp_pvar_t *spv= lex->spcont->find_pvar(&$1.base_name);
sp_instr_set *i= new sp_instr_set(lex->sphead->instructions(),
spv->offset, $3, spv->type);
lex->sphead->add_instr(i);
spv->isset= TRUE;
}
2000-07-31 21:29:14 +02:00
}
2002-07-25 01:00:56 +03:00
| '@' '@' opt_var_ident_type internal_variable_name equal set_expr_or_default
{
LEX *lex=Lex;
lex->var_list.push_back(new set_var((enum_var_type) $3, $4.var,
&$4.base_name, $6));
}
| TRANSACTION_SYM ISOLATION LEVEL_SYM isolation_types
{
LEX *lex=Lex;
LEX_STRING tmp;
tmp.str=0;
tmp.length=0;
lex->var_list.push_back(new set_var(lex->option_type,
find_sys_var("tx_isolation"),
&tmp,
2002-08-28 17:00:58 +03:00
new Item_int((int32) $4)));
}
| charset old_or_new_charset_name_or_default
2000-07-31 21:29:14 +02:00
{
2003-03-18 17:01:32 +04:00
THD *thd= YYTHD;
2003-04-05 18:56:15 +05:00
LEX *lex= Lex;
$2= $2 ? $2: global_system_variables.character_set_client;
lex->var_list.push_back(new set_var_collation_client($2,thd->variables.collation_database,$2));
}
| NAMES_SYM charset_name_or_default opt_collate
{
THD *thd= YYTHD;
2003-04-05 18:56:15 +05:00
LEX *lex= Lex;
$2= $2 ? $2 : global_system_variables.character_set_client;
$3= $3 ? $3 : $2;
if (!my_charset_same($2,$3))
{
net_printf(thd,ER_COLLATION_CHARSET_MISMATCH,$3->name,$2->csname);
YYABORT;
}
lex->var_list.push_back(new set_var_collation_client($3,$3,$3));
}
2000-07-31 21:29:14 +02:00
| PASSWORD equal text_or_password
{
THD *thd=YYTHD;
LEX_USER *user;
if (!(user=(LEX_USER*) thd->alloc(sizeof(LEX_USER))))
YYABORT;
user->host.str=0;
user->user.str=thd->priv_user;
2003-05-05 14:54:37 -04:00
thd->lex->var_list.push_back(new set_var_password(user, $3));
}
| PASSWORD FOR_SYM user equal text_or_password
{
Lex->var_list.push_back(new set_var_password($3,$5));
}
;
internal_variable_name:
ident
{
2003-08-26 17:41:40 +02:00
LEX *lex= Lex;
sp_pcontext *spc= lex->spcont;
sp_pvar_t *spv;
/* We have to lookup here since local vars can shadow sysvars */
if (!spc || !(spv = spc->find_pvar(&$1)))
{ /* Not an SP local variable */
sys_var *tmp=find_sys_var($1.str, $1.length);
if (!tmp)
YYABORT;
$$.var= tmp;
$$.base_name.str=0;
$$.base_name.length=0;
}
else
{ /* An SP local variable */
$$.var= NULL;
$$.base_name= $1;
}
}
| ident '.' ident
{
sys_var *tmp=find_sys_var($3.str, $3.length);
if (!tmp)
YYABORT;
if (!tmp->is_struct())
net_printf(YYTHD, ER_VARIABLE_IS_NOT_STRUCT, $3.str);
$$.var= tmp;
$$.base_name= $1;
}
| DEFAULT '.' ident
{
sys_var *tmp=find_sys_var($3.str, $3.length);
if (!tmp)
YYABORT;
if (!tmp->is_struct())
net_printf(YYTHD, ER_VARIABLE_IS_NOT_STRUCT, $3.str);
$$.var= tmp;
$$.base_name.str= (char*) "default";
$$.base_name.length= 7;
}
2002-10-16 16:55:08 +03:00
;
isolation_types:
READ_SYM UNCOMMITTED_SYM { $$= ISO_READ_UNCOMMITTED; }
| READ_SYM COMMITTED_SYM { $$= ISO_READ_COMMITTED; }
| REPEATABLE_SYM READ_SYM { $$= ISO_REPEATABLE_READ; }
| SERIALIZABLE_SYM { $$= ISO_SERIALIZABLE; }
;
2000-07-31 21:29:14 +02:00
text_or_password:
TEXT_STRING { $$=$1.str;}
| PASSWORD '(' TEXT_STRING ')'
{
$$= $3.length ? YYTHD->variables.old_passwords ?
Item_func_old_password::alloc(YYTHD, $3.str) :
Item_func_password::alloc(YYTHD, $3.str) :
$3.str;
}
| OLD_PASSWORD '(' TEXT_STRING ')'
{
$$= $3.length ? Item_func_old_password::alloc(YYTHD, $3.str) :
$3.str;
2002-10-16 16:55:08 +03:00
}
;
2000-07-31 21:29:14 +02:00
2002-07-25 01:00:56 +03:00
set_expr_or_default:
expr { $$=$1; }
| DEFAULT { $$=0; }
| ON { $$=new Item_string("ON", 2, system_charset_info); }
| ALL { $$=new Item_string("ALL", 3, system_charset_info); }
| BINARY { $$=new Item_string("binary", 6, system_charset_info); }
;
2000-07-31 21:29:14 +02:00
/* Lock function */
lock:
LOCK_SYM table_or_tables
{
Lex->sql_command=SQLCOM_LOCK_TABLES;
}
table_lock_list
{}
;
2000-07-31 21:29:14 +02:00
table_or_tables:
TABLE_SYM
| TABLES;
2000-07-31 21:29:14 +02:00
table_lock_list:
table_lock
| table_lock_list ',' table_lock;
2000-07-31 21:29:14 +02:00
table_lock:
table_ident opt_table_alias lock_option
{
if (!Select->add_table_to_list(YYTHD, $1, $2, 0, (thr_lock_type) $3))
YYABORT;
}
2002-10-16 16:55:08 +03:00
;
2000-07-31 21:29:14 +02:00
lock_option:
READ_SYM { $$=TL_READ_NO_INSERT; }
| WRITE_SYM { $$=YYTHD->update_lock_default; }
2000-07-31 21:29:14 +02:00
| LOW_PRIORITY WRITE_SYM { $$=TL_WRITE_LOW_PRIORITY; }
2002-10-16 16:55:08 +03:00
| READ_SYM LOCAL_SYM { $$= TL_READ; }
;
2000-07-31 21:29:14 +02:00
unlock:
2002-10-16 16:55:08 +03:00
UNLOCK_SYM table_or_tables { Lex->sql_command=SQLCOM_UNLOCK_TABLES; }
;
2000-07-31 21:29:14 +02:00
/*
2001-04-13 16:18:44 +02:00
** Handler: direct access to ISAM functions
*/
handler:
HANDLER_SYM table_ident OPEN_SYM opt_table_alias
{
LEX *lex= Lex;
lex->sql_command = SQLCOM_HA_OPEN;
if (!lex->current_select->add_table_to_list(lex->thd, $2, $4, 0))
YYABORT;
}
| HANDLER_SYM table_ident CLOSE_SYM
{
LEX *lex= Lex;
lex->sql_command = SQLCOM_HA_CLOSE;
if (!lex->current_select->add_table_to_list(lex->thd, $2, 0, 0))
YYABORT;
}
| HANDLER_SYM table_ident READ_SYM
{
2001-11-06 00:05:45 +02:00
LEX *lex=Lex;
lex->sql_command = SQLCOM_HA_READ;
lex->ha_rkey_mode= HA_READ_KEY_EXACT; /* Avoid purify warnings */
lex->current_select->select_limit= 1;
lex->current_select->offset_limit= 0L;
if (!lex->current_select->add_table_to_list(lex->thd, $2, 0, 0))
YYABORT;
2001-04-13 16:18:44 +02:00
}
handler_read_or_scan where_clause opt_limit_clause {}
2002-10-16 16:55:08 +03:00
;
2001-04-13 16:18:44 +02:00
handler_read_or_scan:
handler_scan_function { Lex->backup_dir= 0; }
2002-10-16 16:55:08 +03:00
| ident handler_rkey_function { Lex->backup_dir= $1.str; }
;
2001-04-13 16:18:44 +02:00
handler_scan_function:
FIRST_SYM { Lex->ha_read_mode = RFIRST; }
2002-10-16 16:55:08 +03:00
| NEXT_SYM { Lex->ha_read_mode = RNEXT; }
;
2001-04-13 16:18:44 +02:00
handler_rkey_function:
FIRST_SYM { Lex->ha_read_mode = RFIRST; }
| NEXT_SYM { Lex->ha_read_mode = RNEXT; }
| PREV_SYM { Lex->ha_read_mode = RPREV; }
| LAST_SYM { Lex->ha_read_mode = RLAST; }
| handler_rkey_mode
{
LEX *lex=Lex;
lex->ha_read_mode = RKEY;
2001-11-06 00:05:45 +02:00
lex->ha_rkey_mode=$1;
if (!(lex->insert_list = new List_item))
YYABORT;
2002-10-16 16:55:08 +03:00
} '(' values ')' { }
;
handler_rkey_mode:
2001-11-06 00:05:45 +02:00
EQ { $$=HA_READ_KEY_EXACT; }
| GE { $$=HA_READ_KEY_OR_NEXT; }
| LE { $$=HA_READ_KEY_OR_PREV; }
| GT_SYM { $$=HA_READ_AFTER_KEY; }
2002-10-16 16:55:08 +03:00
| LT { $$=HA_READ_BEFORE_KEY; }
;
2000-07-31 21:29:14 +02:00
/* GRANT / REVOKE */
revoke:
REVOKE
{
LEX *lex=Lex;
lex->sql_command = SQLCOM_REVOKE;
lex->users_list.empty();
lex->columns.empty();
lex->grant= lex->grant_tot_col=0;
lex->select_lex.db=0;
lex->ssl_type= SSL_TYPE_NOT_SPECIFIED;
lex->ssl_cipher= lex->x509_subject= lex->x509_issuer= 0;
bzero((char*) &lex->mqh, sizeof(lex->mqh));
2000-07-31 21:29:14 +02:00
}
revoke_command
{}
;
revoke_command:
grant_privileges ON opt_table FROM user_list
{}
|
ALL PRIVILEGES ',' GRANT FROM user_list
{
Lex->sql_command = SQLCOM_REVOKE_ALL;
}
;
2000-07-31 21:29:14 +02:00
grant:
GRANT
{
LEX *lex=Lex;
lex->users_list.empty();
lex->columns.empty();
lex->sql_command = SQLCOM_GRANT;
lex->grant= lex->grant_tot_col= 0;
lex->select_lex.db= 0;
lex->ssl_type= SSL_TYPE_NOT_SPECIFIED;
lex->ssl_cipher= lex->x509_subject= lex->x509_issuer= 0;
bzero((char *)&(lex->mqh),sizeof(lex->mqh));
2000-07-31 21:29:14 +02:00
}
grant_privileges ON opt_table TO_SYM user_list
require_clause grant_options
{}
;
2000-07-31 21:29:14 +02:00
grant_privileges:
grant_privilege_list {}
| ALL PRIVILEGES { Lex->grant = GLOBAL_ACLS;}
2002-10-16 16:55:08 +03:00
| ALL { Lex->grant = GLOBAL_ACLS;}
;
2000-07-31 21:29:14 +02:00
grant_privilege_list:
grant_privilege
| grant_privilege_list ',' grant_privilege;
2000-07-31 21:29:14 +02:00
grant_privilege:
SELECT_SYM { Lex->which_columns = SELECT_ACL;} opt_column_list {}
| INSERT { Lex->which_columns = INSERT_ACL;} opt_column_list {}
| UPDATE_SYM { Lex->which_columns = UPDATE_ACL; } opt_column_list {}
| REFERENCES { Lex->which_columns = REFERENCES_ACL;} opt_column_list {}
| DELETE_SYM { Lex->grant |= DELETE_ACL;}
| USAGE {}
2000-07-31 21:29:14 +02:00
| INDEX { Lex->grant |= INDEX_ACL;}
| ALTER { Lex->grant |= ALTER_ACL;}
| CREATE { Lex->grant |= CREATE_ACL;}
| DROP { Lex->grant |= DROP_ACL;}
| EXECUTE_SYM { Lex->grant |= EXECUTE_ACL;}
2000-07-31 21:29:14 +02:00
| RELOAD { Lex->grant |= RELOAD_ACL;}
| SHUTDOWN { Lex->grant |= SHUTDOWN_ACL;}
| PROCESS { Lex->grant |= PROCESS_ACL;}
| FILE_SYM { Lex->grant |= FILE_ACL;}
| GRANT OPTION { Lex->grant |= GRANT_ACL;}
| SHOW DATABASES { Lex->grant |= SHOW_DB_ACL;}
| SUPER_SYM { Lex->grant |= SUPER_ACL;}
| CREATE TEMPORARY TABLES { Lex->grant |= CREATE_TMP_ACL;}
| LOCK_SYM TABLES { Lex->grant |= LOCK_TABLES_ACL; }
| REPLICATION SLAVE { Lex->grant |= REPL_SLAVE_ACL;}
| REPLICATION CLIENT_SYM { Lex->grant |= REPL_CLIENT_ACL;}
;
2000-07-31 21:29:14 +02:00
2001-09-19 19:45:13 -06:00
opt_and:
/* empty */ {}
| AND {}
;
require_list:
require_list_element opt_and require_list
| require_list_element
;
require_list_element:
SUBJECT_SYM TEXT_STRING
{
LEX *lex=Lex;
if (lex->x509_subject)
{
net_printf(lex->thd,ER_DUP_ARGUMENT, "SUBJECT");
YYABORT;
}
lex->x509_subject=$2.str;
}
| ISSUER_SYM TEXT_STRING
{
LEX *lex=Lex;
if (lex->x509_issuer)
{
net_printf(lex->thd,ER_DUP_ARGUMENT, "ISSUER");
YYABORT;
}
lex->x509_issuer=$2.str;
}
| CIPHER_SYM TEXT_STRING
{
LEX *lex=Lex;
if (lex->ssl_cipher)
{
net_printf(lex->thd,ER_DUP_ARGUMENT, "CIPHER");
YYABORT;
}
lex->ssl_cipher=$2.str;
}
;
2000-07-31 21:29:14 +02:00
opt_table:
'*'
{
LEX *lex= Lex;
lex->current_select->db= lex->thd->db;
if (lex->grant == GLOBAL_ACLS)
lex->grant = DB_ACLS & ~GRANT_ACL;
else if (lex->columns.elements)
2000-07-31 21:29:14 +02:00
{
send_error(lex->thd,ER_ILLEGAL_GRANT_FOR_TABLE);
YYABORT;
}
2000-07-31 21:29:14 +02:00
}
| ident '.' '*'
{
LEX *lex= Lex;
lex->current_select->db = $1.str;
if (lex->grant == GLOBAL_ACLS)
lex->grant = DB_ACLS & ~GRANT_ACL;
else if (lex->columns.elements)
2000-07-31 21:29:14 +02:00
{
send_error(lex->thd,ER_ILLEGAL_GRANT_FOR_TABLE);
2000-07-31 21:29:14 +02:00
YYABORT;
}
}
| '*' '.' '*'
{
LEX *lex= Lex;
lex->current_select->db = NULL;
if (lex->grant == GLOBAL_ACLS)
lex->grant= GLOBAL_ACLS & ~GRANT_ACL;
else if (lex->columns.elements)
2000-07-31 21:29:14 +02:00
{
send_error(lex->thd,ER_ILLEGAL_GRANT_FOR_TABLE);
2000-07-31 21:29:14 +02:00
YYABORT;
}
}
| table_ident
{
LEX *lex=Lex;
if (!lex->current_select->add_table_to_list(lex->thd, $1,NULL,0))
2000-07-31 21:29:14 +02:00
YYABORT;
if (lex->grant == GLOBAL_ACLS)
lex->grant = TABLE_ACLS & ~GRANT_ACL;
2002-10-16 16:55:08 +03:00
}
;
2000-07-31 21:29:14 +02:00
user_list:
grant_user { if (Lex->users_list.push_back($1)) YYABORT;}
| user_list ',' grant_user
{
if (Lex->users_list.push_back($3))
YYABORT;
}
;
2000-07-31 21:29:14 +02:00
grant_user:
user IDENTIFIED_SYM BY TEXT_STRING
{
$$=$1; $1->password=$4;
if ($4.length)
{
if (YYTHD->variables.old_passwords)
{
char *buff=
(char *) YYTHD->alloc(SCRAMBLED_PASSWORD_CHAR_LENGTH_323+1);
if (buff)
make_scrambled_password_323(buff, $4.str);
$1->password.str= buff;
$1->password.length= SCRAMBLED_PASSWORD_CHAR_LENGTH_323;
}
else
{
char *buff=
(char *) YYTHD->alloc(SCRAMBLED_PASSWORD_CHAR_LENGTH+1);
if (buff)
make_scrambled_password(buff, $4.str);
$1->password.str= buff;
$1->password.length= SCRAMBLED_PASSWORD_CHAR_LENGTH;
}
2000-07-31 21:29:14 +02:00
}
}
| user IDENTIFIED_SYM BY PASSWORD TEXT_STRING
{ $$=$1; $1->password=$5 ; }
| user
2002-10-16 16:55:08 +03:00
{ $$=$1; $1->password.str=NullS; }
;
2000-07-31 21:29:14 +02:00
opt_column_list:
/* empty */
{
LEX *lex=Lex;
lex->grant |= lex->which_columns;
}
| '(' column_list ')';
2000-07-31 21:29:14 +02:00
column_list:
column_list ',' column_list_id
| column_list_id;
2000-07-31 21:29:14 +02:00
column_list_id:
ident
{
String *new_str = new String((const char*) $1.str,$1.length,system_charset_info);
2000-07-31 21:29:14 +02:00
List_iterator <LEX_COLUMN> iter(Lex->columns);
class LEX_COLUMN *point;
LEX *lex=Lex;
2000-07-31 21:29:14 +02:00
while ((point=iter++))
{
if (!my_strcasecmp(system_charset_info,
point->column.ptr(), new_str->ptr()))
2000-07-31 21:29:14 +02:00
break;
}
lex->grant_tot_col|= lex->which_columns;
2000-07-31 21:29:14 +02:00
if (point)
point->rights |= lex->which_columns;
2000-07-31 21:29:14 +02:00
else
lex->columns.push_back(new LEX_COLUMN (*new_str,lex->which_columns));
2002-10-16 16:55:08 +03:00
}
;
2000-07-31 21:29:14 +02:00
2001-09-01 16:29:37 +08:00
require_clause: /* empty */
| REQUIRE_SYM require_list
{
Lex->ssl_type=SSL_TYPE_SPECIFIED;
}
| REQUIRE_SYM SSL_SYM
{
Lex->ssl_type=SSL_TYPE_ANY;
}
| REQUIRE_SYM X509_SYM
{
Lex->ssl_type=SSL_TYPE_X509;
}
| REQUIRE_SYM NONE_SYM
{
Lex->ssl_type=SSL_TYPE_NONE;
}
2002-10-16 16:55:08 +03:00
;
2001-09-01 16:29:37 +08:00
grant_options:
2000-07-31 21:29:14 +02:00
/* empty */ {}
| WITH grant_option_list;
2000-07-31 21:29:14 +02:00
grant_option_list:
grant_option_list grant_option {}
2002-10-16 16:55:08 +03:00
| grant_option {}
;
grant_option:
GRANT OPTION { Lex->grant |= GRANT_ACL;}
| MAX_QUERIES_PER_HOUR ULONG_NUM
{
Lex->mqh.questions=$2;
Lex->mqh.bits |= 1;
}
| MAX_UPDATES_PER_HOUR ULONG_NUM
{
Lex->mqh.updates=$2;
Lex->mqh.bits |= 2;
}
| MAX_CONNECTIONS_PER_HOUR ULONG_NUM
{
Lex->mqh.connections=$2;
Lex->mqh.bits |= 4;
2002-10-16 16:55:08 +03:00
}
;
2000-07-31 21:29:14 +02:00
begin:
BEGIN_SYM { Lex->sql_command = SQLCOM_BEGIN;} opt_work {}
;
2000-07-31 21:29:14 +02:00
opt_work:
/* empty */ {}
2002-10-16 16:55:08 +03:00
| WORK_SYM {;}
;
2000-07-31 21:29:14 +02:00
commit:
COMMIT_SYM { Lex->sql_command = SQLCOM_COMMIT;};
2000-07-31 21:29:14 +02:00
rollback:
2003-08-11 22:44:43 +03:00
ROLLBACK_SYM
{
Lex->sql_command = SQLCOM_ROLLBACK;
}
| ROLLBACK_SYM TO_SYM SAVEPOINT_SYM ident
{
Lex->sql_command = SQLCOM_ROLLBACK_TO_SAVEPOINT;
Lex->savepoint_name = $4.str;
};
savepoint:
SAVEPOINT_SYM ident
{
Lex->sql_command = SQLCOM_SAVEPOINT;
Lex->savepoint_name = $2.str;
};
/*
UNIONS : glue selects together
*/
union_clause:
/* empty */ {}
2002-11-21 22:25:53 +02:00
| union_list
;
union_list:
UNION_SYM union_option
{
LEX *lex=Lex;
if (lex->exchange)
{
/* Only the last SELECT can have INTO...... */
net_printf(lex->thd, ER_WRONG_USAGE, "UNION", "INTO");
YYABORT;
}
if (lex->current_select->linkage == GLOBAL_OPTIONS_TYPE)
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
if (mysql_new_select(lex, 0))
YYABORT;
mysql_init_select(lex);
lex->current_select->linkage=UNION_TYPE;
}
select_init {}
;
union_opt:
2002-10-26 18:28:19 +00:00
union_list {}
2002-11-21 22:25:53 +02:00
| optional_order_or_limit {}
;
optional_order_or_limit:
/* Empty */ {}
|
{
2003-01-14 18:00:34 +02:00
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
LEX *lex= thd->lex;
DBUG_ASSERT(lex->current_select->linkage != GLOBAL_OPTIONS_TYPE);
SELECT_LEX *sel= lex->current_select;
SELECT_LEX_UNIT *unit= sel->master_unit();
SELECT_LEX *fake= unit->fake_select_lex;
if (fake)
{
unit->global_parameters= fake;
fake->no_table_names_allowed= 1;
lex->current_select= fake;
}
2003-01-14 18:00:34 +02:00
thd->where= "global ORDER clause";
}
order_or_limit
2003-01-14 18:00:34 +02:00
{
THD *thd= YYTHD;
2003-05-05 14:54:37 -04:00
thd->lex->current_select->no_table_names_allowed= 0;
2003-01-14 18:00:34 +02:00
thd->where= "";
}
;
order_or_limit:
order_clause opt_limit_clause_init
| limit_clause
;
2001-07-22 13:25:56 +03:00
union_option:
/* empty */ {}
| DISTINCT {}
| ALL {Select->master_unit()->union_option|= UNION_ALL;};
2002-12-19 21:15:09 +02:00
singlerow_subselect:
subselect_start singlerow_subselect_init
subselect_end
{
$$= $2;
};
2002-12-19 21:15:09 +02:00
singlerow_subselect_init:
select_init2
{
$$= new Item_singlerow_subselect(Lex->current_select->
master_unit()->first_select());
};
exists_subselect:
subselect_start exists_subselect_init
subselect_end
{
$$= $2;
};
exists_subselect_init:
select_init2
{
$$= new Item_exists_subselect(Lex->current_select->master_unit()->
first_select());
};
2002-10-27 23:27:00 +02:00
in_subselect:
subselect_start in_subselect_init
subselect_end
{
$$= $2;
};
in_subselect_init:
select_init2
2002-10-27 23:27:00 +02:00
{
2002-11-05 11:59:18 +02:00
$$= Lex->current_select->master_unit()->first_select();
2002-10-27 23:27:00 +02:00
};
subselect_start:
'(' SELECT_SYM
{
LEX *lex=Lex;
if (((int)lex->sql_command >= (int)SQLCOM_HA_OPEN &&
lex->sql_command <= (int)SQLCOM_HA_READ) ||
lex->sql_command == (int)SQLCOM_KILL)
{
send_error(lex->thd, ER_SYNTAX_ERROR);
YYABORT;
}
if (mysql_new_select(Lex, 1))
YYABORT;
};
subselect_end:
')'
{
LEX *lex=Lex;
lex->current_select = lex->current_select->return_after_parsing();
2002-10-11 21:49:10 +03:00
};