2000-07-31 21:29:14 +02:00
|
|
|
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
|
2001-12-06 13:10:51 +01:00
|
|
|
|
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.
|
2001-12-06 13:10:51 +01:00
|
|
|
|
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.
|
2001-12-06 13:10:51 +01:00
|
|
|
|
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 */
|
|
|
|
|
|
|
|
|
|
|
|
/* Classes in mysql */
|
|
|
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
#pragma interface /* gcc class implementation */
|
|
|
|
#endif
|
|
|
|
|
2002-01-20 03:16:52 +01:00
|
|
|
// TODO: create log.h and move all the log header stuff there
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class Query_log_event;
|
|
|
|
class Load_log_event;
|
2001-06-14 22:12:40 +02:00
|
|
|
class Slave_log_event;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2001-05-11 22:26:12 +02:00
|
|
|
enum enum_enable_or_disable { LEAVE_AS_IS, ENABLE, DISABLE };
|
2001-04-07 00:18:33 +02:00
|
|
|
enum enum_ha_read_modes { RFIRST, RNEXT, RPREV, RLAST, RKEY };
|
2000-07-31 21:29:14 +02:00
|
|
|
enum enum_duplicates { DUP_ERROR, DUP_REPLACE, DUP_IGNORE };
|
|
|
|
enum enum_log_type { LOG_CLOSED, LOG_NORMAL, LOG_NEW, LOG_BIN };
|
2002-08-13 01:18:39 +02:00
|
|
|
enum enum_delay_key_write { DELAY_KEY_WRITE_NONE, DELAY_KEY_WRITE_ON,
|
|
|
|
DELAY_KEY_WRITE_ALL };
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
// log info errors
|
|
|
|
#define LOG_INFO_EOF -1
|
|
|
|
#define LOG_INFO_IO -2
|
|
|
|
#define LOG_INFO_INVALID -3
|
|
|
|
#define LOG_INFO_SEEK -4
|
2000-10-27 06:11:55 +02:00
|
|
|
#define LOG_INFO_PURGE_NO_ROTATE -5
|
|
|
|
#define LOG_INFO_MEM -6
|
|
|
|
#define LOG_INFO_FATAL -7
|
|
|
|
#define LOG_INFO_IN_USE -8
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-01-20 03:16:52 +01:00
|
|
|
struct st_relay_log_info;
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
typedef struct st_log_info
|
|
|
|
{
|
|
|
|
char log_file_name[FN_REFLEN];
|
2002-08-08 02:12:02 +02:00
|
|
|
my_off_t index_file_offset, index_file_start_offset;
|
2000-10-27 06:11:55 +02:00
|
|
|
my_off_t pos;
|
|
|
|
bool fatal; // if the purge happens to give us a negative offset
|
|
|
|
pthread_mutex_t lock;
|
2001-03-26 00:05:04 +02:00
|
|
|
st_log_info():fatal(0) { pthread_mutex_init(&lock, MY_MUTEX_INIT_FAST);}
|
2000-10-27 06:11:55 +02:00
|
|
|
~st_log_info() { pthread_mutex_destroy(&lock);}
|
2000-07-31 21:29:14 +02:00
|
|
|
} LOG_INFO;
|
|
|
|
|
2001-09-30 21:04:56 +02:00
|
|
|
class Log_event;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
class MYSQL_LOG {
|
|
|
|
private:
|
|
|
|
pthread_mutex_t LOCK_log, LOCK_index;
|
2002-06-05 22:04:38 +02:00
|
|
|
pthread_cond_t update_cond;
|
|
|
|
ulonglong bytes_written;
|
2000-07-31 21:29:14 +02:00
|
|
|
time_t last_time,query_start;
|
2000-11-15 22:00:06 +01:00
|
|
|
IO_CACHE log_file;
|
2002-08-08 02:12:02 +02:00
|
|
|
IO_CACHE index_file;
|
2000-07-31 21:29:14 +02:00
|
|
|
char *name;
|
|
|
|
char time_buff[20],db[NAME_LEN+1];
|
|
|
|
char log_file_name[FN_REFLEN],index_file_name[FN_REFLEN];
|
2002-06-05 22:04:38 +02:00
|
|
|
// current file sequence number for load data infile binary logging
|
|
|
|
uint file_id;
|
|
|
|
uint open_count; // For replication
|
2002-08-30 11:40:40 +02:00
|
|
|
volatile enum_log_type log_type;
|
|
|
|
enum cache_type io_cache_type;
|
2000-12-12 03:34:56 +01:00
|
|
|
bool write_error,inited;
|
2002-06-05 22:04:38 +02:00
|
|
|
/*
|
|
|
|
For binlog - if log name can never change we should not try to rotate it
|
|
|
|
or write any rotation events. The user should use FLUSH MASTER instead
|
|
|
|
of FLUSH LOGS for purging.
|
|
|
|
*/
|
|
|
|
bool no_rotate;
|
2001-11-11 06:24:12 +01:00
|
|
|
bool need_start_event;
|
2002-01-20 03:16:52 +01:00
|
|
|
bool no_auto_events; // for relay binlog
|
2001-06-19 23:03:48 +02:00
|
|
|
friend class Log_event;
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
public:
|
|
|
|
MYSQL_LOG();
|
|
|
|
~MYSQL_LOG();
|
2002-04-02 06:46:23 +02:00
|
|
|
void reset_bytes_written()
|
2002-06-05 22:04:38 +02:00
|
|
|
{
|
|
|
|
bytes_written = 0;
|
|
|
|
}
|
2002-04-02 06:46:23 +02:00
|
|
|
void harvest_bytes_written(ulonglong* counter)
|
2002-06-05 22:04:38 +02:00
|
|
|
{
|
2002-04-02 06:46:23 +02:00
|
|
|
#ifndef DBUG_OFF
|
2002-06-05 22:04:38 +02:00
|
|
|
char buf1[22],buf2[22];
|
2002-04-02 06:46:23 +02:00
|
|
|
#endif
|
2002-06-05 22:04:38 +02:00
|
|
|
DBUG_ENTER("harvest_bytes_written");
|
|
|
|
(*counter)+=bytes_written;
|
|
|
|
DBUG_PRINT("info",("counter: %s bytes_written: %s", llstr(*counter,buf1),
|
|
|
|
llstr(bytes_written,buf2)));
|
|
|
|
bytes_written=0;
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
2002-01-20 03:16:52 +01:00
|
|
|
void signal_update() { pthread_cond_broadcast(&update_cond);}
|
|
|
|
void wait_for_update(THD* thd);
|
2001-11-11 06:24:12 +01:00
|
|
|
void set_need_start_event() { need_start_event = 1; }
|
|
|
|
void init(enum_log_type log_type_arg,
|
2002-01-20 03:16:52 +01:00
|
|
|
enum cache_type io_cache_type_arg = WRITE_CACHE,
|
|
|
|
bool no_auto_events_arg = 0);
|
2002-08-08 02:12:02 +02:00
|
|
|
bool open(const char *log_name,enum_log_type log_type,
|
|
|
|
const char *new_name, const char *index_file_name_arg,
|
|
|
|
enum cache_type io_cache_type_arg,
|
2002-01-20 03:16:52 +01:00
|
|
|
bool no_auto_events_arg);
|
2002-08-08 02:12:02 +02:00
|
|
|
void new_file(bool need_lock= 1);
|
2002-01-20 03:16:52 +01:00
|
|
|
bool write(THD *thd, enum enum_server_command command,
|
|
|
|
const char *format,...);
|
2000-12-07 13:08:48 +01:00
|
|
|
bool write(THD *thd, const char *query, uint query_length,
|
2000-09-16 03:27:21 +02:00
|
|
|
time_t query_start=0);
|
2001-08-03 23:57:53 +02:00
|
|
|
bool write(Log_event* event_info); // binary log write
|
2002-01-22 21:57:56 +01:00
|
|
|
bool write(THD *thd, IO_CACHE *cache);
|
2002-01-20 03:16:52 +01:00
|
|
|
|
2002-01-30 15:37:47 +01:00
|
|
|
/*
|
|
|
|
v stands for vector
|
|
|
|
invoked as appendv(buf1,len1,buf2,len2,...,bufn,lenn,0)
|
|
|
|
*/
|
2002-01-20 03:16:52 +01:00
|
|
|
bool appendv(const char* buf,uint len,...);
|
2002-01-25 06:49:47 +01:00
|
|
|
bool append(Log_event* ev);
|
2002-01-20 03:16:52 +01:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
int generate_new_name(char *new_name,const char *old_name);
|
|
|
|
void make_log_name(char* buf, const char* log_ident);
|
|
|
|
bool is_active(const char* log_file_name);
|
2000-10-27 06:11:55 +02:00
|
|
|
int purge_logs(THD* thd, const char* to_log);
|
2002-01-20 03:16:52 +01:00
|
|
|
int purge_first_log(struct st_relay_log_info* rli);
|
2002-08-08 02:12:02 +02:00
|
|
|
bool reset_logs(THD* thd);
|
2002-06-05 22:04:38 +02:00
|
|
|
// if we are exiting, we also want to close the index file
|
|
|
|
void close(bool exiting = 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
// iterating through the log index file
|
2002-08-08 02:12:02 +02:00
|
|
|
int find_log_pos(LOG_INFO* linfo, const char* log_name,
|
2002-08-22 15:50:58 +02:00
|
|
|
bool need_mutex);
|
|
|
|
int find_next_log(LOG_INFO* linfo, bool need_mutex);
|
2000-07-31 21:29:14 +02:00
|
|
|
int get_current_log(LOG_INFO* linfo);
|
2001-08-04 13:48:58 +02:00
|
|
|
uint next_file_id();
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2000-09-16 03:27:21 +02:00
|
|
|
inline bool is_open() { return log_type != LOG_CLOSED; }
|
2002-06-05 22:04:38 +02:00
|
|
|
inline char* get_index_fname() { return index_file_name;}
|
|
|
|
inline char* get_log_fname() { return log_file_name; }
|
|
|
|
inline pthread_mutex_t* get_log_lock() { return &LOCK_log; }
|
|
|
|
inline IO_CACHE* get_log_file() { return &log_file; }
|
|
|
|
|
|
|
|
inline void lock_index() { pthread_mutex_lock(&LOCK_index);}
|
|
|
|
inline void unlock_index() { pthread_mutex_unlock(&LOCK_index);}
|
2002-08-08 02:12:02 +02:00
|
|
|
inline IO_CACHE *get_index_file() { return &index_file;}
|
2002-06-05 22:04:38 +02:00
|
|
|
inline uint32 get_open_count() { return open_count; }
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/* character conversion tables */
|
|
|
|
|
2001-01-27 00:20:56 +01:00
|
|
|
class CONVERT;
|
|
|
|
CONVERT *get_convert_set(const char *name_ptr);
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class CONVERT
|
|
|
|
{
|
|
|
|
const uchar *from_map,*to_map;
|
|
|
|
void convert_array(const uchar *mapping,uchar *buff,uint length);
|
|
|
|
public:
|
|
|
|
const char *name;
|
2001-12-02 13:34:01 +01:00
|
|
|
uint numb;
|
|
|
|
CONVERT(const char *name_par,uchar *from_par,uchar *to_par, uint number)
|
|
|
|
:from_map(from_par),to_map(to_par),name(name_par),numb(number) {}
|
2000-07-31 21:29:14 +02:00
|
|
|
friend CONVERT *get_convert_set(const char *name_ptr);
|
|
|
|
inline void convert(char *a,uint length)
|
|
|
|
{
|
|
|
|
convert_array(from_map, (uchar*) a,length);
|
|
|
|
}
|
|
|
|
bool store(String *, const char *,uint);
|
2001-12-02 13:34:01 +01:00
|
|
|
inline uint number() { return numb; }
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct st_copy_info {
|
|
|
|
ha_rows records;
|
|
|
|
ha_rows deleted;
|
|
|
|
ha_rows copied;
|
2002-06-11 10:20:31 +02:00
|
|
|
ha_rows error_count;
|
2000-07-31 21:29:14 +02:00
|
|
|
enum enum_duplicates handle_duplicates;
|
2002-06-11 10:20:31 +02:00
|
|
|
int escape_char, last_errno;
|
2000-07-31 21:29:14 +02:00
|
|
|
} COPY_INFO;
|
|
|
|
|
|
|
|
|
|
|
|
class key_part_spec :public Sql_alloc {
|
|
|
|
public:
|
|
|
|
const char *field_name;
|
|
|
|
uint length;
|
|
|
|
key_part_spec(const char *name,uint len=0) :field_name(name), length(len) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Alter_drop :public Sql_alloc {
|
|
|
|
public:
|
|
|
|
enum drop_type {KEY, COLUMN };
|
|
|
|
const char *name;
|
|
|
|
enum drop_type type;
|
|
|
|
Alter_drop(enum drop_type par_type,const char *par_name)
|
|
|
|
:name(par_name), type(par_type) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Alter_column :public Sql_alloc {
|
|
|
|
public:
|
|
|
|
const char *name;
|
|
|
|
Item *def;
|
|
|
|
Alter_column(const char *par_name,Item *literal)
|
|
|
|
:name(par_name), def(literal) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Key :public Sql_alloc {
|
|
|
|
public:
|
2002-06-02 20:22:20 +02:00
|
|
|
enum Keytype { PRIMARY, UNIQUE, MULTIPLE, FULLTEXT, SPATIAL, FOREIGN_KEY};
|
2000-07-31 21:29:14 +02:00
|
|
|
enum Keytype type;
|
2002-04-12 20:35:46 +02:00
|
|
|
enum ha_key_alg algorithm;
|
2000-07-31 21:29:14 +02:00
|
|
|
List<key_part_spec> columns;
|
2002-06-02 20:22:20 +02:00
|
|
|
const char *name;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-06-02 20:22:20 +02:00
|
|
|
Key(enum Keytype type_par, const char *name_arg, enum ha_key_alg alg_par,
|
|
|
|
List<key_part_spec> &cols)
|
|
|
|
:type(type_par), algorithm(alg_par), columns(cols), name(name_arg)
|
2002-04-12 20:35:46 +02:00
|
|
|
{}
|
2000-07-31 21:29:14 +02:00
|
|
|
~Key() {}
|
|
|
|
};
|
|
|
|
|
2002-06-02 20:22:20 +02:00
|
|
|
class Table_ident;
|
|
|
|
|
|
|
|
class foreign_key: public Key {
|
|
|
|
public:
|
|
|
|
enum fk_match_opt { FK_MATCH_UNDEF, FK_MATCH_FULL,
|
|
|
|
FK_MATCH_PARTIAL, FK_MATCH_SIMPLE};
|
|
|
|
enum fk_option { FK_OPTION_UNDEF, FK_OPTION_RESTRICT, FK_OPTION_CASCADE,
|
|
|
|
FK_OPTION_SET_NULL, FK_OPTION_NO_ACTION, FK_OPTION_DEFAULT};
|
|
|
|
|
|
|
|
Table_ident *ref_table;
|
|
|
|
List<key_part_spec> ref_columns;
|
|
|
|
uint delete_opt, update_opt, match_opt;
|
|
|
|
foreign_key(const char *name_arg, List<key_part_spec> &cols,
|
|
|
|
Table_ident *table, List<key_part_spec> &ref_cols,
|
|
|
|
uint delete_opt_arg, uint update_opt_arg, uint match_opt_arg)
|
|
|
|
:Key(FOREIGN_KEY, name_arg, HA_KEY_ALG_UNDEF, cols),
|
|
|
|
ref_table(table), ref_columns(cols),
|
|
|
|
delete_opt(delete_opt_arg), update_opt(update_opt_arg),
|
|
|
|
match_opt(match_opt_arg)
|
|
|
|
{}
|
|
|
|
};
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
typedef struct st_mysql_lock
|
|
|
|
{
|
|
|
|
TABLE **table;
|
|
|
|
uint table_count,lock_count;
|
|
|
|
THR_LOCK_DATA **locks;
|
|
|
|
} MYSQL_LOCK;
|
|
|
|
|
|
|
|
|
|
|
|
class LEX_COLUMN : public Sql_alloc
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
String column;
|
|
|
|
uint rights;
|
|
|
|
LEX_COLUMN (const String& x,const uint& y ): column (x),rights (y) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
#include "sql_lex.h" /* Must be here */
|
|
|
|
|
2002-06-02 20:22:20 +02:00
|
|
|
/* Needed to be able to have an I_List of char* strings in mysqld.cc. */
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class i_string: public ilink
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
char* ptr;
|
|
|
|
i_string():ptr(0) { }
|
|
|
|
i_string(char* s) : ptr(s) {}
|
|
|
|
};
|
|
|
|
|
2002-06-02 20:22:20 +02:00
|
|
|
/* needed for linked list of two strings for replicate-rewrite-db */
|
2000-10-06 01:58:16 +02:00
|
|
|
class i_string_pair: public ilink
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
char* key;
|
|
|
|
char* val;
|
|
|
|
i_string_pair():key(0),val(0) { }
|
2000-12-18 15:09:19 +01:00
|
|
|
i_string_pair(char* key_arg, char* val_arg) : key(key_arg),val(val_arg) {}
|
2000-10-06 01:58:16 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2002-10-02 12:33:08 +02:00
|
|
|
class MYSQL_ERROR: public Sql_alloc
|
2002-06-12 23:13:12 +02:00
|
|
|
{
|
|
|
|
public:
|
2002-10-02 12:33:08 +02:00
|
|
|
enum enum_warning_level
|
|
|
|
{ WARN_LEVEL_NOTE, WARN_LEVEL_WARN, WARN_LEVEL_ERROR, WARN_LEVEL_END};
|
|
|
|
|
2002-06-12 23:13:12 +02:00
|
|
|
uint code;
|
2002-10-02 12:33:08 +02:00
|
|
|
enum_warning_level level;
|
|
|
|
char *msg;
|
2002-06-12 23:13:12 +02:00
|
|
|
|
2002-10-02 12:33:08 +02:00
|
|
|
MYSQL_ERROR(uint code_arg, enum_warning_level level_arg,
|
|
|
|
const char *msg_arg)
|
|
|
|
:code(code_arg), level(level_arg)
|
2002-06-12 23:13:12 +02:00
|
|
|
{
|
2002-10-02 12:33:08 +02:00
|
|
|
msg=sql_strdup(msg_arg);
|
2002-06-12 23:13:12 +02:00
|
|
|
}
|
|
|
|
};
|
2000-10-06 01:58:16 +02:00
|
|
|
|
2002-10-02 12:33:08 +02:00
|
|
|
|
|
|
|
/* This is a struct as it's allocated in tree_insert */
|
|
|
|
|
|
|
|
typedef struct st_prep_stmt
|
|
|
|
{
|
|
|
|
THD *thd;
|
|
|
|
Item_param *param;
|
|
|
|
Item *free_list;
|
|
|
|
MEM_ROOT mem_root;
|
|
|
|
ulong stmt_id;
|
|
|
|
uint param_count;
|
|
|
|
uint last_errno;
|
|
|
|
char last_error[MYSQL_ERRMSG_SIZE];
|
|
|
|
bool error_in_prepare, long_data_used;
|
|
|
|
} PREP_STMT;
|
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class delayed_insert;
|
2002-09-26 22:08:22 +02:00
|
|
|
class select_result;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-03-30 20:36:05 +01:00
|
|
|
#define THD_SENTRY_MAGIC 0xfeedd1ff
|
|
|
|
#define THD_SENTRY_GONE 0xdeadbeef
|
|
|
|
|
|
|
|
#define THD_CHECK_SENTRY(thd) DBUG_ASSERT(thd->dbug_sentry == THD_SENTRY_MAGIC)
|
|
|
|
|
2002-06-28 18:30:09 +02:00
|
|
|
struct system_variables
|
|
|
|
{
|
2002-07-23 17:31:22 +02:00
|
|
|
ulonglong myisam_max_extra_sort_file_size;
|
|
|
|
ulonglong myisam_max_sort_file_size;
|
|
|
|
ulong bulk_insert_buff_size;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong join_buff_size;
|
|
|
|
ulong long_query_time;
|
2002-07-23 17:31:22 +02:00
|
|
|
ulong max_allowed_packet;
|
2002-10-02 12:33:08 +02:00
|
|
|
ulong max_error_count;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong max_heap_table_size;
|
|
|
|
ulong max_join_size;
|
2002-10-02 12:33:08 +02:00
|
|
|
ulong max_prep_stmt_count;
|
|
|
|
ulong max_sort_length;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong max_tmp_tables;
|
2002-07-23 17:31:22 +02:00
|
|
|
ulong myisam_sort_buff_size;
|
|
|
|
ulong net_buffer_length;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong net_interactive_timeout;
|
2002-07-23 17:31:22 +02:00
|
|
|
ulong net_read_timeout;
|
2002-10-02 12:33:08 +02:00
|
|
|
ulong net_retry_count;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong net_wait_timeout;
|
2002-07-23 17:31:22 +02:00
|
|
|
ulong net_write_timeout;
|
|
|
|
ulong query_cache_type;
|
|
|
|
ulong read_buff_size;
|
|
|
|
ulong read_rnd_buff_size;
|
|
|
|
ulong select_limit;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong sortbuff_size;
|
2002-10-02 12:33:08 +02:00
|
|
|
ulong table_type;
|
2002-06-28 18:30:09 +02:00
|
|
|
ulong tmp_table_size;
|
2002-07-23 17:31:22 +02:00
|
|
|
ulong tx_isolation;
|
2002-06-28 18:30:09 +02:00
|
|
|
|
2002-07-23 17:31:22 +02:00
|
|
|
my_bool log_warnings;
|
|
|
|
my_bool low_priority_updates;
|
2002-06-28 18:30:09 +02:00
|
|
|
|
2002-07-23 17:31:22 +02:00
|
|
|
CONVERT *convert_set;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
For each client connection we create a separate thread with THD serving as
|
|
|
|
a thread/connection descriptor
|
|
|
|
*/
|
2002-01-20 03:16:52 +01:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class THD :public ilink {
|
|
|
|
public:
|
2002-07-23 17:31:22 +02:00
|
|
|
NET net; // client connection descriptor
|
|
|
|
LEX lex; // parse tree descriptor
|
|
|
|
MEM_ROOT mem_root; // 1 command-life memory pool
|
2002-06-12 23:13:12 +02:00
|
|
|
MEM_ROOT con_root; // connection-life memory
|
2002-10-02 12:33:08 +02:00
|
|
|
MEM_ROOT warn_root; // For warnings and errors
|
2002-07-23 17:31:22 +02:00
|
|
|
HASH user_vars; // hash for user variables
|
2002-10-02 12:33:08 +02:00
|
|
|
TREE prepared_statements;
|
2002-07-23 17:31:22 +02:00
|
|
|
String packet; // dynamic buffer for network I/O
|
|
|
|
struct sockaddr_in remote; // client socket address
|
|
|
|
struct rand_struct rand; // used for authentication
|
|
|
|
struct system_variables variables; // Changeable local variables
|
2002-08-22 15:50:58 +02:00
|
|
|
pthread_mutex_t LOCK_delete; // Locked before thd is deleted
|
|
|
|
|
2002-07-23 17:31:22 +02:00
|
|
|
char *query; // Points to the current query,
|
|
|
|
/*
|
|
|
|
A pointer to the stack frame of handle_one_connection(),
|
|
|
|
which is called first in the thread for handling a client
|
|
|
|
*/
|
|
|
|
char *thread_stack;
|
|
|
|
|
2002-01-20 03:16:52 +01:00
|
|
|
/*
|
|
|
|
host - host of the client
|
|
|
|
user - user of the client, set to NULL until the user has been read from
|
|
|
|
the connection
|
2002-06-02 20:22:20 +02:00
|
|
|
priv_user - The user privilege we are using. May be '' for anonymous user.
|
2002-01-20 03:16:52 +01:00
|
|
|
db - currently selected database
|
|
|
|
ip - client IP
|
|
|
|
*/
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
char *host,*user,*priv_user,*db,*ip;
|
2002-06-12 14:04:18 +02:00
|
|
|
/* Points to info-string that will show in SHOW PROCESSLIST */
|
|
|
|
const char *proc_info;
|
|
|
|
/* points to host if host is available, otherwise points to ip */
|
|
|
|
const char *host_or_ip;
|
|
|
|
|
|
|
|
uint client_capabilities; /* What the client supports */
|
|
|
|
/* Determines if which non-standard SQL behaviour should be enabled */
|
|
|
|
uint sql_mode;
|
2002-07-23 17:31:22 +02:00
|
|
|
uint max_client_packet_length;
|
2002-06-12 14:04:18 +02:00
|
|
|
ulong master_access; /* Global privileges from mysql.user */
|
|
|
|
ulong db_access; /* Privileges for current db */
|
2002-01-20 03:16:52 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
open_tables - list of regular tables in use by this thread
|
|
|
|
temporary_tables - list of temp tables in use by this thread
|
|
|
|
handler_tables - list of tables that were opened with HANDLER OPEN
|
|
|
|
and are still in use by this thread
|
2002-06-12 14:04:18 +02:00
|
|
|
*/
|
2001-04-13 16:18:44 +02:00
|
|
|
TABLE *open_tables,*temporary_tables, *handler_tables;
|
2002-01-20 03:16:52 +01:00
|
|
|
// TODO: document the variables below
|
2002-10-02 12:33:08 +02:00
|
|
|
MYSQL_LOCK *lock; /* Current locks */
|
|
|
|
MYSQL_LOCK *locked_tables; /* Tables locked with LOCK */
|
|
|
|
ULL *ull;
|
|
|
|
PREP_STMT *last_prepared_stmt;
|
2002-03-30 20:36:05 +01:00
|
|
|
#ifndef DBUG_OFF
|
|
|
|
uint dbug_sentry; // watch out for memory corruption
|
|
|
|
#endif
|
2000-07-31 21:29:14 +02:00
|
|
|
struct st_my_thread_var *mysys_var;
|
|
|
|
enum enum_server_command command;
|
2002-07-23 17:31:22 +02:00
|
|
|
uint32 server_id;
|
2001-12-05 12:03:00 +01:00
|
|
|
uint32 file_id; // for LOAD DATA INFILE
|
2000-07-31 21:29:14 +02:00
|
|
|
const char *where;
|
2001-12-05 12:03:00 +01:00
|
|
|
time_t start_time,time_after_lock,user_time;
|
|
|
|
time_t connect_time,thr_create_time; // track down slow pthread_create
|
2000-07-31 21:29:14 +02:00
|
|
|
thr_lock_type update_lock_default;
|
|
|
|
delayed_insert *di;
|
|
|
|
struct st_transactions {
|
2000-11-15 22:00:06 +01:00
|
|
|
IO_CACHE trans_log;
|
2001-12-05 12:03:00 +01:00
|
|
|
THD_TRANS all; // Trans since BEGIN WORK
|
|
|
|
THD_TRANS stmt; // Trans for current statement
|
2000-07-31 21:29:14 +02:00
|
|
|
uint bdb_lock_count;
|
2002-03-15 22:57:31 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
Tables changed in transaction (that must be invalidated in query cache).
|
|
|
|
List contain only transactional tables, that not invalidated in query
|
|
|
|
cache (instead of full list of changed in transaction tables).
|
|
|
|
*/
|
|
|
|
CHANGED_TABLE_LIST* changed_tables;
|
|
|
|
MEM_ROOT mem_root; // Transaction-life memory allocation pool
|
|
|
|
void cleanup()
|
|
|
|
{
|
|
|
|
changed_tables = 0;
|
|
|
|
free_root(&mem_root,MYF(MY_KEEP_PREALLOC));
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
} transaction;
|
2001-04-13 16:18:44 +02:00
|
|
|
Item *free_list, *handler_items;
|
2000-07-31 21:29:14 +02:00
|
|
|
Field *dupp_field;
|
|
|
|
#ifndef __WIN__
|
|
|
|
sigset_t signals,block_signals;
|
|
|
|
#endif
|
2001-03-14 07:07:12 +01:00
|
|
|
#ifdef SIGNAL_WITH_VIO_CLOSE
|
|
|
|
Vio* active_vio;
|
2001-03-13 04:17:32 +01:00
|
|
|
#endif
|
2001-12-05 12:03:00 +01:00
|
|
|
ulonglong next_insert_id,last_insert_id,current_insert_id,
|
|
|
|
limit_found_rows;
|
2002-07-23 17:31:22 +02:00
|
|
|
ha_rows select_limit, offset_limit, cuted_fields,
|
|
|
|
sent_row_count, examined_row_count;
|
2001-12-05 12:03:00 +01:00
|
|
|
table_map used_tables;
|
2002-05-15 12:50:38 +02:00
|
|
|
USER_CONN *user_connect;
|
2002-08-30 11:40:40 +02:00
|
|
|
CHARSET_INFO *db_charset;
|
2002-10-02 12:33:08 +02:00
|
|
|
List <MYSQL_ERROR> warn_list;
|
|
|
|
uint warn_count[(uint) MYSQL_ERROR::WARN_LEVEL_END];
|
|
|
|
uint total_warn_count, old_total_warn_count;
|
2002-08-30 11:40:40 +02:00
|
|
|
ulong query_id, version, options, thread_id, col_access;
|
2002-10-02 12:33:08 +02:00
|
|
|
ulong current_stmt_id;
|
2001-12-05 12:03:00 +01:00
|
|
|
long dbug_thread_id;
|
2000-07-31 21:29:14 +02:00
|
|
|
pthread_t real_id;
|
2002-06-12 14:04:18 +02:00
|
|
|
uint current_tablenr,tmp_table,cond_count;
|
2001-12-05 12:03:00 +01:00
|
|
|
uint server_status,open_options;
|
|
|
|
uint32 query_length;
|
2001-12-21 06:00:58 +01:00
|
|
|
uint32 db_length;
|
2002-09-26 22:08:22 +02:00
|
|
|
uint select_number; //number of select (used for EXPLAIN)
|
2002-07-23 17:31:22 +02:00
|
|
|
/* variables.transaction_isolation is reset to this after each commit */
|
|
|
|
enum_tx_isolation session_tx_isolation;
|
2000-07-31 21:29:14 +02:00
|
|
|
char scramble[9];
|
2001-12-05 12:03:00 +01:00
|
|
|
uint8 query_cache_type; // type of query cache processing
|
2001-02-20 21:34:47 +01:00
|
|
|
bool slave_thread;
|
2000-07-31 21:29:14 +02:00
|
|
|
bool set_query_id,locked,count_cuted_fields,some_tables_deleted;
|
2002-10-02 12:33:08 +02:00
|
|
|
bool no_errors, allow_sum_func, password;
|
|
|
|
bool fatal_error;
|
2000-11-16 19:47:28 +01:00
|
|
|
bool query_start_used,last_insert_id_used,insert_id_used;
|
2000-07-31 21:29:14 +02:00
|
|
|
bool system_thread,in_lock_tables,global_read_lock;
|
2001-08-21 19:06:00 +02:00
|
|
|
bool query_error, bootstrap, cleanup_done;
|
2001-12-02 13:34:01 +01:00
|
|
|
bool safe_to_cache_query;
|
2001-02-20 21:34:47 +01:00
|
|
|
bool volatile killed;
|
2002-09-03 08:50:36 +02:00
|
|
|
bool prepare_command;
|
2002-10-02 12:33:08 +02:00
|
|
|
Item_param *params; // Pointer to array of params
|
2002-06-12 23:13:12 +02:00
|
|
|
|
2001-12-05 12:03:00 +01:00
|
|
|
/*
|
|
|
|
If we do a purge of binary logs, log index info of the threads
|
|
|
|
that are currently reading it needs to be adjusted. To do that
|
|
|
|
each thread that is using LOG_INFO needs to adjust the pointer to it
|
|
|
|
*/
|
2000-10-27 06:11:55 +02:00
|
|
|
LOG_INFO* current_linfo;
|
2001-12-05 12:03:00 +01:00
|
|
|
/*
|
|
|
|
In slave thread we need to know in behalf of which
|
|
|
|
thread the query is being run to replicate temp tables properly
|
|
|
|
*/
|
|
|
|
ulong slave_proxy_id;
|
|
|
|
NET* slave_net; // network connection from slave -> m.
|
2002-01-29 17:32:16 +01:00
|
|
|
my_off_t log_pos;
|
2002-07-23 17:31:22 +02:00
|
|
|
/* Used by the sys_var class to store temporary values */
|
|
|
|
union
|
|
|
|
{
|
|
|
|
my_bool my_bool_value;
|
|
|
|
long long_value;
|
|
|
|
} sys_var_tmp;
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
THD();
|
|
|
|
~THD();
|
2001-08-21 19:06:00 +02:00
|
|
|
void cleanup(void);
|
2000-07-31 21:29:14 +02:00
|
|
|
bool store_globals();
|
2001-03-14 07:07:12 +01:00
|
|
|
#ifdef SIGNAL_WITH_VIO_CLOSE
|
|
|
|
inline void set_active_vio(Vio* vio)
|
2001-03-13 04:17:32 +01:00
|
|
|
{
|
2002-08-22 15:50:58 +02:00
|
|
|
pthread_mutex_lock(&LOCK_delete);
|
2001-03-14 07:07:12 +01:00
|
|
|
active_vio = vio;
|
2002-08-22 15:50:58 +02:00
|
|
|
pthread_mutex_unlock(&LOCK_delete);
|
2001-03-13 04:17:32 +01:00
|
|
|
}
|
2001-03-14 07:07:12 +01:00
|
|
|
inline void clear_active_vio()
|
2001-03-13 04:17:32 +01:00
|
|
|
{
|
2002-08-22 15:50:58 +02:00
|
|
|
pthread_mutex_lock(&LOCK_delete);
|
2001-03-14 07:07:12 +01:00
|
|
|
active_vio = 0;
|
2002-08-22 15:50:58 +02:00
|
|
|
pthread_mutex_unlock(&LOCK_delete);
|
2001-03-13 04:17:32 +01:00
|
|
|
}
|
2002-09-05 15:17:08 +02:00
|
|
|
void THD::close_active_vio();
|
2001-03-13 04:17:32 +01:00
|
|
|
#endif
|
2002-01-20 03:16:52 +01:00
|
|
|
void awake(bool prepare_to_die);
|
2001-01-17 13:47:33 +01:00
|
|
|
inline const char* enter_cond(pthread_cond_t *cond, pthread_mutex_t* mutex,
|
|
|
|
const char* msg)
|
|
|
|
{
|
|
|
|
const char* old_msg = proc_info;
|
|
|
|
mysys_var->current_mutex = mutex;
|
|
|
|
mysys_var->current_cond = cond;
|
|
|
|
proc_info = msg;
|
|
|
|
return old_msg;
|
|
|
|
}
|
|
|
|
inline void exit_cond(const char* old_msg)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&mysys_var->mutex);
|
|
|
|
mysys_var->current_mutex = 0;
|
|
|
|
mysys_var->current_cond = 0;
|
|
|
|
proc_info = old_msg;
|
|
|
|
pthread_mutex_unlock(&mysys_var->mutex);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
inline time_t query_start() { query_start_used=1; return start_time; }
|
2000-11-16 19:47:28 +01:00
|
|
|
inline void set_time() { if (user_time) start_time=time_after_lock=user_time; else time_after_lock=time(&start_time); }
|
2000-10-14 02:16:35 +02:00
|
|
|
inline void end_time() { time(&start_time); }
|
2000-11-16 19:47:28 +01:00
|
|
|
inline void set_time(time_t t) { time_after_lock=start_time=user_time=t; }
|
2000-09-16 03:27:21 +02:00
|
|
|
inline void lock_time() { time(&time_after_lock); }
|
2000-07-31 21:29:14 +02:00
|
|
|
inline void insert_id(ulonglong id)
|
|
|
|
{ last_insert_id=id; insert_id_used=1; }
|
|
|
|
inline ulonglong insert_id(void)
|
|
|
|
{
|
|
|
|
if (!last_insert_id_used)
|
|
|
|
{
|
|
|
|
last_insert_id_used=1;
|
|
|
|
current_insert_id=last_insert_id;
|
|
|
|
}
|
|
|
|
return last_insert_id;
|
|
|
|
}
|
2001-04-11 23:54:35 +02:00
|
|
|
inline ulonglong found_rows(void)
|
|
|
|
{
|
|
|
|
return limit_found_rows;
|
|
|
|
}
|
2000-11-24 00:51:18 +01:00
|
|
|
inline bool active_transaction()
|
|
|
|
{
|
2001-01-03 05:46:33 +01:00
|
|
|
#ifdef USING_TRANSACTIONS
|
2000-11-24 00:51:18 +01:00
|
|
|
return (transaction.all.bdb_tid != 0 ||
|
2001-10-10 01:50:28 +02:00
|
|
|
transaction.all.innodb_active_trans != 0);
|
2001-01-03 05:46:33 +01:00
|
|
|
#else
|
|
|
|
return 0;
|
|
|
|
#endif
|
2000-11-24 00:51:18 +01:00
|
|
|
}
|
2000-08-21 02:00:52 +02:00
|
|
|
inline gptr alloc(unsigned int size) { return alloc_root(&mem_root,size); }
|
|
|
|
inline gptr calloc(unsigned int size)
|
|
|
|
{
|
|
|
|
gptr ptr;
|
|
|
|
if ((ptr=alloc_root(&mem_root,size)))
|
|
|
|
bzero((char*) ptr,size);
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
inline char *strdup(const char *str)
|
|
|
|
{ return strdup_root(&mem_root,str); }
|
2001-10-17 18:39:39 +02:00
|
|
|
inline char *strmake(const char *str, uint size)
|
|
|
|
{ return strmake_root(&mem_root,str,size); }
|
|
|
|
inline char *memdup(const char *str, uint size)
|
2000-08-21 02:00:52 +02:00
|
|
|
{ return memdup_root(&mem_root,str,size); }
|
2001-12-21 06:00:58 +01:00
|
|
|
inline char *memdup_w_gap(const char *str, uint size, uint gap)
|
|
|
|
{
|
|
|
|
gptr ptr;
|
|
|
|
if ((ptr=alloc_root(&mem_root,size+gap)))
|
|
|
|
memcpy(ptr,str,size);
|
|
|
|
return ptr;
|
|
|
|
}
|
2002-03-15 22:57:31 +01:00
|
|
|
inline gptr trans_alloc(unsigned int size)
|
|
|
|
{
|
|
|
|
return alloc_root(&transaction.mem_root,size);
|
|
|
|
}
|
|
|
|
void add_changed_table(TABLE *table);
|
2002-09-19 09:36:19 +02:00
|
|
|
void add_changed_table(const char *key, long key_length);
|
|
|
|
CHANGED_TABLE_LIST * changed_table_dup(const char *key, long key_length);
|
2002-09-26 22:08:22 +02:00
|
|
|
int send_explain_fields(select_result *result);
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
2002-03-16 09:36:27 +01:00
|
|
|
/*
|
|
|
|
Used to hold information about file and file structure in exchainge
|
|
|
|
via non-DB file (...INTO OUTFILE..., ...LOAD DATA...)
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
class sql_exchange :public Sql_alloc
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
char *file_name;
|
|
|
|
String *field_term,*enclosed,*line_term,*line_start,*escaped;
|
|
|
|
bool opt_enclosed;
|
|
|
|
bool dumpfile;
|
|
|
|
uint skip_lines;
|
|
|
|
sql_exchange(char *name,bool dumpfile_flag);
|
|
|
|
~sql_exchange() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
#include "log_event.h"
|
|
|
|
|
|
|
|
/*
|
2002-06-02 20:22:20 +02:00
|
|
|
This is used to get result from a select
|
2000-07-31 21:29:14 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-01 12:20:53 +02:00
|
|
|
class JOIN;
|
|
|
|
|
2002-10-02 12:33:08 +02:00
|
|
|
void send_error(THD *thd, uint sql_errno=0, const char *err=0);
|
2001-08-02 05:29:50 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class select_result :public Sql_alloc {
|
|
|
|
protected:
|
|
|
|
THD *thd;
|
2002-05-08 22:14:40 +02:00
|
|
|
SELECT_LEX_UNIT *unit;
|
2000-07-31 21:29:14 +02:00
|
|
|
public:
|
|
|
|
select_result();
|
|
|
|
virtual ~select_result() {};
|
2002-05-08 22:14:40 +02:00
|
|
|
virtual int prepare(List<Item> &list, SELECT_LEX_UNIT *u)
|
|
|
|
{
|
|
|
|
unit= u;
|
|
|
|
return 0;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
virtual bool send_fields(List<Item> &list,uint flag)=0;
|
|
|
|
virtual bool send_data(List<Item> &items)=0;
|
2001-07-11 13:06:41 +02:00
|
|
|
virtual void initialize_tables (JOIN *join=0) {}
|
2001-08-02 05:29:50 +02:00
|
|
|
virtual void send_error(uint errcode,const char *err)
|
|
|
|
{
|
2002-09-03 08:50:36 +02:00
|
|
|
my_message(errcode, err, MYF(0));
|
2001-08-02 05:29:50 +02:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
virtual bool send_eof()=0;
|
|
|
|
virtual void abort() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class select_send :public select_result {
|
|
|
|
public:
|
|
|
|
select_send() {}
|
|
|
|
bool send_fields(List<Item> &list,uint flag);
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
bool send_eof();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class select_export :public select_result {
|
|
|
|
sql_exchange *exchange;
|
|
|
|
File file;
|
|
|
|
IO_CACHE cache;
|
|
|
|
ha_rows row_count;
|
|
|
|
uint field_term_length;
|
|
|
|
int field_sep_char,escape_char,line_sep_char;
|
|
|
|
bool fixed_row_size;
|
|
|
|
public:
|
|
|
|
select_export(sql_exchange *ex) :exchange(ex),file(-1),row_count(0L) {}
|
|
|
|
~select_export();
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2000-07-31 21:29:14 +02:00
|
|
|
bool send_fields(List<Item> &list,
|
|
|
|
uint flag) { return 0; }
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
void send_error(uint errcode,const char *err);
|
|
|
|
bool send_eof();
|
|
|
|
};
|
|
|
|
|
2001-08-02 05:29:50 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class select_dump :public select_result {
|
|
|
|
sql_exchange *exchange;
|
|
|
|
File file;
|
|
|
|
IO_CACHE cache;
|
|
|
|
ha_rows row_count;
|
|
|
|
char path[FN_REFLEN];
|
|
|
|
public:
|
|
|
|
select_dump(sql_exchange *ex) :exchange(ex),file(-1),row_count(0L)
|
|
|
|
{ path[0]=0; }
|
|
|
|
~select_dump();
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2000-07-31 21:29:14 +02:00
|
|
|
bool send_fields(List<Item> &list,
|
|
|
|
uint flag) { return 0; }
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
void send_error(uint errcode,const char *err);
|
|
|
|
bool send_eof();
|
|
|
|
};
|
2001-08-02 05:29:50 +02:00
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class select_insert :public select_result {
|
2001-07-11 13:06:41 +02:00
|
|
|
public:
|
2000-07-31 21:29:14 +02:00
|
|
|
TABLE *table;
|
|
|
|
List<Item> *fields;
|
|
|
|
ulonglong last_insert_id;
|
|
|
|
COPY_INFO info;
|
2001-08-02 05:29:50 +02:00
|
|
|
uint save_time_stamp;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2001-08-02 05:29:50 +02:00
|
|
|
select_insert(TABLE *table_par,List<Item> *fields_par,enum_duplicates duplic)
|
|
|
|
:table(table_par),fields(fields_par), last_insert_id(0), save_time_stamp(0) {
|
|
|
|
bzero((char*) &info,sizeof(info));
|
|
|
|
info.handle_duplicates=duplic;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
~select_insert();
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2001-08-02 05:29:50 +02:00
|
|
|
bool send_fields(List<Item> &list, uint flag)
|
|
|
|
{ return 0; }
|
2000-07-31 21:29:14 +02:00
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
void send_error(uint errcode,const char *err);
|
|
|
|
bool send_eof();
|
|
|
|
};
|
|
|
|
|
2001-08-02 05:29:50 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class select_create: public select_insert {
|
|
|
|
ORDER *group;
|
|
|
|
const char *db;
|
|
|
|
const char *name;
|
|
|
|
List<create_field> *extra_fields;
|
|
|
|
List<Key> *keys;
|
|
|
|
HA_CREATE_INFO *create_info;
|
|
|
|
MYSQL_LOCK *lock;
|
|
|
|
Field **field;
|
|
|
|
public:
|
|
|
|
select_create (const char *db_name, const char *table_name,
|
|
|
|
HA_CREATE_INFO *create_info_par,
|
|
|
|
List<create_field> &fields_par,
|
|
|
|
List<Key> &keys_par,
|
2001-08-02 05:29:50 +02:00
|
|
|
List<Item> &select_fields,enum_duplicates duplic)
|
|
|
|
:select_insert (NULL, &select_fields, duplic), db(db_name),
|
2000-07-31 21:29:14 +02:00
|
|
|
name(table_name), extra_fields(&fields_par),keys(&keys_par),
|
|
|
|
create_info(create_info_par),
|
|
|
|
lock(0)
|
|
|
|
{}
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2000-07-31 21:29:14 +02:00
|
|
|
bool send_data(List<Item> &values);
|
|
|
|
bool send_eof();
|
|
|
|
void abort();
|
|
|
|
};
|
|
|
|
|
2001-08-02 05:29:50 +02:00
|
|
|
class select_union :public select_result {
|
|
|
|
public:
|
|
|
|
TABLE *table;
|
|
|
|
COPY_INFO info;
|
|
|
|
uint save_time_stamp;
|
2002-05-24 14:26:47 +02:00
|
|
|
TMP_TABLE_PARAM *tmp_table_param;
|
2001-08-02 05:29:50 +02:00
|
|
|
|
|
|
|
select_union(TABLE *table_par);
|
|
|
|
~select_union();
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2001-08-02 05:29:50 +02:00
|
|
|
bool send_fields(List<Item> &list, uint flag)
|
|
|
|
{ return 0; }
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
bool send_eof();
|
|
|
|
bool flush();
|
|
|
|
};
|
|
|
|
|
2002-06-19 16:52:44 +02:00
|
|
|
/* Base subselect interface class */
|
2002-05-12 22:46:42 +02:00
|
|
|
class select_subselect :public select_result
|
|
|
|
{
|
2002-06-19 16:52:44 +02:00
|
|
|
protected:
|
2002-05-12 22:46:42 +02:00
|
|
|
Item_subselect *item;
|
|
|
|
public:
|
|
|
|
select_subselect(Item_subselect *item);
|
|
|
|
bool send_fields(List<Item> &list, uint flag) { return 0; };
|
2002-06-19 16:52:44 +02:00
|
|
|
bool send_data(List<Item> &items)=0;
|
2002-05-12 22:46:42 +02:00
|
|
|
bool send_eof() { return 0; };
|
2002-06-19 16:52:44 +02:00
|
|
|
|
2002-05-12 22:46:42 +02:00
|
|
|
friend class Ttem_subselect;
|
|
|
|
};
|
|
|
|
|
2002-06-19 16:52:44 +02:00
|
|
|
/* Single value subselect interface class */
|
|
|
|
class select_singleval_subselect :public select_subselect
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
select_singleval_subselect(Item_subselect *item):select_subselect(item){}
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
};
|
|
|
|
|
|
|
|
/* EXISTS subselect interface class */
|
|
|
|
class select_exists_subselect :public select_subselect
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
select_exists_subselect(Item_subselect *item):select_subselect(item){}
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
};
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Structs used when sorting */
|
|
|
|
|
|
|
|
typedef struct st_sort_field {
|
|
|
|
Field *field; /* Field to sort */
|
|
|
|
Item *item; /* Item if not sorting fields */
|
|
|
|
uint length; /* Length of sort field */
|
|
|
|
my_bool reverse; /* if descending sort */
|
|
|
|
Item_result result_type; /* Type of item */
|
|
|
|
} SORT_FIELD;
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct st_sort_buffer {
|
|
|
|
uint index; /* 0 or 1 */
|
|
|
|
uint sort_orders;
|
|
|
|
uint change_pos; /* If sort-fields changed */
|
|
|
|
char **buff;
|
|
|
|
SORT_FIELD *sortorder;
|
|
|
|
} SORT_BUFFER;
|
|
|
|
|
|
|
|
|
|
|
|
/* Structure for db & table in sql_yacc */
|
|
|
|
|
|
|
|
class Table_ident :public Sql_alloc {
|
|
|
|
public:
|
|
|
|
LEX_STRING db;
|
|
|
|
LEX_STRING table;
|
2002-05-06 23:04:16 +02:00
|
|
|
SELECT_LEX_UNIT *sel;
|
|
|
|
inline Table_ident(LEX_STRING db_arg, LEX_STRING table_arg, bool force)
|
|
|
|
:table(table_arg), sel((SELECT_LEX_UNIT *)0)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (!force && (current_thd->client_capabilities & CLIENT_NO_SCHEMA))
|
|
|
|
db.str=0;
|
|
|
|
else
|
|
|
|
db= db_arg;
|
|
|
|
}
|
2002-05-06 23:04:16 +02:00
|
|
|
inline Table_ident(LEX_STRING table_arg)
|
|
|
|
:table(table_arg), sel((SELECT_LEX_UNIT *)0)
|
|
|
|
{
|
|
|
|
db.str=0;
|
|
|
|
}
|
|
|
|
inline Table_ident(SELECT_LEX_UNIT *s) : sel(s)
|
|
|
|
{
|
|
|
|
db.str=0; table.str=(char *)""; table.length=0;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
inline void change_db(char *db_name)
|
2002-05-06 23:04:16 +02:00
|
|
|
{
|
|
|
|
db.str= db_name; db.length= (uint) strlen(db_name);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// this is needed for user_vars hash
|
|
|
|
class user_var_entry
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
LEX_STRING name;
|
|
|
|
char *value;
|
2001-03-15 10:55:44 +01:00
|
|
|
ulong length, update_query_id;
|
2000-07-31 21:29:14 +02:00
|
|
|
Item_result type;
|
2002-10-01 14:23:28 +02:00
|
|
|
CHARSET_INFO *var_charset;
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
2001-05-23 22:47:08 +02:00
|
|
|
/* Class for unique (removing of duplicates) */
|
|
|
|
|
|
|
|
class Unique :public Sql_alloc
|
|
|
|
{
|
|
|
|
DYNAMIC_ARRAY file_ptrs;
|
|
|
|
ulong max_elements, max_in_memory_size;
|
|
|
|
IO_CACHE file;
|
|
|
|
TREE tree;
|
2001-09-15 15:22:34 +02:00
|
|
|
byte *record_pointers;
|
2001-05-23 22:47:08 +02:00
|
|
|
bool flush();
|
|
|
|
|
|
|
|
public:
|
|
|
|
ulong elements;
|
2001-06-03 16:07:26 +02:00
|
|
|
Unique(qsort_cmp2 comp_func, void * comp_func_fixed_arg,
|
|
|
|
uint size, ulong max_in_memory_size_arg);
|
2001-05-23 22:47:08 +02:00
|
|
|
~Unique();
|
2001-09-07 21:38:45 +02:00
|
|
|
inline bool unique_add(gptr ptr)
|
2001-05-23 22:47:08 +02:00
|
|
|
{
|
|
|
|
if (tree.elements_in_tree > max_elements && flush())
|
|
|
|
return 1;
|
2002-04-25 10:36:55 +02:00
|
|
|
return !tree_insert(&tree, ptr, 0, tree.custom_arg);
|
2001-05-23 22:47:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool get(TABLE *table);
|
|
|
|
|
2001-06-03 16:07:26 +02:00
|
|
|
friend int unique_write_to_file(gptr key, element_count count, Unique *unique);
|
|
|
|
friend int unique_write_to_ptrs(gptr key, element_count count, Unique *unique);
|
2001-05-23 22:47:08 +02:00
|
|
|
};
|
2001-06-07 13:10:58 +02:00
|
|
|
|
|
|
|
class multi_delete : public select_result {
|
|
|
|
TABLE_LIST *delete_tables, *table_being_deleted;
|
|
|
|
#ifdef SINISAS_STRIP
|
|
|
|
IO_CACHE **tempfiles;
|
|
|
|
byte *memory_lane;
|
|
|
|
#else
|
|
|
|
Unique **tempfiles;
|
|
|
|
#endif
|
|
|
|
THD *thd;
|
|
|
|
ha_rows deleted;
|
2001-06-14 14:12:01 +02:00
|
|
|
uint num_of_tables;
|
|
|
|
int error;
|
2001-06-07 13:10:58 +02:00
|
|
|
thr_lock_type lock_option;
|
2002-01-12 18:51:10 +01:00
|
|
|
bool do_delete, not_trans_safe;
|
2001-06-07 13:10:58 +02:00
|
|
|
public:
|
2001-06-15 04:03:15 +02:00
|
|
|
multi_delete(THD *thd, TABLE_LIST *dt, thr_lock_type lock_option_arg,
|
|
|
|
uint num_of_tables);
|
2001-06-07 13:10:58 +02:00
|
|
|
~multi_delete();
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2001-06-07 13:10:58 +02:00
|
|
|
bool send_fields(List<Item> &list,
|
|
|
|
uint flag) { return 0; }
|
|
|
|
bool send_data(List<Item> &items);
|
2001-07-01 12:20:53 +02:00
|
|
|
void initialize_tables (JOIN *join);
|
2001-06-07 13:10:58 +02:00
|
|
|
void send_error(uint errcode,const char *err);
|
|
|
|
int do_deletes (bool from_send_error);
|
|
|
|
bool send_eof();
|
|
|
|
};
|
|
|
|
|
2001-12-26 15:49:10 +01:00
|
|
|
class multi_update : public select_result {
|
|
|
|
TABLE_LIST *update_tables, *table_being_updated;
|
|
|
|
COPY_INFO *infos;
|
|
|
|
TABLE **tmp_tables;
|
|
|
|
THD *thd;
|
|
|
|
ha_rows updated, found;
|
|
|
|
List<Item> fields;
|
|
|
|
List <Item> **fields_by_tables;
|
|
|
|
thr_lock_type lock_option;
|
|
|
|
enum enum_duplicates dupl;
|
|
|
|
uint num_of_tables, num_fields, num_updated, *save_time_stamps, *field_sequence;
|
|
|
|
int error;
|
2002-01-16 21:45:47 +01:00
|
|
|
bool do_update, not_trans_safe;
|
2001-12-26 15:49:10 +01:00
|
|
|
public:
|
|
|
|
multi_update(THD *thd_arg, TABLE_LIST *ut, List<Item> &fs,
|
|
|
|
enum enum_duplicates handle_duplicates,
|
|
|
|
thr_lock_type lock_option_arg, uint num);
|
|
|
|
~multi_update();
|
2002-05-08 22:14:40 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
2001-12-26 15:49:10 +01:00
|
|
|
bool send_fields(List<Item> &list,
|
|
|
|
uint flag) { return 0; }
|
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
void initialize_tables (JOIN *join);
|
|
|
|
void send_error(uint errcode,const char *err);
|
|
|
|
int do_updates (bool from_send_error);
|
|
|
|
bool send_eof();
|
|
|
|
};
|
|
|
|
|
2002-10-11 20:49:10 +02:00
|
|
|
class select_dumpvar :public select_result {
|
|
|
|
ha_rows row_count;
|
|
|
|
public:
|
2002-10-16 15:55:08 +02:00
|
|
|
List<LEX_STRING> var_list;
|
|
|
|
List<Item_func_set_user_var> vars;
|
|
|
|
select_dumpvar(void) { var_list.empty(); vars.empty(); row_count=0;}
|
2002-10-11 20:49:10 +02:00
|
|
|
~select_dumpvar() {}
|
2002-10-16 15:55:08 +02:00
|
|
|
int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
|
|
|
|
bool send_fields(List<Item> &list, uint flag) {return 0;}
|
2002-10-11 20:49:10 +02:00
|
|
|
bool send_data(List<Item> &items);
|
|
|
|
bool send_eof();
|
|
|
|
};
|