2003-06-23 20:03:59 +03:00
|
|
|
/* Copyright (C) 2000-2003 MySQL AB
|
2001-12-06 14:10:51 +02: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
|
2006-12-23 20:17:15 +01:00
|
|
|
the Free Software Foundation; version 2 of the License.
|
2001-12-06 14:10:51 +02: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 14:10:51 +02: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 */
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
The privileges are saved in the following tables:
|
2004-10-07 01:45:06 +03:00
|
|
|
mysql/user ; super user who are allowed to do almost anything
|
|
|
|
mysql/host ; host privileges. This is used if host is empty in mysql/db.
|
2000-07-31 21:29:14 +02:00
|
|
|
mysql/db ; database privileges / user
|
|
|
|
|
|
|
|
data in tables is sorted according to how many not-wild-cards there is
|
|
|
|
in the relevant fields. Empty strings comes last.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "mysql_priv.h"
|
|
|
|
#include "hash_filo.h"
|
|
|
|
#include <m_ctype.h>
|
|
|
|
#include <stdarg.h>
|
2004-12-23 10:46:24 +00:00
|
|
|
#include "sp_head.h"
|
|
|
|
#include "sp.h"
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2006-02-14 16:20:48 +01:00
|
|
|
time_t mysql_db_table_last_check= 0L;
|
|
|
|
|
|
|
|
TABLE_FIELD_W_TYPE mysql_db_table_fields[MYSQL_DB_FIELD_COUNT] = {
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Host") },
|
|
|
|
{ C_STRING_WITH_LEN("char(60)") },
|
2006-02-14 16:20:48 +01:00
|
|
|
{NULL, 0}
|
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Db") },
|
|
|
|
{ C_STRING_WITH_LEN("char(64)") },
|
2006-02-14 16:20:48 +01:00
|
|
|
{NULL, 0}
|
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("User") },
|
|
|
|
{ C_STRING_WITH_LEN("char(16)") },
|
2006-02-14 16:20:48 +01:00
|
|
|
{NULL, 0}
|
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Select_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Insert_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Update_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Delete_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Create_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Drop_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Grant_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("References_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Index_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Alter_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Create_tmp_table_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Lock_tables_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Create_view_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Show_view_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Create_routine_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Alter_routine_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Execute_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Event_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
},
|
|
|
|
{
|
2006-08-17 18:13:45 +02:00
|
|
|
{ C_STRING_WITH_LEN("Trigger_priv") },
|
|
|
|
{ C_STRING_WITH_LEN("enum('N','Y')") },
|
|
|
|
{ C_STRING_WITH_LEN("utf8") }
|
2006-02-14 16:20:48 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-02-14 22:21:23 +01:00
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
|
|
|
|
#define FIRST_NON_YN_FIELD 26
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
class acl_entry :public hash_filo_element
|
|
|
|
{
|
|
|
|
public:
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong access;
|
2000-07-31 21:29:14 +02:00
|
|
|
uint16 length;
|
|
|
|
char key[1]; // Key will be stored here
|
|
|
|
};
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
static byte* acl_entry_get_key(acl_entry *entry,uint *length,
|
|
|
|
my_bool not_used __attribute__((unused)))
|
|
|
|
{
|
|
|
|
*length=(uint) entry->length;
|
|
|
|
return (byte*) entry->key;
|
|
|
|
}
|
|
|
|
|
2003-12-21 21:18:59 +01:00
|
|
|
#define IP_ADDR_STRLEN (3+1+3+1+3+1+3)
|
2006-09-27 19:21:29 +05:00
|
|
|
#define ACL_KEY_LENGTH (IP_ADDR_STRLEN+1+NAME_LEN+1+USERNAME_LENGTH+1)
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
static DYNAMIC_ARRAY acl_hosts,acl_users,acl_dbs;
|
|
|
|
static MEM_ROOT mem, memex;
|
|
|
|
static bool initialized=0;
|
|
|
|
static bool allow_all_hosts=1;
|
2005-05-17 19:54:20 +01:00
|
|
|
static HASH acl_check_hosts, column_priv_hash, proc_priv_hash, func_priv_hash;
|
2000-07-31 21:29:14 +02:00
|
|
|
static DYNAMIC_ARRAY acl_wild_hosts;
|
|
|
|
static hash_filo *acl_cache;
|
2005-09-01 23:25:29 +04:00
|
|
|
static uint grant_version=0; /* Version of priv tables. incremented by acl_load */
|
2004-07-19 16:01:53 +02:00
|
|
|
static ulong get_access(TABLE *form,uint fieldnr, uint *next_field=0);
|
2000-07-31 21:29:14 +02:00
|
|
|
static int acl_compare(ACL_ACCESS *a,ACL_ACCESS *b);
|
|
|
|
static ulong get_sort(uint count,...);
|
|
|
|
static void init_check_host(void);
|
2005-12-28 14:43:50 +01:00
|
|
|
static void rebuild_check_host(void);
|
2005-08-22 15:48:50 -07:00
|
|
|
static ACL_USER *find_acl_user(const char *host, const char *user,
|
|
|
|
my_bool exact);
|
2005-09-01 16:52:59 +04:00
|
|
|
static bool update_user_table(THD *thd, TABLE *table,
|
|
|
|
const char *host, const char *user,
|
2003-07-01 23:40:59 +04:00
|
|
|
const char *new_password, uint new_password_len);
|
2000-07-31 21:29:14 +02:00
|
|
|
static void update_hostname(acl_host_and_ip *host, const char *hostname);
|
2003-04-29 00:15:18 +02:00
|
|
|
static bool compare_hostname(const acl_host_and_ip *host,const char *hostname,
|
2000-07-31 21:29:14 +02:00
|
|
|
const char *ip);
|
2005-09-01 16:52:59 +04:00
|
|
|
static my_bool acl_load(THD *thd, TABLE_LIST *tables);
|
|
|
|
static my_bool grant_load(TABLE_LIST *tables);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
/*
|
|
|
|
Convert scrambled password to binary form, according to scramble type,
|
|
|
|
Binary form is stored in user.salt.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static
|
|
|
|
void
|
|
|
|
set_user_salt(ACL_USER *acl_user, const char *password, uint password_len)
|
|
|
|
{
|
|
|
|
if (password_len == SCRAMBLED_PASSWORD_CHAR_LENGTH)
|
|
|
|
{
|
|
|
|
get_salt_from_password(acl_user->salt, password);
|
|
|
|
acl_user->salt_len= SCRAMBLE_LENGTH;
|
|
|
|
}
|
2003-07-08 02:36:14 +04:00
|
|
|
else if (password_len == SCRAMBLED_PASSWORD_CHAR_LENGTH_323)
|
2003-07-01 23:40:59 +04:00
|
|
|
{
|
|
|
|
get_salt_from_password_323((ulong *) acl_user->salt, password);
|
2003-07-08 02:36:14 +04:00
|
|
|
acl_user->salt_len= SCRAMBLE_LENGTH_323;
|
2003-07-01 23:40:59 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
acl_user->salt_len= 0;
|
|
|
|
}
|
|
|
|
|
2003-07-08 02:36:14 +04:00
|
|
|
/*
|
|
|
|
This after_update function is used when user.password is less than
|
|
|
|
SCRAMBLE_LENGTH bytes.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void restrict_update_of_old_passwords_var(THD *thd,
|
|
|
|
enum_var_type var_type)
|
|
|
|
{
|
|
|
|
if (var_type == OPT_GLOBAL)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&LOCK_global_system_variables);
|
|
|
|
global_system_variables.old_passwords= 1;
|
|
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
thd->variables.old_passwords= 1;
|
|
|
|
}
|
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
|
2002-09-05 16:17:08 +03:00
|
|
|
/*
|
2005-09-01 16:52:59 +04:00
|
|
|
Initialize structures responsible for user/db-level privilege checking and
|
|
|
|
load privilege information for them from tables in the 'mysql' database.
|
2002-09-05 16:17:08 +03:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
acl_init()
|
2005-09-01 16:52:59 +04:00
|
|
|
dont_read_acl_tables TRUE if we want to skip loading data from
|
|
|
|
privilege tables and disable privilege checking.
|
|
|
|
|
|
|
|
NOTES
|
|
|
|
This function is mostly responsible for preparatory steps, main work
|
|
|
|
on initialization and grants loading is done in acl_reload().
|
2002-09-05 16:17:08 +03:00
|
|
|
|
|
|
|
RETURN VALUES
|
|
|
|
0 ok
|
|
|
|
1 Could not initialize grant's
|
|
|
|
*/
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool acl_init(bool dont_read_acl_tables)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-10-29 21:59:03 +02:00
|
|
|
THD *thd;
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool return_val;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("acl_init");
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
acl_cache= new hash_filo(ACL_CACHE_SIZE, 0, 0,
|
|
|
|
(hash_get_key) acl_entry_get_key,
|
2006-02-27 16:41:58 +01:00
|
|
|
(hash_free_key) free,
|
2006-05-04 11:55:09 +05:00
|
|
|
lower_case_file_system ?
|
|
|
|
system_charset_info : &my_charset_bin);
|
2000-07-31 21:29:14 +02:00
|
|
|
if (dont_read_acl_tables)
|
2002-10-19 23:18:31 +04:00
|
|
|
{
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(0); /* purecov: tested */
|
2002-11-30 20:58:53 +03:00
|
|
|
}
|
|
|
|
|
2002-09-05 16:17:08 +03:00
|
|
|
/*
|
|
|
|
To be able to run this from boot, we allocate a temporary THD
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!(thd=new THD))
|
|
|
|
DBUG_RETURN(1); /* purecov: inspected */
|
2005-11-23 21:18:10 +03:00
|
|
|
thd->thread_stack= (char*) &thd;
|
2002-09-05 16:17:08 +03:00
|
|
|
thd->store_globals();
|
2005-09-01 16:52:59 +04:00
|
|
|
/*
|
|
|
|
It is safe to call acl_reload() since acl_* arrays and hashes which
|
|
|
|
will be freed there are global static objects and thus are initialized
|
|
|
|
by zeros at startup.
|
|
|
|
*/
|
|
|
|
return_val= acl_reload(thd);
|
|
|
|
delete thd;
|
|
|
|
/* Remember that we don't have a THD */
|
|
|
|
my_pthread_setspecific_ptr(THR_THD, 0);
|
|
|
|
DBUG_RETURN(return_val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Initialize structures responsible for user/db-level privilege checking
|
|
|
|
and load information about grants from open privilege tables.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
acl_load()
|
|
|
|
thd Current thread
|
|
|
|
tables List containing open "mysql.host", "mysql.user" and
|
|
|
|
"mysql.db" tables.
|
|
|
|
|
|
|
|
RETURN VALUES
|
|
|
|
FALSE Success
|
|
|
|
TRUE Error
|
|
|
|
*/
|
|
|
|
|
|
|
|
static my_bool acl_load(THD *thd, TABLE_LIST *tables)
|
|
|
|
{
|
|
|
|
TABLE *table;
|
|
|
|
READ_RECORD read_record_info;
|
|
|
|
my_bool return_val= 1;
|
|
|
|
bool check_no_resolve= specialflag & SPECIAL_NO_RESOLVE;
|
2006-09-27 19:21:29 +05:00
|
|
|
char tmp_name[NAME_LEN+1];
|
2005-09-20 16:57:58 -07:00
|
|
|
int password_length;
|
2005-09-01 16:52:59 +04:00
|
|
|
DBUG_ENTER("acl_load");
|
|
|
|
|
2005-09-01 23:25:29 +04:00
|
|
|
grant_version++; /* Privileges updated */
|
2002-09-05 16:17:08 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_cache->clear(1); // Clear locked hostname cache
|
|
|
|
|
2003-10-11 22:00:24 +03:00
|
|
|
init_sql_alloc(&mem, ACL_ALLOC_BLOCK_SIZE, 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
init_read_record(&read_record_info,thd,table= tables[0].table,NULL,1,0);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2002-04-28 21:22:37 +00:00
|
|
|
VOID(my_init_dynamic_array(&acl_hosts,sizeof(ACL_HOST),20,50));
|
2000-07-31 21:29:14 +02:00
|
|
|
while (!(read_record_info.read_record(&read_record_info)))
|
|
|
|
{
|
|
|
|
ACL_HOST host;
|
2003-02-12 21:55:37 +02:00
|
|
|
update_hostname(&host.host,get_field(&mem, table->field[0]));
|
|
|
|
host.db= get_field(&mem, table->field[1]);
|
2005-05-26 14:02:18 -05:00
|
|
|
if (lower_case_table_names && host.db)
|
2005-02-15 13:36:46 -08:00
|
|
|
{
|
|
|
|
/*
|
2005-02-21 14:47:57 +02:00
|
|
|
convert db to lower case and give a warning if the db wasn't
|
|
|
|
already in lower case
|
2005-02-15 13:36:46 -08:00
|
|
|
*/
|
2005-02-21 14:47:57 +02:00
|
|
|
(void) strmov(tmp_name, host.db);
|
|
|
|
my_casedn_str(files_charset_info, host.db);
|
2005-02-15 13:36:46 -08:00
|
|
|
if (strcmp(host.db, tmp_name) != 0)
|
|
|
|
sql_print_warning("'host' entry '%s|%s' had database in mixed "
|
|
|
|
"case that has been forced to lowercase because "
|
2005-02-18 12:47:33 -08:00
|
|
|
"lower_case_table_names is set. It will not be "
|
|
|
|
"possible to remove this privilege using REVOKE.",
|
2006-06-06 14:21:07 +03:00
|
|
|
host.host.hostname ? host.host.hostname : "",
|
|
|
|
host.db ? host.db : "");
|
2005-02-15 13:36:46 -08:00
|
|
|
}
|
2002-06-12 15:04:18 +03:00
|
|
|
host.access= get_access(table,2);
|
|
|
|
host.access= fix_rights_for_db(host.access);
|
2003-06-23 20:03:59 +03:00
|
|
|
host.sort= get_sort(2,host.host.hostname,host.db);
|
2003-11-20 12:55:48 +04:00
|
|
|
if (check_no_resolve && hostname_requires_resolving(host.host.hostname))
|
|
|
|
{
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_warning("'host' entry '%s|%s' "
|
2003-11-26 13:38:32 +04:00
|
|
|
"ignored in --skip-name-resolve mode.",
|
2006-06-06 14:21:07 +03:00
|
|
|
host.host.hostname ? host.host.hostname : "",
|
|
|
|
host.db ? host.db : "");
|
2003-11-20 12:55:48 +04:00
|
|
|
continue;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
#ifndef TO_BE_REMOVED
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields == 8)
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // Without grant
|
|
|
|
if (host.access & CREATE_ACL)
|
2002-09-16 15:55:19 +03:00
|
|
|
host.access|=REFERENCES_ACL | INDEX_ACL | ALTER_ACL | CREATE_TMP_ACL;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
VOID(push_dynamic(&acl_hosts,(gptr) &host));
|
|
|
|
}
|
|
|
|
qsort((gptr) dynamic_element(&acl_hosts,0,ACL_HOST*),acl_hosts.elements,
|
|
|
|
sizeof(ACL_HOST),(qsort_cmp) acl_compare);
|
|
|
|
end_read_record(&read_record_info);
|
|
|
|
freeze_size(&acl_hosts);
|
|
|
|
|
|
|
|
init_read_record(&read_record_info,thd,table=tables[1].table,NULL,1,0);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2002-04-28 21:22:37 +00:00
|
|
|
VOID(my_init_dynamic_array(&acl_users,sizeof(ACL_USER),50,100));
|
2005-09-20 16:57:58 -07:00
|
|
|
password_length= table->field[2]->field_length /
|
|
|
|
table->field[2]->charset()->mbmaxlen;
|
|
|
|
if (password_length < SCRAMBLED_PASSWORD_CHAR_LENGTH_323)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-07-08 02:36:14 +04:00
|
|
|
sql_print_error("Fatal error: mysql.user table is damaged or in "
|
|
|
|
"unsupported 3.20 format.");
|
|
|
|
goto end;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2003-01-18 16:39:21 +02:00
|
|
|
DBUG_PRINT("info",("user table fields: %d, password length: %d",
|
2005-10-19 13:47:05 -07:00
|
|
|
table->s->fields, password_length));
|
2005-03-22 15:54:18 +01:00
|
|
|
|
2003-07-08 02:36:14 +04:00
|
|
|
pthread_mutex_lock(&LOCK_global_system_variables);
|
2005-09-20 16:57:58 -07:00
|
|
|
if (password_length < SCRAMBLED_PASSWORD_CHAR_LENGTH)
|
2003-01-18 16:39:21 +02:00
|
|
|
{
|
2003-07-08 02:36:14 +04:00
|
|
|
if (opt_secure_auth)
|
|
|
|
{
|
|
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
|
|
|
sql_print_error("Fatal error: mysql.user table is in old format, "
|
|
|
|
"but server started with --secure-auth option.");
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
sys_old_passwords.after_update= restrict_update_of_old_passwords_var;
|
|
|
|
if (global_system_variables.old_passwords)
|
|
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
global_system_variables.old_passwords= 1;
|
|
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
2004-08-30 08:37:36 +02:00
|
|
|
sql_print_warning("mysql.user table is not updated to new password format; "
|
|
|
|
"Disabling new password usage until "
|
|
|
|
"mysql_fix_privilege_tables is run");
|
2003-07-08 02:36:14 +04:00
|
|
|
}
|
|
|
|
thd->variables.old_passwords= 1;
|
|
|
|
}
|
|
|
|
else
|
2003-01-18 16:39:21 +02:00
|
|
|
{
|
2003-07-08 02:36:14 +04:00
|
|
|
sys_old_passwords.after_update= 0;
|
|
|
|
pthread_mutex_unlock(&LOCK_global_system_variables);
|
2003-01-18 16:39:21 +02:00
|
|
|
}
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
allow_all_hosts=0;
|
|
|
|
while (!(read_record_info.read_record(&read_record_info)))
|
|
|
|
{
|
|
|
|
ACL_USER user;
|
2003-07-01 23:40:59 +04:00
|
|
|
update_hostname(&user.host, get_field(&mem, table->field[0]));
|
|
|
|
user.user= get_field(&mem, table->field[1]);
|
2003-11-20 12:55:48 +04:00
|
|
|
if (check_no_resolve && hostname_requires_resolving(user.host.hostname))
|
|
|
|
{
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_warning("'user' entry '%s@%s' "
|
|
|
|
"ignored in --skip-name-resolve mode.",
|
2006-06-06 14:21:07 +03:00
|
|
|
user.user ? user.user : "",
|
|
|
|
user.host.hostname ? user.host.hostname : "");
|
2003-11-20 12:55:48 +04:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
const char *password= get_field(&mem, table->field[2]);
|
|
|
|
uint password_len= password ? strlen(password) : 0;
|
|
|
|
set_user_salt(&user, password, password_len);
|
|
|
|
if (user.salt_len == 0 && password_len != 0)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-07-01 23:40:59 +04:00
|
|
|
switch (password_len) {
|
|
|
|
case 45: /* 4.1: to be removed */
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_warning("Found 4.1 style password for user '%s@%s'. "
|
|
|
|
"Ignoring user. "
|
|
|
|
"You should change password for this user.",
|
|
|
|
user.user ? user.user : "",
|
|
|
|
user.host.hostname ? user.host.hostname : "");
|
2003-07-01 23:40:59 +04:00
|
|
|
break;
|
|
|
|
default:
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_warning("Found invalid password for user: '%s@%s'; "
|
|
|
|
"Ignoring user", user.user ? user.user : "",
|
|
|
|
user.host.hostname ? user.host.hostname : "");
|
2003-07-01 23:40:59 +04:00
|
|
|
break;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
else // password is correct
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-07-19 16:01:53 +02:00
|
|
|
uint next_field;
|
|
|
|
user.access= get_access(table,3,&next_field) & GLOBAL_ACLS;
|
2004-10-10 18:51:02 +03:00
|
|
|
/*
|
|
|
|
if it is pre 5.0.1 privilege table then map CREATE privilege on
|
|
|
|
CREATE VIEW & SHOW VIEW privileges
|
|
|
|
*/
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields <= 31 && (user.access & CREATE_ACL))
|
2004-10-10 18:51:02 +03:00
|
|
|
user.access|= (CREATE_VIEW_ACL | SHOW_VIEW_ACL);
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
if it is pre 5.0.2 privilege table then map CREATE/ALTER privilege on
|
|
|
|
CREATE PROCEDURE & ALTER PROCEDURE privileges
|
|
|
|
*/
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields <= 33 && (user.access & CREATE_ACL))
|
2004-12-23 10:46:24 +00:00
|
|
|
user.access|= CREATE_PROC_ACL;
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields <= 33 && (user.access & ALTER_ACL))
|
2004-12-23 10:46:24 +00:00
|
|
|
user.access|= ALTER_PROC_ACL;
|
|
|
|
|
2005-03-22 15:54:18 +01:00
|
|
|
/*
|
|
|
|
pre 5.0.3 did not have CREATE_USER_ACL
|
|
|
|
*/
|
|
|
|
if (table->s->fields <= 36 && (user.access & GRANT_ACL))
|
|
|
|
user.access|= CREATE_USER_ACL;
|
|
|
|
|
2005-12-05 11:45:04 +01:00
|
|
|
|
|
|
|
/*
|
2006-02-24 00:29:50 +01:00
|
|
|
if it is pre 5.1.6 privilege table then map CREATE privilege on
|
2005-12-05 11:45:04 +01:00
|
|
|
CREATE|ALTER|DROP|EXECUTE EVENT
|
|
|
|
*/
|
2006-02-24 00:29:50 +01:00
|
|
|
if (table->s->fields <= 37 && (user.access & SUPER_ACL))
|
2005-12-05 11:45:04 +01:00
|
|
|
user.access|= EVENT_ACL;
|
|
|
|
|
2006-02-01 13:28:45 +03:00
|
|
|
/*
|
|
|
|
if it is pre 5.1.6 privilege then map TRIGGER privilege on CREATE.
|
|
|
|
*/
|
|
|
|
if (table->s->fields <= 38 && (user.access & SUPER_ACL))
|
|
|
|
user.access|= TRIGGER_ACL;
|
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
user.sort= get_sort(2,user.host.hostname,user.user);
|
|
|
|
user.hostname_length= (user.host.hostname ?
|
|
|
|
(uint) strlen(user.host.hostname) : 0);
|
2004-07-16 01:15:55 +03:00
|
|
|
|
2005-01-06 13:00:13 +02:00
|
|
|
/* Starting from 4.0.2 we have more fields */
|
|
|
|
if (table->s->fields >= 31)
|
2002-11-24 17:07:53 +03:00
|
|
|
{
|
2004-07-19 16:01:53 +02:00
|
|
|
char *ssl_type=get_field(&mem, table->field[next_field++]);
|
2003-07-01 23:40:59 +04:00
|
|
|
if (!ssl_type)
|
|
|
|
user.ssl_type=SSL_TYPE_NONE;
|
|
|
|
else if (!strcmp(ssl_type, "ANY"))
|
|
|
|
user.ssl_type=SSL_TYPE_ANY;
|
|
|
|
else if (!strcmp(ssl_type, "X509"))
|
|
|
|
user.ssl_type=SSL_TYPE_X509;
|
|
|
|
else /* !strcmp(ssl_type, "SPECIFIED") */
|
|
|
|
user.ssl_type=SSL_TYPE_SPECIFIED;
|
|
|
|
|
2004-07-19 16:01:53 +02:00
|
|
|
user.ssl_cipher= get_field(&mem, table->field[next_field++]);
|
|
|
|
user.x509_issuer= get_field(&mem, table->field[next_field++]);
|
|
|
|
user.x509_subject= get_field(&mem, table->field[next_field++]);
|
2003-07-01 23:40:59 +04:00
|
|
|
|
2004-07-19 16:01:53 +02:00
|
|
|
char *ptr = get_field(&mem, table->field[next_field++]);
|
|
|
|
user.user_resource.questions=ptr ? atoi(ptr) : 0;
|
|
|
|
ptr = get_field(&mem, table->field[next_field++]);
|
|
|
|
user.user_resource.updates=ptr ? atoi(ptr) : 0;
|
|
|
|
ptr = get_field(&mem, table->field[next_field++]);
|
2004-12-29 20:30:37 +03:00
|
|
|
user.user_resource.conn_per_hour= ptr ? atoi(ptr) : 0;
|
2003-07-01 23:40:59 +04:00
|
|
|
if (user.user_resource.questions || user.user_resource.updates ||
|
2004-12-29 20:30:37 +03:00
|
|
|
user.user_resource.conn_per_hour)
|
2003-07-01 23:40:59 +04:00
|
|
|
mqh_used=1;
|
2004-12-29 20:30:37 +03:00
|
|
|
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields >= 36)
|
2004-12-29 20:30:37 +03:00
|
|
|
{
|
|
|
|
/* Starting from 5.0.3 we have max_user_connections field */
|
|
|
|
ptr= get_field(&mem, table->field[next_field++]);
|
|
|
|
user.user_resource.user_conn= ptr ? atoi(ptr) : 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
user.user_resource.user_conn= 0;
|
2002-11-24 17:07:53 +03:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
else
|
|
|
|
{
|
|
|
|
user.ssl_type=SSL_TYPE_NONE;
|
2003-09-03 14:12:10 +04:00
|
|
|
bzero((char *)&(user.user_resource),sizeof(user.user_resource));
|
2000-07-31 21:29:14 +02:00
|
|
|
#ifndef TO_BE_REMOVED
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields <= 13)
|
2003-07-01 23:40:59 +04:00
|
|
|
{ // Without grant
|
|
|
|
if (user.access & CREATE_ACL)
|
|
|
|
user.access|=REFERENCES_ACL | INDEX_ACL | ALTER_ACL;
|
|
|
|
}
|
|
|
|
/* Convert old privileges */
|
|
|
|
user.access|= LOCK_TABLES_ACL | CREATE_TMP_ACL | SHOW_DB_ACL;
|
|
|
|
if (user.access & FILE_ACL)
|
|
|
|
user.access|= REPL_CLIENT_ACL | REPL_SLAVE_ACL;
|
|
|
|
if (user.access & PROCESS_ACL)
|
|
|
|
user.access|= SUPER_ACL | EXECUTE_ACL;
|
2000-07-31 21:29:14 +02:00
|
|
|
#endif
|
2003-07-01 23:40:59 +04:00
|
|
|
}
|
|
|
|
VOID(push_dynamic(&acl_users,(gptr) &user));
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!user.host.hostname ||
|
|
|
|
(user.host.hostname[0] == wild_many && !user.host.hostname[1]))
|
2003-07-01 23:40:59 +04:00
|
|
|
allow_all_hosts=1; // Anyone can connect
|
2002-06-12 15:04:18 +03:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
qsort((gptr) dynamic_element(&acl_users,0,ACL_USER*),acl_users.elements,
|
|
|
|
sizeof(ACL_USER),(qsort_cmp) acl_compare);
|
|
|
|
end_read_record(&read_record_info);
|
|
|
|
freeze_size(&acl_users);
|
2002-11-30 20:58:53 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
init_read_record(&read_record_info,thd,table=tables[2].table,NULL,1,0);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2002-04-28 21:22:37 +00:00
|
|
|
VOID(my_init_dynamic_array(&acl_dbs,sizeof(ACL_DB),50,100));
|
2000-07-31 21:29:14 +02:00
|
|
|
while (!(read_record_info.read_record(&read_record_info)))
|
|
|
|
{
|
|
|
|
ACL_DB db;
|
2006-02-14 16:20:48 +01:00
|
|
|
update_hostname(&db.host,get_field(&mem, table->field[MYSQL_DB_FIELD_HOST]));
|
|
|
|
db.db=get_field(&mem, table->field[MYSQL_DB_FIELD_DB]);
|
2002-08-08 00:53:00 +03:00
|
|
|
if (!db.db)
|
|
|
|
{
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_warning("Found an entry in the 'db' table with empty database name; Skipped");
|
2002-08-07 21:57:32 +03:00
|
|
|
continue;
|
2002-08-08 00:53:00 +03:00
|
|
|
}
|
2006-02-14 16:20:48 +01:00
|
|
|
db.user=get_field(&mem, table->field[MYSQL_DB_FIELD_USER]);
|
2003-11-20 12:55:48 +04:00
|
|
|
if (check_no_resolve && hostname_requires_resolving(db.host.hostname))
|
|
|
|
{
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_warning("'db' entry '%s %s@%s' "
|
|
|
|
"ignored in --skip-name-resolve mode.",
|
2006-06-06 14:21:07 +03:00
|
|
|
db.db,
|
|
|
|
db.user ? db.user : "",
|
|
|
|
db.host.hostname ? db.host.hostname : "");
|
2003-11-20 12:55:48 +04:00
|
|
|
continue;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
db.access=get_access(table,3);
|
|
|
|
db.access=fix_rights_for_db(db.access);
|
2005-02-15 13:36:46 -08:00
|
|
|
if (lower_case_table_names)
|
|
|
|
{
|
|
|
|
/*
|
2005-02-21 14:47:57 +02:00
|
|
|
convert db to lower case and give a warning if the db wasn't
|
|
|
|
already in lower case
|
2005-02-15 13:36:46 -08:00
|
|
|
*/
|
|
|
|
(void)strmov(tmp_name, db.db);
|
2005-02-21 14:47:57 +02:00
|
|
|
my_casedn_str(files_charset_info, db.db);
|
2005-02-15 13:36:46 -08:00
|
|
|
if (strcmp(db.db, tmp_name) != 0)
|
|
|
|
{
|
|
|
|
sql_print_warning("'db' entry '%s %s@%s' had database in mixed "
|
|
|
|
"case that has been forced to lowercase because "
|
2005-02-18 12:47:33 -08:00
|
|
|
"lower_case_table_names is set. It will not be "
|
|
|
|
"possible to remove this privilege using REVOKE.",
|
2006-06-06 14:21:07 +03:00
|
|
|
db.db,
|
|
|
|
db.user ? db.user : "",
|
|
|
|
db.host.hostname ? db.host.hostname : "");
|
2005-02-15 13:36:46 -08:00
|
|
|
}
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
db.sort=get_sort(3,db.host.hostname,db.db,db.user);
|
|
|
|
#ifndef TO_BE_REMOVED
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields <= 9)
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // Without grant
|
|
|
|
if (db.access & CREATE_ACL)
|
|
|
|
db.access|=REFERENCES_ACL | INDEX_ACL | ALTER_ACL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
VOID(push_dynamic(&acl_dbs,(gptr) &db));
|
|
|
|
}
|
|
|
|
qsort((gptr) dynamic_element(&acl_dbs,0,ACL_DB*),acl_dbs.elements,
|
|
|
|
sizeof(ACL_DB),(qsort_cmp) acl_compare);
|
|
|
|
end_read_record(&read_record_info);
|
|
|
|
freeze_size(&acl_dbs);
|
|
|
|
init_check_host();
|
|
|
|
|
2002-09-05 16:17:08 +03:00
|
|
|
initialized=1;
|
|
|
|
return_val=0;
|
|
|
|
|
|
|
|
end:
|
|
|
|
DBUG_RETURN(return_val);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void acl_free(bool end)
|
|
|
|
{
|
2000-09-12 03:02:33 +03:00
|
|
|
free_root(&mem,MYF(0));
|
2000-07-31 21:29:14 +02:00
|
|
|
delete_dynamic(&acl_hosts);
|
|
|
|
delete_dynamic(&acl_users);
|
|
|
|
delete_dynamic(&acl_dbs);
|
|
|
|
delete_dynamic(&acl_wild_hosts);
|
|
|
|
hash_free(&acl_check_hosts);
|
|
|
|
if (!end)
|
|
|
|
acl_cache->clear(1); /* purecov: inspected */
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delete acl_cache;
|
|
|
|
acl_cache=0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
|
|
|
|
/*
|
2005-09-01 16:52:59 +04:00
|
|
|
Forget current user/db-level privileges and read new privileges
|
|
|
|
from the privilege tables.
|
2003-06-23 20:03:59 +03:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
acl_reload()
|
2005-09-01 16:52:59 +04:00
|
|
|
thd Current thread
|
|
|
|
|
|
|
|
NOTE
|
|
|
|
All tables of calling thread which were open and locked by LOCK TABLES
|
|
|
|
statement will be unlocked and closed.
|
|
|
|
This function is also used for initialization of structures responsible
|
|
|
|
for user/db-level privilege checking.
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
FALSE Success
|
|
|
|
TRUE Failure
|
2003-06-23 20:03:59 +03:00
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool acl_reload(THD *thd)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-09-01 16:52:59 +04:00
|
|
|
TABLE_LIST tables[3];
|
2000-07-31 21:29:14 +02:00
|
|
|
DYNAMIC_ARRAY old_acl_hosts,old_acl_users,old_acl_dbs;
|
|
|
|
MEM_ROOT old_mem;
|
|
|
|
bool old_initialized;
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool return_val= 1;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("acl_reload");
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
if (thd->locked_tables)
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // Can't have locked tables here
|
2002-09-05 16:17:08 +03:00
|
|
|
thd->lock=thd->locked_tables;
|
|
|
|
thd->locked_tables=0;
|
|
|
|
close_thread_tables(thd);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2005-09-01 16:52:59 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
To avoid deadlocks we should obtain table locks before
|
|
|
|
obtaining acl_cache->lock mutex.
|
|
|
|
*/
|
|
|
|
bzero((char*) tables, sizeof(tables));
|
2005-09-02 01:01:33 +04:00
|
|
|
tables[0].alias= tables[0].table_name= (char*) "host";
|
|
|
|
tables[1].alias= tables[1].table_name= (char*) "user";
|
|
|
|
tables[2].alias= tables[2].table_name= (char*) "db";
|
|
|
|
tables[0].db=tables[1].db=tables[2].db=(char*) "mysql";
|
|
|
|
tables[0].next_local= tables[0].next_global= tables+1;
|
|
|
|
tables[1].next_local= tables[1].next_global= tables+2;
|
2005-09-01 16:52:59 +04:00
|
|
|
tables[0].lock_type=tables[1].lock_type=tables[2].lock_type=TL_READ;
|
|
|
|
|
|
|
|
if (simple_open_n_lock_tables(thd, tables))
|
|
|
|
{
|
|
|
|
sql_print_error("Fatal error: Can't open and lock privilege tables: %s",
|
|
|
|
thd->net.last_error);
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
if ((old_initialized=initialized))
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
|
|
|
old_acl_hosts=acl_hosts;
|
|
|
|
old_acl_users=acl_users;
|
|
|
|
old_acl_dbs=acl_dbs;
|
|
|
|
old_mem=mem;
|
|
|
|
delete_dynamic(&acl_wild_hosts);
|
|
|
|
hash_free(&acl_check_hosts);
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
if ((return_val= acl_load(thd, tables)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // Error. Revert to old list
|
2003-01-18 16:39:21 +02:00
|
|
|
DBUG_PRINT("error",("Reverting to old privileges"));
|
2002-09-05 16:17:08 +03:00
|
|
|
acl_free(); /* purecov: inspected */
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_hosts=old_acl_hosts;
|
|
|
|
acl_users=old_acl_users;
|
|
|
|
acl_dbs=old_acl_dbs;
|
|
|
|
mem=old_mem;
|
|
|
|
init_check_host();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-09-12 03:02:33 +03:00
|
|
|
free_root(&old_mem,MYF(0));
|
2000-07-31 21:29:14 +02:00
|
|
|
delete_dynamic(&old_acl_hosts);
|
|
|
|
delete_dynamic(&old_acl_users);
|
|
|
|
delete_dynamic(&old_acl_dbs);
|
|
|
|
}
|
|
|
|
if (old_initialized)
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2005-09-01 16:52:59 +04:00
|
|
|
end:
|
|
|
|
close_thread_tables(thd);
|
|
|
|
DBUG_RETURN(return_val);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
/*
|
|
|
|
Get all access bits from table after fieldnr
|
2004-07-19 16:01:53 +02:00
|
|
|
|
|
|
|
IMPLEMENTATION
|
2002-06-12 15:04:18 +03:00
|
|
|
We know that the access privileges ends when there is no more fields
|
|
|
|
or the field is not an enum with two elements.
|
2004-07-19 16:01:53 +02:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
get_access()
|
|
|
|
form an open table to read privileges from.
|
|
|
|
The record should be already read in table->record[0]
|
|
|
|
fieldnr number of the first privilege (that is ENUM('N','Y') field
|
|
|
|
next_field on return - number of the field next to the last ENUM
|
|
|
|
(unless next_field == 0)
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
privilege mask
|
2002-06-12 15:04:18 +03:00
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-07-19 16:01:53 +02:00
|
|
|
static ulong get_access(TABLE *form, uint fieldnr, uint *next_field)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong access_bits=0,bit;
|
2000-07-31 21:29:14 +02:00
|
|
|
char buff[2];
|
2003-01-29 17:31:20 +04:00
|
|
|
String res(buff,sizeof(buff),&my_charset_latin1);
|
2000-07-31 21:29:14 +02:00
|
|
|
Field **pos;
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
for (pos=form->field+fieldnr, bit=1;
|
2006-12-01 17:26:52 -08:00
|
|
|
*pos && (*pos)->real_type() == MYSQL_TYPE_ENUM &&
|
2002-06-12 15:04:18 +03:00
|
|
|
((Field_enum*) (*pos))->typelib->count == 2 ;
|
2004-07-19 16:01:53 +02:00
|
|
|
pos++, fieldnr++, bit<<=1)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-04-06 21:35:26 +02:00
|
|
|
(*pos)->val_str(&res);
|
2003-01-29 17:31:20 +04:00
|
|
|
if (my_toupper(&my_charset_latin1, res[0]) == 'Y')
|
2002-06-12 15:04:18 +03:00
|
|
|
access_bits|= bit;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-07-19 16:01:53 +02:00
|
|
|
if (next_field)
|
|
|
|
*next_field=fieldnr;
|
2000-07-31 21:29:14 +02:00
|
|
|
return access_bits;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2002-06-12 15:04:18 +03:00
|
|
|
Return a number which, if sorted 'desc', puts strings in this order:
|
|
|
|
no wildcards
|
|
|
|
wildcards
|
|
|
|
empty string
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
static ulong get_sort(uint count,...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args,count);
|
|
|
|
ulong sort=0;
|
|
|
|
|
2004-08-12 09:41:35 +05:00
|
|
|
/* Should not use this function with more than 4 arguments for compare. */
|
|
|
|
DBUG_ASSERT(count <= 4);
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
while (count--)
|
|
|
|
{
|
2004-08-12 09:41:35 +05:00
|
|
|
char *start, *str= va_arg(args,char*);
|
|
|
|
uint chars= 0;
|
|
|
|
uint wild_pos= 0; /* first wildcard position */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-08-26 18:26:38 +03:00
|
|
|
if ((start= str))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
for (; *str ; str++)
|
|
|
|
{
|
|
|
|
if (*str == wild_many || *str == wild_one || *str == wild_prefix)
|
2004-08-12 09:41:35 +05:00
|
|
|
{
|
2004-08-26 18:26:38 +03:00
|
|
|
wild_pos= (uint) (str - start) + 1;
|
2004-08-12 09:41:35 +05:00
|
|
|
break;
|
|
|
|
}
|
2004-08-26 18:26:38 +03:00
|
|
|
chars= 128; // Marker that chars existed
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
2004-08-26 18:26:38 +03:00
|
|
|
sort= (sort << 8) + (wild_pos ? min(wild_pos, 127) : chars);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
va_end(args);
|
|
|
|
return sort;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int acl_compare(ACL_ACCESS *a,ACL_ACCESS *b)
|
|
|
|
{
|
|
|
|
if (a->sort > b->sort)
|
|
|
|
return -1;
|
|
|
|
if (a->sort < b->sort)
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-02-12 21:55:37 +02:00
|
|
|
|
2002-11-24 17:07:53 +03:00
|
|
|
/*
|
2003-09-11 19:06:23 +03:00
|
|
|
Seek ACL entry for a user, check password, SSL cypher, and if
|
|
|
|
everything is OK, update THD user data and USER_RESOURCES struct.
|
2002-11-24 17:07:53 +03:00
|
|
|
|
2003-09-11 19:06:23 +03:00
|
|
|
IMPLEMENTATION
|
|
|
|
This function does not check if the user has any sensible privileges:
|
|
|
|
only user's existence and validity is checked.
|
|
|
|
Note, that entire operation is protected by acl_cache_lock.
|
2002-11-30 20:58:53 +03:00
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
SYNOPSIS
|
2003-07-18 18:25:54 +04:00
|
|
|
acl_getroot()
|
|
|
|
thd thread handle. If all checks are OK,
|
2005-09-28 18:43:46 +04:00
|
|
|
thd->security_ctx->priv_user/master_access are updated.
|
|
|
|
thd->security_ctx->host/ip/user are used for checks.
|
2003-07-18 18:25:54 +04:00
|
|
|
mqh user resources; on success mqh is reset, else
|
|
|
|
unchanged
|
2004-10-07 01:45:06 +03:00
|
|
|
passwd scrambled & crypted password, received from client
|
2003-07-18 18:25:54 +04:00
|
|
|
(to check): thd->scramble or thd->scramble_323 is
|
|
|
|
used to decrypt passwd, so they must contain
|
|
|
|
original random string,
|
|
|
|
passwd_len length of passwd, must be one of 0, 8,
|
|
|
|
SCRAMBLE_LENGTH_323, SCRAMBLE_LENGTH
|
|
|
|
'thd' and 'mqh' are updated on success; other params are IN.
|
|
|
|
|
2003-09-11 19:06:23 +03:00
|
|
|
RETURN VALUE
|
2003-07-18 18:25:54 +04:00
|
|
|
0 success: thd->priv_user, thd->priv_host, thd->master_access, mqh are
|
|
|
|
updated
|
2004-10-07 01:45:06 +03:00
|
|
|
1 user not found or authentication failure
|
2003-07-04 20:52:04 +04:00
|
|
|
2 user found, has long (4.1.1) salt, but passwd is in old (3.23) format.
|
2003-07-01 23:40:59 +04:00
|
|
|
-1 user found, has short (3.23) salt, but passwd is in new (4.1.1) format.
|
2002-01-29 18:32:16 +02:00
|
|
|
*/
|
|
|
|
|
2003-07-08 02:36:14 +04:00
|
|
|
int acl_getroot(THD *thd, USER_RESOURCES *mqh,
|
|
|
|
const char *passwd, uint passwd_len)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-09-03 19:53:08 +03:00
|
|
|
ulong user_access= NO_ACCESS;
|
|
|
|
int res= 1;
|
|
|
|
ACL_USER *acl_user= 0;
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2002-08-08 03:12:02 +03:00
|
|
|
DBUG_ENTER("acl_getroot");
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (!initialized)
|
2002-08-08 03:12:02 +03:00
|
|
|
{
|
2003-07-18 18:25:54 +04:00
|
|
|
/*
|
|
|
|
here if mysqld's been started with --skip-grant-tables option.
|
|
|
|
*/
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->skip_grants();
|
2003-09-03 19:53:08 +03:00
|
|
|
bzero((char*) mqh, sizeof(*mqh));
|
2003-07-18 18:25:54 +04:00
|
|
|
DBUG_RETURN(0);
|
2002-08-08 03:12:02 +03:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
2002-11-30 20:58:53 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/*
|
2003-07-01 23:40:59 +04:00
|
|
|
Find acl entry in user database. Note, that find_acl_user is not the same,
|
|
|
|
because it doesn't take into account the case when user is not empty,
|
|
|
|
but acl_user->user is empty
|
2000-07-31 21:29:14 +02:00
|
|
|
*/
|
2002-11-30 20:58:53 +03:00
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
for (uint i=0 ; i < acl_users.elements ; i++)
|
2003-02-14 11:47:41 +02:00
|
|
|
{
|
2003-09-11 19:06:23 +03:00
|
|
|
ACL_USER *acl_user_tmp= dynamic_element(&acl_users,i,ACL_USER*);
|
2005-09-15 22:29:07 +03:00
|
|
|
if (!acl_user_tmp->user || !strcmp(sctx->user, acl_user_tmp->user))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-09-15 22:29:07 +03:00
|
|
|
if (compare_hostname(&acl_user_tmp->host, sctx->host, sctx->ip))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-07-01 23:40:59 +04:00
|
|
|
/* check password: it should be empty or valid */
|
2003-09-11 19:06:23 +03:00
|
|
|
if (passwd_len == acl_user_tmp->salt_len)
|
2002-11-30 16:31:58 +03:00
|
|
|
{
|
2003-09-11 19:06:23 +03:00
|
|
|
if (acl_user_tmp->salt_len == 0 ||
|
2004-06-02 00:27:59 +02:00
|
|
|
(acl_user_tmp->salt_len == SCRAMBLE_LENGTH ?
|
|
|
|
check_scramble(passwd, thd->scramble, acl_user_tmp->salt) :
|
2003-07-18 18:25:54 +04:00
|
|
|
check_scramble_323(passwd, thd->scramble,
|
2004-06-02 00:27:59 +02:00
|
|
|
(ulong *) acl_user_tmp->salt)) == 0)
|
2003-07-01 23:40:59 +04:00
|
|
|
{
|
2003-09-11 19:06:23 +03:00
|
|
|
acl_user= acl_user_tmp;
|
2003-07-01 23:40:59 +04:00
|
|
|
res= 0;
|
|
|
|
}
|
2002-11-30 20:58:53 +03:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
else if (passwd_len == SCRAMBLE_LENGTH &&
|
2003-09-11 19:06:23 +03:00
|
|
|
acl_user_tmp->salt_len == SCRAMBLE_LENGTH_323)
|
2003-07-01 23:40:59 +04:00
|
|
|
res= -1;
|
2003-07-04 20:52:04 +04:00
|
|
|
else if (passwd_len == SCRAMBLE_LENGTH_323 &&
|
2003-09-11 19:06:23 +03:00
|
|
|
acl_user_tmp->salt_len == SCRAMBLE_LENGTH)
|
2003-07-04 20:52:04 +04:00
|
|
|
res= 2;
|
2003-07-01 23:40:59 +04:00
|
|
|
/* linear search complete: */
|
|
|
|
break;
|
2002-11-30 16:31:58 +03:00
|
|
|
}
|
2002-11-30 20:58:53 +03:00
|
|
|
}
|
2003-02-14 11:47:41 +02:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
/*
|
|
|
|
This was moved to separate tree because of heavy HAVE_OPENSSL case.
|
|
|
|
If acl_user is not null, res is 0.
|
|
|
|
*/
|
2002-11-30 20:58:53 +03:00
|
|
|
|
|
|
|
if (acl_user)
|
|
|
|
{
|
2003-07-01 23:40:59 +04:00
|
|
|
/* OK. User found and password checked continue validation */
|
2003-08-27 02:51:39 +03:00
|
|
|
#ifdef HAVE_OPENSSL
|
2003-11-20 22:06:25 +02:00
|
|
|
Vio *vio=thd->net.vio;
|
2003-08-29 13:44:35 +03:00
|
|
|
SSL *ssl= (SSL*) vio->ssl_arg;
|
2003-08-27 02:51:39 +03:00
|
|
|
#endif
|
2003-09-03 19:53:08 +03:00
|
|
|
|
2002-12-06 22:01:12 +02:00
|
|
|
/*
|
2003-05-19 16:35:49 +03:00
|
|
|
At this point we know that user is allowed to connect
|
2002-12-06 22:01:12 +02:00
|
|
|
from given host by given username/password pair. Now
|
|
|
|
we check if SSL is required, if user is using SSL and
|
|
|
|
if X509 certificate attributes are OK
|
|
|
|
*/
|
|
|
|
switch (acl_user->ssl_type) {
|
|
|
|
case SSL_TYPE_NOT_SPECIFIED: // Impossible
|
2003-09-03 19:53:08 +03:00
|
|
|
case SSL_TYPE_NONE: // SSL is not required
|
|
|
|
user_access= acl_user->access;
|
2002-12-06 22:01:12 +02:00
|
|
|
break;
|
2003-07-23 18:50:18 +04:00
|
|
|
#ifdef HAVE_OPENSSL
|
2003-09-03 19:53:08 +03:00
|
|
|
case SSL_TYPE_ANY: // Any kind of SSL is ok
|
2002-12-06 22:01:12 +02:00
|
|
|
if (vio_type(vio) == VIO_TYPE_SSL)
|
2003-09-03 19:53:08 +03:00
|
|
|
user_access= acl_user->access;
|
2002-12-06 22:01:12 +02:00
|
|
|
break;
|
|
|
|
case SSL_TYPE_X509: /* Client should have any valid certificate. */
|
|
|
|
/*
|
|
|
|
Connections with non-valid certificates are dropped already
|
|
|
|
in sslaccept() anyway, so we do not check validity here.
|
2003-09-03 19:53:08 +03:00
|
|
|
|
2003-05-19 16:35:49 +03:00
|
|
|
We need to check for absence of SSL because without SSL
|
|
|
|
we should reject connection.
|
2002-12-06 22:01:12 +02:00
|
|
|
*/
|
2003-08-11 22:44:43 +03:00
|
|
|
if (vio_type(vio) == VIO_TYPE_SSL &&
|
2003-08-29 13:44:35 +03:00
|
|
|
SSL_get_verify_result(ssl) == X509_V_OK &&
|
|
|
|
SSL_get_peer_certificate(ssl))
|
2003-09-03 19:53:08 +03:00
|
|
|
user_access= acl_user->access;
|
2002-12-06 22:01:12 +02:00
|
|
|
break;
|
|
|
|
case SSL_TYPE_SPECIFIED: /* Client should have specified attrib */
|
|
|
|
/*
|
|
|
|
We do not check for absence of SSL because without SSL it does
|
|
|
|
not pass all checks here anyway.
|
|
|
|
If cipher name is specified, we compare it to actual cipher in
|
|
|
|
use.
|
|
|
|
*/
|
2003-12-17 17:35:34 +02:00
|
|
|
X509 *cert;
|
2003-08-11 22:44:43 +03:00
|
|
|
if (vio_type(vio) != VIO_TYPE_SSL ||
|
2003-08-29 13:44:35 +03:00
|
|
|
SSL_get_verify_result(ssl) != X509_V_OK)
|
2003-05-19 16:35:49 +03:00
|
|
|
break;
|
2002-12-06 22:01:12 +02:00
|
|
|
if (acl_user->ssl_cipher)
|
2002-11-30 16:31:58 +03:00
|
|
|
{
|
2002-12-06 22:01:12 +02:00
|
|
|
DBUG_PRINT("info",("comparing ciphers: '%s' and '%s'",
|
2003-08-29 13:44:35 +03:00
|
|
|
acl_user->ssl_cipher,SSL_get_cipher(ssl)));
|
|
|
|
if (!strcmp(acl_user->ssl_cipher,SSL_get_cipher(ssl)))
|
2003-09-03 19:53:08 +03:00
|
|
|
user_access= acl_user->access;
|
2002-12-06 22:01:12 +02:00
|
|
|
else
|
|
|
|
{
|
2003-05-19 16:35:49 +03:00
|
|
|
if (global_system_variables.log_warnings)
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_information("X509 ciphers mismatch: should be '%s' but is '%s'",
|
|
|
|
acl_user->ssl_cipher,
|
|
|
|
SSL_get_cipher(ssl));
|
2002-12-06 22:01:12 +02:00
|
|
|
break;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2002-12-06 22:01:12 +02:00
|
|
|
/* Prepare certificate (if exists) */
|
|
|
|
DBUG_PRINT("info",("checkpoint 1"));
|
2003-12-17 17:35:34 +02:00
|
|
|
if (!(cert= SSL_get_peer_certificate(ssl)))
|
|
|
|
{
|
|
|
|
user_access=NO_ACCESS;
|
|
|
|
break;
|
|
|
|
}
|
2002-12-06 22:01:12 +02:00
|
|
|
DBUG_PRINT("info",("checkpoint 2"));
|
2004-10-07 01:45:06 +03:00
|
|
|
/* If X509 issuer is specified, we check it... */
|
2002-12-06 22:01:12 +02:00
|
|
|
if (acl_user->x509_issuer)
|
2002-11-30 16:31:58 +03:00
|
|
|
{
|
2003-07-31 17:11:52 +04:00
|
|
|
DBUG_PRINT("info",("checkpoint 3"));
|
2006-03-10 16:41:14 +01:00
|
|
|
char *ptr = X509_NAME_oneline(X509_get_issuer_name(cert), 0, 0);
|
|
|
|
DBUG_PRINT("info",("comparing issuers: '%s' and '%s'",
|
2002-12-06 22:01:12 +02:00
|
|
|
acl_user->x509_issuer, ptr));
|
2003-07-31 17:11:52 +04:00
|
|
|
if (strcmp(acl_user->x509_issuer, ptr))
|
2003-07-01 23:40:59 +04:00
|
|
|
{
|
2003-07-31 17:11:52 +04:00
|
|
|
if (global_system_variables.log_warnings)
|
2004-09-04 20:17:09 +02:00
|
|
|
sql_print_information("X509 issuer mismatch: should be '%s' "
|
|
|
|
"but is '%s'", acl_user->x509_issuer, ptr);
|
2003-07-01 23:40:59 +04:00
|
|
|
free(ptr);
|
2006-08-22 14:29:48 +02:00
|
|
|
user_access=NO_ACCESS;
|
2003-07-31 17:11:52 +04:00
|
|
|
break;
|
2003-07-01 23:40:59 +04:00
|
|
|
}
|
2003-09-03 19:53:08 +03:00
|
|
|
user_access= acl_user->access;
|
2003-07-31 17:11:52 +04:00
|
|
|
free(ptr);
|
2002-11-30 16:31:58 +03:00
|
|
|
}
|
2002-12-06 22:01:12 +02:00
|
|
|
DBUG_PRINT("info",("checkpoint 4"));
|
|
|
|
/* X509 subject is specified, we check it .. */
|
|
|
|
if (acl_user->x509_subject)
|
|
|
|
{
|
2003-07-31 17:11:52 +04:00
|
|
|
char *ptr= X509_NAME_oneline(X509_get_subject_name(cert), 0, 0);
|
|
|
|
DBUG_PRINT("info",("comparing subjects: '%s' and '%s'",
|
|
|
|
acl_user->x509_subject, ptr));
|
|
|
|
if (strcmp(acl_user->x509_subject,ptr))
|
2003-07-01 23:40:59 +04:00
|
|
|
{
|
2003-07-31 17:11:52 +04:00
|
|
|
if (global_system_variables.log_warnings)
|
2006-08-22 14:29:48 +02:00
|
|
|
sql_print_information("X509 subject mismatch: should be '%s' but is '%s'",
|
2003-07-31 17:11:52 +04:00
|
|
|
acl_user->x509_subject, ptr);
|
2006-08-22 14:29:48 +02:00
|
|
|
free(ptr);
|
|
|
|
user_access=NO_ACCESS;
|
|
|
|
break;
|
2003-07-01 23:40:59 +04:00
|
|
|
}
|
2006-08-22 14:29:48 +02:00
|
|
|
user_access= acl_user->access;
|
2003-07-31 17:11:52 +04:00
|
|
|
free(ptr);
|
2002-12-06 22:01:12 +02:00
|
|
|
}
|
|
|
|
break;
|
2002-11-30 16:31:58 +03:00
|
|
|
#else /* HAVE_OPENSSL */
|
2003-08-11 22:44:43 +03:00
|
|
|
default:
|
2003-07-23 18:50:18 +04:00
|
|
|
/*
|
2003-07-31 17:11:52 +04:00
|
|
|
If we don't have SSL but SSL is required for this user the
|
|
|
|
authentication should fail.
|
|
|
|
*/
|
2003-07-23 18:50:18 +04:00
|
|
|
break;
|
|
|
|
#endif /* HAVE_OPENSSL */
|
2002-11-30 16:31:58 +03:00
|
|
|
}
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->master_access= user_access;
|
|
|
|
sctx->priv_user= acl_user->user ? sctx->user : (char *) "";
|
2003-07-01 23:40:59 +04:00
|
|
|
*mqh= acl_user->user_resource;
|
2003-07-23 18:50:18 +04:00
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
if (acl_user->host.hostname)
|
2005-09-15 22:29:07 +03:00
|
|
|
strmake(sctx->priv_host, acl_user->host.hostname, MAX_HOSTNAME);
|
2003-07-01 23:40:59 +04:00
|
|
|
else
|
2005-09-15 22:29:07 +03:00
|
|
|
*sctx->priv_host= 0;
|
2003-07-01 23:40:59 +04:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2003-07-01 23:40:59 +04:00
|
|
|
DBUG_RETURN(res);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-13 16:40:52 +01:00
|
|
|
/*
|
2005-09-20 21:20:38 +03:00
|
|
|
This is like acl_getroot() above, but it doesn't check password,
|
|
|
|
and we don't care about the user resources.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
acl_getroot_no_password()
|
|
|
|
sctx Context which should be initialized
|
|
|
|
user user name
|
|
|
|
host host name
|
|
|
|
ip IP
|
|
|
|
db current data base name
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE OK
|
|
|
|
TRUE Error
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool acl_getroot_no_password(Security_context *sctx, char *user, char *host,
|
|
|
|
char *ip, char *db)
|
2003-12-13 16:40:52 +01:00
|
|
|
{
|
|
|
|
int res= 1;
|
2004-03-02 11:52:19 +01:00
|
|
|
uint i;
|
2003-12-13 16:40:52 +01:00
|
|
|
ACL_USER *acl_user= 0;
|
|
|
|
DBUG_ENTER("acl_getroot_no_password");
|
|
|
|
|
2005-10-28 00:18:23 +03:00
|
|
|
DBUG_PRINT("enter", ("Host: '%s', Ip: '%s', User: '%s', db: '%s'",
|
|
|
|
(host ? host : "(NULL)"), (ip ? ip : "(NULL)"),
|
2006-01-03 17:54:54 +01:00
|
|
|
user, (db ? db : "(NULL)")));
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->user= user;
|
|
|
|
sctx->host= host;
|
|
|
|
sctx->ip= ip;
|
|
|
|
sctx->host_or_ip= host ? host : (ip ? ip : "");
|
|
|
|
|
2003-12-13 16:40:52 +01:00
|
|
|
if (!initialized)
|
|
|
|
{
|
2005-09-15 22:29:07 +03:00
|
|
|
/*
|
2003-12-13 16:40:52 +01:00
|
|
|
here if mysqld's been started with --skip-grant-tables option.
|
|
|
|
*/
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->skip_grants();
|
2005-09-20 21:20:38 +03:00
|
|
|
DBUG_RETURN(FALSE);
|
2003-12-13 16:40:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->master_access= 0;
|
|
|
|
sctx->db_access= 0;
|
2006-05-26 11:49:39 +03:00
|
|
|
sctx->priv_user= (char *) "";
|
|
|
|
*sctx->priv_host= 0;
|
2004-03-02 11:52:19 +01:00
|
|
|
|
2003-12-13 16:40:52 +01:00
|
|
|
/*
|
|
|
|
Find acl entry in user database.
|
|
|
|
This is specially tailored to suit the check we do for CALL of
|
2005-09-15 22:29:07 +03:00
|
|
|
a stored procedure; user is set to what is actually a
|
2003-12-13 16:40:52 +01:00
|
|
|
priv_user, which can be ''.
|
|
|
|
*/
|
2004-03-02 11:52:19 +01:00
|
|
|
for (i=0 ; i < acl_users.elements ; i++)
|
2003-12-13 16:40:52 +01:00
|
|
|
{
|
|
|
|
acl_user= dynamic_element(&acl_users,i,ACL_USER*);
|
2006-01-03 17:54:54 +01:00
|
|
|
if ((!acl_user->user && !user[0]) ||
|
2005-09-15 22:29:07 +03:00
|
|
|
(acl_user->user && strcmp(user, acl_user->user) == 0))
|
2003-12-13 16:40:52 +01:00
|
|
|
{
|
2005-09-15 22:29:07 +03:00
|
|
|
if (compare_hostname(&acl_user->host, host, ip))
|
2003-12-13 16:40:52 +01:00
|
|
|
{
|
|
|
|
res= 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (acl_user)
|
|
|
|
{
|
2004-03-02 11:52:19 +01:00
|
|
|
for (i=0 ; i < acl_dbs.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_DB *acl_db= dynamic_element(&acl_dbs, i, ACL_DB*);
|
|
|
|
if (!acl_db->user ||
|
2005-09-15 22:29:07 +03:00
|
|
|
(user && user[0] && !strcmp(user, acl_db->user)))
|
2004-03-02 11:52:19 +01:00
|
|
|
{
|
2005-09-15 22:29:07 +03:00
|
|
|
if (compare_hostname(&acl_db->host, host, ip))
|
2004-03-02 11:52:19 +01:00
|
|
|
{
|
2005-12-15 15:23:16 +01:00
|
|
|
if (!acl_db->db || (db && !wild_compare(db, acl_db->db, 0)))
|
2004-03-02 11:52:19 +01:00
|
|
|
{
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->db_access= acl_db->access;
|
2004-03-02 11:52:19 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->master_access= acl_user->access;
|
|
|
|
sctx->priv_user= acl_user->user ? user : (char *) "";
|
2003-12-13 16:40:52 +01:00
|
|
|
|
|
|
|
if (acl_user->host.hostname)
|
2005-09-15 22:29:07 +03:00
|
|
|
strmake(sctx->priv_host, acl_user->host.hostname, MAX_HOSTNAME);
|
2003-12-13 16:40:52 +01:00
|
|
|
else
|
2005-09-15 22:29:07 +03:00
|
|
|
*sctx->priv_host= 0;
|
2003-12-13 16:40:52 +01:00
|
|
|
}
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
DBUG_RETURN(res);
|
|
|
|
}
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
static byte* check_get_key(ACL_USER *buff,uint *length,
|
|
|
|
my_bool not_used __attribute__((unused)))
|
|
|
|
{
|
|
|
|
*length=buff->hostname_length;
|
|
|
|
return (byte*) buff->host.hostname;
|
|
|
|
}
|
|
|
|
|
2003-07-01 23:40:59 +04:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
static void acl_update_user(const char *user, const char *host,
|
2003-07-01 23:40:59 +04:00
|
|
|
const char *password, uint password_len,
|
2001-09-30 10:46:20 +08:00
|
|
|
enum SSL_type ssl_type,
|
|
|
|
const char *ssl_cipher,
|
|
|
|
const char *x509_issuer,
|
|
|
|
const char *x509_subject,
|
2002-11-30 20:58:53 +03:00
|
|
|
USER_RESOURCES *mqh,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong privileges)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-05-06 11:25:59 +04:00
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
for (uint i=0 ; i < acl_users.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_USER *acl_user=dynamic_element(&acl_users,i,ACL_USER*);
|
|
|
|
if (!acl_user->user && !user[0] ||
|
2006-06-06 14:21:07 +03:00
|
|
|
acl_user->user && !strcmp(user,acl_user->user))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (!acl_user->host.hostname && !host[0] ||
|
2002-09-25 20:42:55 +03:00
|
|
|
acl_user->host.hostname &&
|
2004-11-25 21:55:49 +01:00
|
|
|
!my_strcasecmp(system_charset_info, host, acl_user->host.hostname))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
acl_user->access=privileges;
|
2004-12-29 20:30:37 +03:00
|
|
|
if (mqh->specified_limits & USER_RESOURCES::QUERIES_PER_HOUR)
|
2002-06-20 16:46:25 +03:00
|
|
|
acl_user->user_resource.questions=mqh->questions;
|
2004-12-29 20:30:37 +03:00
|
|
|
if (mqh->specified_limits & USER_RESOURCES::UPDATES_PER_HOUR)
|
2002-06-20 16:46:25 +03:00
|
|
|
acl_user->user_resource.updates=mqh->updates;
|
2004-12-29 20:30:37 +03:00
|
|
|
if (mqh->specified_limits & USER_RESOURCES::CONNECTIONS_PER_HOUR)
|
|
|
|
acl_user->user_resource.conn_per_hour= mqh->conn_per_hour;
|
|
|
|
if (mqh->specified_limits & USER_RESOURCES::USER_CONNECTIONS)
|
|
|
|
acl_user->user_resource.user_conn= mqh->user_conn;
|
2002-09-05 16:17:08 +03:00
|
|
|
if (ssl_type != SSL_TYPE_NOT_SPECIFIED)
|
|
|
|
{
|
|
|
|
acl_user->ssl_type= ssl_type;
|
|
|
|
acl_user->ssl_cipher= (ssl_cipher ? strdup_root(&mem,ssl_cipher) :
|
|
|
|
0);
|
|
|
|
acl_user->x509_issuer= (x509_issuer ? strdup_root(&mem,x509_issuer) :
|
|
|
|
0);
|
|
|
|
acl_user->x509_subject= (x509_subject ?
|
|
|
|
strdup_root(&mem,x509_subject) : 0);
|
|
|
|
}
|
2004-03-01 21:54:08 +02:00
|
|
|
if (password)
|
|
|
|
set_user_salt(acl_user, password, password_len);
|
2003-07-01 23:40:59 +04:00
|
|
|
/* search complete: */
|
2000-07-31 21:29:14 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void acl_insert_user(const char *user, const char *host,
|
2003-07-01 23:40:59 +04:00
|
|
|
const char *password, uint password_len,
|
2001-09-30 10:46:20 +08:00
|
|
|
enum SSL_type ssl_type,
|
|
|
|
const char *ssl_cipher,
|
|
|
|
const char *x509_issuer,
|
|
|
|
const char *x509_subject,
|
2002-05-15 13:50:38 +03:00
|
|
|
USER_RESOURCES *mqh,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong privileges)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
ACL_USER acl_user;
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
|
|
|
|
2003-07-21 13:46:57 +02:00
|
|
|
acl_user.user=*user ? strdup_root(&mem,user) : 0;
|
2004-05-05 17:05:24 +03:00
|
|
|
update_hostname(&acl_user.host, *host ? strdup_root(&mem, host): 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_user.access=privileges;
|
2002-05-15 13:50:38 +03:00
|
|
|
acl_user.user_resource = *mqh;
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_user.sort=get_sort(2,acl_user.host.hostname,acl_user.user);
|
2004-04-28 17:45:08 +03:00
|
|
|
acl_user.hostname_length=(uint) strlen(host);
|
2002-09-05 16:17:08 +03:00
|
|
|
acl_user.ssl_type= (ssl_type != SSL_TYPE_NOT_SPECIFIED ?
|
|
|
|
ssl_type : SSL_TYPE_NONE);
|
|
|
|
acl_user.ssl_cipher= ssl_cipher ? strdup_root(&mem,ssl_cipher) : 0;
|
|
|
|
acl_user.x509_issuer= x509_issuer ? strdup_root(&mem,x509_issuer) : 0;
|
|
|
|
acl_user.x509_subject=x509_subject ? strdup_root(&mem,x509_subject) : 0;
|
2003-07-01 23:40:59 +04:00
|
|
|
|
|
|
|
set_user_salt(&acl_user, password, password_len);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
VOID(push_dynamic(&acl_users,(gptr) &acl_user));
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!acl_user.host.hostname ||
|
|
|
|
(acl_user.host.hostname[0] == wild_many && !acl_user.host.hostname[1]))
|
2003-05-19 16:35:49 +03:00
|
|
|
allow_all_hosts=1; // Anyone can connect /* purecov: tested */
|
2000-07-31 21:29:14 +02:00
|
|
|
qsort((gptr) dynamic_element(&acl_users,0,ACL_USER*),acl_users.elements,
|
|
|
|
sizeof(ACL_USER),(qsort_cmp) acl_compare);
|
|
|
|
|
2005-12-28 14:43:50 +01:00
|
|
|
/* Rebuild 'acl_check_hosts' since 'acl_users' has been modified */
|
|
|
|
rebuild_check_host();
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void acl_update_db(const char *user, const char *host, const char *db,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong privileges)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-05-06 11:25:59 +04:00
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
for (uint i=0 ; i < acl_dbs.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_DB *acl_db=dynamic_element(&acl_dbs,i,ACL_DB*);
|
|
|
|
if (!acl_db->user && !user[0] ||
|
|
|
|
acl_db->user &&
|
|
|
|
!strcmp(user,acl_db->user))
|
|
|
|
{
|
|
|
|
if (!acl_db->host.hostname && !host[0] ||
|
2002-10-03 16:54:26 +03:00
|
|
|
acl_db->host.hostname &&
|
2004-11-25 21:55:49 +01:00
|
|
|
!my_strcasecmp(system_charset_info, host, acl_db->host.hostname))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (!acl_db->db && !db[0] ||
|
|
|
|
acl_db->db && !strcmp(db,acl_db->db))
|
|
|
|
{
|
|
|
|
if (privileges)
|
|
|
|
acl_db->access=privileges;
|
|
|
|
else
|
|
|
|
delete_dynamic_element(&acl_dbs,i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-22 10:35:15 +03:00
|
|
|
/*
|
|
|
|
Insert a user/db/host combination into the global acl_cache
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
acl_insert_db()
|
|
|
|
user User name
|
|
|
|
host Host name
|
|
|
|
db Database name
|
|
|
|
privileges Bitmap of privileges
|
|
|
|
|
|
|
|
NOTES
|
|
|
|
acl_cache->lock must be locked when calling this
|
|
|
|
*/
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
static void acl_insert_db(const char *user, const char *host, const char *db,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong privileges)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
ACL_DB acl_db;
|
2002-09-22 10:35:15 +03:00
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_db.user=strdup_root(&mem,user);
|
2006-06-06 14:21:07 +03:00
|
|
|
update_hostname(&acl_db.host, *host ? strdup_root(&mem,host) : 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_db.db=strdup_root(&mem,db);
|
|
|
|
acl_db.access=privileges;
|
|
|
|
acl_db.sort=get_sort(3,acl_db.host.hostname,acl_db.db,acl_db.user);
|
|
|
|
VOID(push_dynamic(&acl_dbs,(gptr) &acl_db));
|
|
|
|
qsort((gptr) dynamic_element(&acl_dbs,0,ACL_DB*),acl_dbs.elements,
|
|
|
|
sizeof(ACL_DB),(qsort_cmp) acl_compare);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
Get privilege for a host, user and db combination
|
2005-03-26 19:46:42 +01:00
|
|
|
|
|
|
|
as db_is_pattern changes the semantics of comparison,
|
|
|
|
acl_cache is not used if db_is_pattern is set.
|
2003-06-23 20:03:59 +03:00
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-10-24 23:27:21 +02:00
|
|
|
ulong acl_get(const char *host, const char *ip,
|
2004-06-02 00:09:14 +03:00
|
|
|
const char *user, const char *db, my_bool db_is_pattern)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-06-01 21:56:30 -07:00
|
|
|
ulong host_access= ~(ulong)0, db_access= 0;
|
2002-06-12 15:04:18 +03:00
|
|
|
uint i,key_length;
|
2001-12-13 02:31:19 +02:00
|
|
|
char key[ACL_KEY_LENGTH],*tmp_db,*end;
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_entry *entry;
|
2004-11-12 17:44:17 +02:00
|
|
|
DBUG_ENTER("acl_get");
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
2003-10-24 23:27:21 +02:00
|
|
|
end=strmov((tmp_db=strmov(strmov(key, ip ? ip : "")+1,user)+1),db);
|
2001-12-13 02:31:19 +02:00
|
|
|
if (lower_case_table_names)
|
|
|
|
{
|
2004-05-22 23:41:58 +04:00
|
|
|
my_casedn_str(files_charset_info, tmp_db);
|
2001-12-13 02:31:19 +02:00
|
|
|
db=tmp_db;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
key_length=(uint) (end-key);
|
2005-03-26 19:46:42 +01:00
|
|
|
if (!db_is_pattern && (entry=(acl_entry*) acl_cache->search(key,key_length)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
db_access=entry->access;
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2004-11-12 17:44:17 +02:00
|
|
|
DBUG_PRINT("exit", ("access: 0x%lx", db_access));
|
|
|
|
DBUG_RETURN(db_access);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Check if there are some access rights for database and user
|
|
|
|
*/
|
|
|
|
for (i=0 ; i < acl_dbs.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_DB *acl_db=dynamic_element(&acl_dbs,i,ACL_DB*);
|
|
|
|
if (!acl_db->user || !strcmp(user,acl_db->user))
|
|
|
|
{
|
|
|
|
if (compare_hostname(&acl_db->host,host,ip))
|
|
|
|
{
|
2003-07-22 22:21:23 +02:00
|
|
|
if (!acl_db->db || !wild_compare(db,acl_db->db,db_is_pattern))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
db_access=acl_db->access;
|
|
|
|
if (acl_db->host.hostname)
|
|
|
|
goto exit; // Fully specified. Take it
|
|
|
|
break; /* purecov: tested */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!db_access)
|
|
|
|
goto exit; // Can't be better
|
|
|
|
|
|
|
|
/*
|
|
|
|
No host specified for user. Get hostdata from host table
|
|
|
|
*/
|
|
|
|
host_access=0; // Host must be found
|
|
|
|
for (i=0 ; i < acl_hosts.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_HOST *acl_host=dynamic_element(&acl_hosts,i,ACL_HOST*);
|
|
|
|
if (compare_hostname(&acl_host->host,host,ip))
|
|
|
|
{
|
2003-07-22 22:21:23 +02:00
|
|
|
if (!acl_host->db || !wild_compare(db,acl_host->db,db_is_pattern))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
host_access=acl_host->access; // Fully specified. Take it
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
exit:
|
|
|
|
/* Save entry in cache for quick retrieval */
|
2005-03-26 19:46:42 +01:00
|
|
|
if (!db_is_pattern &&
|
|
|
|
(entry= (acl_entry*) malloc(sizeof(acl_entry)+key_length)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
entry->access=(db_access & host_access);
|
|
|
|
entry->length=key_length;
|
|
|
|
memcpy((gptr) entry->key,key,key_length);
|
|
|
|
acl_cache->add(entry);
|
|
|
|
}
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2004-11-12 17:44:17 +02:00
|
|
|
DBUG_PRINT("exit", ("access: 0x%lx", db_access & host_access));
|
|
|
|
DBUG_RETURN(db_access & host_access);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
|
|
|
Check if there are any possible matching entries for this host
|
|
|
|
|
|
|
|
NOTES
|
|
|
|
All host names without wild cards are stored in a hash table,
|
|
|
|
entries with wildcards are stored in a dynamic array
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
static void init_check_host(void)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("init_check_host");
|
2002-04-28 21:22:37 +00:00
|
|
|
VOID(my_init_dynamic_array(&acl_wild_hosts,sizeof(struct acl_host_and_ip),
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_users.elements,1));
|
2004-11-25 21:55:49 +01:00
|
|
|
VOID(hash_init(&acl_check_hosts,system_charset_info,acl_users.elements,0,0,
|
2003-04-01 12:45:16 +05:00
|
|
|
(hash_get_key) check_get_key,0,0));
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!allow_all_hosts)
|
|
|
|
{
|
|
|
|
for (uint i=0 ; i < acl_users.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_USER *acl_user=dynamic_element(&acl_users,i,ACL_USER*);
|
|
|
|
if (strchr(acl_user->host.hostname,wild_many) ||
|
|
|
|
strchr(acl_user->host.hostname,wild_one) ||
|
|
|
|
acl_user->host.ip_mask)
|
|
|
|
{ // Has wildcard
|
|
|
|
uint j;
|
|
|
|
for (j=0 ; j < acl_wild_hosts.elements ; j++)
|
|
|
|
{ // Check if host already exists
|
|
|
|
acl_host_and_ip *acl=dynamic_element(&acl_wild_hosts,j,
|
|
|
|
acl_host_and_ip *);
|
2004-11-25 21:55:49 +01:00
|
|
|
if (!my_strcasecmp(system_charset_info,
|
2002-03-12 21:37:58 +04:00
|
|
|
acl_user->host.hostname, acl->hostname))
|
2000-07-31 21:29:14 +02:00
|
|
|
break; // already stored
|
|
|
|
}
|
|
|
|
if (j == acl_wild_hosts.elements) // If new
|
|
|
|
(void) push_dynamic(&acl_wild_hosts,(char*) &acl_user->host);
|
|
|
|
}
|
2005-12-28 14:43:50 +01:00
|
|
|
else if (!hash_search(&acl_check_hosts,(byte*) acl_user->host.hostname,
|
2000-08-22 00:18:32 +03:00
|
|
|
(uint) strlen(acl_user->host.hostname)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-09-19 14:44:31 +05:00
|
|
|
if (my_hash_insert(&acl_check_hosts,(byte*) acl_user))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // End of memory
|
|
|
|
allow_all_hosts=1; // Should never happen
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
freeze_size(&acl_wild_hosts);
|
|
|
|
freeze_size(&acl_check_hosts.array);
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-28 14:43:50 +01:00
|
|
|
/*
|
|
|
|
Rebuild lists used for checking of allowed hosts
|
|
|
|
|
|
|
|
We need to rebuild 'acl_check_hosts' and 'acl_wild_hosts' after adding,
|
|
|
|
dropping or renaming user, since they contain pointers to elements of
|
|
|
|
'acl_user' array, which are invalidated by drop operation, and use
|
|
|
|
ACL_USER::host::hostname as a key, which is changed by rename.
|
|
|
|
*/
|
|
|
|
void rebuild_check_host(void)
|
|
|
|
{
|
|
|
|
delete_dynamic(&acl_wild_hosts);
|
|
|
|
hash_free(&acl_check_hosts);
|
|
|
|
init_check_host();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Return true if there is no users that can match the given host */
|
|
|
|
|
|
|
|
bool acl_check_host(const char *host, const char *ip)
|
|
|
|
{
|
|
|
|
if (allow_all_hosts)
|
|
|
|
return 0;
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
2000-08-22 00:18:32 +03:00
|
|
|
if (host && hash_search(&acl_check_hosts,(byte*) host,(uint) strlen(host)) ||
|
|
|
|
ip && hash_search(&acl_check_hosts,(byte*) ip,(uint) strlen(ip)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
return 0; // Found host
|
|
|
|
}
|
|
|
|
for (uint i=0 ; i < acl_wild_hosts.elements ; i++)
|
|
|
|
{
|
|
|
|
acl_host_and_ip *acl=dynamic_element(&acl_wild_hosts,i,acl_host_and_ip*);
|
|
|
|
if (compare_hostname(acl, host, ip))
|
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
return 0; // Host ok
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
return 1; // Host is not allowed
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-23 18:31:22 +03:00
|
|
|
/*
|
|
|
|
Check if the user is allowed to change password
|
|
|
|
|
|
|
|
SYNOPSIS:
|
|
|
|
check_change_password()
|
|
|
|
thd THD
|
|
|
|
host hostname for the user
|
|
|
|
user user name
|
2005-01-24 15:48:25 +01:00
|
|
|
new_password new password
|
|
|
|
|
|
|
|
NOTE:
|
|
|
|
new_password cannot be NULL
|
2002-03-12 12:21:51 +02:00
|
|
|
|
2002-07-23 18:31:22 +03:00
|
|
|
RETURN VALUE
|
2003-06-23 20:03:59 +03:00
|
|
|
0 OK
|
|
|
|
1 ERROR ; In this case the error is sent to the client.
|
2002-07-23 18:31:22 +03:00
|
|
|
*/
|
|
|
|
|
2004-07-30 22:05:08 +02:00
|
|
|
bool check_change_password(THD *thd, const char *host, const char *user,
|
2005-01-24 15:48:25 +01:00
|
|
|
char *new_password, uint new_password_len)
|
2002-07-23 18:31:22 +03:00
|
|
|
{
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!initialized)
|
|
|
|
{
|
2004-10-20 04:04:37 +03:00
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--skip-grant-tables");
|
2004-07-30 22:05:08 +02:00
|
|
|
return(1);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2002-03-12 11:38:22 +02:00
|
|
|
if (!thd->slave_thread &&
|
2005-09-15 22:29:07 +03:00
|
|
|
(strcmp(thd->security_ctx->user, user) ||
|
|
|
|
my_strcasecmp(system_charset_info, host,
|
|
|
|
thd->security_ctx->priv_host)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-09-13 16:07:38 +05:00
|
|
|
if (check_access(thd, UPDATE_ACL, "mysql",0,1,0,0))
|
2002-07-23 18:31:22 +03:00
|
|
|
return(1);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2005-09-15 22:29:07 +03:00
|
|
|
if (!thd->slave_thread && !thd->security_ctx->user[0])
|
2002-03-12 11:38:22 +02:00
|
|
|
{
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_PASSWORD_ANONYMOUS_USER, ER(ER_PASSWORD_ANONYMOUS_USER),
|
|
|
|
MYF(0));
|
2002-07-23 18:31:22 +03:00
|
|
|
return(1);
|
2002-03-12 11:38:22 +02:00
|
|
|
}
|
2004-07-30 22:05:08 +02:00
|
|
|
uint len=strlen(new_password);
|
2004-08-01 17:21:55 +02:00
|
|
|
if (len && len != SCRAMBLED_PASSWORD_CHAR_LENGTH &&
|
2004-07-30 22:05:08 +02:00
|
|
|
len != SCRAMBLED_PASSWORD_CHAR_LENGTH_323)
|
|
|
|
{
|
2004-10-20 04:04:37 +03:00
|
|
|
my_error(ER_PASSWD_LENGTH, MYF(0), SCRAMBLED_PASSWORD_CHAR_LENGTH);
|
2004-07-30 22:05:08 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2002-07-23 18:31:22 +03:00
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-22 18:02:39 +03:00
|
|
|
/*
|
|
|
|
Change a password for a user
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
change_password()
|
|
|
|
thd Thread handle
|
|
|
|
host Hostname
|
|
|
|
user User name
|
|
|
|
new_password New password for host@user
|
|
|
|
|
|
|
|
RETURN VALUES
|
|
|
|
0 ok
|
|
|
|
1 ERROR; In this case the error is sent to the client.
|
2002-11-30 20:58:53 +03:00
|
|
|
*/
|
2002-09-22 18:02:39 +03:00
|
|
|
|
2002-07-23 18:31:22 +03:00
|
|
|
bool change_password(THD *thd, const char *host, const char *user,
|
|
|
|
char *new_password)
|
|
|
|
{
|
2005-09-01 16:52:59 +04:00
|
|
|
TABLE_LIST tables;
|
|
|
|
TABLE *table;
|
|
|
|
/* Buffer should be extended when password length is extended. */
|
|
|
|
char buff[512];
|
|
|
|
ulong query_length;
|
2005-01-24 15:48:25 +01:00
|
|
|
uint new_password_len= strlen(new_password);
|
2005-09-01 16:52:59 +04:00
|
|
|
bool result= 1;
|
2002-07-23 18:31:22 +03:00
|
|
|
DBUG_ENTER("change_password");
|
|
|
|
DBUG_PRINT("enter",("host: '%s' user: '%s' new_password: '%s'",
|
|
|
|
host,user,new_password));
|
|
|
|
DBUG_ASSERT(host != 0); // Ensured by parent
|
|
|
|
|
2005-01-24 15:48:25 +01:00
|
|
|
if (check_change_password(thd, host, user, new_password, new_password_len))
|
2002-07-23 18:31:22 +03:00
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
bzero((char*) &tables, sizeof(tables));
|
2005-09-02 01:01:33 +04:00
|
|
|
tables.alias= tables.table_name= (char*) "user";
|
2005-09-01 16:52:59 +04:00
|
|
|
tables.db= (char*) "mysql";
|
|
|
|
|
|
|
|
#ifdef HAVE_REPLICATION
|
|
|
|
/*
|
|
|
|
GRANT and REVOKE are applied the slave in/exclusion rules as they are
|
|
|
|
some kind of updates to the mysql.% tables.
|
|
|
|
*/
|
2005-09-02 08:59:59 +02:00
|
|
|
if (thd->slave_thread && rpl_filter->is_on())
|
2005-09-01 16:52:59 +04:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
The tables must be marked "updating" so that tables_ok() takes them into
|
|
|
|
account in tests. It's ok to leave 'updating' set after tables_ok.
|
|
|
|
*/
|
|
|
|
tables.updating= 1;
|
|
|
|
/* Thanks to bzero, tables.next==0 */
|
2005-11-19 15:56:55 +01:00
|
|
|
if (!(thd->spcont || rpl_filter->tables_ok(0, &tables)))
|
2005-09-01 16:52:59 +04:00
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!(table= open_ltable(thd, &tables, TL_WRITE)))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
ACL_USER *acl_user;
|
2005-08-22 15:48:50 -07:00
|
|
|
if (!(acl_user= find_acl_user(host, user, TRUE)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_PASSWORD_NO_MATCH, ER(ER_PASSWORD_NO_MATCH), MYF(0));
|
2005-09-01 16:52:59 +04:00
|
|
|
goto end;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
/* update loaded acl entry: */
|
|
|
|
set_user_salt(acl_user, new_password, new_password_len);
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
if (update_user_table(thd, table,
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_user->host.hostname ? acl_user->host.hostname : "",
|
2002-03-12 11:38:22 +02:00
|
|
|
acl_user->user ? acl_user->user : "",
|
2003-07-01 23:40:59 +04:00
|
|
|
new_password, new_password_len))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock)); /* purecov: deadcode */
|
2005-09-01 16:52:59 +04:00
|
|
|
goto end;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2002-11-30 20:58:53 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_cache->clear(1); // Clear locked hostname cache
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2005-09-01 16:52:59 +04:00
|
|
|
result= 0;
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
2005-09-02 01:01:33 +04:00
|
|
|
query_length=
|
|
|
|
my_sprintf(buff,
|
2006-11-30 11:08:23 +04:00
|
|
|
(buff,"SET PASSWORD FOR '%-.120s'@'%-.120s'='%-.120s'",
|
2005-09-02 01:01:33 +04:00
|
|
|
acl_user->user ? acl_user->user : "",
|
|
|
|
acl_user->host.hostname ? acl_user->host.hostname : "",
|
|
|
|
new_password));
|
2005-09-01 16:52:59 +04:00
|
|
|
thd->clear_error();
|
2005-12-22 06:39:02 +01:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE, buff, query_length, FALSE, FALSE);
|
2005-09-01 16:52:59 +04:00
|
|
|
}
|
|
|
|
end:
|
|
|
|
close_thread_tables(thd);
|
|
|
|
DBUG_RETURN(result);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-09-14 10:53:09 +03:00
|
|
|
/*
|
|
|
|
Find user in ACL
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
is_acl_user()
|
|
|
|
host host name
|
|
|
|
user user name
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE user not fond
|
|
|
|
TRUE there are such user
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool is_acl_user(const char *host, const char *user)
|
|
|
|
{
|
|
|
|
bool res;
|
2005-10-18 21:42:26 +03:00
|
|
|
|
|
|
|
/* --skip-grants */
|
|
|
|
if (!initialized)
|
|
|
|
return TRUE;
|
|
|
|
|
2005-09-14 10:53:09 +03:00
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
2005-09-15 14:32:01 -05:00
|
|
|
res= find_acl_user(host, user, TRUE) != NULL;
|
2005-09-14 10:53:09 +03:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/*
|
|
|
|
Find first entry that matches the current user
|
|
|
|
*/
|
|
|
|
|
|
|
|
static ACL_USER *
|
2005-08-22 15:48:50 -07:00
|
|
|
find_acl_user(const char *host, const char *user, my_bool exact)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2001-11-27 02:21:44 +02:00
|
|
|
DBUG_ENTER("find_acl_user");
|
2002-06-11 11:20:31 +03:00
|
|
|
DBUG_PRINT("enter",("host: '%s' user: '%s'",host,user));
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
for (uint i=0 ; i < acl_users.elements ; i++)
|
|
|
|
{
|
|
|
|
ACL_USER *acl_user=dynamic_element(&acl_users,i,ACL_USER*);
|
2001-11-27 02:21:44 +02:00
|
|
|
DBUG_PRINT("info",("strcmp('%s','%s'), compare_hostname('%s','%s'),",
|
2006-06-06 14:21:07 +03:00
|
|
|
user, acl_user->user ? acl_user->user : "",
|
|
|
|
host,
|
|
|
|
acl_user->host.hostname ? acl_user->host.hostname :
|
|
|
|
""));
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!acl_user->user && !user[0] ||
|
|
|
|
acl_user->user && !strcmp(user,acl_user->user))
|
|
|
|
{
|
2006-05-06 11:25:59 +04:00
|
|
|
if (exact ? !my_strcasecmp(system_charset_info, host,
|
2006-01-09 10:31:07 +01:00
|
|
|
acl_user->host.hostname ?
|
|
|
|
acl_user->host.hostname : "") :
|
2005-08-22 15:48:50 -07:00
|
|
|
compare_hostname(&acl_user->host,host,host))
|
2001-11-27 02:21:44 +02:00
|
|
|
{
|
|
|
|
DBUG_RETURN(acl_user);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
2001-11-27 02:21:44 +02:00
|
|
|
DBUG_RETURN(0);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
|
|
|
Comparing of hostnames
|
|
|
|
|
|
|
|
NOTES
|
|
|
|
A hostname may be of type:
|
|
|
|
hostname (May include wildcards); monty.pp.sci.fi
|
|
|
|
ip (May include wildcards); 192.168.0.0
|
|
|
|
ip/netmask 192.168.0.0/255.255.255.0
|
|
|
|
|
|
|
|
A net mask of 0.0.0.0 is not allowed.
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
static const char *calc_ip(const char *ip, long *val, char end)
|
|
|
|
{
|
|
|
|
long ip_val,tmp;
|
|
|
|
if (!(ip=str2int(ip,10,0,255,&ip_val)) || *ip != '.')
|
|
|
|
return 0;
|
|
|
|
ip_val<<=24;
|
|
|
|
if (!(ip=str2int(ip+1,10,0,255,&tmp)) || *ip != '.')
|
|
|
|
return 0;
|
|
|
|
ip_val+=tmp<<16;
|
|
|
|
if (!(ip=str2int(ip+1,10,0,255,&tmp)) || *ip != '.')
|
|
|
|
return 0;
|
|
|
|
ip_val+=tmp<<8;
|
|
|
|
if (!(ip=str2int(ip+1,10,0,255,&tmp)) || *ip != end)
|
|
|
|
return 0;
|
|
|
|
*val=ip_val+tmp;
|
|
|
|
return ip;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void update_hostname(acl_host_and_ip *host, const char *hostname)
|
|
|
|
{
|
2006-06-06 14:21:07 +03:00
|
|
|
host->hostname=(char*) hostname; // This will not be modified!
|
2003-12-12 22:26:58 +02:00
|
|
|
if (!hostname ||
|
2000-07-31 21:29:14 +02:00
|
|
|
(!(hostname=calc_ip(hostname,&host->ip,'/')) ||
|
|
|
|
!(hostname=calc_ip(hostname+1,&host->ip_mask,'\0'))))
|
|
|
|
{
|
2003-12-12 22:26:58 +02:00
|
|
|
host->ip= host->ip_mask=0; // Not a masked ip
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool compare_hostname(const acl_host_and_ip *host, const char *hostname,
|
|
|
|
const char *ip)
|
|
|
|
{
|
|
|
|
long tmp;
|
|
|
|
if (host->ip_mask && ip && calc_ip(ip,&tmp,'\0'))
|
|
|
|
{
|
|
|
|
return (tmp & host->ip_mask) == host->ip;
|
|
|
|
}
|
|
|
|
return (!host->hostname ||
|
2004-11-25 21:55:49 +01:00
|
|
|
(hostname && !wild_case_compare(system_charset_info,
|
2006-06-06 14:21:07 +03:00
|
|
|
hostname, host->hostname)) ||
|
|
|
|
(ip && !wild_compare(ip, host->hostname, 0)));
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2003-11-20 12:55:48 +04:00
|
|
|
bool hostname_requires_resolving(const char *hostname)
|
|
|
|
{
|
|
|
|
char cur;
|
|
|
|
if (!hostname)
|
2004-03-30 02:32:41 +03:00
|
|
|
return FALSE;
|
2003-11-20 12:55:48 +04:00
|
|
|
int namelen= strlen(hostname);
|
|
|
|
int lhlen= strlen(my_localhost);
|
|
|
|
if ((namelen == lhlen) &&
|
2004-11-25 21:55:49 +01:00
|
|
|
!my_strnncoll(system_charset_info, (const uchar *)hostname, namelen,
|
2003-11-20 12:55:48 +04:00
|
|
|
(const uchar *)my_localhost, strlen(my_localhost)))
|
2004-03-30 02:32:41 +03:00
|
|
|
return FALSE;
|
2003-11-20 12:55:48 +04:00
|
|
|
for (; (cur=*hostname); hostname++)
|
|
|
|
{
|
2005-02-14 20:33:14 -08:00
|
|
|
if ((cur != '%') && (cur != '_') && (cur != '.') && (cur != '/') &&
|
2003-11-20 12:55:48 +04:00
|
|
|
((cur < '0') || (cur > '9')))
|
2004-03-30 02:32:41 +03:00
|
|
|
return TRUE;
|
2003-11-20 12:55:48 +04:00
|
|
|
}
|
2004-03-30 02:32:41 +03:00
|
|
|
return FALSE;
|
2003-11-20 12:55:48 +04:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
2005-09-01 16:52:59 +04:00
|
|
|
Update record for user in mysql.user privilege table with new password.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
update_user_table()
|
|
|
|
thd Thread handle
|
|
|
|
table Pointer to TABLE object for open mysql.user table
|
|
|
|
host/user Hostname/username pair identifying user for which
|
|
|
|
new password should be set
|
|
|
|
new_password New password
|
|
|
|
new_password_len Length of new password
|
2003-06-23 20:03:59 +03:00
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
static bool update_user_table(THD *thd, TABLE *table,
|
|
|
|
const char *host, const char *user,
|
2003-07-01 23:40:59 +04:00
|
|
|
const char *new_password, uint new_password_len)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-06 02:00:37 +02:00
|
|
|
char user_key[MAX_KEY_LENGTH];
|
2005-09-01 16:52:59 +04:00
|
|
|
int error;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("update_user_table");
|
|
|
|
DBUG_PRINT("enter",("user: %s host: %s",user,host));
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[0]->store(host,(uint) strlen(host), system_charset_info);
|
|
|
|
table->field[1]->store(user,(uint) strlen(user), system_charset_info);
|
2005-01-26 15:19:20 +01:00
|
|
|
key_copy((byte *) user_key, table->record[0], table->key_info,
|
2004-12-06 02:00:37 +02:00
|
|
|
table->key_info->key_length);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-12-06 02:00:37 +02:00
|
|
|
if (table->file->index_read_idx(table->record[0], 0,
|
2007-03-17 00:13:25 +01:00
|
|
|
(byte *) user_key, HA_WHOLE_KEY,
|
2000-07-31 21:29:14 +02:00
|
|
|
HA_READ_KEY_EXACT))
|
|
|
|
{
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_PASSWORD_NO_MATCH, ER(ER_PASSWORD_NO_MATCH),
|
|
|
|
MYF(0)); /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(1); /* purecov: deadcode */
|
|
|
|
}
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]);
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[2]->store(new_password, new_password_len, system_charset_info);
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((error=table->file->ha_update_row(table->record[1],table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: deadcode */
|
2005-09-01 16:52:59 +04:00
|
|
|
DBUG_RETURN(1);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2005-09-01 16:52:59 +04:00
|
|
|
DBUG_RETURN(0);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2001-08-10 17:37:37 +03:00
|
|
|
|
2005-03-22 15:54:18 +01:00
|
|
|
/*
|
|
|
|
Return 1 if we are allowed to create new users
|
|
|
|
the logic here is: INSERT_ACL is sufficient.
|
|
|
|
It's also a requirement in opt_safe_user_create,
|
|
|
|
otherwise CREATE_USER_ACL is enough.
|
|
|
|
*/
|
2001-08-10 17:37:37 +03:00
|
|
|
|
|
|
|
static bool test_if_create_new_users(THD *thd)
|
|
|
|
{
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2005-09-15 22:29:07 +03:00
|
|
|
bool create_new_users= test(sctx->master_access & INSERT_ACL) ||
|
2005-03-22 15:54:18 +01:00
|
|
|
(!opt_safe_user_create &&
|
2005-09-15 22:29:07 +03:00
|
|
|
test(sctx->master_access & CREATE_USER_ACL));
|
2005-03-22 15:54:18 +01:00
|
|
|
if (!create_new_users)
|
2001-08-10 17:37:37 +03:00
|
|
|
{
|
|
|
|
TABLE_LIST tl;
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong db_access;
|
2001-08-10 17:37:37 +03:00
|
|
|
bzero((char*) &tl,sizeof(tl));
|
|
|
|
tl.db= (char*) "mysql";
|
2005-01-06 13:00:13 +02:00
|
|
|
tl.table_name= (char*) "user";
|
2005-03-22 15:54:18 +01:00
|
|
|
create_new_users= 1;
|
2004-04-07 13:25:24 +03:00
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
db_access=acl_get(sctx->host, sctx->ip,
|
|
|
|
sctx->priv_user, tl.db, 0);
|
2001-08-10 17:37:37 +03:00
|
|
|
if (!(db_access & INSERT_ACL))
|
|
|
|
{
|
2004-04-10 01:14:32 +03:00
|
|
|
if (check_grant(thd, INSERT_ACL, &tl, 0, UINT_MAX, 1))
|
2001-08-10 17:37:37 +03:00
|
|
|
create_new_users=0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return create_new_users;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/****************************************************************************
|
2003-06-23 20:03:59 +03:00
|
|
|
Handle GRANT commands
|
2000-07-31 21:29:14 +02:00
|
|
|
****************************************************************************/
|
|
|
|
|
2001-09-30 10:46:20 +08:00
|
|
|
static int replace_user_table(THD *thd, TABLE *table, const LEX_USER &combo,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong rights, bool revoke_grant,
|
2005-03-23 19:18:25 +01:00
|
|
|
bool can_create_user, bool no_auto_create)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
int error = -1;
|
2001-08-10 17:37:37 +03:00
|
|
|
bool old_row_exists=0;
|
2003-07-18 18:25:54 +04:00
|
|
|
const char *password= "";
|
2003-07-01 23:40:59 +04:00
|
|
|
uint password_len= 0;
|
2002-06-12 15:04:18 +03:00
|
|
|
char what= (revoke_grant) ? 'N' : 'Y';
|
2004-12-06 02:00:37 +02:00
|
|
|
byte user_key[MAX_KEY_LENGTH];
|
2004-11-11 21:57:55 +03:00
|
|
|
LEX *lex= thd->lex;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("replace_user_table");
|
2004-12-06 19:18:35 +02:00
|
|
|
|
2002-09-22 10:35:15 +03:00
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (combo.password.str && combo.password.str[0])
|
2002-05-15 13:50:38 +03:00
|
|
|
{
|
2003-07-01 23:40:59 +04:00
|
|
|
if (combo.password.length != SCRAMBLED_PASSWORD_CHAR_LENGTH &&
|
|
|
|
combo.password.length != SCRAMBLED_PASSWORD_CHAR_LENGTH_323)
|
2002-05-15 13:50:38 +03:00
|
|
|
{
|
2004-11-13 19:35:51 +02:00
|
|
|
my_error(ER_PASSWD_LENGTH, MYF(0), SCRAMBLED_PASSWORD_CHAR_LENGTH);
|
2002-05-16 16:32:51 +03:00
|
|
|
DBUG_RETURN(-1);
|
2002-05-15 13:50:38 +03:00
|
|
|
}
|
2003-07-01 23:40:59 +04:00
|
|
|
password_len= combo.password.length;
|
2000-07-31 21:29:14 +02:00
|
|
|
password=combo.password.str;
|
2002-05-15 13:50:38 +03:00
|
|
|
}
|
2002-11-30 20:58:53 +03:00
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
|
|
|
table->field[0]->store(combo.host.str,combo.host.length,
|
|
|
|
system_charset_info);
|
|
|
|
table->field[1]->store(combo.user.str,combo.user.length,
|
|
|
|
system_charset_info);
|
2004-12-06 02:00:37 +02:00
|
|
|
key_copy(user_key, table->record[0], table->key_info,
|
|
|
|
table->key_info->key_length);
|
|
|
|
|
2007-03-17 00:13:25 +01:00
|
|
|
if (table->file->index_read_idx(table->record[0], 0, user_key, HA_WHOLE_KEY,
|
2004-12-06 02:00:37 +02:00
|
|
|
HA_READ_KEY_EXACT))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-11-02 17:45:26 +03:00
|
|
|
/* what == 'N' means revoke */
|
|
|
|
if (what == 'N')
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-11-02 17:45:26 +03:00
|
|
|
my_error(ER_NONEXISTING_GRANT, MYF(0), combo.user.str, combo.host.str);
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
/*
|
2005-03-22 15:54:18 +01:00
|
|
|
There are four options which affect the process of creation of
|
|
|
|
a new user (mysqld option --safe-create-user, 'insert' privilege
|
2004-11-02 17:45:26 +03:00
|
|
|
on 'mysql.user' table, using 'GRANT' with 'IDENTIFIED BY' and
|
|
|
|
SQL_MODE flag NO_AUTO_CREATE_USER). Below is the simplified rule
|
|
|
|
how it should work.
|
|
|
|
if (safe-user-create && ! INSERT_priv) => reject
|
|
|
|
else if (identified_by) => create
|
|
|
|
else if (no_auto_create_user) => reject
|
|
|
|
else create
|
2005-03-22 15:54:18 +01:00
|
|
|
|
|
|
|
see also test_if_create_new_users()
|
2004-11-02 17:45:26 +03:00
|
|
|
*/
|
2005-03-23 19:18:25 +01:00
|
|
|
else if (!password_len && no_auto_create)
|
|
|
|
{
|
|
|
|
my_error(ER_PASSWORD_NO_MATCH, MYF(0), combo.user.str, combo.host.str);
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
else if (!can_create_user)
|
2004-11-02 17:45:26 +03:00
|
|
|
{
|
2005-03-22 15:57:24 +02:00
|
|
|
my_error(ER_CANT_CREATE_USER_WITH_GRANT, MYF(0),
|
2005-09-15 22:29:07 +03:00
|
|
|
thd->security_ctx->user, thd->security_ctx->host_or_ip);
|
2000-07-31 21:29:14 +02:00
|
|
|
goto end;
|
|
|
|
}
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 0;
|
2005-01-06 13:00:13 +02:00
|
|
|
restore_record(table,s->default_values);
|
2003-07-01 23:40:59 +04:00
|
|
|
table->field[0]->store(combo.host.str,combo.host.length,
|
2004-11-25 21:55:49 +01:00
|
|
|
system_charset_info);
|
2003-07-01 23:40:59 +04:00
|
|
|
table->field[1]->store(combo.user.str,combo.user.length,
|
2004-11-25 21:55:49 +01:00
|
|
|
system_charset_info);
|
2003-07-01 23:40:59 +04:00
|
|
|
table->field[2]->store(password, password_len,
|
2004-11-25 21:55:49 +01:00
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 1;
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]); // Save copy for update
|
2000-07-31 21:29:14 +02:00
|
|
|
if (combo.password.str) // If password given
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[2]->store(password, password_len, system_charset_info);
|
2004-11-11 21:57:55 +03:00
|
|
|
else if (!rights && !revoke_grant &&
|
2004-12-29 20:30:37 +03:00
|
|
|
lex->ssl_type == SSL_TYPE_NOT_SPECIFIED &&
|
|
|
|
!lex->mqh.specified_limits)
|
2004-03-01 21:54:08 +02:00
|
|
|
{
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
/* Update table columns with new privileges */
|
|
|
|
|
|
|
|
Field **tmp_field;
|
|
|
|
ulong priv;
|
2004-12-23 21:59:36 +03:00
|
|
|
uint next_field;
|
2002-06-12 15:04:18 +03:00
|
|
|
for (tmp_field= table->field+3, priv = SELECT_ACL;
|
2006-12-01 17:26:52 -08:00
|
|
|
*tmp_field && (*tmp_field)->real_type() == MYSQL_TYPE_ENUM &&
|
2002-06-12 15:04:18 +03:00
|
|
|
((Field_enum*) (*tmp_field))->typelib->count == 2 ;
|
|
|
|
tmp_field++, priv <<= 1)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
if (priv & rights) // set requested privileges
|
2003-01-29 17:31:20 +04:00
|
|
|
(*tmp_field)->store(&what, 1, &my_charset_latin1);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-12-23 21:59:36 +03:00
|
|
|
rights= get_access(table, 3, &next_field);
|
2005-01-06 13:00:13 +02:00
|
|
|
DBUG_PRINT("info",("table fields: %d",table->s->fields));
|
|
|
|
if (table->s->fields >= 31) /* From 4.0.0 we have more fields */
|
2002-02-18 23:54:45 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
/* We write down SSL related ACL stuff */
|
2004-11-11 21:57:55 +03:00
|
|
|
switch (lex->ssl_type) {
|
2001-09-30 05:47:34 +03:00
|
|
|
case SSL_TYPE_ANY:
|
2005-11-20 20:47:07 +02:00
|
|
|
table->field[next_field]->store(STRING_WITH_LEN("ANY"),
|
|
|
|
&my_charset_latin1);
|
2004-12-23 22:16:43 +03:00
|
|
|
table->field[next_field+1]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+2]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+3]->store("", 0, &my_charset_latin1);
|
2001-09-30 05:47:34 +03:00
|
|
|
break;
|
|
|
|
case SSL_TYPE_X509:
|
2005-11-20 20:47:07 +02:00
|
|
|
table->field[next_field]->store(STRING_WITH_LEN("X509"),
|
|
|
|
&my_charset_latin1);
|
2004-12-23 22:16:43 +03:00
|
|
|
table->field[next_field+1]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+2]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+3]->store("", 0, &my_charset_latin1);
|
2001-09-30 05:47:34 +03:00
|
|
|
break;
|
|
|
|
case SSL_TYPE_SPECIFIED:
|
2005-11-20 20:47:07 +02:00
|
|
|
table->field[next_field]->store(STRING_WITH_LEN("SPECIFIED"),
|
|
|
|
&my_charset_latin1);
|
2004-12-23 22:16:43 +03:00
|
|
|
table->field[next_field+1]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+2]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+3]->store("", 0, &my_charset_latin1);
|
2004-11-11 21:57:55 +03:00
|
|
|
if (lex->ssl_cipher)
|
2004-12-31 15:26:24 +01:00
|
|
|
table->field[next_field+1]->store(lex->ssl_cipher,
|
|
|
|
strlen(lex->ssl_cipher), system_charset_info);
|
2004-11-11 21:57:55 +03:00
|
|
|
if (lex->x509_issuer)
|
2004-12-31 15:26:24 +01:00
|
|
|
table->field[next_field+2]->store(lex->x509_issuer,
|
|
|
|
strlen(lex->x509_issuer), system_charset_info);
|
2004-11-11 21:57:55 +03:00
|
|
|
if (lex->x509_subject)
|
2004-12-31 15:26:24 +01:00
|
|
|
table->field[next_field+3]->store(lex->x509_subject,
|
|
|
|
strlen(lex->x509_subject), system_charset_info);
|
2001-09-30 05:47:34 +03:00
|
|
|
break;
|
2002-09-11 06:40:08 +03:00
|
|
|
case SSL_TYPE_NOT_SPECIFIED:
|
2002-10-30 18:27:15 +04:00
|
|
|
break;
|
|
|
|
case SSL_TYPE_NONE:
|
2004-12-23 22:16:43 +03:00
|
|
|
table->field[next_field]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+1]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+2]->store("", 0, &my_charset_latin1);
|
|
|
|
table->field[next_field+3]->store("", 0, &my_charset_latin1);
|
2002-10-30 18:27:15 +04:00
|
|
|
break;
|
2001-09-30 10:46:20 +08:00
|
|
|
}
|
2004-12-31 15:26:24 +01:00
|
|
|
next_field+=4;
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2004-11-11 21:57:55 +03:00
|
|
|
USER_RESOURCES mqh= lex->mqh;
|
2004-12-29 20:30:37 +03:00
|
|
|
if (mqh.specified_limits & USER_RESOURCES::QUERIES_PER_HOUR)
|
2005-09-14 01:41:44 +03:00
|
|
|
table->field[next_field]->store((longlong) mqh.questions, TRUE);
|
2004-12-29 20:30:37 +03:00
|
|
|
if (mqh.specified_limits & USER_RESOURCES::UPDATES_PER_HOUR)
|
2005-09-14 01:41:44 +03:00
|
|
|
table->field[next_field+1]->store((longlong) mqh.updates, TRUE);
|
2004-12-29 20:30:37 +03:00
|
|
|
if (mqh.specified_limits & USER_RESOURCES::CONNECTIONS_PER_HOUR)
|
2005-09-14 01:41:44 +03:00
|
|
|
table->field[next_field+2]->store((longlong) mqh.conn_per_hour, TRUE);
|
2005-01-06 13:00:13 +02:00
|
|
|
if (table->s->fields >= 36 &&
|
2004-12-29 20:30:37 +03:00
|
|
|
(mqh.specified_limits & USER_RESOURCES::USER_CONNECTIONS))
|
2006-06-16 12:17:20 +02:00
|
|
|
table->field[next_field+3]->store((longlong) mqh.user_conn, TRUE);
|
2004-12-29 20:30:37 +03:00
|
|
|
mqh_used= mqh_used || mqh.questions || mqh.updates || mqh.conn_per_hour;
|
2001-12-26 16:49:10 +02:00
|
|
|
}
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
We should NEVER delete from the user table, as a uses can still
|
|
|
|
use mysqld even if he doesn't have any privileges in the user table!
|
|
|
|
*/
|
2003-05-03 01:16:56 +02:00
|
|
|
if (cmp_record(table,record[1]) &&
|
2005-12-22 06:39:02 +01:00
|
|
|
(error=table->file->ha_update_row(table->record[1],table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // This should never happen
|
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: deadcode */
|
|
|
|
error= -1; /* purecov: deadcode */
|
|
|
|
goto end; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
}
|
2005-12-22 06:39:02 +01:00
|
|
|
else if ((error=table->file->ha_write_row(table->record[0]))) // insert
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // This should never happen
|
2006-07-01 00:01:37 -04:00
|
|
|
if (table->file->is_fatal_error(error, HA_CHECK_DUP))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: deadcode */
|
|
|
|
error= -1; /* purecov: deadcode */
|
|
|
|
goto end; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
error=0; // Privileges granted / revoked
|
|
|
|
|
2001-09-30 05:47:34 +03:00
|
|
|
end:
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!error)
|
|
|
|
{
|
|
|
|
acl_cache->clear(1); // Clear privilege cache
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2004-05-04 15:02:38 +03:00
|
|
|
acl_update_user(combo.user.str, combo.host.str,
|
|
|
|
combo.password.str, password_len,
|
2004-11-11 21:57:55 +03:00
|
|
|
lex->ssl_type,
|
|
|
|
lex->ssl_cipher,
|
|
|
|
lex->x509_issuer,
|
|
|
|
lex->x509_subject,
|
|
|
|
&lex->mqh,
|
2001-09-30 05:47:34 +03:00
|
|
|
rights);
|
2000-07-31 21:29:14 +02:00
|
|
|
else
|
2003-07-01 23:40:59 +04:00
|
|
|
acl_insert_user(combo.user.str, combo.host.str, password, password_len,
|
2004-11-11 21:57:55 +03:00
|
|
|
lex->ssl_type,
|
|
|
|
lex->ssl_cipher,
|
|
|
|
lex->x509_issuer,
|
|
|
|
lex->x509_subject,
|
|
|
|
&lex->mqh,
|
2001-09-30 05:47:34 +03:00
|
|
|
rights);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2002-06-12 15:04:18 +03:00
|
|
|
change grants in the mysql.db table
|
2000-07-31 21:29:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
static int replace_db_table(TABLE *table, const char *db,
|
|
|
|
const LEX_USER &combo,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong rights, bool revoke_grant)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
uint i;
|
|
|
|
ulong priv,store_rights;
|
2001-08-10 17:37:37 +03:00
|
|
|
bool old_row_exists=0;
|
2000-07-31 21:29:14 +02:00
|
|
|
int error;
|
2002-06-12 15:04:18 +03:00
|
|
|
char what= (revoke_grant) ? 'N' : 'Y';
|
2004-12-06 02:00:37 +02:00
|
|
|
byte user_key[MAX_KEY_LENGTH];
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("replace_db_table");
|
|
|
|
|
2004-01-26 20:31:38 +04:00
|
|
|
if (!initialized)
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--skip-grant-tables");
|
2004-01-26 20:31:38 +04:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* Check if there is such a user in user table in memory? */
|
2005-08-22 15:48:50 -07:00
|
|
|
if (!find_acl_user(combo.host.str,combo.user.str, FALSE))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_PASSWORD_NO_MATCH, ER(ER_PASSWORD_NO_MATCH), MYF(0));
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
|
|
|
table->field[0]->store(combo.host.str,combo.host.length,
|
|
|
|
system_charset_info);
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[1]->store(db,(uint) strlen(db), system_charset_info);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->field[2]->store(combo.user.str,combo.user.length,
|
|
|
|
system_charset_info);
|
2004-12-06 02:00:37 +02:00
|
|
|
key_copy(user_key, table->record[0], table->key_info,
|
|
|
|
table->key_info->key_length);
|
|
|
|
|
2007-03-17 00:13:25 +01:00
|
|
|
if (table->file->index_read_idx(table->record[0],0, user_key, HA_WHOLE_KEY,
|
2004-12-06 02:00:37 +02:00
|
|
|
HA_READ_KEY_EXACT))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (what == 'N')
|
|
|
|
{ // no row, no revoke
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_NONEXISTING_GRANT, MYF(0), combo.user.str, combo.host.str);
|
2000-07-31 21:29:14 +02:00
|
|
|
goto abort;
|
|
|
|
}
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 0;
|
2005-01-06 13:00:13 +02:00
|
|
|
restore_record(table, s->default_values);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->field[0]->store(combo.host.str,combo.host.length,
|
|
|
|
system_charset_info);
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[1]->store(db,(uint) strlen(db), system_charset_info);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->field[2]->store(combo.user.str,combo.user.length,
|
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 1;
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
store_rights=get_rights_for_db(rights);
|
2005-01-06 13:00:13 +02:00
|
|
|
for (i= 3, priv= 1; i < table->s->fields; i++, priv <<= 1)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
if (priv & store_rights) // do it if priv is chosen
|
2003-01-29 17:31:20 +04:00
|
|
|
table->field [i]->store(&what,1, &my_charset_latin1);// set requested privileges
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
rights=get_access(table,3);
|
|
|
|
rights=fix_rights_for_db(rights);
|
|
|
|
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
/* update old existing row */
|
2000-07-31 21:29:14 +02:00
|
|
|
if (rights)
|
|
|
|
{
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
if ((error= table->file->ha_update_row(table->record[1],
|
|
|
|
table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto table_error; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
else /* must have been a revoke of all privileges */
|
|
|
|
{
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
if ((error= table->file->ha_delete_row(table->record[1])))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto table_error; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
}
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
else if (rights && (error= table->file->ha_write_row(table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-07-01 00:01:37 -04:00
|
|
|
if (table->file->is_fatal_error(error, HA_CHECK_DUP_KEY))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto table_error; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
|
|
|
|
acl_cache->clear(1); // Clear privilege cache
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_update_db(combo.user.str,combo.host.str,db,rights);
|
|
|
|
else
|
2004-11-11 20:59:03 +00:00
|
|
|
if (rights)
|
2000-07-31 21:29:14 +02:00
|
|
|
acl_insert_db(combo.user.str,combo.host.str,db,rights);
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
/* This could only happen if the grant tables got corrupted */
|
2003-06-23 20:03:59 +03:00
|
|
|
table_error:
|
2000-07-31 21:29:14 +02:00
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: deadcode */
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
abort:
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class GRANT_COLUMN :public Sql_alloc
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
char *column;
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong rights;
|
|
|
|
uint key_length;
|
|
|
|
GRANT_COLUMN(String &c, ulong y) :rights (y)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
column= memdup_root(&memex,c.ptr(), key_length=c.length());
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
static byte* get_key_column(GRANT_COLUMN *buff,uint *length,
|
|
|
|
my_bool not_used __attribute__((unused)))
|
|
|
|
{
|
|
|
|
*length=buff->key_length;
|
|
|
|
return (byte*) buff->column;
|
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
class GRANT_NAME :public Sql_alloc
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
public:
|
2005-03-02 16:30:24 -08:00
|
|
|
acl_host_and_ip host;
|
|
|
|
char *db, *user, *tname, *hash_key;
|
2004-12-23 10:46:24 +00:00
|
|
|
ulong privs;
|
2003-09-02 11:18:13 +02:00
|
|
|
ulong sort;
|
2002-06-12 15:04:18 +03:00
|
|
|
uint key_length;
|
2004-12-23 10:46:24 +00:00
|
|
|
GRANT_NAME(const char *h, const char *d,const char *u,
|
|
|
|
const char *t, ulong p);
|
|
|
|
GRANT_NAME (TABLE *form);
|
|
|
|
virtual ~GRANT_NAME() {};
|
|
|
|
virtual bool ok() { return privs != 0; }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class GRANT_TABLE :public GRANT_NAME
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
ulong cols;
|
2000-07-31 21:29:14 +02:00
|
|
|
HASH hash_columns;
|
2004-05-05 17:05:24 +03:00
|
|
|
|
|
|
|
GRANT_TABLE(const char *h, const char *d,const char *u,
|
|
|
|
const char *t, ulong p, ulong c);
|
|
|
|
GRANT_TABLE (TABLE *form, TABLE *col_privs);
|
2004-11-25 21:55:49 +01:00
|
|
|
~GRANT_TABLE();
|
2004-04-28 17:45:08 +03:00
|
|
|
bool ok() { return privs != 0 || cols != 0; }
|
|
|
|
};
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-04-28 17:45:08 +03:00
|
|
|
|
2004-05-05 17:05:24 +03:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
GRANT_NAME::GRANT_NAME(const char *h, const char *d,const char *u,
|
|
|
|
const char *t, ulong p)
|
|
|
|
:privs(p)
|
2004-04-28 17:45:08 +03:00
|
|
|
{
|
|
|
|
/* Host given by user */
|
2005-03-02 16:30:24 -08:00
|
|
|
update_hostname(&host, strdup_root(&memex, h));
|
2004-04-28 17:45:08 +03:00
|
|
|
db = strdup_root(&memex,d);
|
|
|
|
user = strdup_root(&memex,u);
|
2005-03-02 16:30:24 -08:00
|
|
|
sort= get_sort(3,host.hostname,db,user);
|
2004-04-28 17:45:08 +03:00
|
|
|
tname= strdup_root(&memex,t);
|
|
|
|
if (lower_case_table_names)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-05-22 23:41:58 +04:00
|
|
|
my_casedn_str(files_charset_info, db);
|
|
|
|
my_casedn_str(files_charset_info, tname);
|
2004-04-28 17:45:08 +03:00
|
|
|
}
|
|
|
|
key_length =(uint) strlen(d)+(uint) strlen(u)+(uint) strlen(t)+3;
|
|
|
|
hash_key = (char*) alloc_root(&memex,key_length);
|
|
|
|
strmov(strmov(strmov(hash_key,user)+1,db)+1,tname);
|
2004-12-23 10:46:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GRANT_TABLE::GRANT_TABLE(const char *h, const char *d,const char *u,
|
|
|
|
const char *t, ulong p, ulong c)
|
|
|
|
:GRANT_NAME(h,d,u,t,p), cols(c)
|
|
|
|
{
|
2004-11-25 21:55:49 +01:00
|
|
|
(void) hash_init(&hash_columns,system_charset_info,
|
2004-05-05 17:05:24 +03:00
|
|
|
0,0,0, (hash_get_key) get_key_column,0,0);
|
2004-04-28 17:45:08 +03:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-04-28 17:45:08 +03:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
GRANT_NAME::GRANT_NAME(TABLE *form)
|
2004-04-28 17:45:08 +03:00
|
|
|
{
|
2005-03-02 16:30:24 -08:00
|
|
|
update_hostname(&host, get_field(&memex, form->field[0]));
|
2004-05-05 17:05:24 +03:00
|
|
|
db= get_field(&memex,form->field[1]);
|
|
|
|
user= get_field(&memex,form->field[2]);
|
2004-04-28 17:45:08 +03:00
|
|
|
if (!user)
|
|
|
|
user= (char*) "";
|
2005-03-02 16:30:24 -08:00
|
|
|
sort= get_sort(3, host.hostname, db, user);
|
2004-05-05 17:05:24 +03:00
|
|
|
tname= get_field(&memex,form->field[3]);
|
2004-04-28 17:45:08 +03:00
|
|
|
if (!db || !tname)
|
|
|
|
{
|
|
|
|
/* Wrong table row; Ignore it */
|
2004-12-23 10:46:24 +00:00
|
|
|
privs= 0;
|
2004-04-28 17:45:08 +03:00
|
|
|
return; /* purecov: inspected */
|
|
|
|
}
|
|
|
|
if (lower_case_table_names)
|
|
|
|
{
|
2004-05-22 23:41:58 +04:00
|
|
|
my_casedn_str(files_charset_info, db);
|
|
|
|
my_casedn_str(files_charset_info, tname);
|
2004-04-28 17:45:08 +03:00
|
|
|
}
|
|
|
|
key_length = ((uint) strlen(db) + (uint) strlen(user) +
|
|
|
|
(uint) strlen(tname) + 3);
|
|
|
|
hash_key = (char*) alloc_root(&memex,key_length);
|
|
|
|
strmov(strmov(strmov(hash_key,user)+1,db)+1,tname);
|
|
|
|
privs = (ulong) form->field[6]->val_int();
|
|
|
|
privs = fix_rights_for_table(privs);
|
2004-12-23 10:46:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GRANT_TABLE::GRANT_TABLE(TABLE *form, TABLE *col_privs)
|
|
|
|
:GRANT_NAME(form)
|
|
|
|
{
|
|
|
|
byte key[MAX_KEY_LENGTH];
|
|
|
|
|
|
|
|
if (!db || !tname)
|
|
|
|
{
|
|
|
|
/* Wrong table row; Ignore it */
|
|
|
|
hash_clear(&hash_columns); /* allow for destruction */
|
|
|
|
cols= 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
cols= (ulong) form->field[7]->val_int();
|
2004-04-28 17:45:08 +03:00
|
|
|
cols = fix_rights_for_column(cols);
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
(void) hash_init(&hash_columns,system_charset_info,
|
2004-05-05 17:05:24 +03:00
|
|
|
0,0,0, (hash_get_key) get_key_column,0,0);
|
2004-04-28 17:45:08 +03:00
|
|
|
if (cols)
|
|
|
|
{
|
2004-12-06 02:00:37 +02:00
|
|
|
uint key_prefix_len;
|
|
|
|
KEY_PART_INFO *key_part= col_privs->key_info->key_part;
|
2005-06-20 10:21:35 -07:00
|
|
|
col_privs->field[0]->store(host.hostname,
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
host.hostname ? (uint) strlen(host.hostname) :
|
|
|
|
0,
|
2004-11-25 21:55:49 +01:00
|
|
|
system_charset_info);
|
|
|
|
col_privs->field[1]->store(db,(uint) strlen(db), system_charset_info);
|
|
|
|
col_privs->field[2]->store(user,(uint) strlen(user), system_charset_info);
|
|
|
|
col_privs->field[3]->store(tname,(uint) strlen(tname), system_charset_info);
|
2004-12-06 02:00:37 +02:00
|
|
|
|
|
|
|
key_prefix_len= (key_part[0].store_length +
|
|
|
|
key_part[1].store_length +
|
|
|
|
key_part[2].store_length +
|
|
|
|
key_part[3].store_length);
|
|
|
|
key_copy(key, col_privs->record[0], col_privs->key_info, key_prefix_len);
|
2004-05-05 17:05:24 +03:00
|
|
|
col_privs->field[4]->store("",0, &my_charset_latin1);
|
2004-12-06 02:00:37 +02:00
|
|
|
|
2005-07-18 13:31:02 +02:00
|
|
|
col_privs->file->ha_index_init(0, 1);
|
2007-01-29 10:40:26 +01:00
|
|
|
if (col_privs->file->index_read(col_privs->record[0], (byte*) key,
|
2007-03-17 00:13:25 +01:00
|
|
|
(key_part_map)15, HA_READ_KEY_EXACT))
|
2001-12-13 02:31:19 +02:00
|
|
|
{
|
2004-04-28 17:45:08 +03:00
|
|
|
cols = 0; /* purecov: deadcode */
|
2004-07-06 16:30:50 +01:00
|
|
|
col_privs->file->ha_index_end();
|
2004-04-28 17:45:08 +03:00
|
|
|
return;
|
2001-12-13 02:31:19 +02:00
|
|
|
}
|
2004-04-28 17:45:08 +03:00
|
|
|
do
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-04-28 17:45:08 +03:00
|
|
|
String *res,column_name;
|
|
|
|
GRANT_COLUMN *mem_check;
|
|
|
|
/* As column name is a string, we don't have to supply a buffer */
|
2004-05-05 17:05:24 +03:00
|
|
|
res=col_privs->field[4]->val_str(&column_name);
|
2004-04-28 17:45:08 +03:00
|
|
|
ulong priv= (ulong) col_privs->field[6]->val_int();
|
|
|
|
if (!(mem_check = new GRANT_COLUMN(*res,
|
|
|
|
fix_rights_for_column(priv))))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-04-28 17:45:08 +03:00
|
|
|
/* Don't use this entry */
|
|
|
|
privs = cols = 0; /* purecov: deadcode */
|
|
|
|
return; /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-05-05 17:05:24 +03:00
|
|
|
my_hash_insert(&hash_columns, (byte *) mem_check);
|
2004-04-28 17:45:08 +03:00
|
|
|
} while (!col_privs->file->index_next(col_privs->record[0]) &&
|
2004-12-06 02:00:37 +02:00
|
|
|
!key_cmp_if_same(col_privs,key,0,key_prefix_len));
|
2004-07-06 16:30:50 +01:00
|
|
|
col_privs->file->ha_index_end();
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-04-28 17:45:08 +03:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
GRANT_TABLE::~GRANT_TABLE()
|
|
|
|
{
|
|
|
|
hash_free(&hash_columns);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
static byte* get_grant_table(GRANT_NAME *buff,uint *length,
|
2000-07-31 21:29:14 +02:00
|
|
|
my_bool not_used __attribute__((unused)))
|
|
|
|
{
|
|
|
|
*length=buff->key_length;
|
|
|
|
return (byte*) buff->hash_key;
|
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
void free_grant_table(GRANT_TABLE *grant_table)
|
|
|
|
{
|
|
|
|
hash_free(&grant_table->hash_columns);
|
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Search after a matching grant. Prefer exact grants before not exact ones */
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
static GRANT_NAME *name_hash_search(HASH *name_hash,
|
2005-11-23 22:45:02 +02:00
|
|
|
const char *host,const char* ip,
|
|
|
|
const char *db,
|
|
|
|
const char *user, const char *tname,
|
|
|
|
bool exact)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-09-27 19:21:29 +05:00
|
|
|
char helping [NAME_LEN*2+USERNAME_LENGTH+3];
|
2000-07-31 21:29:14 +02:00
|
|
|
uint len;
|
2004-12-23 10:46:24 +00:00
|
|
|
GRANT_NAME *grant_name,*found=0;
|
2006-01-04 17:35:30 +03:00
|
|
|
HASH_SEARCH_STATE state;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
len = (uint) (strmov(strmov(strmov(helping,user)+1,db)+1,tname)-helping)+ 1;
|
2006-01-04 17:49:45 +03:00
|
|
|
for (grant_name= (GRANT_NAME*) hash_first(name_hash, (byte*) helping,
|
|
|
|
len, &state);
|
2004-12-23 10:46:24 +00:00
|
|
|
grant_name ;
|
|
|
|
grant_name= (GRANT_NAME*) hash_next(name_hash,(byte*) helping,
|
2006-01-04 17:49:45 +03:00
|
|
|
len, &state))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (exact)
|
|
|
|
{
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!grant_name->host.hostname ||
|
|
|
|
(host &&
|
2006-03-06 14:03:40 +04:00
|
|
|
!my_strcasecmp(system_charset_info, host,
|
2006-03-06 14:38:31 +04:00
|
|
|
grant_name->host.hostname)) ||
|
|
|
|
(ip && !strcmp(ip, grant_name->host.hostname)))
|
2004-12-23 10:46:24 +00:00
|
|
|
return grant_name;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-03-03 17:44:28 -08:00
|
|
|
if (compare_hostname(&grant_name->host, host, ip) &&
|
2004-12-23 10:46:24 +00:00
|
|
|
(!found || found->sort < grant_name->sort))
|
|
|
|
found=grant_name; // Host ok
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return found;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
inline GRANT_NAME *
|
2005-05-17 19:54:20 +01:00
|
|
|
routine_hash_search(const char *host, const char *ip, const char *db,
|
|
|
|
const char *user, const char *tname, bool proc, bool exact)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
return (GRANT_TABLE*)
|
|
|
|
name_hash_search(proc ? &proc_priv_hash : &func_priv_hash,
|
|
|
|
host, ip, db, user, tname, exact);
|
2004-12-23 10:46:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline GRANT_TABLE *
|
|
|
|
table_hash_search(const char *host, const char *ip, const char *db,
|
|
|
|
const char *user, const char *tname, bool exact)
|
|
|
|
{
|
|
|
|
return (GRANT_TABLE*) name_hash_search(&column_priv_hash, host, ip, db,
|
|
|
|
user, tname, exact);
|
|
|
|
}
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2001-01-25 22:38:26 +02:00
|
|
|
inline GRANT_COLUMN *
|
2002-06-12 15:04:18 +03:00
|
|
|
column_hash_search(GRANT_TABLE *t, const char *cname, uint length)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
return (GRANT_COLUMN*) hash_search(&t->hash_columns, (byte*) cname,length);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int replace_column_table(GRANT_TABLE *g_t,
|
|
|
|
TABLE *table, const LEX_USER &combo,
|
|
|
|
List <LEX_COLUMN> &columns,
|
|
|
|
const char *db, const char *table_name,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong rights, bool revoke_grant)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
int error=0,result=0;
|
|
|
|
byte key[MAX_KEY_LENGTH];
|
2004-12-06 02:00:37 +02:00
|
|
|
uint key_prefix_length;
|
|
|
|
KEY_PART_INFO *key_part= table->key_info->key_part;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("replace_column_table");
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2005-05-06 11:39:30 +03:00
|
|
|
table->field[0]->store(combo.host.str,combo.host.length,
|
|
|
|
system_charset_info);
|
|
|
|
table->field[1]->store(db,(uint) strlen(db),
|
|
|
|
system_charset_info);
|
|
|
|
table->field[2]->store(combo.user.str,combo.user.length,
|
|
|
|
system_charset_info);
|
|
|
|
table->field[3]->store(table_name,(uint) strlen(table_name),
|
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2007-01-29 10:40:26 +01:00
|
|
|
/* Get length of 4 first key parts */
|
2004-12-06 02:00:37 +02:00
|
|
|
key_prefix_length= (key_part[0].store_length + key_part[1].store_length +
|
|
|
|
key_part[2].store_length + key_part[3].store_length);
|
|
|
|
key_copy(key, table->record[0], table->key_info, key_prefix_length);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-12-06 02:00:37 +02:00
|
|
|
rights&= COL_ACLS; // Only ACL for columns
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* first fix privileges for all columns in column list */
|
|
|
|
|
|
|
|
List_iterator <LEX_COLUMN> iter(columns);
|
2005-05-06 11:39:30 +03:00
|
|
|
class LEX_COLUMN *column;
|
2005-07-18 13:31:02 +02:00
|
|
|
table->file->ha_index_init(0, 1);
|
2005-05-06 11:39:30 +03:00
|
|
|
while ((column= iter++))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-05-06 11:39:30 +03:00
|
|
|
ulong privileges= column->rights;
|
2001-08-10 17:37:37 +03:00
|
|
|
bool old_row_exists=0;
|
2004-12-06 02:00:37 +02:00
|
|
|
byte user_key[MAX_KEY_LENGTH];
|
|
|
|
|
|
|
|
key_restore(table->record[0],key,table->key_info,
|
|
|
|
key_prefix_length);
|
2005-05-06 11:39:30 +03:00
|
|
|
table->field[4]->store(column->column.ptr(), column->column.length(),
|
2004-11-25 21:55:49 +01:00
|
|
|
system_charset_info);
|
2004-12-06 02:00:37 +02:00
|
|
|
/* Get key for the first 4 columns */
|
|
|
|
key_copy(user_key, table->record[0], table->key_info,
|
|
|
|
table->key_info->key_length);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2007-03-17 00:13:25 +01:00
|
|
|
if (table->file->index_read(table->record[0], user_key, HA_WHOLE_KEY,
|
2004-12-06 02:00:37 +02:00
|
|
|
HA_READ_KEY_EXACT))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_NONEXISTING_TABLE_GRANT, MYF(0),
|
2004-11-13 19:35:51 +02:00
|
|
|
combo.user.str, combo.host.str,
|
2005-05-06 11:39:30 +03:00
|
|
|
table_name); /* purecov: inspected */
|
|
|
|
result= -1; /* purecov: inspected */
|
|
|
|
continue; /* purecov: inspected */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 0;
|
2005-01-06 13:00:13 +02:00
|
|
|
restore_record(table, s->default_values); // Get empty record
|
2004-12-06 02:00:37 +02:00
|
|
|
key_restore(table->record[0],key,table->key_info,
|
|
|
|
key_prefix_length);
|
2005-05-06 11:39:30 +03:00
|
|
|
table->field[4]->store(column->column.ptr(),column->column.length(),
|
2004-11-25 21:55:49 +01:00
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong tmp= (ulong) table->field[6]->val_int();
|
2000-07-31 21:29:14 +02:00
|
|
|
tmp=fix_rights_for_column(tmp);
|
|
|
|
|
|
|
|
if (revoke_grant)
|
|
|
|
privileges = tmp & ~(privileges | rights);
|
|
|
|
else
|
|
|
|
privileges |= tmp;
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 1;
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]); // copy original row
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2005-09-14 01:41:44 +03:00
|
|
|
table->field[6]->store((longlong) get_rights_for_column(privileges), TRUE);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-05-06 11:39:30 +03:00
|
|
|
GRANT_COLUMN *grant_column;
|
2000-07-31 21:29:14 +02:00
|
|
|
if (privileges)
|
2005-12-22 06:39:02 +01:00
|
|
|
error=table->file->ha_update_row(table->record[1],table->record[0]);
|
2000-07-31 21:29:14 +02:00
|
|
|
else
|
2005-12-22 06:39:02 +01:00
|
|
|
error=table->file->ha_delete_row(table->record[1]);
|
2000-07-31 21:29:14 +02:00
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: inspected */
|
|
|
|
result= -1; /* purecov: inspected */
|
|
|
|
goto end; /* purecov: inspected */
|
|
|
|
}
|
2005-05-06 11:39:30 +03:00
|
|
|
grant_column= column_hash_search(g_t, column->column.ptr(),
|
|
|
|
column->column.length());
|
2000-07-31 21:29:14 +02:00
|
|
|
if (grant_column) // Should always be true
|
2005-05-06 11:39:30 +03:00
|
|
|
grant_column->rights= privileges; // Update hash
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else // new grant
|
|
|
|
{
|
2005-05-06 11:39:30 +03:00
|
|
|
GRANT_COLUMN *grant_column;
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((error=table->file->ha_write_row(table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: inspected */
|
|
|
|
result= -1; /* purecov: inspected */
|
|
|
|
goto end; /* purecov: inspected */
|
|
|
|
}
|
2005-05-06 11:39:30 +03:00
|
|
|
grant_column= new GRANT_COLUMN(column->column,privileges);
|
2003-09-19 14:44:31 +05:00
|
|
|
my_hash_insert(&g_t->hash_columns,(byte*) grant_column);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
If revoke of privileges on the table level, remove all such privileges
|
|
|
|
for all columns
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
2004-12-06 02:00:37 +02:00
|
|
|
byte user_key[MAX_KEY_LENGTH];
|
|
|
|
key_copy(user_key, table->record[0], table->key_info,
|
2004-12-06 19:18:35 +02:00
|
|
|
key_prefix_length);
|
|
|
|
|
2007-03-17 00:13:25 +01:00
|
|
|
if (table->file->index_read(table->record[0], user_key, (key_part_map)15,
|
2004-12-06 02:00:37 +02:00
|
|
|
HA_READ_KEY_EXACT))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto end;
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* Scan through all rows with the same host,db,user and table */
|
2000-07-31 21:29:14 +02:00
|
|
|
do
|
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong privileges = (ulong) table->field[6]->val_int();
|
2000-07-31 21:29:14 +02:00
|
|
|
privileges=fix_rights_for_column(privileges);
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (privileges & rights) // is in this record the priv to be revoked ??
|
|
|
|
{
|
|
|
|
GRANT_COLUMN *grant_column = NULL;
|
|
|
|
char colum_name_buf[HOSTNAME_LENGTH+1];
|
2004-12-06 02:00:37 +02:00
|
|
|
String column_name(colum_name_buf,sizeof(colum_name_buf),
|
2004-12-06 19:18:35 +02:00
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
privileges&= ~rights;
|
|
|
|
table->field[6]->store((longlong)
|
2005-09-14 01:41:44 +03:00
|
|
|
get_rights_for_column(privileges), TRUE);
|
2004-04-06 21:35:26 +02:00
|
|
|
table->field[4]->val_str(&column_name);
|
2000-07-31 21:29:14 +02:00
|
|
|
grant_column = column_hash_search(g_t,
|
|
|
|
column_name.ptr(),
|
|
|
|
column_name.length());
|
|
|
|
if (privileges)
|
|
|
|
{
|
|
|
|
int tmp_error;
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((tmp_error=table->file->ha_update_row(table->record[1],
|
|
|
|
table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ /* purecov: deadcode */
|
|
|
|
table->file->print_error(tmp_error,MYF(0)); /* purecov: deadcode */
|
|
|
|
result= -1; /* purecov: deadcode */
|
|
|
|
goto end; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
if (grant_column)
|
|
|
|
grant_column->rights = privileges; // Update hash
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int tmp_error;
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((tmp_error = table->file->ha_delete_row(table->record[1])))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ /* purecov: deadcode */
|
|
|
|
table->file->print_error(tmp_error,MYF(0)); /* purecov: deadcode */
|
|
|
|
result= -1; /* purecov: deadcode */
|
|
|
|
goto end; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
if (grant_column)
|
|
|
|
hash_delete(&g_t->hash_columns,(byte*) grant_column);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (!table->file->index_next(table->record[0]) &&
|
2004-12-06 02:00:37 +02:00
|
|
|
!key_cmp_if_same(table, key, 0, key_prefix_length));
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
end:
|
2004-06-23 12:29:05 +02:00
|
|
|
table->file->ha_index_end();
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int replace_table_table(THD *thd, GRANT_TABLE *grant_table,
|
|
|
|
TABLE *table, const LEX_USER &combo,
|
|
|
|
const char *db, const char *table_name,
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong rights, ulong col_rights,
|
|
|
|
bool revoke_grant)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-01-11 02:07:40 +03:00
|
|
|
char grantor[USER_HOST_BUFF_SIZE];
|
2001-08-10 17:37:37 +03:00
|
|
|
int old_row_exists = 1;
|
2000-07-31 21:29:14 +02:00
|
|
|
int error=0;
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong store_table_rights, store_col_rights;
|
2004-12-06 02:00:37 +02:00
|
|
|
byte user_key[MAX_KEY_LENGTH];
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("replace_table_table");
|
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
strxmov(grantor, thd->security_ctx->user, "@",
|
|
|
|
thd->security_ctx->host_or_ip, NullS);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
/*
|
|
|
|
The following should always succeed as new users are created before
|
|
|
|
this function is called!
|
|
|
|
*/
|
2005-08-22 15:48:50 -07:00
|
|
|
if (!find_acl_user(combo.host.str,combo.user.str, FALSE))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_PASSWORD_NO_MATCH, ER(ER_PASSWORD_NO_MATCH),
|
|
|
|
MYF(0)); /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(-1); /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2005-01-06 13:00:13 +02:00
|
|
|
restore_record(table, s->default_values); // Get empty record
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->field[0]->store(combo.host.str,combo.host.length,
|
|
|
|
system_charset_info);
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[1]->store(db,(uint) strlen(db), system_charset_info);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->field[2]->store(combo.user.str,combo.user.length,
|
|
|
|
system_charset_info);
|
|
|
|
table->field[3]->store(table_name,(uint) strlen(table_name),
|
|
|
|
system_charset_info);
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]); // store at pos 1
|
2004-12-06 02:00:37 +02:00
|
|
|
key_copy(user_key, table->record[0], table->key_info,
|
|
|
|
table->key_info->key_length);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2007-03-17 00:13:25 +01:00
|
|
|
if (table->file->index_read_idx(table->record[0], 0, user_key, HA_WHOLE_KEY,
|
2000-07-31 21:29:14 +02:00
|
|
|
HA_READ_KEY_EXACT))
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
The following should never happen as we first check the in memory
|
|
|
|
grant tables for the user. There is however always a small change that
|
|
|
|
the user has modified the grant tables directly.
|
|
|
|
*/
|
|
|
|
if (revoke_grant)
|
|
|
|
{ // no row, no revoke
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_NONEXISTING_TABLE_GRANT, MYF(0),
|
|
|
|
combo.user.str, combo.host.str,
|
2004-11-13 19:35:51 +02:00
|
|
|
table_name); /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(-1); /* purecov: deadcode */
|
|
|
|
}
|
2001-08-10 17:37:37 +03:00
|
|
|
old_row_exists = 0;
|
2003-05-03 01:16:56 +02:00
|
|
|
restore_record(table,record[1]); // Get saved record
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
store_table_rights= get_rights_for_table(rights);
|
|
|
|
store_col_rights= get_rights_for_column(col_rights);
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong j,k;
|
2003-05-03 01:16:56 +02:00
|
|
|
store_record(table,record[1]);
|
2002-06-12 15:04:18 +03:00
|
|
|
j = (ulong) table->field[6]->val_int();
|
|
|
|
k = (ulong) table->field[7]->val_int();
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
/* column rights are already fixed in mysql_table_grant */
|
2000-07-31 21:29:14 +02:00
|
|
|
store_table_rights=j & ~store_table_rights;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
store_table_rights|= j;
|
|
|
|
store_col_rights|= k;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
table->field[4]->store(grantor,(uint) strlen(grantor), system_charset_info);
|
2005-09-14 01:41:44 +03:00
|
|
|
table->field[6]->store((longlong) store_table_rights, TRUE);
|
|
|
|
table->field[7]->store((longlong) store_col_rights, TRUE);
|
2000-07-31 21:29:14 +02:00
|
|
|
rights=fix_rights_for_table(store_table_rights);
|
2002-06-12 15:04:18 +03:00
|
|
|
col_rights=fix_rights_for_column(store_col_rights);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2001-08-10 17:37:37 +03:00
|
|
|
if (old_row_exists)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (store_table_rights || store_col_rights)
|
|
|
|
{
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((error=table->file->ha_update_row(table->record[1],table->record[0])))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto table_error; /* purecov: deadcode */
|
|
|
|
}
|
2005-12-22 06:39:02 +01:00
|
|
|
else if ((error = table->file->ha_delete_row(table->record[1])))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto table_error; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-12-22 06:39:02 +01:00
|
|
|
error=table->file->ha_write_row(table->record[0]);
|
2006-07-01 00:01:37 -04:00
|
|
|
if (table->file->is_fatal_error(error, HA_CHECK_DUP_KEY))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto table_error; /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
if (rights | col_rights)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
grant_table->privs= rights;
|
2003-06-23 20:03:59 +03:00
|
|
|
grant_table->cols= col_rights;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-01-18 16:39:21 +02:00
|
|
|
hash_delete(&column_priv_hash,(byte*) grant_table);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* This should never happen */
|
|
|
|
table_error:
|
2000-07-31 21:29:14 +02:00
|
|
|
table->file->print_error(error,MYF(0)); /* purecov: deadcode */
|
|
|
|
DBUG_RETURN(-1); /* purecov: deadcode */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
static int replace_routine_table(THD *thd, GRANT_NAME *grant_name,
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE *table, const LEX_USER &combo,
|
2005-05-17 19:54:20 +01:00
|
|
|
const char *db, const char *routine_name,
|
|
|
|
bool is_proc, ulong rights, bool revoke_grant)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
2006-01-11 02:07:40 +03:00
|
|
|
char grantor[USER_HOST_BUFF_SIZE];
|
2004-12-23 10:46:24 +00:00
|
|
|
int old_row_exists= 1;
|
|
|
|
int error=0;
|
|
|
|
ulong store_proc_rights;
|
2005-05-17 19:54:20 +01:00
|
|
|
DBUG_ENTER("replace_routine_table");
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (!initialized)
|
|
|
|
{
|
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--skip-grant-tables");
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
strxmov(grantor, thd->security_ctx->user, "@",
|
|
|
|
thd->security_ctx->host_or_ip, NullS);
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
The following should always succeed as new users are created before
|
|
|
|
this function is called!
|
|
|
|
*/
|
2005-08-25 09:38:06 +03:00
|
|
|
if (!find_acl_user(combo.host.str, combo.user.str, FALSE))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
my_error(ER_PASSWORD_NO_MATCH,MYF(0));
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2005-01-06 13:00:13 +02:00
|
|
|
restore_record(table, s->default_values); // Get empty record
|
2004-12-23 10:46:24 +00:00
|
|
|
table->field[0]->store(combo.host.str,combo.host.length, &my_charset_latin1);
|
|
|
|
table->field[1]->store(db,(uint) strlen(db), &my_charset_latin1);
|
|
|
|
table->field[2]->store(combo.user.str,combo.user.length, &my_charset_latin1);
|
2005-05-17 19:54:20 +01:00
|
|
|
table->field[3]->store(routine_name,(uint) strlen(routine_name),
|
|
|
|
&my_charset_latin1);
|
2007-03-17 00:13:25 +01:00
|
|
|
table->field[4]->store((longlong)(is_proc ?
|
2005-09-14 01:41:44 +03:00
|
|
|
TYPE_ENUM_PROCEDURE : TYPE_ENUM_FUNCTION),
|
|
|
|
TRUE);
|
2004-12-23 10:46:24 +00:00
|
|
|
store_record(table,record[1]); // store at pos 1
|
|
|
|
|
2007-01-29 10:40:26 +01:00
|
|
|
if (table->file->index_read_idx(table->record[0], 0,
|
2007-03-17 00:13:25 +01:00
|
|
|
(byte*) table->field[0]->ptr, HA_WHOLE_KEY,
|
2004-12-23 10:46:24 +00:00
|
|
|
HA_READ_KEY_EXACT))
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
The following should never happen as we first check the in memory
|
|
|
|
grant tables for the user. There is however always a small change that
|
|
|
|
the user has modified the grant tables directly.
|
|
|
|
*/
|
|
|
|
if (revoke_grant)
|
|
|
|
{ // no row, no revoke
|
|
|
|
my_error(ER_NONEXISTING_PROC_GRANT, MYF(0),
|
2005-05-17 19:54:20 +01:00
|
|
|
combo.user.str, combo.host.str, routine_name);
|
2004-12-23 10:46:24 +00:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
old_row_exists= 0;
|
|
|
|
restore_record(table,record[1]); // Get saved record
|
|
|
|
}
|
|
|
|
|
|
|
|
store_proc_rights= get_rights_for_procedure(rights);
|
|
|
|
if (old_row_exists)
|
|
|
|
{
|
|
|
|
ulong j;
|
|
|
|
store_record(table,record[1]);
|
|
|
|
j= (ulong) table->field[6]->val_int();
|
|
|
|
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
|
|
|
/* column rights are already fixed in mysql_table_grant */
|
|
|
|
store_proc_rights=j & ~store_proc_rights;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
store_proc_rights|= j;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
table->field[5]->store(grantor,(uint) strlen(grantor), &my_charset_latin1);
|
2005-09-14 01:41:44 +03:00
|
|
|
table->field[6]->store((longlong) store_proc_rights, TRUE);
|
2004-12-23 10:46:24 +00:00
|
|
|
rights=fix_rights_for_procedure(store_proc_rights);
|
|
|
|
|
|
|
|
if (old_row_exists)
|
|
|
|
{
|
|
|
|
if (store_proc_rights)
|
|
|
|
{
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((error=table->file->ha_update_row(table->record[1],table->record[0])))
|
2004-12-23 10:46:24 +00:00
|
|
|
goto table_error;
|
|
|
|
}
|
2005-12-22 06:39:02 +01:00
|
|
|
else if ((error= table->file->ha_delete_row(table->record[1])))
|
2004-12-23 10:46:24 +00:00
|
|
|
goto table_error;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-12-22 06:39:02 +01:00
|
|
|
error=table->file->ha_write_row(table->record[0]);
|
2006-07-01 00:01:37 -04:00
|
|
|
if (table->file->is_fatal_error(error, HA_CHECK_DUP_KEY))
|
2004-12-23 10:46:24 +00:00
|
|
|
goto table_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rights)
|
|
|
|
{
|
|
|
|
grant_name->privs= rights;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
hash_delete(is_proc ? &proc_priv_hash : &func_priv_hash,(byte*) grant_name);
|
2004-12-23 10:46:24 +00:00
|
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
/* This should never happen */
|
|
|
|
table_error:
|
|
|
|
table->file->print_error(error,MYF(0));
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
|
|
|
Store table level and column level grants in the privilege tables
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_table_grant()
|
|
|
|
thd Thread handle
|
|
|
|
table_list List of tables to give grant
|
|
|
|
user_list List of users to give grant
|
|
|
|
columns List of columns to give grant
|
|
|
|
rights Table level grant
|
|
|
|
revoke_grant Set to 1 if this is a REVOKE command
|
|
|
|
|
|
|
|
RETURN
|
2004-10-20 04:04:37 +03:00
|
|
|
FALSE ok
|
|
|
|
TRUE error
|
2003-06-23 20:03:59 +03:00
|
|
|
*/
|
|
|
|
|
2004-10-20 04:04:37 +03:00
|
|
|
bool mysql_table_grant(THD *thd, TABLE_LIST *table_list,
|
2003-04-28 10:32:56 +03:00
|
|
|
List <LEX_USER> &user_list,
|
|
|
|
List <LEX_COLUMN> &columns, ulong rights,
|
|
|
|
bool revoke_grant)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
ulong column_priv= 0;
|
2000-07-31 21:29:14 +02:00
|
|
|
List_iterator <LEX_USER> str_list (user_list);
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *Str, *tmp_Str;
|
2000-07-31 21:29:14 +02:00
|
|
|
TABLE_LIST tables[3];
|
2001-09-15 15:24:56 +03:00
|
|
|
bool create_new_users=0;
|
2005-01-06 13:00:13 +02:00
|
|
|
char *db_name, *table_name;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("mysql_table_grant");
|
|
|
|
|
|
|
|
if (!initialized)
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0),
|
|
|
|
"--skip-grant-tables"); /* purecov: inspected */
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE); /* purecov: inspected */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
if (rights & ~TABLE_ACLS)
|
|
|
|
{
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_ILLEGAL_GRANT_FOR_TABLE, ER(ER_ILLEGAL_GRANT_FOR_TABLE),
|
|
|
|
MYF(0));
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2004-12-30 21:01:11 +01:00
|
|
|
if (!revoke_grant)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-31 15:26:24 +01:00
|
|
|
if (columns.elements)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-30 21:01:11 +01:00
|
|
|
class LEX_COLUMN *column;
|
|
|
|
List_iterator <LEX_COLUMN> column_iter(columns);
|
2004-12-31 15:26:24 +01:00
|
|
|
|
|
|
|
if (open_and_lock_tables(thd, table_list))
|
|
|
|
DBUG_RETURN(TRUE);
|
2004-12-30 21:01:11 +01:00
|
|
|
|
|
|
|
while ((column = column_iter++))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-30 21:11:40 +01:00
|
|
|
uint unused_field_idx= NO_CACHED_FIELD_INDEX;
|
2005-08-12 17:57:19 +03:00
|
|
|
TABLE_LIST *dummy;
|
|
|
|
Field *f=find_field_in_table_ref(thd, table_list, column->column.ptr(),
|
2005-11-28 21:57:50 +02:00
|
|
|
column->column.length(),
|
2005-08-12 17:57:19 +03:00
|
|
|
column->column.ptr(), NULL, NULL,
|
2005-11-30 21:27:11 +02:00
|
|
|
NULL, TRUE, FALSE,
|
2005-08-12 17:57:19 +03:00
|
|
|
&unused_field_idx, FALSE, &dummy);
|
2004-12-31 15:26:24 +01:00
|
|
|
if (f == (Field*)0)
|
2004-12-30 21:01:11 +01:00
|
|
|
{
|
2004-12-30 21:11:40 +01:00
|
|
|
my_error(ER_BAD_FIELD_ERROR, MYF(0),
|
|
|
|
column->column.c_ptr(), table_list->alias);
|
2004-12-31 15:26:24 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
2004-12-30 21:01:11 +01:00
|
|
|
}
|
2004-12-31 15:26:24 +01:00
|
|
|
if (f == (Field *)-1)
|
|
|
|
DBUG_RETURN(TRUE);
|
2004-12-30 21:01:11 +01:00
|
|
|
column_priv|= column->rights;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-12-30 21:01:11 +01:00
|
|
|
close_thread_tables(thd);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-12-30 21:01:11 +01:00
|
|
|
else
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-30 21:01:11 +01:00
|
|
|
if (!(rights & CREATE_ACL))
|
|
|
|
{
|
|
|
|
char buf[FN_REFLEN];
|
2005-12-31 09:01:26 +04:00
|
|
|
build_table_filename(buf, sizeof(buf), table_list->db,
|
2006-08-02 17:57:06 +02:00
|
|
|
table_list->table_name, reg_ext, 0);
|
2005-12-31 09:01:26 +04:00
|
|
|
fn_format(buf, buf, "", "", MY_UNPACK_FILENAME | MY_RESOLVE_SYMLINKS |
|
|
|
|
MY_RETURN_REAL_PATH | MY_APPEND_EXT);
|
2004-12-30 21:01:11 +01:00
|
|
|
if (access(buf,F_OK))
|
|
|
|
{
|
2004-12-30 21:11:40 +01:00
|
|
|
my_error(ER_NO_SUCH_TABLE, MYF(0), table_list->db, table_list->alias);
|
2004-12-31 15:26:24 +01:00
|
|
|
DBUG_RETURN(TRUE);
|
2004-12-30 21:01:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (table_list->grant.want_privilege)
|
|
|
|
{
|
|
|
|
char command[128];
|
|
|
|
get_privilege_desc(command, sizeof(command),
|
|
|
|
table_list->grant.want_privilege);
|
|
|
|
my_error(ER_TABLEACCESS_DENIED_ERROR, MYF(0),
|
2005-09-15 22:29:07 +03:00
|
|
|
command, thd->security_ctx->priv_user,
|
|
|
|
thd->security_ctx->host_or_ip, table_list->alias);
|
2004-12-30 21:01:11 +01:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* open the mysql.tables_priv and mysql.columns_priv tables */
|
|
|
|
|
|
|
|
bzero((char*) &tables,sizeof(tables));
|
2005-01-06 13:00:13 +02:00
|
|
|
tables[0].alias=tables[0].table_name= (char*) "user";
|
|
|
|
tables[1].alias=tables[1].table_name= (char*) "tables_priv";
|
|
|
|
tables[2].alias=tables[2].table_name= (char*) "columns_priv";
|
2004-07-16 01:15:55 +03:00
|
|
|
tables[0].next_local= tables[0].next_global= tables+1;
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Don't open column table if we don't need it ! */
|
2004-07-16 01:15:55 +03:00
|
|
|
tables[1].next_local=
|
|
|
|
tables[1].next_global= ((column_priv ||
|
|
|
|
(revoke_grant &&
|
|
|
|
((rights & COL_ACLS) || columns.elements)))
|
|
|
|
? tables+2 : 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
tables[0].lock_type=tables[1].lock_type=tables[2].lock_type=TL_WRITE;
|
|
|
|
tables[0].db=tables[1].db=tables[2].db=(char*) "mysql";
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2003-04-01 22:49:40 +02:00
|
|
|
#ifdef HAVE_REPLICATION
|
|
|
|
/*
|
|
|
|
GRANT and REVOKE are applied the slave in/exclusion rules as they are
|
|
|
|
some kind of updates to the mysql.% tables.
|
|
|
|
*/
|
2005-03-08 21:12:35 +01:00
|
|
|
if (thd->slave_thread && rpl_filter->is_on())
|
2003-08-02 16:10:26 +02:00
|
|
|
{
|
2003-08-11 22:44:43 +03:00
|
|
|
/*
|
|
|
|
The tables must be marked "updating" so that tables_ok() takes them into
|
|
|
|
account in tests.
|
2003-08-02 16:10:26 +02:00
|
|
|
*/
|
2003-08-07 20:16:37 +03:00
|
|
|
tables[0].updating= tables[1].updating= tables[2].updating= 1;
|
2005-05-09 11:43:09 +02:00
|
|
|
if (!(thd->spcont || rpl_filter->tables_ok(0, tables)))
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(FALSE);
|
2003-08-02 16:10:26 +02:00
|
|
|
}
|
2003-04-01 22:49:40 +02:00
|
|
|
#endif
|
|
|
|
|
2004-02-01 15:30:32 +02:00
|
|
|
if (simple_open_n_lock_tables(thd,tables))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // Should never happen
|
|
|
|
close_thread_tables(thd); /* purecov: deadcode */
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE); /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2001-09-15 15:24:56 +03:00
|
|
|
if (!revoke_grant)
|
|
|
|
create_new_users= test_if_create_new_users(thd);
|
2004-10-20 04:04:37 +03:00
|
|
|
bool result= FALSE;
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_wrlock(&LOCK_grant);
|
2006-05-06 11:25:59 +04:00
|
|
|
pthread_mutex_lock(&acl_cache->lock);
|
2004-11-08 01:13:54 +02:00
|
|
|
MEM_ROOT *old_root= thd->mem_root;
|
|
|
|
thd->mem_root= &memex;
|
2005-05-07 13:55:46 +00:00
|
|
|
grant_version++;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_Str = str_list++))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-09-21 21:36:23 +03:00
|
|
|
int error;
|
2000-07-31 21:29:14 +02:00
|
|
|
GRANT_TABLE *grant_table;
|
2006-06-29 15:50:44 +05:00
|
|
|
if (!(Str= get_current_user(thd, tmp_Str)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Create user if needed */
|
2002-09-22 10:59:33 +03:00
|
|
|
error=replace_user_table(thd, tables[0].table, *Str,
|
2005-03-23 19:18:25 +01:00
|
|
|
0, revoke_grant, create_new_users,
|
2005-03-29 12:37:36 +03:00
|
|
|
test(thd->variables.sql_mode &
|
|
|
|
MODE_NO_AUTO_CREATE_USER));
|
2002-09-21 21:36:23 +03:00
|
|
|
if (error)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-10-20 04:04:37 +03:00
|
|
|
result= TRUE; // Remember error
|
2000-07-31 21:29:14 +02:00
|
|
|
continue; // Add next user
|
|
|
|
}
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
db_name= (table_list->view_db.length ?
|
|
|
|
table_list->view_db.str :
|
|
|
|
table_list->db);
|
2005-01-06 13:00:13 +02:00
|
|
|
table_name= (table_list->view_name.length ?
|
2004-07-16 01:15:55 +03:00
|
|
|
table_list->view_name.str :
|
2005-01-06 13:00:13 +02:00
|
|
|
table_list->table_name);
|
2004-07-16 01:15:55 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/* Find/create cached table grant */
|
2004-07-16 01:15:55 +03:00
|
|
|
grant_table= table_hash_search(Str->host.str, NullS, db_name,
|
2005-01-06 13:00:13 +02:00
|
|
|
Str->user.str, table_name, 1);
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!grant_table)
|
|
|
|
{
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_NONEXISTING_TABLE_GRANT, MYF(0),
|
2005-01-06 13:00:13 +02:00
|
|
|
Str->user.str, Str->host.str, table_list->table_name);
|
2004-10-20 04:04:37 +03:00
|
|
|
result= TRUE;
|
2000-07-31 21:29:14 +02:00
|
|
|
continue;
|
|
|
|
}
|
2004-07-16 01:15:55 +03:00
|
|
|
grant_table = new GRANT_TABLE (Str->host.str, db_name,
|
2005-01-06 13:00:13 +02:00
|
|
|
Str->user.str, table_name,
|
2000-07-31 21:29:14 +02:00
|
|
|
rights,
|
|
|
|
column_priv);
|
|
|
|
if (!grant_table) // end of memory
|
|
|
|
{
|
2004-10-20 04:04:37 +03:00
|
|
|
result= TRUE; /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
continue; /* purecov: deadcode */
|
|
|
|
}
|
2003-09-19 14:44:31 +05:00
|
|
|
my_hash_insert(&column_priv_hash,(byte*) grant_table);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If revoke_grant, calculate the new column privilege for tables_priv */
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
class LEX_COLUMN *column;
|
|
|
|
List_iterator <LEX_COLUMN> column_iter(columns);
|
2000-07-31 21:29:14 +02:00
|
|
|
GRANT_COLUMN *grant_column;
|
|
|
|
|
|
|
|
/* Fix old grants */
|
2003-06-23 20:03:59 +03:00
|
|
|
while ((column = column_iter++))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
grant_column = column_hash_search(grant_table,
|
2003-06-23 20:03:59 +03:00
|
|
|
column->column.ptr(),
|
|
|
|
column->column.length());
|
2000-07-31 21:29:14 +02:00
|
|
|
if (grant_column)
|
2003-06-23 20:03:59 +03:00
|
|
|
grant_column->rights&= ~(column->rights | rights);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
/* scan trough all columns to get new column grant */
|
2003-06-23 20:03:59 +03:00
|
|
|
column_priv= 0;
|
2000-07-31 21:29:14 +02:00
|
|
|
for (uint idx=0 ; idx < grant_table->hash_columns.records ; idx++)
|
|
|
|
{
|
|
|
|
grant_column= (GRANT_COLUMN*) hash_element(&grant_table->hash_columns,
|
|
|
|
idx);
|
|
|
|
grant_column->rights&= ~rights; // Fix other columns
|
|
|
|
column_priv|= grant_column->rights;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
column_priv|= grant_table->cols;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* update table and columns */
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
if (replace_table_table(thd, grant_table, tables[1].table, *Str,
|
2005-01-06 13:00:13 +02:00
|
|
|
db_name, table_name,
|
2000-07-31 21:29:14 +02:00
|
|
|
rights, column_priv, revoke_grant))
|
2004-10-20 01:28:42 +03:00
|
|
|
{
|
|
|
|
/* Should only happen if table is crashed */
|
2004-10-20 04:04:37 +03:00
|
|
|
result= TRUE; /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else if (tables[2].table)
|
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
if ((replace_column_table(grant_table, tables[2].table, *Str,
|
2000-07-31 21:29:14 +02:00
|
|
|
columns,
|
2005-01-06 13:00:13 +02:00
|
|
|
db_name, table_name,
|
2000-07-31 21:29:14 +02:00
|
|
|
rights, revoke_grant)))
|
|
|
|
{
|
2004-10-20 04:04:37 +03:00
|
|
|
result= TRUE;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
grant_option=TRUE;
|
2004-11-08 01:13:54 +02:00
|
|
|
thd->mem_root= old_root;
|
2006-05-06 11:25:59 +04:00
|
|
|
pthread_mutex_unlock(&acl_cache->lock);
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (!result) /* success */
|
|
|
|
{
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
|
|
|
thd->clear_error();
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (!result) /* success */
|
2002-10-02 13:33:08 +03:00
|
|
|
send_ok(thd);
|
2006-10-03 13:38:25 -04:00
|
|
|
|
2001-09-30 10:46:20 +08:00
|
|
|
/* Tables are automatically closed */
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
/*
|
2005-05-17 19:54:20 +01:00
|
|
|
Store routine level grants in the privilege tables
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
SYNOPSIS
|
2005-05-17 19:54:20 +01:00
|
|
|
mysql_routine_grant()
|
2004-12-23 10:46:24 +00:00
|
|
|
thd Thread handle
|
2005-05-17 19:54:20 +01:00
|
|
|
table_list List of routines to give grant
|
|
|
|
is_proc true indicates routine list are procedures
|
2004-12-23 10:46:24 +00:00
|
|
|
user_list List of users to give grant
|
|
|
|
rights Table level grant
|
|
|
|
revoke_grant Set to 1 if this is a REVOKE command
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 ok
|
|
|
|
1 error
|
|
|
|
*/
|
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
bool mysql_routine_grant(THD *thd, TABLE_LIST *table_list, bool is_proc,
|
|
|
|
List <LEX_USER> &user_list, ulong rights,
|
|
|
|
bool revoke_grant, bool no_error)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
List_iterator <LEX_USER> str_list (user_list);
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *Str, *tmp_Str;
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE_LIST tables[2];
|
|
|
|
bool create_new_users=0, result=0;
|
2005-01-06 13:00:13 +02:00
|
|
|
char *db_name, *table_name;
|
2005-05-17 19:54:20 +01:00
|
|
|
DBUG_ENTER("mysql_routine_grant");
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (!initialized)
|
|
|
|
{
|
|
|
|
if (!no_error)
|
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0),
|
|
|
|
"--skip-grant-tables");
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
if (rights & ~PROC_ACLS)
|
|
|
|
{
|
|
|
|
if (!no_error)
|
|
|
|
my_message(ER_ILLEGAL_GRANT_FOR_TABLE, ER(ER_ILLEGAL_GRANT_FOR_TABLE),
|
|
|
|
MYF(0));
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!revoke_grant)
|
|
|
|
{
|
2006-01-26 13:29:46 +01:00
|
|
|
if (sp_exist_routines(thd, table_list, is_proc, no_error)<0)
|
2004-12-23 10:46:24 +00:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* open the mysql.user and mysql.procs_priv tables */
|
|
|
|
|
|
|
|
bzero((char*) &tables,sizeof(tables));
|
2005-01-06 13:00:13 +02:00
|
|
|
tables[0].alias=tables[0].table_name= (char*) "user";
|
|
|
|
tables[1].alias=tables[1].table_name= (char*) "procs_priv";
|
2004-12-23 10:46:24 +00:00
|
|
|
tables[0].next_local= tables[0].next_global= tables+1;
|
|
|
|
tables[0].lock_type=tables[1].lock_type=TL_WRITE;
|
|
|
|
tables[0].db=tables[1].db=(char*) "mysql";
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
#ifdef HAVE_REPLICATION
|
|
|
|
/*
|
|
|
|
GRANT and REVOKE are applied the slave in/exclusion rules as they are
|
|
|
|
some kind of updates to the mysql.% tables.
|
|
|
|
*/
|
2005-03-08 21:12:35 +01:00
|
|
|
if (thd->slave_thread && rpl_filter->is_on())
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
The tables must be marked "updating" so that tables_ok() takes them into
|
|
|
|
account in tests.
|
|
|
|
*/
|
|
|
|
tables[0].updating= tables[1].updating= 1;
|
2005-05-09 11:43:09 +02:00
|
|
|
if (!(thd->spcont || rpl_filter->tables_ok(0, tables)))
|
2004-12-23 10:46:24 +00:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (simple_open_n_lock_tables(thd,tables))
|
|
|
|
{ // Should never happen
|
|
|
|
close_thread_tables(thd);
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!revoke_grant)
|
|
|
|
create_new_users= test_if_create_new_users(thd);
|
|
|
|
rw_wrlock(&LOCK_grant);
|
2006-05-06 11:25:59 +04:00
|
|
|
pthread_mutex_lock(&acl_cache->lock);
|
2004-12-23 10:46:24 +00:00
|
|
|
MEM_ROOT *old_root= thd->mem_root;
|
|
|
|
thd->mem_root= &memex;
|
|
|
|
|
|
|
|
DBUG_PRINT("info",("now time to iterate and add users"));
|
|
|
|
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_Str= str_list++))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
int error;
|
|
|
|
GRANT_NAME *grant_name;
|
2006-06-29 15:50:44 +05:00
|
|
|
if (!(Str= get_current_user(thd, tmp_Str)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
2004-12-23 10:46:24 +00:00
|
|
|
/* Create user if needed */
|
|
|
|
error=replace_user_table(thd, tables[0].table, *Str,
|
2005-03-23 19:18:25 +01:00
|
|
|
0, revoke_grant, create_new_users,
|
2005-03-29 12:37:36 +03:00
|
|
|
test(thd->variables.sql_mode &
|
|
|
|
MODE_NO_AUTO_CREATE_USER));
|
2004-12-23 10:46:24 +00:00
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
result= TRUE; // Remember error
|
|
|
|
continue; // Add next user
|
|
|
|
}
|
|
|
|
|
|
|
|
db_name= table_list->db;
|
2005-01-06 13:00:13 +02:00
|
|
|
table_name= table_list->table_name;
|
2004-12-23 10:46:24 +00:00
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
grant_name= routine_hash_search(Str->host.str, NullS, db_name,
|
|
|
|
Str->user.str, table_name, is_proc, 1);
|
2004-12-23 10:46:24 +00:00
|
|
|
if (!grant_name)
|
|
|
|
{
|
|
|
|
if (revoke_grant)
|
|
|
|
{
|
|
|
|
if (!no_error)
|
|
|
|
my_error(ER_NONEXISTING_PROC_GRANT, MYF(0),
|
2005-01-06 13:00:13 +02:00
|
|
|
Str->user.str, Str->host.str, table_name);
|
2004-12-23 10:46:24 +00:00
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
grant_name= new GRANT_NAME(Str->host.str, db_name,
|
2005-01-06 13:00:13 +02:00
|
|
|
Str->user.str, table_name,
|
2004-12-23 10:46:24 +00:00
|
|
|
rights);
|
|
|
|
if (!grant_name)
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
2005-05-17 19:54:20 +01:00
|
|
|
my_hash_insert(is_proc ? &proc_priv_hash : &func_priv_hash,(byte*) grant_name);
|
2004-12-23 10:46:24 +00:00
|
|
|
}
|
2005-03-22 15:54:18 +01:00
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
if (replace_routine_table(thd, grant_name, tables[1].table, *Str,
|
|
|
|
db_name, table_name, is_proc, rights, revoke_grant))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
grant_option=TRUE;
|
|
|
|
thd->mem_root= old_root;
|
2006-05-06 11:25:59 +04:00
|
|
|
pthread_mutex_unlock(&acl_cache->lock);
|
2006-10-03 13:38:25 -04:00
|
|
|
if (!result && !no_error)
|
|
|
|
{
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
|
|
|
thd->clear_error();
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
rw_unlock(&LOCK_grant);
|
2006-10-03 13:38:25 -04:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
if (!result && !no_error)
|
|
|
|
send_ok(thd);
|
2006-10-03 13:38:25 -04:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
/* Tables are automatically closed */
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-10-20 04:04:37 +03:00
|
|
|
bool mysql_grant(THD *thd, const char *db, List <LEX_USER> &list,
|
|
|
|
ulong rights, bool revoke_grant)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
List_iterator <LEX_USER> str_list (list);
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *Str, *tmp_Str;
|
2006-09-27 19:21:29 +05:00
|
|
|
char tmp_db[NAME_LEN+1];
|
2001-08-10 17:37:37 +03:00
|
|
|
bool create_new_users=0;
|
2000-07-31 21:29:14 +02:00
|
|
|
TABLE_LIST tables[2];
|
|
|
|
DBUG_ENTER("mysql_grant");
|
|
|
|
if (!initialized)
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0),
|
|
|
|
"--skip-grant-tables"); /* purecov: tested */
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE); /* purecov: tested */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2001-12-13 02:31:19 +02:00
|
|
|
if (lower_case_table_names && db)
|
|
|
|
{
|
|
|
|
strmov(tmp_db,db);
|
2004-05-22 23:41:58 +04:00
|
|
|
my_casedn_str(files_charset_info, tmp_db);
|
2001-12-13 02:31:19 +02:00
|
|
|
db=tmp_db;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* open the mysql.user and mysql.db tables */
|
2004-02-01 15:30:32 +02:00
|
|
|
bzero((char*) &tables,sizeof(tables));
|
2005-01-06 13:00:13 +02:00
|
|
|
tables[0].alias=tables[0].table_name=(char*) "user";
|
|
|
|
tables[1].alias=tables[1].table_name=(char*) "db";
|
2004-07-16 01:15:55 +03:00
|
|
|
tables[0].next_local= tables[0].next_global= tables+1;
|
2000-07-31 21:29:14 +02:00
|
|
|
tables[0].lock_type=tables[1].lock_type=TL_WRITE;
|
|
|
|
tables[0].db=tables[1].db=(char*) "mysql";
|
2003-04-01 22:49:40 +02:00
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2003-04-01 22:49:40 +02:00
|
|
|
#ifdef HAVE_REPLICATION
|
|
|
|
/*
|
|
|
|
GRANT and REVOKE are applied the slave in/exclusion rules as they are
|
|
|
|
some kind of updates to the mysql.% tables.
|
|
|
|
*/
|
2005-03-08 21:12:35 +01:00
|
|
|
if (thd->slave_thread && rpl_filter->is_on())
|
2003-08-02 16:10:26 +02:00
|
|
|
{
|
2003-08-11 22:44:43 +03:00
|
|
|
/*
|
|
|
|
The tables must be marked "updating" so that tables_ok() takes them into
|
|
|
|
account in tests.
|
2003-08-02 16:10:26 +02:00
|
|
|
*/
|
2003-08-07 20:16:37 +03:00
|
|
|
tables[0].updating= tables[1].updating= 1;
|
2005-05-09 11:43:09 +02:00
|
|
|
if (!(thd->spcont || rpl_filter->tables_ok(0, tables)))
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(FALSE);
|
2003-08-02 16:10:26 +02:00
|
|
|
}
|
2003-04-01 22:49:40 +02:00
|
|
|
#endif
|
|
|
|
|
2004-02-01 15:30:32 +02:00
|
|
|
if (simple_open_n_lock_tables(thd,tables))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // This should never happen
|
|
|
|
close_thread_tables(thd); /* purecov: deadcode */
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE); /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2001-08-10 17:37:37 +03:00
|
|
|
if (!revoke_grant)
|
|
|
|
create_new_users= test_if_create_new_users(thd);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* go through users in user_list */
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_wrlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
grant_version++;
|
|
|
|
|
|
|
|
int result=0;
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_Str = str_list++))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-06-29 15:50:44 +05:00
|
|
|
if (!(Str= get_current_user(thd, tmp_Str)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
2006-09-27 19:21:29 +05:00
|
|
|
}
|
2005-03-23 19:18:25 +01:00
|
|
|
if (replace_user_table(thd, tables[0].table, *Str,
|
|
|
|
(!db ? rights : 0), revoke_grant, create_new_users,
|
2005-03-29 12:37:36 +03:00
|
|
|
test(thd->variables.sql_mode &
|
|
|
|
MODE_NO_AUTO_CREATE_USER)))
|
2001-07-17 21:04:01 +03:00
|
|
|
result= -1;
|
2003-05-31 21:35:20 +03:00
|
|
|
else if (db)
|
2001-08-10 17:37:37 +03:00
|
|
|
{
|
2003-05-31 21:35:20 +03:00
|
|
|
ulong db_rights= rights & DB_ACLS;
|
|
|
|
if (db_rights == rights)
|
|
|
|
{
|
|
|
|
if (replace_db_table(tables[1].table, db, *Str, db_rights,
|
|
|
|
revoke_grant))
|
|
|
|
result= -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_WRONG_USAGE, MYF(0), "DB GRANT", "GLOBAL PRIVILEGES");
|
2003-09-11 19:06:23 +03:00
|
|
|
result= -1;
|
2003-05-31 21:35:20 +03:00
|
|
|
}
|
2001-08-10 17:37:37 +03:00
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
|
|
|
thd->clear_error();
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
close_thread_tables(thd);
|
|
|
|
|
|
|
|
if (!result)
|
2002-10-02 13:33:08 +03:00
|
|
|
send_ok(thd);
|
2006-10-03 13:38:25 -04:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
|
|
|
|
/* Free grant array if possible */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
void grant_free(void)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("grant_free");
|
|
|
|
grant_option = FALSE;
|
2003-01-18 16:39:21 +02:00
|
|
|
hash_free(&column_priv_hash);
|
2004-12-23 10:46:24 +00:00
|
|
|
hash_free(&proc_priv_hash);
|
2005-05-18 19:00:21 +03:00
|
|
|
hash_free(&func_priv_hash);
|
2000-09-12 03:02:33 +03:00
|
|
|
free_root(&memex,MYF(0));
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
/*
|
|
|
|
Initialize structures responsible for table/column-level privilege checking
|
|
|
|
and load information for them from tables in the 'mysql' database.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
grant_init()
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
RETURN VALUES
|
|
|
|
0 ok
|
|
|
|
1 Could not initialize grant's
|
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool grant_init()
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-10-29 21:59:03 +02:00
|
|
|
THD *thd;
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool return_val;
|
|
|
|
DBUG_ENTER("grant_init");
|
|
|
|
|
|
|
|
if (!(thd= new THD))
|
|
|
|
DBUG_RETURN(1); /* purecov: deadcode */
|
2005-11-23 21:18:10 +03:00
|
|
|
thd->thread_stack= (char*) &thd;
|
2005-09-01 16:52:59 +04:00
|
|
|
thd->store_globals();
|
|
|
|
return_val= grant_reload(thd);
|
|
|
|
delete thd;
|
|
|
|
/* Remember that we don't have a THD */
|
|
|
|
my_pthread_setspecific_ptr(THR_THD, 0);
|
|
|
|
DBUG_RETURN(return_val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Initialize structures responsible for table/column-level privilege
|
|
|
|
checking and load information about grants from open privilege tables.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
grant_load()
|
|
|
|
thd Current thread
|
|
|
|
tables List containing open "mysql.tables_priv" and
|
|
|
|
"mysql.columns_priv" tables.
|
|
|
|
|
|
|
|
RETURN VALUES
|
|
|
|
FALSE - success
|
|
|
|
TRUE - error
|
|
|
|
*/
|
|
|
|
|
|
|
|
static my_bool grant_load(TABLE_LIST *tables)
|
|
|
|
{
|
2004-11-08 01:13:54 +02:00
|
|
|
MEM_ROOT *memex_ptr;
|
2002-09-05 16:17:08 +03:00
|
|
|
my_bool return_val= 1;
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE *t_table, *c_table, *p_table;
|
2003-11-20 12:55:48 +04:00
|
|
|
bool check_no_resolve= specialflag & SPECIAL_NO_RESOLVE;
|
2005-09-01 16:52:59 +04:00
|
|
|
MEM_ROOT **save_mem_root_ptr= my_pthread_getspecific_ptr(MEM_ROOT**,
|
|
|
|
THR_MALLOC);
|
|
|
|
DBUG_ENTER("grant_load");
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
grant_option = FALSE;
|
2004-11-25 21:55:49 +01:00
|
|
|
(void) hash_init(&column_priv_hash,system_charset_info,
|
2002-03-14 21:44:42 +04:00
|
|
|
0,0,0, (hash_get_key) get_grant_table,
|
2000-07-31 21:29:14 +02:00
|
|
|
(hash_free_key) free_grant_table,0);
|
2004-12-23 10:46:24 +00:00
|
|
|
(void) hash_init(&proc_priv_hash,system_charset_info,
|
|
|
|
0,0,0, (hash_get_key) get_grant_table,
|
|
|
|
0,0);
|
2005-05-17 19:54:20 +01:00
|
|
|
(void) hash_init(&func_priv_hash,system_charset_info,
|
|
|
|
0,0,0, (hash_get_key) get_grant_table,
|
|
|
|
0,0);
|
2003-10-11 22:00:24 +03:00
|
|
|
init_sql_alloc(&memex, ACL_ALLOC_BLOCK_SIZE, 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
t_table = tables[0].table;
|
|
|
|
c_table = tables[1].table;
|
2004-12-23 10:46:24 +00:00
|
|
|
p_table= tables[2].table;
|
2005-07-18 13:31:02 +02:00
|
|
|
t_table->file->ha_index_init(0, 1);
|
|
|
|
p_table->file->ha_index_init(0, 1);
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
t_table->use_all_columns();
|
|
|
|
c_table->use_all_columns();
|
|
|
|
p_table->use_all_columns();
|
2004-12-23 10:46:24 +00:00
|
|
|
if (!t_table->file->index_first(t_table->record[0]))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-23 10:46:24 +00:00
|
|
|
memex_ptr= &memex;
|
|
|
|
my_pthread_setspecific_ptr(THR_MALLOC, &memex_ptr);
|
|
|
|
do
|
|
|
|
{
|
|
|
|
GRANT_TABLE *mem_check;
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
if (!(mem_check=new (memex_ptr) GRANT_TABLE(t_table,c_table)))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
/* This could only happen if we are out memory */
|
|
|
|
grant_option= FALSE;
|
|
|
|
goto end_unlock;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
if (check_no_resolve)
|
|
|
|
{
|
2005-03-03 17:44:28 -08:00
|
|
|
if (hostname_requires_resolving(mem_check->host.hostname))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
sql_print_warning("'tables_priv' entry '%s %s@%s' "
|
|
|
|
"ignored in --skip-name-resolve mode.",
|
2006-06-06 14:21:07 +03:00
|
|
|
mem_check->tname,
|
|
|
|
mem_check->user ? mem_check->user : "",
|
|
|
|
mem_check->host.hostname ?
|
|
|
|
mem_check->host.hostname : "");
|
2004-12-23 10:46:24 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! mem_check->ok())
|
|
|
|
delete mem_check;
|
|
|
|
else if (my_hash_insert(&column_priv_hash,(byte*) mem_check))
|
|
|
|
{
|
|
|
|
delete mem_check;
|
|
|
|
grant_option= FALSE;
|
|
|
|
goto end_unlock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (!t_table->file->index_next(t_table->record[0]));
|
|
|
|
}
|
|
|
|
if (!p_table->file->index_first(p_table->record[0]))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-23 10:46:24 +00:00
|
|
|
memex_ptr= &memex;
|
|
|
|
my_pthread_setspecific_ptr(THR_MALLOC, &memex_ptr);
|
|
|
|
do
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-23 10:46:24 +00:00
|
|
|
GRANT_NAME *mem_check;
|
2005-05-17 19:54:20 +01:00
|
|
|
HASH *hash;
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
if (!(mem_check=new (&memex) GRANT_NAME(p_table)))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
/* This could only happen if we are out memory */
|
|
|
|
grant_option= FALSE;
|
|
|
|
goto end_unlock;
|
|
|
|
}
|
2003-11-20 12:55:48 +04:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
if (check_no_resolve)
|
2003-11-20 12:55:48 +04:00
|
|
|
{
|
2005-03-03 15:01:46 -08:00
|
|
|
if (hostname_requires_resolving(mem_check->host.hostname))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
sql_print_warning("'procs_priv' entry '%s %s@%s' "
|
|
|
|
"ignored in --skip-name-resolve mode.",
|
|
|
|
mem_check->tname, mem_check->user,
|
2006-06-06 14:21:07 +03:00
|
|
|
mem_check->host.hostname ?
|
|
|
|
mem_check->host.hostname : "");
|
2004-12-23 10:46:24 +00:00
|
|
|
continue;
|
|
|
|
}
|
2003-11-20 12:55:48 +04:00
|
|
|
}
|
2005-05-17 19:54:20 +01:00
|
|
|
if (p_table->field[4]->val_int() == TYPE_ENUM_PROCEDURE)
|
|
|
|
{
|
|
|
|
hash= &proc_priv_hash;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if (p_table->field[4]->val_int() == TYPE_ENUM_FUNCTION)
|
|
|
|
{
|
|
|
|
hash= &func_priv_hash;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sql_print_warning("'procs_priv' entry '%s' "
|
|
|
|
"ignored, bad routine type",
|
|
|
|
mem_check->tname);
|
|
|
|
continue;
|
|
|
|
}
|
2003-11-20 12:55:48 +04:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
mem_check->privs= fix_rights_for_procedure(mem_check->privs);
|
|
|
|
if (! mem_check->ok())
|
|
|
|
delete mem_check;
|
2005-05-17 19:54:20 +01:00
|
|
|
else if (my_hash_insert(hash, (byte*) mem_check))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
delete mem_check;
|
|
|
|
grant_option= FALSE;
|
|
|
|
goto end_unlock;
|
|
|
|
}
|
2003-11-20 12:55:48 +04:00
|
|
|
}
|
2004-12-23 10:46:24 +00:00
|
|
|
while (!p_table->file->index_next(p_table->record[0]));
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-12-23 10:46:24 +00:00
|
|
|
grant_option= TRUE;
|
2002-09-05 16:17:08 +03:00
|
|
|
return_val=0; // Return ok
|
|
|
|
|
|
|
|
end_unlock:
|
2004-06-23 12:29:05 +02:00
|
|
|
t_table->file->ha_index_end();
|
2004-12-23 10:46:24 +00:00
|
|
|
p_table->file->ha_index_end();
|
2005-09-01 16:52:59 +04:00
|
|
|
my_pthread_setspecific_ptr(THR_MALLOC, save_mem_root_ptr);
|
2002-09-05 16:17:08 +03:00
|
|
|
DBUG_RETURN(return_val);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
2005-09-01 16:52:59 +04:00
|
|
|
Reload information about table and column level privileges if possible.
|
2003-06-23 20:03:59 +03:00
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
grant_reload()
|
2005-09-01 16:52:59 +04:00
|
|
|
thd Current thread
|
2003-06-23 20:03:59 +03:00
|
|
|
|
|
|
|
NOTES
|
2005-09-01 16:52:59 +04:00
|
|
|
Locked tables are checked by acl_reload() and doesn't have to be checked
|
|
|
|
in this call.
|
|
|
|
This function is also used for initialization of structures responsible
|
|
|
|
for table/column-level privilege checking.
|
|
|
|
|
|
|
|
RETURN VALUE
|
|
|
|
FALSE Success
|
|
|
|
TRUE Error
|
2003-06-23 20:03:59 +03:00
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool grant_reload(THD *thd)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-09-02 01:01:33 +04:00
|
|
|
TABLE_LIST tables[3];
|
2005-05-17 19:54:20 +01:00
|
|
|
HASH old_column_priv_hash, old_proc_priv_hash, old_func_priv_hash;
|
2003-01-18 16:39:21 +02:00
|
|
|
bool old_grant_option;
|
2000-07-31 21:29:14 +02:00
|
|
|
MEM_ROOT old_mem;
|
2005-09-01 16:52:59 +04:00
|
|
|
my_bool return_val= 1;
|
2000-07-31 21:29:14 +02:00
|
|
|
DBUG_ENTER("grant_reload");
|
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
/* Don't do anything if running with --skip-grant-tables */
|
|
|
|
if (!initialized)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
bzero((char*) tables, sizeof(tables));
|
2005-09-02 01:01:33 +04:00
|
|
|
tables[0].alias= tables[0].table_name= (char*) "tables_priv";
|
|
|
|
tables[1].alias= tables[1].table_name= (char*) "columns_priv";
|
|
|
|
tables[2].alias= tables[2].table_name= (char*) "procs_priv";
|
|
|
|
tables[0].db= tables[1].db= tables[2].db= (char *) "mysql";
|
|
|
|
tables[0].next_local= tables[0].next_global= tables+1;
|
|
|
|
tables[1].next_local= tables[1].next_global= tables+2;
|
|
|
|
tables[0].lock_type= tables[1].lock_type= tables[2].lock_type= TL_READ;
|
2005-09-01 16:52:59 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
To avoid deadlocks we should obtain table locks before
|
|
|
|
obtaining LOCK_grant rwlock.
|
|
|
|
*/
|
|
|
|
if (simple_open_n_lock_tables(thd, tables))
|
|
|
|
goto end;
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_wrlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
grant_version++;
|
2003-01-18 16:39:21 +02:00
|
|
|
old_column_priv_hash= column_priv_hash;
|
2004-12-23 10:46:24 +00:00
|
|
|
old_proc_priv_hash= proc_priv_hash;
|
2005-05-17 19:54:20 +01:00
|
|
|
old_func_priv_hash= func_priv_hash;
|
2003-04-28 10:32:56 +03:00
|
|
|
old_grant_option= grant_option;
|
2003-05-19 16:35:49 +03:00
|
|
|
old_mem= memex;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-01 16:52:59 +04:00
|
|
|
if ((return_val= grant_load(tables)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{ // Error. Revert to old hash
|
2003-01-18 16:39:21 +02:00
|
|
|
DBUG_PRINT("error",("Reverting to old privileges"));
|
2000-07-31 21:29:14 +02:00
|
|
|
grant_free(); /* purecov: deadcode */
|
2003-01-18 16:39:21 +02:00
|
|
|
column_priv_hash= old_column_priv_hash; /* purecov: deadcode */
|
2004-12-23 10:46:24 +00:00
|
|
|
proc_priv_hash= old_proc_priv_hash;
|
2005-05-17 19:54:20 +01:00
|
|
|
func_priv_hash= old_func_priv_hash;
|
2003-04-28 10:32:56 +03:00
|
|
|
grant_option= old_grant_option; /* purecov: deadcode */
|
2003-05-19 16:35:49 +03:00
|
|
|
memex= old_mem; /* purecov: deadcode */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-01-18 16:39:21 +02:00
|
|
|
hash_free(&old_column_priv_hash);
|
2004-12-23 10:46:24 +00:00
|
|
|
hash_free(&old_proc_priv_hash);
|
2005-05-17 19:54:20 +01:00
|
|
|
hash_free(&old_func_priv_hash);
|
2000-09-12 03:02:33 +03:00
|
|
|
free_root(&old_mem,MYF(0));
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2005-09-01 16:52:59 +04:00
|
|
|
end:
|
|
|
|
close_thread_tables(thd);
|
|
|
|
DBUG_RETURN(return_val);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
2004-04-28 17:45:08 +03:00
|
|
|
Check table level grants
|
2004-09-03 21:43:04 +03:00
|
|
|
|
2004-10-07 01:45:06 +03:00
|
|
|
SYNOPSIS
|
2004-09-03 21:43:04 +03:00
|
|
|
bool check_grant()
|
|
|
|
thd Thread handler
|
|
|
|
want_access Bits of privileges user needs to have
|
|
|
|
tables List of tables to check. The user should have 'want_access'
|
|
|
|
to all tables in list.
|
|
|
|
show_table <> 0 if we are in show table. In this case it's enough to have
|
|
|
|
any privilege for the table
|
|
|
|
number Check at most this number of tables.
|
|
|
|
no_errors If 0 then we write an error. The error is sent directly to
|
|
|
|
the client
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 ok
|
2004-10-07 01:45:06 +03:00
|
|
|
1 Error: User did not have the requested privileges
|
2006-03-20 13:42:02 +04:00
|
|
|
|
|
|
|
NOTE
|
|
|
|
This functions assumes that either number of tables to be inspected
|
|
|
|
by it is limited explicitly (i.e. is is not UINT_MAX) or table list
|
|
|
|
used and thd->lex->query_tables_own_last value correspond to each
|
|
|
|
other (the latter should be either 0 or point to next_global member
|
|
|
|
of one of elements of this table list).
|
2000-07-31 21:29:14 +02:00
|
|
|
****************************************************************************/
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
bool check_grant(THD *thd, ulong want_access, TABLE_LIST *tables,
|
2004-04-10 01:14:32 +03:00
|
|
|
uint show_table, uint number, bool no_errors)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-10-28 00:18:23 +03:00
|
|
|
TABLE_LIST *table, *first_not_own_table= thd->lex->first_not_own_table();
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2005-10-28 00:18:23 +03:00
|
|
|
uint i;
|
2006-05-26 11:47:53 +03:00
|
|
|
ulong orig_want_access= want_access;
|
2004-09-03 21:43:04 +03:00
|
|
|
DBUG_ENTER("check_grant");
|
|
|
|
DBUG_ASSERT(number > 0);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-10-28 00:18:23 +03:00
|
|
|
/*
|
2005-10-28 00:56:44 +03:00
|
|
|
Walk through the list of tables that belong to the query and save the
|
|
|
|
requested access (orig_want_privilege) to be able to use it when
|
|
|
|
checking access rights to the underlying tables of a view. Our grant
|
|
|
|
system gradually eliminates checked bits from want_privilege and thus
|
|
|
|
after all checks are done we can no longer use it.
|
|
|
|
The check that first_not_own_table is not reached is for the case when
|
|
|
|
the given table list refers to the list for prelocking (contains tables
|
|
|
|
of other queries). For simple queries first_not_own_table is 0.
|
2005-10-28 00:18:23 +03:00
|
|
|
*/
|
|
|
|
for (i= 0, table= tables;
|
2006-01-06 00:47:49 +02:00
|
|
|
table != first_not_own_table && i < number;
|
2005-10-28 00:18:23 +03:00
|
|
|
table= table->next_global, i++)
|
|
|
|
{
|
|
|
|
/* Remove SHOW_VIEW_ACL, because it will be checked during making view */
|
|
|
|
table->grant.orig_want_privilege= (want_access & ~SHOW_VIEW_ACL);
|
|
|
|
}
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2005-10-28 00:18:23 +03:00
|
|
|
for (table= tables;
|
|
|
|
table && number-- && table != first_not_own_table;
|
|
|
|
table= table->next_global)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-09-03 21:43:04 +03:00
|
|
|
GRANT_TABLE *grant_table;
|
2006-05-26 11:47:53 +03:00
|
|
|
sctx = test(table->security_ctx) ?
|
|
|
|
table->security_ctx : thd->security_ctx;
|
|
|
|
|
|
|
|
want_access= orig_want_access;
|
|
|
|
want_access&= ~sctx->master_access;
|
|
|
|
if (!want_access)
|
|
|
|
continue; // ok
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
if (!(~table->grant.privilege & want_access) ||
|
2006-05-26 11:47:53 +03:00
|
|
|
table->derived || table->schema_table)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
/*
|
|
|
|
It is subquery in the FROM clause. VIEW set table->derived after
|
|
|
|
table opening, but this function always called before table opening.
|
|
|
|
*/
|
2005-10-28 00:18:23 +03:00
|
|
|
if (!table->referencing_view)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
If it's a temporary table created for a subquery in the FROM
|
|
|
|
clause, or an INFORMATION_SCHEMA table, drop the request for
|
|
|
|
a privilege.
|
|
|
|
*/
|
|
|
|
table->grant.want_privilege= 0;
|
|
|
|
}
|
|
|
|
continue;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2005-09-15 22:29:07 +03:00
|
|
|
if (!(grant_table= table_hash_search(sctx->host, sctx->ip,
|
|
|
|
table->db, sctx->priv_user,
|
|
|
|
table->table_name,0)))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
want_access &= ~table->grant.privilege;
|
|
|
|
goto err; // No grants
|
|
|
|
}
|
2000-10-06 21:15:03 +03:00
|
|
|
if (show_table)
|
|
|
|
continue; // We have some priv on this
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
table->grant.grant_table=grant_table; // Remember for column test
|
|
|
|
table->grant.version=grant_version;
|
|
|
|
table->grant.privilege|= grant_table->privs;
|
|
|
|
table->grant.want_privilege= ((want_access & COL_ACLS)
|
|
|
|
& ~table->grant.privilege);
|
|
|
|
|
|
|
|
if (!(~table->grant.privilege & want_access))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (want_access & ~(grant_table->cols | table->grant.privilege))
|
|
|
|
{
|
|
|
|
want_access &= ~(grant_table->cols | table->grant.privilege);
|
|
|
|
goto err; // impossible
|
|
|
|
}
|
|
|
|
}
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2004-09-03 21:43:04 +03:00
|
|
|
DBUG_RETURN(0);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
err:
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2001-08-10 17:37:37 +03:00
|
|
|
if (!no_errors) // Not a silent skip of table
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-12-30 21:01:11 +01:00
|
|
|
char command[128];
|
|
|
|
get_privilege_desc(command, sizeof(command), want_access);
|
2004-11-13 19:35:51 +02:00
|
|
|
my_error(ER_TABLEACCESS_DENIED_ERROR, MYF(0),
|
|
|
|
command,
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->priv_user,
|
|
|
|
sctx->host_or_ip,
|
2005-01-06 13:00:13 +02:00
|
|
|
table ? table->table_name : "unknown");
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-09-03 21:43:04 +03:00
|
|
|
DBUG_RETURN(1);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-30 21:27:11 +02:00
|
|
|
/*
|
|
|
|
Check column rights in given security context
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
check_grant_column()
|
|
|
|
thd thread handler
|
|
|
|
grant grant information structure
|
|
|
|
db_name db name
|
|
|
|
table_name table name
|
|
|
|
name column name
|
|
|
|
length column name length
|
|
|
|
sctx security context
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE OK
|
|
|
|
TRUE access denied
|
|
|
|
*/
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
bool check_grant_column(THD *thd, GRANT_INFO *grant,
|
2005-01-06 13:00:13 +02:00
|
|
|
const char *db_name, const char *table_name,
|
2005-11-30 21:27:11 +02:00
|
|
|
const char *name, uint length, Security_context *sctx)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
GRANT_TABLE *grant_table;
|
|
|
|
GRANT_COLUMN *grant_column;
|
2004-07-16 01:15:55 +03:00
|
|
|
ulong want_access= grant->want_privilege & ~grant->privilege;
|
2004-12-31 00:44:00 +02:00
|
|
|
DBUG_ENTER("check_grant_column");
|
2006-11-20 22:42:06 +02:00
|
|
|
DBUG_PRINT("enter", ("table: %s want_access: %lu", table_name, want_access));
|
2004-12-31 00:44:00 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!want_access)
|
2004-12-31 00:44:00 +02:00
|
|
|
DBUG_RETURN(0); // Already checked
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* reload table if someone has modified any grants */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
if (grant->version != grant_version)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
grant->grant_table=
|
2005-09-15 22:29:07 +03:00
|
|
|
table_hash_search(sctx->host, sctx->ip, db_name,
|
|
|
|
sctx->priv_user,
|
2004-12-31 00:44:00 +02:00
|
|
|
table_name, 0); /* purecov: inspected */
|
2004-07-16 01:15:55 +03:00
|
|
|
grant->version= grant_version; /* purecov: inspected */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-07-16 01:15:55 +03:00
|
|
|
if (!(grant_table= grant->grant_table))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto err; /* purecov: deadcode */
|
|
|
|
|
|
|
|
grant_column=column_hash_search(grant_table, name, length);
|
|
|
|
if (grant_column && !(~grant_column->rights & want_access))
|
|
|
|
{
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2004-12-31 00:44:00 +02:00
|
|
|
DBUG_RETURN(0);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2003-05-07 23:59:24 +03:00
|
|
|
err:
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2005-11-30 21:27:11 +02:00
|
|
|
char command[128];
|
|
|
|
get_privilege_desc(command, sizeof(command), want_access);
|
|
|
|
my_error(ER_COLUMNACCESS_DENIED_ERROR, MYF(0),
|
|
|
|
command,
|
|
|
|
sctx->priv_user,
|
|
|
|
sctx->host_or_ip,
|
|
|
|
name,
|
|
|
|
table_name);
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Check the access right to a column depending on the type of table.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
check_column_grant_in_table_ref()
|
|
|
|
thd thread handler
|
|
|
|
table_ref table reference where to check the field
|
|
|
|
name name of field to check
|
|
|
|
length length of name
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Check the access rights to a column depending on the type of table
|
|
|
|
reference where the column is checked. The function provides a
|
|
|
|
generic interface to check column access rights that hides the
|
|
|
|
heterogeneity of the column representation - whether it is a view
|
|
|
|
or a stored table colum.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE OK
|
|
|
|
TRUE access denied
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool check_column_grant_in_table_ref(THD *thd, TABLE_LIST * table_ref,
|
|
|
|
const char *name, uint length)
|
|
|
|
{
|
|
|
|
GRANT_INFO *grant;
|
|
|
|
const char *db_name;
|
|
|
|
const char *table_name;
|
|
|
|
Security_context *sctx= test(table_ref->security_ctx) ?
|
|
|
|
table_ref->security_ctx : thd->security_ctx;
|
|
|
|
|
|
|
|
if (table_ref->view || table_ref->field_translation)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-11-30 21:27:11 +02:00
|
|
|
/* View or derived information schema table. */
|
2006-07-25 17:23:25 +05:00
|
|
|
ulong view_privs;
|
2005-11-30 21:27:11 +02:00
|
|
|
grant= &(table_ref->grant);
|
|
|
|
db_name= table_ref->view_db.str;
|
|
|
|
table_name= table_ref->view_name.str;
|
2006-07-25 17:23:25 +05:00
|
|
|
if (table_ref->belong_to_view &&
|
|
|
|
(thd->lex->sql_command == SQLCOM_SHOW_FIELDS ||
|
|
|
|
thd->lex->sql_command == SQLCOM_SHOW_CREATE))
|
|
|
|
{
|
|
|
|
view_privs= get_column_grant(thd, grant, db_name, table_name, name);
|
|
|
|
if (view_privs & VIEW_ANY_ACL)
|
|
|
|
{
|
|
|
|
table_ref->belong_to_view->allowed_show= TRUE;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
table_ref->belong_to_view->allowed_show= FALSE;
|
|
|
|
my_message(ER_VIEW_NO_EXPLAIN, ER(ER_VIEW_NO_EXPLAIN), MYF(0));
|
|
|
|
return TRUE;
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2005-11-30 21:27:11 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Normal or temporary table. */
|
|
|
|
TABLE *table= table_ref->table;
|
|
|
|
grant= &(table->grant);
|
2005-12-07 12:58:33 +01:00
|
|
|
db_name= table->s->db.str;
|
|
|
|
table_name= table->s->table_name.str;
|
2005-11-30 21:27:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (grant->want_privilege)
|
|
|
|
return check_grant_column(thd, grant, db_name, table_name, name,
|
|
|
|
length, sctx);
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
bool check_grant_all_columns(THD *thd, ulong want_access, GRANT_INFO *grant,
|
2005-01-06 13:00:13 +02:00
|
|
|
const char* db_name, const char *table_name,
|
2004-07-16 01:15:55 +03:00
|
|
|
Field_iterator *fields)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2000-07-31 21:29:14 +02:00
|
|
|
GRANT_TABLE *grant_table;
|
|
|
|
GRANT_COLUMN *grant_column;
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
want_access &= ~grant->privilege;
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!want_access)
|
2003-05-21 21:39:58 +03:00
|
|
|
return 0; // Already checked
|
2003-05-19 16:35:49 +03:00
|
|
|
if (!grant_option)
|
|
|
|
goto err2;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* reload table if someone has modified any grants */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
if (grant->version != grant_version)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
grant->grant_table=
|
2005-09-15 22:29:07 +03:00
|
|
|
table_hash_search(sctx->host, sctx->ip, db_name,
|
|
|
|
sctx->priv_user,
|
2004-07-16 01:15:55 +03:00
|
|
|
table_name, 0); /* purecov: inspected */
|
|
|
|
grant->version= grant_version; /* purecov: inspected */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2003-06-23 20:03:59 +03:00
|
|
|
/* The following should always be true */
|
2004-07-16 01:15:55 +03:00
|
|
|
if (!(grant_table= grant->grant_table))
|
2000-07-31 21:29:14 +02:00
|
|
|
goto err; /* purecov: inspected */
|
|
|
|
|
2004-09-03 21:43:04 +03:00
|
|
|
for (; !fields->end_of_fields(); fields->next())
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
const char *field_name= fields->name();
|
|
|
|
grant_column= column_hash_search(grant_table, field_name,
|
|
|
|
(uint) strlen(field_name));
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!grant_column || (~grant_column->rights & want_access))
|
|
|
|
goto err;
|
|
|
|
}
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
return 0;
|
|
|
|
|
2003-05-19 16:35:49 +03:00
|
|
|
err:
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2003-05-19 16:35:49 +03:00
|
|
|
err2:
|
2004-12-30 21:01:11 +01:00
|
|
|
char command[128];
|
|
|
|
get_privilege_desc(command, sizeof(command), want_access);
|
2004-11-13 19:35:51 +02:00
|
|
|
my_error(ER_COLUMNACCESS_DENIED_ERROR, MYF(0),
|
|
|
|
command,
|
2005-09-15 22:29:07 +03:00
|
|
|
sctx->priv_user,
|
|
|
|
sctx->host_or_ip,
|
2004-11-13 19:35:51 +02:00
|
|
|
fields->name(),
|
|
|
|
table_name);
|
2000-07-31 21:29:14 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-23 14:12:11 +03:00
|
|
|
static bool check_grant_db_routine(THD *thd, const char *db, HASH *hash)
|
|
|
|
{
|
|
|
|
Security_context *sctx= thd->security_ctx;
|
|
|
|
|
|
|
|
for (uint idx= 0; idx < hash->records; ++idx)
|
|
|
|
{
|
|
|
|
GRANT_NAME *item= (GRANT_NAME*) hash_element(hash, idx);
|
|
|
|
|
|
|
|
if (strcmp(item->user, sctx->priv_user) == 0 &&
|
|
|
|
strcmp(item->db, db) == 0 &&
|
|
|
|
compare_hostname(&item->host, sctx->host, sctx->ip))
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
2002-06-12 15:04:18 +03:00
|
|
|
Check if a user has the right to access a database
|
2004-12-23 10:46:24 +00:00
|
|
|
Access is accepted if he has a grant for any table/routine in the database
|
2002-06-12 15:04:18 +03:00
|
|
|
Return 1 if access is denied
|
2003-06-23 20:03:59 +03:00
|
|
|
*/
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
bool check_grant_db(THD *thd,const char *db)
|
|
|
|
{
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2006-09-27 19:21:29 +05:00
|
|
|
char helping [NAME_LEN+USERNAME_LENGTH+2];
|
2000-07-31 21:29:14 +02:00
|
|
|
uint len;
|
2007-03-23 14:12:11 +03:00
|
|
|
bool error= TRUE;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
len= (uint) (strmov(strmov(helping, sctx->priv_user) + 1, db) - helping) + 1;
|
2007-03-23 14:12:11 +03:00
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-01-18 16:39:21 +02:00
|
|
|
for (uint idx=0 ; idx < column_priv_hash.records ; idx++)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-01-18 16:39:21 +02:00
|
|
|
GRANT_TABLE *grant_table= (GRANT_TABLE*) hash_element(&column_priv_hash,
|
|
|
|
idx);
|
2000-07-31 21:29:14 +02:00
|
|
|
if (len < grant_table->key_length &&
|
|
|
|
!memcmp(grant_table->hash_key,helping,len) &&
|
2005-09-15 22:29:07 +03:00
|
|
|
compare_hostname(&grant_table->host, sctx->host, sctx->ip))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2007-03-23 14:12:11 +03:00
|
|
|
error= FALSE; /* Found match. */
|
2000-07-31 21:29:14 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-03-23 14:12:11 +03:00
|
|
|
|
|
|
|
if (error)
|
|
|
|
error= check_grant_db_routine(thd, db, &proc_priv_hash) &&
|
|
|
|
check_grant_db_routine(thd, db, &func_priv_hash);
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2007-03-23 14:12:11 +03:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
/****************************************************************************
|
2005-05-17 19:54:20 +01:00
|
|
|
Check routine level grants
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
SYNPOSIS
|
2005-05-17 19:54:20 +01:00
|
|
|
bool check_grant_routine()
|
2004-12-23 10:46:24 +00:00
|
|
|
thd Thread handler
|
|
|
|
want_access Bits of privileges user needs to have
|
2005-05-17 19:54:20 +01:00
|
|
|
procs List of routines to check. The user should have 'want_access'
|
|
|
|
is_proc True if the list is all procedures, else functions
|
2004-12-23 10:46:24 +00:00
|
|
|
no_errors If 0 then we write an error. The error is sent directly to
|
|
|
|
the client
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 ok
|
|
|
|
1 Error: User did not have the requested privielges
|
|
|
|
****************************************************************************/
|
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
bool check_grant_routine(THD *thd, ulong want_access,
|
2005-05-17 19:54:20 +01:00
|
|
|
TABLE_LIST *procs, bool is_proc, bool no_errors)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
TABLE_LIST *table;
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2005-09-15 22:29:07 +03:00
|
|
|
char *user= sctx->priv_user;
|
|
|
|
char *host= sctx->priv_host;
|
2005-05-17 19:54:20 +01:00
|
|
|
DBUG_ENTER("check_grant_routine");
|
2004-12-23 10:46:24 +00:00
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
want_access&= ~sctx->master_access;
|
2004-12-23 10:46:24 +00:00
|
|
|
if (!want_access)
|
|
|
|
DBUG_RETURN(0); // ok
|
|
|
|
|
|
|
|
rw_rdlock(&LOCK_grant);
|
|
|
|
for (table= procs; table; table= table->next_global)
|
|
|
|
{
|
|
|
|
GRANT_NAME *grant_proc;
|
2005-09-15 22:29:07 +03:00
|
|
|
if ((grant_proc= routine_hash_search(host, sctx->ip, table->db, user,
|
2005-05-17 19:54:20 +01:00
|
|
|
table->table_name, is_proc, 0)))
|
2004-12-23 10:46:24 +00:00
|
|
|
table->grant.privilege|= grant_proc->privs;
|
|
|
|
|
|
|
|
if (want_access & ~table->grant.privilege)
|
|
|
|
{
|
|
|
|
want_access &= ~table->grant.privilege;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
err:
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
if (!no_errors)
|
|
|
|
{
|
|
|
|
char buff[1024];
|
|
|
|
const char *command="";
|
|
|
|
if (table)
|
2005-01-06 13:00:13 +02:00
|
|
|
strxmov(buff, table->db, ".", table->table_name, NullS);
|
2004-12-23 10:46:24 +00:00
|
|
|
if (want_access & EXECUTE_ACL)
|
|
|
|
command= "execute";
|
|
|
|
else if (want_access & ALTER_PROC_ACL)
|
2005-05-17 19:54:20 +01:00
|
|
|
command= "alter routine";
|
2004-12-23 10:46:24 +00:00
|
|
|
else if (want_access & GRANT_ACL)
|
|
|
|
command= "grant";
|
|
|
|
my_error(ER_PROCACCESS_DENIED_ERROR, MYF(0),
|
|
|
|
command, user, host, table ? buff : "unknown");
|
|
|
|
}
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-05 14:35:32 +03:00
|
|
|
/*
|
|
|
|
Check if routine has any of the
|
2005-05-17 19:54:20 +01:00
|
|
|
routine level grants
|
2005-03-05 14:35:32 +03:00
|
|
|
|
|
|
|
SYNPOSIS
|
|
|
|
bool check_routine_level_acl()
|
|
|
|
thd Thread handler
|
|
|
|
db Database name
|
|
|
|
name Routine name
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 Ok
|
2005-03-15 16:07:28 +02:00
|
|
|
1 error
|
2005-03-05 14:35:32 +03:00
|
|
|
*/
|
|
|
|
|
2005-05-17 23:49:01 +01:00
|
|
|
bool check_routine_level_acl(THD *thd, const char *db, const char *name,
|
|
|
|
bool is_proc)
|
2005-03-05 14:35:32 +03:00
|
|
|
{
|
|
|
|
bool no_routine_acl= 1;
|
|
|
|
if (grant_option)
|
|
|
|
{
|
|
|
|
GRANT_NAME *grant_proc;
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2005-03-05 14:35:32 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2005-09-20 21:20:38 +03:00
|
|
|
if ((grant_proc= routine_hash_search(sctx->priv_host,
|
|
|
|
sctx->ip, db,
|
|
|
|
sctx->priv_user,
|
2005-09-15 22:29:07 +03:00
|
|
|
name, is_proc, 0)))
|
2005-03-05 14:35:32 +03:00
|
|
|
no_routine_acl= !(grant_proc->privs & SHOW_PROC_ACLS);
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
}
|
|
|
|
return no_routine_acl;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/*****************************************************************************
|
2002-06-12 15:04:18 +03:00
|
|
|
Functions to retrieve the grant for a table/column (for SHOW functions)
|
2000-07-31 21:29:14 +02:00
|
|
|
*****************************************************************************/
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong get_table_grant(THD *thd, TABLE_LIST *table)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-10-02 17:55:12 +03:00
|
|
|
ulong privilege;
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2000-07-31 21:29:14 +02:00
|
|
|
const char *db = table->db ? table->db : thd->db;
|
|
|
|
GRANT_TABLE *grant_table;
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2002-06-17 16:24:51 +05:00
|
|
|
#ifdef EMBEDDED_LIBRARY
|
|
|
|
grant_table= NULL;
|
|
|
|
#else
|
2005-09-15 22:29:07 +03:00
|
|
|
grant_table= table_hash_search(sctx->host, sctx->ip, db, sctx->priv_user,
|
2005-01-06 13:00:13 +02:00
|
|
|
table->table_name, 0);
|
2002-06-17 16:24:51 +05:00
|
|
|
#endif
|
2000-07-31 21:29:14 +02:00
|
|
|
table->grant.grant_table=grant_table; // Remember for column test
|
|
|
|
table->grant.version=grant_version;
|
|
|
|
if (grant_table)
|
|
|
|
table->grant.privilege|= grant_table->privs;
|
2002-09-21 21:36:23 +03:00
|
|
|
privilege= table->grant.privilege;
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2002-09-21 21:36:23 +03:00
|
|
|
return privilege;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-08-12 17:57:19 +03:00
|
|
|
/*
|
|
|
|
Determine the access priviliges for a field.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
get_column_grant()
|
|
|
|
thd thread handler
|
|
|
|
grant grants table descriptor
|
|
|
|
db_name name of database that the field belongs to
|
|
|
|
table_name name of table that the field belongs to
|
|
|
|
field_name name of field
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
The procedure may also modify: grant->grant_table and grant->version.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
The access priviliges for the field db_name.table_name.field_name
|
|
|
|
*/
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
ulong get_column_grant(THD *thd, GRANT_INFO *grant,
|
|
|
|
const char *db_name, const char *table_name,
|
|
|
|
const char *field_name)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
GRANT_TABLE *grant_table;
|
|
|
|
GRANT_COLUMN *grant_column;
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong priv;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2003-06-23 20:03:59 +03:00
|
|
|
/* reload table if someone has modified any grants */
|
2004-07-16 01:15:55 +03:00
|
|
|
if (grant->version != grant_version)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2004-07-16 01:15:55 +03:00
|
|
|
grant->grant_table=
|
2005-09-20 21:20:38 +03:00
|
|
|
table_hash_search(sctx->host, sctx->ip,
|
|
|
|
db_name, sctx->priv_user,
|
2004-07-16 01:15:55 +03:00
|
|
|
table_name, 0); /* purecov: inspected */
|
|
|
|
grant->version= grant_version; /* purecov: inspected */
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
if (!(grant_table= grant->grant_table))
|
|
|
|
priv= grant->privilege;
|
2000-07-31 21:29:14 +02:00
|
|
|
else
|
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
grant_column= column_hash_search(grant_table, field_name,
|
|
|
|
(uint) strlen(field_name));
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!grant_column)
|
2005-05-07 13:55:46 +00:00
|
|
|
priv= (grant->privilege | grant_table->privs);
|
2000-07-31 21:29:14 +02:00
|
|
|
else
|
2005-05-07 13:55:46 +00:00
|
|
|
priv= (grant->privilege | grant_table->privs | grant_column->rights);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2002-10-02 13:33:08 +03:00
|
|
|
rw_unlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
return priv;
|
|
|
|
}
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/* Help function for mysql_show_grants */
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
static void add_user_option(String *grant, ulong value, const char *name)
|
|
|
|
{
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
char buff[22], *p; // just as in int2str
|
|
|
|
grant->append(' ');
|
|
|
|
grant->append(name, strlen(name));
|
|
|
|
grant->append(' ');
|
|
|
|
p=int10_to_str(value, buff, 10);
|
|
|
|
grant->append(buff,p-buff);
|
|
|
|
}
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
static const char *command_array[]=
|
2002-06-12 15:04:18 +03:00
|
|
|
{
|
2004-07-16 01:15:55 +03:00
|
|
|
"SELECT", "INSERT", "UPDATE", "DELETE", "CREATE", "DROP", "RELOAD",
|
|
|
|
"SHUTDOWN", "PROCESS","FILE", "GRANT", "REFERENCES", "INDEX",
|
|
|
|
"ALTER", "SHOW DATABASES", "SUPER", "CREATE TEMPORARY TABLES",
|
|
|
|
"LOCK TABLES", "EXECUTE", "REPLICATION SLAVE", "REPLICATION CLIENT",
|
2004-12-23 10:46:24 +00:00
|
|
|
"CREATE VIEW", "SHOW VIEW", "CREATE ROUTINE", "ALTER ROUTINE",
|
2006-02-01 13:28:45 +03:00
|
|
|
"CREATE USER", "EVENT", "TRIGGER"
|
2002-06-12 15:04:18 +03:00
|
|
|
};
|
2003-06-23 20:03:59 +03:00
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
static uint command_lengths[]=
|
|
|
|
{
|
2005-03-22 15:54:18 +01:00
|
|
|
6, 6, 6, 6, 6, 4, 6, 8, 7, 4, 5, 10, 5, 5, 14, 5, 23, 11, 7, 17, 18, 11, 9,
|
2006-02-01 13:28:45 +03:00
|
|
|
14, 13, 11, 5, 7
|
2002-06-12 15:04:18 +03:00
|
|
|
};
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
static int show_routine_grants(THD *thd, LEX_USER *lex_user, HASH *hash,
|
|
|
|
const char *type, int typelen,
|
|
|
|
char *buff, int buffsize);
|
|
|
|
|
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
/*
|
|
|
|
SHOW GRANTS; Send grants for a user to the client
|
|
|
|
|
|
|
|
IMPLEMENTATION
|
|
|
|
Send to client grant-like strings depicting user@host privileges
|
|
|
|
*/
|
|
|
|
|
2004-10-20 04:04:37 +03:00
|
|
|
bool mysql_show_grants(THD *thd,LEX_USER *lex_user)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong want_access;
|
|
|
|
uint counter,index;
|
2000-07-31 21:29:14 +02:00
|
|
|
int error = 0;
|
2003-08-19 00:08:08 +03:00
|
|
|
ACL_USER *acl_user;
|
|
|
|
ACL_DB *acl_db;
|
2000-07-31 21:29:14 +02:00
|
|
|
char buff[1024];
|
2002-12-11 09:17:51 +02:00
|
|
|
Protocol *protocol= thd->protocol;
|
2001-08-30 04:28:55 +08:00
|
|
|
DBUG_ENTER("mysql_show_grants");
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
LINT_INIT(acl_user);
|
|
|
|
if (!initialized)
|
|
|
|
{
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--skip-grant-tables");
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
2004-05-05 21:24:21 +03:00
|
|
|
|
2006-05-06 11:25:59 +04:00
|
|
|
rw_rdlock(&LOCK_grant);
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
|
|
|
acl_user= find_acl_user(lex_user->host.str, lex_user->user.str, TRUE);
|
|
|
|
if (!acl_user)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-05-06 11:25:59 +04:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
|
2004-02-21 15:15:26 +01:00
|
|
|
my_error(ER_NONEXISTING_GRANT, MYF(0),
|
|
|
|
lex_user->user.str, lex_user->host.str);
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(TRUE);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2003-01-29 17:31:20 +04:00
|
|
|
Item_string *field=new Item_string("",0,&my_charset_latin1);
|
2000-07-31 21:29:14 +02:00
|
|
|
List<Item> field_list;
|
|
|
|
field->name=buff;
|
|
|
|
field->max_length=1024;
|
|
|
|
strxmov(buff,"Grants for ",lex_user->user.str,"@",
|
|
|
|
lex_user->host.str,NullS);
|
|
|
|
field_list.push_back(field);
|
2004-08-03 03:32:21 -07:00
|
|
|
if (protocol->send_fields(&field_list,
|
|
|
|
Protocol::SEND_NUM_ROWS | Protocol::SEND_EOF))
|
2006-05-06 11:25:59 +04:00
|
|
|
{
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
rw_unlock(&LOCK_grant);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2006-05-06 11:25:59 +04:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* Add first global access grants */
|
|
|
|
{
|
2004-05-14 12:16:27 +05:00
|
|
|
String global(buff,sizeof(buff),system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
global.length(0);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("GRANT "));
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-08-19 00:08:08 +03:00
|
|
|
want_access= acl_user->access;
|
2000-07-31 21:29:14 +02:00
|
|
|
if (test_all_bits(want_access, (GLOBAL_ACLS & ~ GRANT_ACL)))
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("ALL PRIVILEGES"));
|
2000-07-31 21:29:14 +02:00
|
|
|
else if (!(want_access & ~GRANT_ACL))
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("USAGE"));
|
2002-11-30 20:58:53 +03:00
|
|
|
else
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
bool found=0;
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong j,test_access= want_access & ~GRANT_ACL;
|
2000-07-31 21:29:14 +02:00
|
|
|
for (counter=0, j = SELECT_ACL;j <= GLOBAL_ACLS;counter++,j <<= 1)
|
|
|
|
{
|
2002-11-30 20:58:53 +03:00
|
|
|
if (test_access & j)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (found)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(", "));
|
2000-07-31 21:29:14 +02:00
|
|
|
found=1;
|
|
|
|
global.append(command_array[counter],command_lengths[counter]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append (STRING_WITH_LEN(" ON *.* TO '"));
|
2004-05-14 12:16:27 +05:00
|
|
|
global.append(lex_user->user.str, lex_user->user.length,
|
|
|
|
system_charset_info);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append (STRING_WITH_LEN("'@'"));
|
2004-11-25 21:55:49 +01:00
|
|
|
global.append(lex_user->host.str,lex_user->host.length,
|
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
global.append ('\'');
|
2003-07-01 23:40:59 +04:00
|
|
|
if (acl_user->salt_len)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-07-01 23:40:59 +04:00
|
|
|
char passwd_buff[SCRAMBLED_PASSWORD_CHAR_LENGTH+1];
|
|
|
|
if (acl_user->salt_len == SCRAMBLE_LENGTH)
|
|
|
|
make_password_from_salt(passwd_buff, acl_user->salt);
|
|
|
|
else
|
|
|
|
make_password_from_salt_323(passwd_buff, (ulong *) acl_user->salt);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" IDENTIFIED BY PASSWORD '"));
|
2003-07-01 23:40:59 +04:00
|
|
|
global.append(passwd_buff);
|
2000-07-31 21:29:14 +02:00
|
|
|
global.append('\'');
|
|
|
|
}
|
2001-09-30 05:47:34 +03:00
|
|
|
/* "show grants" SSL related stuff */
|
|
|
|
if (acl_user->ssl_type == SSL_TYPE_ANY)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" REQUIRE SSL"));
|
2002-06-11 11:20:31 +03:00
|
|
|
else if (acl_user->ssl_type == SSL_TYPE_X509)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" REQUIRE X509"));
|
2002-06-11 11:20:31 +03:00
|
|
|
else if (acl_user->ssl_type == SSL_TYPE_SPECIFIED)
|
2001-09-30 10:46:20 +08:00
|
|
|
{
|
2002-06-11 11:20:31 +03:00
|
|
|
int ssl_options = 0;
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" REQUIRE "));
|
2001-09-30 05:47:34 +03:00
|
|
|
if (acl_user->x509_issuer)
|
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
ssl_options++;
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("ISSUER \'"));
|
2003-06-23 20:03:59 +03:00
|
|
|
global.append(acl_user->x509_issuer,strlen(acl_user->x509_issuer));
|
2002-06-11 11:20:31 +03:00
|
|
|
global.append('\'');
|
2001-09-30 10:46:20 +08:00
|
|
|
}
|
2001-09-30 05:47:34 +03:00
|
|
|
if (acl_user->x509_subject)
|
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
if (ssl_options++)
|
|
|
|
global.append(' ');
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("SUBJECT \'"));
|
2004-11-25 21:55:49 +01:00
|
|
|
global.append(acl_user->x509_subject,strlen(acl_user->x509_subject),
|
|
|
|
system_charset_info);
|
2002-06-11 11:20:31 +03:00
|
|
|
global.append('\'');
|
2001-08-30 04:28:55 +08:00
|
|
|
}
|
2001-09-30 05:47:34 +03:00
|
|
|
if (acl_user->ssl_cipher)
|
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
if (ssl_options++)
|
|
|
|
global.append(' ');
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("CIPHER '"));
|
2004-11-25 21:55:49 +01:00
|
|
|
global.append(acl_user->ssl_cipher,strlen(acl_user->ssl_cipher),
|
|
|
|
system_charset_info);
|
2002-06-11 11:20:31 +03:00
|
|
|
global.append('\'');
|
2001-09-30 10:46:20 +08:00
|
|
|
}
|
|
|
|
}
|
2002-06-12 15:04:18 +03:00
|
|
|
if ((want_access & GRANT_ACL) ||
|
2004-12-29 20:30:37 +03:00
|
|
|
(acl_user->user_resource.questions ||
|
|
|
|
acl_user->user_resource.updates ||
|
|
|
|
acl_user->user_resource.conn_per_hour ||
|
|
|
|
acl_user->user_resource.user_conn))
|
2002-05-15 13:50:38 +03:00
|
|
|
{
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" WITH"));
|
2002-06-12 15:04:18 +03:00
|
|
|
if (want_access & GRANT_ACL)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" GRANT OPTION"));
|
2003-06-23 20:03:59 +03:00
|
|
|
add_user_option(&global, acl_user->user_resource.questions,
|
|
|
|
"MAX_QUERIES_PER_HOUR");
|
|
|
|
add_user_option(&global, acl_user->user_resource.updates,
|
|
|
|
"MAX_UPDATES_PER_HOUR");
|
2004-12-29 20:30:37 +03:00
|
|
|
add_user_option(&global, acl_user->user_resource.conn_per_hour,
|
2003-06-23 20:03:59 +03:00
|
|
|
"MAX_CONNECTIONS_PER_HOUR");
|
2004-12-29 20:30:37 +03:00
|
|
|
add_user_option(&global, acl_user->user_resource.user_conn,
|
|
|
|
"MAX_USER_CONNECTIONS");
|
2002-02-23 14:00:07 +02:00
|
|
|
}
|
2002-12-11 09:17:51 +02:00
|
|
|
protocol->prepare_for_resend();
|
2003-03-17 13:14:04 +04:00
|
|
|
protocol->store(global.ptr(),global.length(),global.charset());
|
2002-12-11 09:17:51 +02:00
|
|
|
if (protocol->write())
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-09-11 19:06:23 +03:00
|
|
|
error= -1;
|
2002-12-11 09:17:51 +02:00
|
|
|
goto end;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add database access */
|
|
|
|
for (counter=0 ; counter < acl_dbs.elements ; counter++)
|
|
|
|
{
|
2004-05-06 11:42:23 +03:00
|
|
|
const char *user, *host;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
acl_db=dynamic_element(&acl_dbs,counter,ACL_DB*);
|
|
|
|
if (!(user=acl_db->user))
|
2004-05-05 17:05:24 +03:00
|
|
|
user= "";
|
2000-07-31 21:29:14 +02:00
|
|
|
if (!(host=acl_db->host.hostname))
|
2004-05-05 17:05:24 +03:00
|
|
|
host= "";
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (!strcmp(lex_user->user.str,user) &&
|
2004-11-25 21:55:49 +01:00
|
|
|
!my_strcasecmp(system_charset_info, lex_user->host.str, host))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
want_access=acl_db->access;
|
2002-11-30 20:58:53 +03:00
|
|
|
if (want_access)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-05-14 12:16:27 +05:00
|
|
|
String db(buff,sizeof(buff),system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
db.length(0);
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append(STRING_WITH_LEN("GRANT "));
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (test_all_bits(want_access,(DB_ACLS & ~GRANT_ACL)))
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append(STRING_WITH_LEN("ALL PRIVILEGES"));
|
2003-01-11 18:02:10 +02:00
|
|
|
else if (!(want_access & ~GRANT_ACL))
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append(STRING_WITH_LEN("USAGE"));
|
2000-07-31 21:29:14 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
int found=0, cnt;
|
2002-06-12 15:04:18 +03:00
|
|
|
ulong j,test_access= want_access & ~GRANT_ACL;
|
2000-07-31 21:29:14 +02:00
|
|
|
for (cnt=0, j = SELECT_ACL; j <= DB_ACLS; cnt++,j <<= 1)
|
|
|
|
{
|
|
|
|
if (test_access & j)
|
|
|
|
{
|
|
|
|
if (found)
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append(STRING_WITH_LEN(", "));
|
2000-07-31 21:29:14 +02:00
|
|
|
found = 1;
|
|
|
|
db.append(command_array[cnt],command_lengths[cnt]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append (STRING_WITH_LEN(" ON "));
|
2004-03-17 16:19:09 +04:00
|
|
|
append_identifier(thd, &db, acl_db->db, strlen(acl_db->db));
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append (STRING_WITH_LEN(".* TO '"));
|
2004-05-14 12:16:27 +05:00
|
|
|
db.append(lex_user->user.str, lex_user->user.length,
|
|
|
|
system_charset_info);
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append (STRING_WITH_LEN("'@'"));
|
2004-11-25 21:55:49 +01:00
|
|
|
db.append(lex_user->host.str, lex_user->host.length,
|
|
|
|
system_charset_info);
|
2002-11-30 20:58:53 +03:00
|
|
|
db.append ('\'');
|
2000-07-31 21:29:14 +02:00
|
|
|
if (want_access & GRANT_ACL)
|
2005-11-20 20:47:07 +02:00
|
|
|
db.append(STRING_WITH_LEN(" WITH GRANT OPTION"));
|
2002-12-11 09:17:51 +02:00
|
|
|
protocol->prepare_for_resend();
|
2003-03-17 13:14:04 +04:00
|
|
|
protocol->store(db.ptr(),db.length(),db.charset());
|
2002-12-11 09:17:51 +02:00
|
|
|
if (protocol->write())
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-09-11 19:06:23 +03:00
|
|
|
error= -1;
|
2000-07-31 21:29:14 +02:00
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-04-28 17:45:08 +03:00
|
|
|
/* Add table & column access */
|
2003-01-18 16:39:21 +02:00
|
|
|
for (index=0 ; index < column_priv_hash.records ; index++)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2006-06-06 14:21:07 +03:00
|
|
|
const char *user, *host;
|
2003-01-18 16:39:21 +02:00
|
|
|
GRANT_TABLE *grant_table= (GRANT_TABLE*) hash_element(&column_priv_hash,
|
|
|
|
index);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (!(user=grant_table->user))
|
2004-04-28 17:45:08 +03:00
|
|
|
user= "";
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!(host= grant_table->host.hostname))
|
|
|
|
host= "";
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
if (!strcmp(lex_user->user.str,user) &&
|
2006-06-06 14:21:07 +03:00
|
|
|
!my_strcasecmp(system_charset_info, lex_user->host.str, host))
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
ulong table_access= grant_table->privs;
|
|
|
|
if ((table_access | grant_table->cols) != 0)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-05-14 12:16:27 +05:00
|
|
|
String global(buff, sizeof(buff), system_charset_info);
|
2003-09-09 20:06:50 +03:00
|
|
|
ulong test_access= (table_access | grant_table->cols) & ~GRANT_ACL;
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
global.length(0);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("GRANT "));
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
if (test_all_bits(table_access, (TABLE_ACLS & ~GRANT_ACL)))
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("ALL PRIVILEGES"));
|
2003-09-11 20:24:14 +03:00
|
|
|
else if (!test_access)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("USAGE"));
|
2002-11-30 20:58:53 +03:00
|
|
|
else
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2004-04-28 17:45:08 +03:00
|
|
|
/* Add specific column access */
|
2003-06-23 20:03:59 +03:00
|
|
|
int found= 0;
|
2003-09-09 20:06:50 +03:00
|
|
|
ulong j;
|
2000-07-31 21:29:14 +02:00
|
|
|
|
2003-06-23 20:03:59 +03:00
|
|
|
for (counter= 0, j= SELECT_ACL; j <= TABLE_ACLS; counter++, j<<= 1)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-11-30 20:58:53 +03:00
|
|
|
if (test_access & j)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
|
|
|
if (found)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(", "));
|
2003-06-23 20:03:59 +03:00
|
|
|
found= 1;
|
2000-07-31 21:29:14 +02:00
|
|
|
global.append(command_array[counter],command_lengths[counter]);
|
|
|
|
|
2002-11-30 20:58:53 +03:00
|
|
|
if (grant_table->cols)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
uint found_col= 0;
|
2000-07-31 21:29:14 +02:00
|
|
|
for (uint col_index=0 ;
|
|
|
|
col_index < grant_table->hash_columns.records ;
|
|
|
|
col_index++)
|
|
|
|
{
|
|
|
|
GRANT_COLUMN *grant_column = (GRANT_COLUMN*)
|
|
|
|
hash_element(&grant_table->hash_columns,col_index);
|
2002-11-30 20:58:53 +03:00
|
|
|
if (grant_column->rights & j)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-11-30 20:58:53 +03:00
|
|
|
if (!found_col)
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2003-06-23 20:03:59 +03:00
|
|
|
found_col= 1;
|
|
|
|
/*
|
|
|
|
If we have a duplicated table level privilege, we
|
|
|
|
must write the access privilege name again.
|
|
|
|
*/
|
|
|
|
if (table_access & j)
|
|
|
|
{
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(", "));
|
2003-06-23 20:03:59 +03:00
|
|
|
global.append(command_array[counter],
|
|
|
|
command_lengths[counter]);
|
|
|
|
}
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" ("));
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
else
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(", "));
|
2000-07-31 21:29:14 +02:00
|
|
|
global.append(grant_column->column,
|
2004-05-14 12:16:27 +05:00
|
|
|
grant_column->key_length,
|
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (found_col)
|
|
|
|
global.append(')');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" ON "));
|
2004-03-17 16:19:09 +04:00
|
|
|
append_identifier(thd, &global, grant_table->db,
|
|
|
|
strlen(grant_table->db));
|
|
|
|
global.append('.');
|
|
|
|
append_identifier(thd, &global, grant_table->tname,
|
|
|
|
strlen(grant_table->tname));
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" TO '"));
|
2004-05-14 12:16:27 +05:00
|
|
|
global.append(lex_user->user.str, lex_user->user.length,
|
|
|
|
system_charset_info);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("'@'"));
|
2004-11-25 21:55:49 +01:00
|
|
|
global.append(lex_user->host.str,lex_user->host.length,
|
|
|
|
system_charset_info);
|
2000-07-31 21:29:14 +02:00
|
|
|
global.append('\'');
|
2003-06-23 20:03:59 +03:00
|
|
|
if (table_access & GRANT_ACL)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" WITH GRANT OPTION"));
|
2002-12-11 09:17:51 +02:00
|
|
|
protocol->prepare_for_resend();
|
2003-03-17 13:14:04 +04:00
|
|
|
protocol->store(global.ptr(),global.length(),global.charset());
|
2002-12-11 09:17:51 +02:00
|
|
|
if (protocol->write())
|
2000-07-31 21:29:14 +02:00
|
|
|
{
|
2002-10-02 17:55:12 +03:00
|
|
|
error= -1;
|
2002-09-21 21:36:23 +03:00
|
|
|
break;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-12-23 10:46:24 +00:00
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
if (show_routine_grants(thd, lex_user, &proc_priv_hash,
|
2005-11-20 20:47:07 +02:00
|
|
|
STRING_WITH_LEN("PROCEDURE"), buff, sizeof(buff)))
|
2005-05-17 19:54:20 +01:00
|
|
|
{
|
|
|
|
error= -1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (show_routine_grants(thd, lex_user, &func_priv_hash,
|
2005-11-20 20:47:07 +02:00
|
|
|
STRING_WITH_LEN("FUNCTION"), buff, sizeof(buff)))
|
2005-05-17 19:54:20 +01:00
|
|
|
{
|
|
|
|
error= -1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
|
|
|
|
send_eof(thd);
|
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int show_routine_grants(THD* thd, LEX_USER *lex_user, HASH *hash,
|
|
|
|
const char *type, int typelen,
|
|
|
|
char *buff, int buffsize)
|
|
|
|
{
|
|
|
|
uint counter, index;
|
|
|
|
int error= 0;
|
|
|
|
Protocol *protocol= thd->protocol;
|
|
|
|
/* Add routine access */
|
|
|
|
for (index=0 ; index < hash->records ; index++)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
2006-06-06 14:21:07 +03:00
|
|
|
const char *user, *host;
|
2005-05-17 19:54:20 +01:00
|
|
|
GRANT_NAME *grant_proc= (GRANT_NAME*) hash_element(hash, index);
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (!(user=grant_proc->user))
|
|
|
|
user= "";
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!(host= grant_proc->host.hostname))
|
|
|
|
host= "";
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (!strcmp(lex_user->user.str,user) &&
|
2006-06-06 14:21:07 +03:00
|
|
|
!my_strcasecmp(system_charset_info, lex_user->host.str, host))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
ulong proc_access= grant_proc->privs;
|
|
|
|
if (proc_access != 0)
|
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
String global(buff, buffsize, system_charset_info);
|
2004-12-23 10:46:24 +00:00
|
|
|
ulong test_access= proc_access & ~GRANT_ACL;
|
|
|
|
|
|
|
|
global.length(0);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("GRANT "));
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (!test_access)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("USAGE"));
|
2004-12-23 10:46:24 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Add specific procedure access */
|
|
|
|
int found= 0;
|
|
|
|
ulong j;
|
|
|
|
|
|
|
|
for (counter= 0, j= SELECT_ACL; j <= PROC_ACLS; counter++, j<<= 1)
|
|
|
|
{
|
|
|
|
if (test_access & j)
|
|
|
|
{
|
|
|
|
if (found)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(", "));
|
2004-12-23 10:46:24 +00:00
|
|
|
found= 1;
|
|
|
|
global.append(command_array[counter],command_lengths[counter]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" ON "));
|
2005-05-17 19:54:20 +01:00
|
|
|
global.append(type,typelen);
|
|
|
|
global.append(' ');
|
2004-12-23 10:46:24 +00:00
|
|
|
append_identifier(thd, &global, grant_proc->db,
|
|
|
|
strlen(grant_proc->db));
|
|
|
|
global.append('.');
|
|
|
|
append_identifier(thd, &global, grant_proc->tname,
|
|
|
|
strlen(grant_proc->tname));
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" TO '"));
|
2004-12-23 10:46:24 +00:00
|
|
|
global.append(lex_user->user.str, lex_user->user.length,
|
|
|
|
system_charset_info);
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN("'@'"));
|
2004-12-23 10:46:24 +00:00
|
|
|
global.append(lex_user->host.str,lex_user->host.length,
|
|
|
|
system_charset_info);
|
|
|
|
global.append('\'');
|
|
|
|
if (proc_access & GRANT_ACL)
|
2005-11-20 20:47:07 +02:00
|
|
|
global.append(STRING_WITH_LEN(" WITH GRANT OPTION"));
|
2004-12-23 10:46:24 +00:00
|
|
|
protocol->prepare_for_resend();
|
|
|
|
protocol->store(global.ptr(),global.length(),global.charset());
|
|
|
|
if (protocol->write())
|
|
|
|
{
|
|
|
|
error= -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-05-17 19:54:20 +01:00
|
|
|
return error;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
|
2002-06-12 15:04:18 +03:00
|
|
|
/*
|
|
|
|
Make a clear-text version of the requested privilege.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void get_privilege_desc(char *to, uint max_length, ulong access)
|
|
|
|
{
|
|
|
|
uint pos;
|
|
|
|
char *start=to;
|
|
|
|
DBUG_ASSERT(max_length >= 30); // For end ',' removal
|
|
|
|
|
|
|
|
if (access)
|
|
|
|
{
|
|
|
|
max_length--; // Reserve place for end-zero
|
|
|
|
for (pos=0 ; access ; pos++, access>>=1)
|
|
|
|
{
|
|
|
|
if ((access & 1) &&
|
|
|
|
command_lengths[pos] + (uint) (to-start) < max_length)
|
|
|
|
{
|
|
|
|
to= strmov(to, command_array[pos]);
|
|
|
|
*to++=',';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
to--; // Remove end ','
|
|
|
|
}
|
|
|
|
*to=0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-05-15 13:50:38 +03:00
|
|
|
void get_mqh(const char *user, const char *host, USER_CONN *uc)
|
2002-02-01 21:45:23 +02:00
|
|
|
{
|
|
|
|
ACL_USER *acl_user;
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
pthread_mutex_lock(&acl_cache->lock);
|
|
|
|
|
2005-08-22 15:48:50 -07:00
|
|
|
if (initialized && (acl_user= find_acl_user(host,user, FALSE)))
|
2002-05-15 13:50:38 +03:00
|
|
|
uc->user_resources= acl_user->user_resource;
|
|
|
|
else
|
|
|
|
bzero((char*) &uc->user_resources, sizeof(uc->user_resources));
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
pthread_mutex_unlock(&acl_cache->lock);
|
2002-02-01 21:45:23 +02:00
|
|
|
}
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
Open the grant tables.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
open_grant_tables()
|
|
|
|
thd The current thread.
|
|
|
|
tables (out) The 4 elements array for the opened tables.
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Tables are numbered as follows:
|
|
|
|
0 user
|
|
|
|
1 db
|
|
|
|
2 tables_priv
|
|
|
|
3 columns_priv
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
1 Skip GRANT handling during replication.
|
|
|
|
0 OK.
|
|
|
|
< 0 Error.
|
|
|
|
*/
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
#define GRANT_TABLES 5
|
2003-06-06 17:43:23 +05:00
|
|
|
int open_grant_tables(THD *thd, TABLE_LIST *tables)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("open_grant_tables");
|
|
|
|
|
|
|
|
if (!initialized)
|
|
|
|
{
|
2004-10-20 04:04:37 +03:00
|
|
|
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--skip-grant-tables");
|
2003-06-06 17:43:23 +05:00
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
bzero((char*) tables, GRANT_TABLES*sizeof(*tables));
|
2005-01-06 13:00:13 +02:00
|
|
|
tables->alias= tables->table_name= (char*) "user";
|
|
|
|
(tables+1)->alias= (tables+1)->table_name= (char*) "db";
|
|
|
|
(tables+2)->alias= (tables+2)->table_name= (char*) "tables_priv";
|
|
|
|
(tables+3)->alias= (tables+3)->table_name= (char*) "columns_priv";
|
|
|
|
(tables+4)->alias= (tables+4)->table_name= (char*) "procs_priv";
|
2004-07-16 01:15:55 +03:00
|
|
|
tables->next_local= tables->next_global= tables+1;
|
|
|
|
(tables+1)->next_local= (tables+1)->next_global= tables+2;
|
|
|
|
(tables+2)->next_local= (tables+2)->next_global= tables+3;
|
2004-12-23 10:46:24 +00:00
|
|
|
(tables+3)->next_local= (tables+3)->next_global= tables+4;
|
2003-06-06 17:43:23 +05:00
|
|
|
tables->lock_type= (tables+1)->lock_type=
|
2004-12-23 10:46:24 +00:00
|
|
|
(tables+2)->lock_type= (tables+3)->lock_type=
|
|
|
|
(tables+4)->lock_type= TL_WRITE;
|
|
|
|
tables->db= (tables+1)->db= (tables+2)->db=
|
|
|
|
(tables+3)->db= (tables+4)->db= (char*) "mysql";
|
2003-06-06 17:43:23 +05:00
|
|
|
|
|
|
|
#ifdef HAVE_REPLICATION
|
|
|
|
/*
|
|
|
|
GRANT and REVOKE are applied the slave in/exclusion rules as they are
|
|
|
|
some kind of updates to the mysql.% tables.
|
|
|
|
*/
|
2005-03-08 21:12:35 +01:00
|
|
|
if (thd->slave_thread && rpl_filter->is_on())
|
2003-08-02 17:23:39 +02:00
|
|
|
{
|
2003-08-11 22:44:43 +03:00
|
|
|
/*
|
|
|
|
The tables must be marked "updating" so that tables_ok() takes them into
|
|
|
|
account in tests.
|
2003-08-02 17:23:39 +02:00
|
|
|
*/
|
2004-12-23 10:46:24 +00:00
|
|
|
tables[0].updating=tables[1].updating=tables[2].updating=
|
|
|
|
tables[3].updating=tables[4].updating=1;
|
2005-05-09 11:43:09 +02:00
|
|
|
if (!(thd->spcont || rpl_filter->tables_ok(0, tables)))
|
2003-08-02 17:23:39 +02:00
|
|
|
DBUG_RETURN(1);
|
2004-12-23 10:46:24 +00:00
|
|
|
tables[0].updating=tables[1].updating=tables[2].updating=
|
|
|
|
tables[3].updating=tables[4].updating=0;;
|
2003-08-02 17:23:39 +02:00
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
#endif
|
|
|
|
|
2004-02-01 15:30:32 +02:00
|
|
|
if (simple_open_n_lock_tables(thd, tables))
|
2003-06-06 17:43:23 +05:00
|
|
|
{ // This should never happen
|
|
|
|
close_thread_tables(thd);
|
|
|
|
DBUG_RETURN(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ACL_USER *check_acl_user(LEX_USER *user_name,
|
2003-09-03 19:53:08 +03:00
|
|
|
uint *acl_acl_userdx)
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
|
|
|
ACL_USER *acl_user= 0;
|
|
|
|
uint counter;
|
|
|
|
|
2006-04-27 14:54:36 +04:00
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
for (counter= 0 ; counter < acl_users.elements ; counter++)
|
|
|
|
{
|
|
|
|
const char *user,*host;
|
|
|
|
acl_user= dynamic_element(&acl_users, counter, ACL_USER*);
|
|
|
|
if (!(user=acl_user->user))
|
2004-05-05 17:05:24 +03:00
|
|
|
user= "";
|
2003-06-06 17:43:23 +05:00
|
|
|
if (!(host=acl_user->host.hostname))
|
2006-01-09 10:31:07 +01:00
|
|
|
host= "";
|
2003-06-06 17:43:23 +05:00
|
|
|
if (!strcmp(user_name->user.str,user) &&
|
|
|
|
!my_strcasecmp(system_charset_info, user_name->host.str, host))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (counter == acl_users.elements)
|
|
|
|
return 0;
|
|
|
|
|
2003-09-03 19:53:08 +03:00
|
|
|
*acl_acl_userdx= counter;
|
2003-08-11 22:44:43 +03:00
|
|
|
return acl_user;
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2003-08-11 22:44:43 +03:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
Modify a privilege table.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
modify_grant_table()
|
|
|
|
table The table to modify.
|
|
|
|
host_field The host name field.
|
|
|
|
user_field The user name field.
|
|
|
|
user_to The new name for the user if to be renamed,
|
|
|
|
NULL otherwise.
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Update user/host in the current record if user_to is not NULL.
|
|
|
|
Delete the current record if user_to is NULL.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 OK.
|
|
|
|
!= 0 Error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int modify_grant_table(TABLE *table, Field *host_field,
|
|
|
|
Field *user_field, LEX_USER *user_to)
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-25 21:55:49 +01:00
|
|
|
int error;
|
|
|
|
DBUG_ENTER("modify_grant_table");
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
if (user_to)
|
|
|
|
{
|
|
|
|
/* rename */
|
|
|
|
store_record(table, record[1]);
|
|
|
|
host_field->store(user_to->host.str, user_to->host.length,
|
|
|
|
system_charset_info);
|
|
|
|
user_field->store(user_to->user.str, user_to->user.length,
|
|
|
|
system_charset_info);
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((error= table->file->ha_update_row(table->record[1], table->record[0])))
|
2004-11-25 21:55:49 +01:00
|
|
|
table->file->print_error(error, MYF(0));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* delete */
|
2005-12-22 06:39:02 +01:00
|
|
|
if ((error=table->file->ha_delete_row(table->record[0])))
|
2004-11-25 21:55:49 +01:00
|
|
|
table->file->print_error(error, MYF(0));
|
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_RETURN(error);
|
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
Handle a privilege table.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
handle_grant_table()
|
|
|
|
tables The array with the four open tables.
|
2004-12-23 10:46:24 +00:00
|
|
|
table_no The number of the table to handle (0..4).
|
2004-11-25 21:55:49 +01:00
|
|
|
drop If user_from is to be dropped.
|
|
|
|
user_from The the user to be searched/dropped/renamed.
|
|
|
|
user_to The new name for the user if to be renamed,
|
|
|
|
NULL otherwise.
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Scan through all records in a grant table and apply the requested
|
|
|
|
operation. For the "user" table, a single index access is sufficient,
|
|
|
|
since there is an unique index on (host, user).
|
|
|
|
Delete from grant table if drop is true.
|
|
|
|
Update in grant table if drop is false and user_to is not NULL.
|
|
|
|
Search in grant table if drop is false and user_to is NULL.
|
|
|
|
Tables are numbered as follows:
|
|
|
|
0 user
|
|
|
|
1 db
|
|
|
|
2 tables_priv
|
|
|
|
3 columns_priv
|
2004-12-23 10:46:24 +00:00
|
|
|
4 procs_priv
|
2004-11-25 21:55:49 +01:00
|
|
|
|
|
|
|
RETURN
|
|
|
|
> 0 At least one record matched.
|
|
|
|
0 OK, but no record matched.
|
|
|
|
< 0 Error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int handle_grant_table(TABLE_LIST *tables, uint table_no, bool drop,
|
|
|
|
LEX_USER *user_from, LEX_USER *user_to)
|
|
|
|
{
|
|
|
|
int result= 0;
|
|
|
|
int error;
|
|
|
|
TABLE *table= tables[table_no].table;
|
|
|
|
Field *host_field= table->field[0];
|
|
|
|
Field *user_field= table->field[table_no ? 2 : 1];
|
|
|
|
char *host_str= user_from->host.str;
|
|
|
|
char *user_str= user_from->user.str;
|
|
|
|
const char *host;
|
|
|
|
const char *user;
|
2004-12-07 15:47:00 +02:00
|
|
|
byte user_key[MAX_KEY_LENGTH];
|
2004-12-06 19:18:35 +02:00
|
|
|
uint key_prefix_length;
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_ENTER("handle_grant_table");
|
|
|
|
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
table->use_all_columns();
|
2004-11-27 23:07:30 +01:00
|
|
|
if (! table_no) // mysql.user table
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
The 'user' table has an unique index on (host, user).
|
|
|
|
Thus, we can handle everything with a single index access.
|
|
|
|
The host- and user fields are consecutive in the user table records.
|
|
|
|
So we set host- and user fields of table->record[0] and use the
|
|
|
|
pointer to the host field as key.
|
|
|
|
index_read_idx() will replace table->record[0] (its first argument)
|
|
|
|
by the searched record, if it exists.
|
|
|
|
*/
|
|
|
|
DBUG_PRINT("info",("read table: '%s' search: '%s'@'%s'",
|
2005-11-23 22:45:02 +02:00
|
|
|
table->s->table_name.str, user_str, host_str));
|
2004-11-25 21:55:49 +01:00
|
|
|
host_field->store(host_str, user_from->host.length, system_charset_info);
|
|
|
|
user_field->store(user_str, user_from->user.length, system_charset_info);
|
2004-12-06 19:18:35 +02:00
|
|
|
|
|
|
|
key_prefix_length= (table->key_info->key_part[0].store_length +
|
|
|
|
table->key_info->key_part[1].store_length);
|
|
|
|
key_copy(user_key, table->record[0], table->key_info, key_prefix_length);
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
if ((error= table->file->index_read_idx(table->record[0], 0,
|
2007-03-17 00:13:25 +01:00
|
|
|
user_key, (key_part_map)3,
|
2004-11-25 21:55:49 +01:00
|
|
|
HA_READ_KEY_EXACT)))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2005-07-18 13:31:02 +02:00
|
|
|
if (error != HA_ERR_KEY_NOT_FOUND && error != HA_ERR_END_OF_FILE)
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
|
|
|
table->file->print_error(error, MYF(0));
|
|
|
|
result= -1;
|
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-11-25 21:55:49 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* If requested, delete or update the record. */
|
|
|
|
result= ((drop || user_to) &&
|
|
|
|
modify_grant_table(table, host_field, user_field, user_to)) ?
|
|
|
|
-1 : 1; /* Error or found. */
|
|
|
|
}
|
|
|
|
DBUG_PRINT("info",("read result: %d", result));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
The non-'user' table do not have indexes on (host, user).
|
|
|
|
And their host- and user fields are not consecutive.
|
|
|
|
Thus, we need to do a table scan to find all matching records.
|
|
|
|
*/
|
|
|
|
if ((error= table->file->ha_rnd_init(1)))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-25 21:55:49 +01:00
|
|
|
table->file->print_error(error, MYF(0));
|
2003-06-06 17:43:23 +05:00
|
|
|
result= -1;
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
#ifdef EXTRA_DEBUG
|
|
|
|
DBUG_PRINT("info",("scan table: '%s' search: '%s'@'%s'",
|
2005-11-23 22:45:02 +02:00
|
|
|
table->s->table_name.str, user_str, host_str));
|
2004-11-25 21:55:49 +01:00
|
|
|
#endif
|
|
|
|
while ((error= table->file->rnd_next(table->record[0])) !=
|
|
|
|
HA_ERR_END_OF_FILE)
|
|
|
|
{
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
/* Most probable 'deleted record'. */
|
|
|
|
DBUG_PRINT("info",("scan error: %d", error));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (! (host= get_field(&mem, host_field)))
|
|
|
|
host= "";
|
|
|
|
if (! (user= get_field(&mem, user_field)))
|
|
|
|
user= "";
|
|
|
|
|
|
|
|
#ifdef EXTRA_DEBUG
|
|
|
|
DBUG_PRINT("loop",("scan fields: '%s'@'%s' '%s' '%s' '%s'",
|
|
|
|
user, host,
|
|
|
|
get_field(&mem, table->field[1]) /*db*/,
|
|
|
|
get_field(&mem, table->field[3]) /*table*/,
|
|
|
|
get_field(&mem, table->field[4]) /*column*/));
|
|
|
|
#endif
|
|
|
|
if (strcmp(user_str, user) ||
|
|
|
|
my_strcasecmp(system_charset_info, host_str, host))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* If requested, delete or update the record. */
|
|
|
|
result= ((drop || user_to) &&
|
|
|
|
modify_grant_table(table, host_field, user_field, user_to)) ?
|
|
|
|
-1 : result ? result : 1; /* Error or keep result or found. */
|
|
|
|
/* If search is requested, we do not need to search further. */
|
|
|
|
if (! drop && ! user_to)
|
|
|
|
break ;
|
|
|
|
}
|
|
|
|
(void) table->file->ha_rnd_end();
|
|
|
|
DBUG_PRINT("info",("scan result: %d", result));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Handle an in-memory privilege structure.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
handle_grant_struct()
|
2005-01-06 13:00:13 +02:00
|
|
|
struct_no The number of the structure to handle (0..3).
|
2004-11-25 21:55:49 +01:00
|
|
|
drop If user_from is to be dropped.
|
|
|
|
user_from The the user to be searched/dropped/renamed.
|
|
|
|
user_to The new name for the user if to be renamed,
|
|
|
|
NULL otherwise.
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Scan through all elements in an in-memory grant structure and apply
|
|
|
|
the requested operation.
|
|
|
|
Delete from grant structure if drop is true.
|
|
|
|
Update in grant structure if drop is false and user_to is not NULL.
|
|
|
|
Search in grant structure if drop is false and user_to is NULL.
|
|
|
|
Structures are numbered as follows:
|
|
|
|
0 acl_users
|
|
|
|
1 acl_dbs
|
|
|
|
2 column_priv_hash
|
2004-12-23 10:46:24 +00:00
|
|
|
3 procs_priv_hash
|
2004-11-25 21:55:49 +01:00
|
|
|
|
|
|
|
RETURN
|
|
|
|
> 0 At least one element matched.
|
|
|
|
0 OK, but no element matched.
|
2005-01-06 13:00:13 +02:00
|
|
|
-1 Wrong arguments to function
|
2004-11-25 21:55:49 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
static int handle_grant_struct(uint struct_no, bool drop,
|
|
|
|
LEX_USER *user_from, LEX_USER *user_to)
|
|
|
|
{
|
|
|
|
int result= 0;
|
|
|
|
uint idx;
|
|
|
|
uint elements;
|
|
|
|
const char *user;
|
|
|
|
const char *host;
|
|
|
|
ACL_USER *acl_user;
|
|
|
|
ACL_DB *acl_db;
|
2004-12-23 10:46:24 +00:00
|
|
|
GRANT_NAME *grant_name;
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_ENTER("handle_grant_struct");
|
2005-08-25 09:38:06 +03:00
|
|
|
DBUG_PRINT("info",("scan struct: %u search: '%s'@'%s'",
|
|
|
|
struct_no, user_from->user.str, user_from->host.str));
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
LINT_INIT(acl_user);
|
|
|
|
LINT_INIT(acl_db);
|
2004-12-23 11:23:36 +00:00
|
|
|
LINT_INIT(grant_name);
|
2006-03-29 14:27:36 +03:00
|
|
|
LINT_INIT(user);
|
|
|
|
LINT_INIT(host);
|
2004-11-25 21:55:49 +01:00
|
|
|
|
2006-05-06 11:25:59 +04:00
|
|
|
safe_mutex_assert_owner(&acl_cache->lock);
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/* Get the number of elements in the in-memory structure. */
|
2005-01-06 13:00:13 +02:00
|
|
|
switch (struct_no) {
|
2004-11-25 21:55:49 +01:00
|
|
|
case 0:
|
|
|
|
elements= acl_users.elements;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
elements= acl_dbs.elements;
|
|
|
|
break;
|
2004-12-23 10:46:24 +00:00
|
|
|
case 2:
|
2004-11-25 21:55:49 +01:00
|
|
|
elements= column_priv_hash.records;
|
2004-12-23 10:46:24 +00:00
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
elements= proc_priv_hash.records;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -1;
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef EXTRA_DEBUG
|
|
|
|
DBUG_PRINT("loop",("scan struct: %u search user: '%s' host: '%s'",
|
|
|
|
struct_no, user_from->user.str, user_from->host.str));
|
|
|
|
#endif
|
|
|
|
/* Loop over all elements. */
|
|
|
|
for (idx= 0; idx < elements; idx++)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Get a pointer to the element.
|
|
|
|
*/
|
2005-01-06 13:00:13 +02:00
|
|
|
switch (struct_no) {
|
2004-11-25 21:55:49 +01:00
|
|
|
case 0:
|
|
|
|
acl_user= dynamic_element(&acl_users, idx, ACL_USER*);
|
|
|
|
user= acl_user->user;
|
2006-06-06 14:21:07 +03:00
|
|
|
host= acl_user->host.hostname;
|
|
|
|
break;
|
2004-11-25 21:55:49 +01:00
|
|
|
|
|
|
|
case 1:
|
|
|
|
acl_db= dynamic_element(&acl_dbs, idx, ACL_DB*);
|
|
|
|
user= acl_db->user;
|
2006-06-06 14:21:07 +03:00
|
|
|
host= acl_db->host.hostname;
|
2004-11-25 21:55:49 +01:00
|
|
|
break;
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
case 2:
|
|
|
|
grant_name= (GRANT_NAME*) hash_element(&column_priv_hash, idx);
|
|
|
|
user= grant_name->user;
|
2006-06-06 14:21:07 +03:00
|
|
|
host= grant_name->host.hostname;
|
2004-12-23 10:46:24 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
grant_name= (GRANT_NAME*) hash_element(&proc_priv_hash, idx);
|
|
|
|
user= grant_name->user;
|
2006-06-06 14:21:07 +03:00
|
|
|
host= grant_name->host.hostname;
|
2004-12-23 10:46:24 +00:00
|
|
|
break;
|
2007-02-23 13:13:55 +02:00
|
|
|
default:
|
|
|
|
assert(0);
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
|
|
|
if (! user)
|
2005-01-06 13:00:13 +02:00
|
|
|
user= "";
|
2006-06-06 14:21:07 +03:00
|
|
|
if (! host)
|
|
|
|
host= "";
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
#ifdef EXTRA_DEBUG
|
|
|
|
DBUG_PRINT("loop",("scan struct: %u index: %u user: '%s' host: '%s'",
|
|
|
|
struct_no, idx, user, host));
|
|
|
|
#endif
|
|
|
|
if (strcmp(user_from->user.str, user) ||
|
|
|
|
my_strcasecmp(system_charset_info, user_from->host.str, host))
|
2003-06-06 17:43:23 +05:00
|
|
|
continue;
|
2004-11-25 21:55:49 +01:00
|
|
|
|
|
|
|
result= 1; /* At least one element found. */
|
|
|
|
if ( drop )
|
|
|
|
{
|
2006-03-29 14:27:36 +03:00
|
|
|
switch ( struct_no ) {
|
2004-11-25 21:55:49 +01:00
|
|
|
case 0:
|
|
|
|
delete_dynamic_element(&acl_users, idx);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
delete_dynamic_element(&acl_dbs, idx);
|
|
|
|
break;
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
case 2:
|
|
|
|
hash_delete(&column_priv_hash, (byte*) grant_name);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
hash_delete(&proc_priv_hash, (byte*) grant_name);
|
|
|
|
break;
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
|
|
|
elements--;
|
|
|
|
idx--;
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-11-25 21:55:49 +01:00
|
|
|
else if ( user_to )
|
|
|
|
{
|
2005-01-06 13:00:13 +02:00
|
|
|
switch ( struct_no ) {
|
2004-11-25 21:55:49 +01:00
|
|
|
case 0:
|
|
|
|
acl_user->user= strdup_root(&mem, user_to->user.str);
|
|
|
|
acl_user->host.hostname= strdup_root(&mem, user_to->host.str);
|
|
|
|
break;
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
case 1:
|
|
|
|
acl_db->user= strdup_root(&mem, user_to->user.str);
|
|
|
|
acl_db->host.hostname= strdup_root(&mem, user_to->host.str);
|
|
|
|
break;
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
grant_name->user= strdup_root(&mem, user_to->user.str);
|
2005-03-03 17:44:28 -08:00
|
|
|
update_hostname(&grant_name->host,
|
|
|
|
strdup_root(&mem, user_to->host.str));
|
2004-12-23 10:46:24 +00:00
|
|
|
break;
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-25 21:55:49 +01:00
|
|
|
/* If search is requested, we do not need to search further. */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef EXTRA_DEBUG
|
|
|
|
DBUG_PRINT("loop",("scan struct: %u result %d", struct_no, result));
|
|
|
|
#endif
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Handle all privilege tables and in-memory privilege structures.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
handle_grant_data()
|
|
|
|
tables The array with the four open tables.
|
|
|
|
drop If user_from is to be dropped.
|
|
|
|
user_from The the user to be searched/dropped/renamed.
|
|
|
|
user_to The new name for the user if to be renamed,
|
|
|
|
NULL otherwise.
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Go through all grant tables and in-memory grant structures and apply
|
|
|
|
the requested operation.
|
|
|
|
Delete from grant data if drop is true.
|
|
|
|
Update in grant data if drop is false and user_to is not NULL.
|
|
|
|
Search in grant data if drop is false and user_to is NULL.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
> 0 At least one element matched.
|
|
|
|
0 OK, but no element matched.
|
|
|
|
< 0 Error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int handle_grant_data(TABLE_LIST *tables, bool drop,
|
|
|
|
LEX_USER *user_from, LEX_USER *user_to)
|
|
|
|
{
|
|
|
|
int result= 0;
|
|
|
|
int found;
|
|
|
|
DBUG_ENTER("handle_grant_data");
|
|
|
|
|
|
|
|
/* Handle user table. */
|
|
|
|
if ((found= handle_grant_table(tables, 0, drop, user_from, user_to)) < 0)
|
|
|
|
{
|
|
|
|
/* Handle of table failed, don't touch the in-memory array. */
|
|
|
|
result= -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Handle user array. */
|
2004-12-06 19:18:35 +02:00
|
|
|
if ((handle_grant_struct(0, drop, user_from, user_to) && ! result) ||
|
|
|
|
found)
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
|
|
|
result= 1; /* At least one record/element found. */
|
|
|
|
/* If search is requested, we do not need to search further. */
|
|
|
|
if (! drop && ! user_to)
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handle db table. */
|
|
|
|
if ((found= handle_grant_table(tables, 1, drop, user_from, user_to)) < 0)
|
|
|
|
{
|
|
|
|
/* Handle of table failed, don't touch the in-memory array. */
|
|
|
|
result= -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Handle db array. */
|
|
|
|
if (((handle_grant_struct(1, drop, user_from, user_to) && ! result) ||
|
|
|
|
found) && ! result)
|
|
|
|
{
|
|
|
|
result= 1; /* At least one record/element found. */
|
|
|
|
/* If search is requested, we do not need to search further. */
|
|
|
|
if (! drop && ! user_to)
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
/* Handle procedures table. */
|
|
|
|
if ((found= handle_grant_table(tables, 4, drop, user_from, user_to)) < 0)
|
|
|
|
{
|
|
|
|
/* Handle of table failed, don't touch in-memory array. */
|
|
|
|
result= -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Handle procs array. */
|
|
|
|
if (((handle_grant_struct(3, drop, user_from, user_to) && ! result) ||
|
|
|
|
found) && ! result)
|
|
|
|
{
|
|
|
|
result= 1; /* At least one record/element found. */
|
|
|
|
/* If search is requested, we do not need to search further. */
|
|
|
|
if (! drop && ! user_to)
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/* Handle tables table. */
|
|
|
|
if ((found= handle_grant_table(tables, 2, drop, user_from, user_to)) < 0)
|
|
|
|
{
|
|
|
|
/* Handle of table failed, don't touch columns and in-memory array. */
|
|
|
|
result= -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (found && ! result)
|
|
|
|
{
|
|
|
|
result= 1; /* At least one record found. */
|
|
|
|
/* If search is requested, we do not need to search further. */
|
|
|
|
if (! drop && ! user_to)
|
|
|
|
goto end;
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-11-25 21:55:49 +01:00
|
|
|
|
|
|
|
/* Handle columns table. */
|
|
|
|
if ((found= handle_grant_table(tables, 3, drop, user_from, user_to)) < 0)
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-25 21:55:49 +01:00
|
|
|
/* Handle of table failed, don't touch the in-memory array. */
|
2003-06-06 17:43:23 +05:00
|
|
|
result= -1;
|
|
|
|
}
|
2004-11-25 21:55:49 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Handle columns hash. */
|
|
|
|
if (((handle_grant_struct(2, drop, user_from, user_to) && ! result) ||
|
|
|
|
found) && ! result)
|
|
|
|
result= 1; /* At least one record/element found. */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end:
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
2004-12-06 19:18:35 +02:00
|
|
|
|
2004-11-27 23:07:30 +01:00
|
|
|
static void append_user(String *str, LEX_USER *user)
|
|
|
|
{
|
|
|
|
if (str->length())
|
|
|
|
str->append(',');
|
|
|
|
str->append('\'');
|
|
|
|
str->append(user->user.str);
|
2005-11-20 20:47:07 +02:00
|
|
|
str->append(STRING_WITH_LEN("'@'"));
|
2004-11-27 23:07:30 +01:00
|
|
|
str->append(user->host.str);
|
|
|
|
str->append('\'');
|
|
|
|
}
|
2004-11-25 21:55:49 +01:00
|
|
|
|
2004-12-06 19:18:35 +02:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
Create a list of users.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_create_user()
|
|
|
|
thd The current thread.
|
|
|
|
list The users to create.
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
RETURN
|
|
|
|
FALSE OK.
|
|
|
|
TRUE Error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool mysql_create_user(THD *thd, List <LEX_USER> &list)
|
|
|
|
{
|
|
|
|
int result;
|
2004-11-27 23:07:30 +01:00
|
|
|
String wrong_users;
|
2004-11-25 21:55:49 +01:00
|
|
|
ulong sql_mode;
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *user_name, *tmp_user_name;
|
2004-11-25 21:55:49 +01:00
|
|
|
List_iterator <LEX_USER> user_list(list);
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE_LIST tables[GRANT_TABLES];
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_ENTER("mysql_create_user");
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/* CREATE USER may be skipped on replication client. */
|
|
|
|
if ((result= open_grant_tables(thd, tables)))
|
|
|
|
DBUG_RETURN(result != 1);
|
|
|
|
|
|
|
|
rw_wrlock(&LOCK_grant);
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_user_name= user_list++))
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
2006-06-29 15:50:44 +05:00
|
|
|
if (!(user_name= get_current_user(thd, tmp_user_name)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
2006-07-24 16:45:26 -07:00
|
|
|
}
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
Search all in-memory structures and grant tables
|
|
|
|
for a mention of the new user name.
|
|
|
|
*/
|
2005-02-25 16:53:22 +02:00
|
|
|
if (handle_grant_data(tables, 0, user_name, NULL))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-27 23:07:30 +01:00
|
|
|
append_user(&wrong_users, user_name);
|
2004-11-25 21:55:49 +01:00
|
|
|
result= TRUE;
|
2004-11-27 23:07:30 +01:00
|
|
|
continue;
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
sql_mode= thd->variables.sql_mode;
|
2005-03-23 19:18:25 +01:00
|
|
|
if (replace_user_table(thd, tables[0].table, *user_name, 0, 0, 1, 0))
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
2004-11-27 23:07:30 +01:00
|
|
|
append_user(&wrong_users, user_name);
|
2004-11-25 21:55:49 +01:00
|
|
|
result= TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
close_thread_tables(thd);
|
|
|
|
if (result)
|
2005-01-12 03:38:53 +02:00
|
|
|
my_error(ER_CANNOT_USER, MYF(0), "CREATE USER", wrong_users.c_ptr_safe());
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Drop a list of users and all their privileges.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_drop_user()
|
|
|
|
thd The current thread.
|
|
|
|
list The users to drop.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE OK.
|
|
|
|
TRUE Error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool mysql_drop_user(THD *thd, List <LEX_USER> &list)
|
|
|
|
{
|
|
|
|
int result;
|
2004-11-27 23:07:30 +01:00
|
|
|
String wrong_users;
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *user_name, *tmp_user_name;
|
2004-11-25 21:55:49 +01:00
|
|
|
List_iterator <LEX_USER> user_list(list);
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE_LIST tables[GRANT_TABLES];
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_ENTER("mysql_drop_user");
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2004-11-27 23:07:30 +01:00
|
|
|
/* DROP USER may be skipped on replication client. */
|
2004-11-25 21:55:49 +01:00
|
|
|
if ((result= open_grant_tables(thd, tables)))
|
|
|
|
DBUG_RETURN(result != 1);
|
|
|
|
|
|
|
|
rw_wrlock(&LOCK_grant);
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_user_name= user_list++))
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
2006-06-29 15:50:44 +05:00
|
|
|
user_name= get_current_user(thd, tmp_user_name);
|
|
|
|
if (!(user_name= get_current_user(thd, tmp_user_name)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
2005-02-25 16:53:22 +02:00
|
|
|
if (handle_grant_data(tables, 1, user_name, NULL) <= 0)
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-11-27 23:07:30 +01:00
|
|
|
append_user(&wrong_users, user_name);
|
2004-11-25 21:55:49 +01:00
|
|
|
result= TRUE;
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-11-25 21:55:49 +01:00
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
|
2005-12-28 14:43:50 +01:00
|
|
|
/* Rebuild 'acl_check_hosts' since 'acl_users' has been modified */
|
|
|
|
rebuild_check_host();
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
close_thread_tables(thd);
|
|
|
|
if (result)
|
2004-12-31 03:47:56 +02:00
|
|
|
my_error(ER_CANNOT_USER, MYF(0), "DROP USER", wrong_users.c_ptr_safe());
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Rename a user.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_rename_user()
|
|
|
|
thd The current thread.
|
|
|
|
list The user name pairs: (from, to).
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE OK.
|
|
|
|
TRUE Error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool mysql_rename_user(THD *thd, List <LEX_USER> &list)
|
|
|
|
{
|
2005-12-28 14:43:50 +01:00
|
|
|
int result;
|
2004-11-27 23:07:30 +01:00
|
|
|
String wrong_users;
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *user_from, *tmp_user_from;
|
|
|
|
LEX_USER *user_to, *tmp_user_to;
|
2004-11-25 21:55:49 +01:00
|
|
|
List_iterator <LEX_USER> user_list(list);
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE_LIST tables[GRANT_TABLES];
|
2004-11-25 21:55:49 +01:00
|
|
|
DBUG_ENTER("mysql_rename_user");
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2004-11-27 23:07:30 +01:00
|
|
|
/* RENAME USER may be skipped on replication client. */
|
2004-11-25 21:55:49 +01:00
|
|
|
if ((result= open_grant_tables(thd, tables)))
|
|
|
|
DBUG_RETURN(result != 1);
|
|
|
|
|
|
|
|
rw_wrlock(&LOCK_grant);
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_user_from= user_list++))
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
2006-06-29 15:50:44 +05:00
|
|
|
if (!(user_from= get_current_user(thd, tmp_user_from)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
tmp_user_to= user_list++;
|
|
|
|
if (!(user_to= get_current_user(thd, tmp_user_to)))
|
|
|
|
{
|
|
|
|
result= TRUE;
|
|
|
|
continue;
|
|
|
|
}
|
2005-02-25 16:53:22 +02:00
|
|
|
DBUG_ASSERT(user_to != 0); /* Syntax enforces pairs of users. */
|
2004-11-25 21:55:49 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
Search all in-memory structures and grant tables
|
|
|
|
for a mention of the new user name.
|
|
|
|
*/
|
2004-11-27 23:07:30 +01:00
|
|
|
if (handle_grant_data(tables, 0, user_to, NULL) ||
|
|
|
|
handle_grant_data(tables, 0, user_from, user_to) <= 0)
|
2004-11-25 21:55:49 +01:00
|
|
|
{
|
2004-11-27 23:07:30 +01:00
|
|
|
append_user(&wrong_users, user_from);
|
2004-11-25 21:55:49 +01:00
|
|
|
result= TRUE;
|
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2006-08-16 19:19:49 -04:00
|
|
|
|
2005-12-28 14:43:50 +01:00
|
|
|
/* Rebuild 'acl_check_hosts' since 'acl_users' has been modified */
|
|
|
|
rebuild_check_host();
|
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
close_thread_tables(thd);
|
|
|
|
if (result)
|
2005-01-12 03:38:53 +02:00
|
|
|
my_error(ER_CANNOT_USER, MYF(0), "RENAME USER", wrong_users.c_ptr_safe());
|
2003-06-06 17:43:23 +05:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
2004-12-06 02:00:37 +02:00
|
|
|
|
2004-11-25 21:55:49 +01:00
|
|
|
/*
|
|
|
|
Revoke all privileges from a list of users.
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
mysql_revoke_all()
|
|
|
|
thd The current thread.
|
|
|
|
list The users to revoke all privileges from.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
> 0 Error. Error message already sent.
|
|
|
|
0 OK.
|
|
|
|
< 0 Error. Error message not yet sent.
|
|
|
|
*/
|
|
|
|
|
2004-10-20 04:04:37 +03:00
|
|
|
bool mysql_revoke_all(THD *thd, List <LEX_USER> &list)
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
uint counter, revoked, is_proc;
|
2003-06-06 17:43:23 +05:00
|
|
|
int result;
|
2003-11-20 22:06:25 +02:00
|
|
|
ACL_DB *acl_db;
|
2004-12-23 10:46:24 +00:00
|
|
|
TABLE_LIST tables[GRANT_TABLES];
|
2003-06-06 17:43:23 +05:00
|
|
|
DBUG_ENTER("mysql_revoke_all");
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
if ((result= open_grant_tables(thd, tables)))
|
2004-10-20 04:04:37 +03:00
|
|
|
DBUG_RETURN(result != 1);
|
2003-06-06 17:43:23 +05:00
|
|
|
|
|
|
|
rw_wrlock(&LOCK_grant);
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
2006-06-29 15:50:44 +05:00
|
|
|
LEX_USER *lex_user, *tmp_lex_user;
|
2003-06-06 17:43:23 +05:00
|
|
|
List_iterator <LEX_USER> user_list(list);
|
2006-06-29 15:50:44 +05:00
|
|
|
while ((tmp_lex_user= user_list++))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2006-06-29 15:50:44 +05:00
|
|
|
if (!(lex_user= get_current_user(thd, tmp_lex_user)))
|
|
|
|
{
|
|
|
|
result= -1;
|
|
|
|
continue;
|
|
|
|
}
|
2006-05-06 11:25:59 +04:00
|
|
|
if (!find_acl_user(lex_user->host.str, lex_user->user.str, TRUE))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
|
|
|
result= -1;
|
|
|
|
continue;
|
|
|
|
}
|
2003-08-11 22:44:43 +03:00
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
if (replace_user_table(thd, tables[0].table,
|
2005-06-01 21:56:30 -07:00
|
|
|
*lex_user, ~(ulong)0, 1, 0, 0))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
|
|
|
result= -1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove db access privileges */
|
2004-10-18 20:25:47 -05:00
|
|
|
/*
|
|
|
|
Because acl_dbs and column_priv_hash shrink and may re-order
|
|
|
|
as privileges are removed, removal occurs in a repeated loop
|
|
|
|
until no more privileges are revoked.
|
|
|
|
*/
|
2004-10-22 13:41:40 -05:00
|
|
|
do
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-10-22 13:41:40 -05:00
|
|
|
for (counter= 0, revoked= 0 ; counter < acl_dbs.elements ; )
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-10-18 20:25:47 -05:00
|
|
|
const char *user,*host;
|
2004-11-27 23:07:30 +01:00
|
|
|
|
2004-10-18 20:25:47 -05:00
|
|
|
acl_db=dynamic_element(&acl_dbs,counter,ACL_DB*);
|
|
|
|
if (!(user=acl_db->user))
|
|
|
|
user= "";
|
|
|
|
if (!(host=acl_db->host.hostname))
|
|
|
|
host= "";
|
2004-11-27 23:07:30 +01:00
|
|
|
|
2004-10-18 20:25:47 -05:00
|
|
|
if (!strcmp(lex_user->user.str,user) &&
|
|
|
|
!my_strcasecmp(system_charset_info, lex_user->host.str, host))
|
|
|
|
{
|
This changeset is largely a handler cleanup changeset (WL#3281), but includes fixes and cleanups that was found necessary while testing the handler changes
Changes that requires code changes in other code of other storage engines.
(Note that all changes are very straightforward and one should find all issues
by compiling a --debug build and fixing all compiler errors and all
asserts in field.cc while running the test suite),
- New optional handler function introduced: reset()
This is called after every DML statement to make it easy for a handler to
statement specific cleanups.
(The only case it's not called is if force the file to be closed)
- handler::extra(HA_EXTRA_RESET) is removed. Code that was there before
should be moved to handler::reset()
- table->read_set contains a bitmap over all columns that are needed
in the query. read_row() and similar functions only needs to read these
columns
- table->write_set contains a bitmap over all columns that will be updated
in the query. write_row() and update_row() only needs to update these
columns.
The above bitmaps should now be up to date in all context
(including ALTER TABLE, filesort()).
The handler is informed of any changes to the bitmap after
fix_fields() by calling the virtual function
handler::column_bitmaps_signal(). If the handler does caching of
these bitmaps (instead of using table->read_set, table->write_set),
it should redo the caching in this code. as the signal() may be sent
several times, it's probably best to set a variable in the signal
and redo the caching on read_row() / write_row() if the variable was
set.
- Removed the read_set and write_set bitmap objects from the handler class
- Removed all column bit handling functions from the handler class.
(Now one instead uses the normal bitmap functions in my_bitmap.c instead
of handler dedicated bitmap functions)
- field->query_id is removed. One should instead instead check
table->read_set and table->write_set if a field is used in the query.
- handler::extra(HA_EXTRA_RETRIVE_ALL_COLS) and
handler::extra(HA_EXTRA_RETRIEVE_PRIMARY_KEY) are removed. One should now
instead use table->read_set to check for which columns to retrieve.
- If a handler needs to call Field->val() or Field->store() on columns
that are not used in the query, one should install a temporary
all-columns-used map while doing so. For this, we provide the following
functions:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->read_set);
field->val();
dbug_tmp_restore_column_map(table->read_set, old_map);
and similar for the write map:
my_bitmap_map *old_map= dbug_tmp_use_all_columns(table, table->write_set);
field->val();
dbug_tmp_restore_column_map(table->write_set, old_map);
If this is not done, you will sooner or later hit a DBUG_ASSERT
in the field store() / val() functions.
(For not DBUG binaries, the dbug_tmp_restore_column_map() and
dbug_tmp_restore_column_map() are inline dummy functions and should
be optimized away be the compiler).
- If one needs to temporary set the column map for all binaries (and not
just to avoid the DBUG_ASSERT() in the Field::store() / Field::val()
methods) one should use the functions tmp_use_all_columns() and
tmp_restore_column_map() instead of the above dbug_ variants.
- All 'status' fields in the handler base class (like records,
data_file_length etc) are now stored in a 'stats' struct. This makes
it easier to know what status variables are provided by the base
handler. This requires some trivial variable names in the extra()
function.
- New virtual function handler::records(). This is called to optimize
COUNT(*) if (handler::table_flags() & HA_HAS_RECORDS()) is true.
(stats.records is not supposed to be an exact value. It's only has to
be 'reasonable enough' for the optimizer to be able to choose a good
optimization path).
- Non virtual handler::init() function added for caching of virtual
constants from engine.
- Removed has_transactions() virtual method. Now one should instead return
HA_NO_TRANSACTIONS in table_flags() if the table handler DOES NOT support
transactions.
- The 'xxxx_create_handler()' function now has a MEM_ROOT_root argument
that is to be used with 'new handler_name()' to allocate the handler
in the right area. The xxxx_create_handler() function is also
responsible for any initialization of the object before returning.
For example, one should change:
static handler *myisam_create_handler(TABLE_SHARE *table)
{
return new ha_myisam(table);
}
->
static handler *myisam_create_handler(TABLE_SHARE *table, MEM_ROOT *mem_root)
{
return new (mem_root) ha_myisam(table);
}
- New optional virtual function: use_hidden_primary_key().
This is called in case of an update/delete when
(table_flags() and HA_PRIMARY_KEY_REQUIRED_FOR_DELETE) is defined
but we don't have a primary key. This allows the handler to take precisions
in remembering any hidden primary key to able to update/delete any
found row. The default handler marks all columns to be read.
- handler::table_flags() now returns a ulonglong (to allow for more flags).
- New/changed table_flags()
- HA_HAS_RECORDS Set if ::records() is supported
- HA_NO_TRANSACTIONS Set if engine doesn't support transactions
- HA_PRIMARY_KEY_REQUIRED_FOR_DELETE
Set if we should mark all primary key columns for
read when reading rows as part of a DELETE
statement. If there is no primary key,
all columns are marked for read.
- HA_PARTIAL_COLUMN_READ Set if engine will not read all columns in some
cases (based on table->read_set)
- HA_PRIMARY_KEY_ALLOW_RANDOM_ACCESS
Renamed to HA_PRIMARY_KEY_REQUIRED_FOR_POSITION.
- HA_DUPP_POS Renamed to HA_DUPLICATE_POS
- HA_REQUIRES_KEY_COLUMNS_FOR_DELETE
Set this if we should mark ALL key columns for
read when when reading rows as part of a DELETE
statement. In case of an update we will mark
all keys for read for which key part changed
value.
- HA_STATS_RECORDS_IS_EXACT
Set this if stats.records is exact.
(This saves us some extra records() calls
when optimizing COUNT(*))
- Removed table_flags()
- HA_NOT_EXACT_COUNT Now one should instead use HA_HAS_RECORDS if
handler::records() gives an exact count() and
HA_STATS_RECORDS_IS_EXACT if stats.records is exact.
- HA_READ_RND_SAME Removed (no one supported this one)
- Removed not needed functions ha_retrieve_all_cols() and ha_retrieve_all_pk()
- Renamed handler::dupp_pos to handler::dup_pos
- Removed not used variable handler::sortkey
Upper level handler changes:
- ha_reset() now does some overall checks and calls ::reset()
- ha_table_flags() added. This is a cached version of table_flags(). The
cache is updated on engine creation time and updated on open.
MySQL level changes (not obvious from the above):
- DBUG_ASSERT() added to check that column usage matches what is set
in the column usage bit maps. (This found a LOT of bugs in current
column marking code).
- In 5.1 before, all used columns was marked in read_set and only updated
columns was marked in write_set. Now we only mark columns for which we
need a value in read_set.
- Column bitmaps are created in open_binary_frm() and open_table_from_share().
(Before this was in table.cc)
- handler::table_flags() calls are replaced with handler::ha_table_flags()
- For calling field->val() you must have the corresponding bit set in
table->read_set. For calling field->store() you must have the
corresponding bit set in table->write_set. (There are asserts in
all store()/val() functions to catch wrong usage)
- thd->set_query_id is renamed to thd->mark_used_columns and instead
of setting this to an integer value, this has now the values:
MARK_COLUMNS_NONE, MARK_COLUMNS_READ, MARK_COLUMNS_WRITE
Changed also all variables named 'set_query_id' to mark_used_columns.
- In filesort() we now inform the handler of exactly which columns are needed
doing the sort and choosing the rows.
- The TABLE_SHARE object has a 'all_set' column bitmap one can use
when one needs a column bitmap with all columns set.
(This is used for table->use_all_columns() and other places)
- The TABLE object has 3 column bitmaps:
- def_read_set Default bitmap for columns to be read
- def_write_set Default bitmap for columns to be written
- tmp_set Can be used as a temporary bitmap when needed.
The table object has also two pointer to bitmaps read_set and write_set
that the handler should use to find out which columns are used in which way.
- count() optimization now calls handler::records() instead of using
handler->stats.records (if (table_flags() & HA_HAS_RECORDS) is true).
- Added extra argument to Item::walk() to indicate if we should also
traverse sub queries.
- Added TABLE parameter to cp_buffer_from_ref()
- Don't close tables created with CREATE ... SELECT but keep them in
the table cache. (Faster usage of newly created tables).
New interfaces:
- table->clear_column_bitmaps() to initialize the bitmaps for tables
at start of new statements.
- table->column_bitmaps_set() to set up new column bitmaps and signal
the handler about this.
- table->column_bitmaps_set_no_signal() for some few cases where we need
to setup new column bitmaps but don't signal the handler (as the handler
has already been signaled about these before). Used for the momement
only in opt_range.cc when doing ROR scans.
- table->use_all_columns() to install a bitmap where all columns are marked
as use in the read and the write set.
- table->default_column_bitmaps() to install the normal read and write
column bitmaps, but not signaling the handler about this.
This is mainly used when creating TABLE instances.
- table->mark_columns_needed_for_delete(),
table->mark_columns_needed_for_delete() and
table->mark_columns_needed_for_insert() to allow us to put additional
columns in column usage maps if handler so requires.
(The handler indicates what it neads in handler->table_flags())
- table->prepare_for_position() to allow us to tell handler that it
needs to read primary key parts to be able to store them in
future table->position() calls.
(This replaces the table->file->ha_retrieve_all_pk function)
- table->mark_auto_increment_column() to tell handler are going to update
columns part of any auto_increment key.
- table->mark_columns_used_by_index() to mark all columns that is part of
an index. It will also send extra(HA_EXTRA_KEYREAD) to handler to allow
it to quickly know that it only needs to read colums that are part
of the key. (The handler can also use the column map for detecting this,
but simpler/faster handler can just monitor the extra() call).
- table->mark_columns_used_by_index_no_reset() to in addition to other columns,
also mark all columns that is used by the given key.
- table->restore_column_maps_after_mark_index() to restore to default
column maps after a call to table->mark_columns_used_by_index().
- New item function register_field_in_read_map(), for marking used columns
in table->read_map. Used by filesort() to mark all used columns
- Maintain in TABLE->merge_keys set of all keys that are used in query.
(Simplices some optimization loops)
- Maintain Field->part_of_key_not_clustered which is like Field->part_of_key
but the field in the clustered key is not assumed to be part of all index.
(used in opt_range.cc for faster loops)
- dbug_tmp_use_all_columns(), dbug_tmp_restore_column_map()
tmp_use_all_columns() and tmp_restore_column_map() functions to temporally
mark all columns as usable. The 'dbug_' version is primarily intended
inside a handler when it wants to just call Field:store() & Field::val()
functions, but don't need the column maps set for any other usage.
(ie:: bitmap_is_set() is never called)
- We can't use compare_records() to skip updates for handlers that returns
a partial column set and the read_set doesn't cover all columns in the
write set. The reason for this is that if we have a column marked only for
write we can't in the MySQL level know if the value changed or not.
The reason this worked before was that MySQL marked all to be written
columns as also to be read. The new 'optimal' bitmaps exposed this 'hidden
bug'.
- open_table_from_share() does not anymore setup temporary MEM_ROOT
object as a thread specific variable for the handler. Instead we
send the to-be-used MEMROOT to get_new_handler().
(Simpler, faster code)
Bugs fixed:
- Column marking was not done correctly in a lot of cases.
(ALTER TABLE, when using triggers, auto_increment fields etc)
(Could potentially result in wrong values inserted in table handlers
relying on that the old column maps or field->set_query_id was correct)
Especially when it comes to triggers, there may be cases where the
old code would cause lost/wrong values for NDB and/or InnoDB tables.
- Split thd->options flag OPTION_STATUS_NO_TRANS_UPDATE to two flags:
OPTION_STATUS_NO_TRANS_UPDATE and OPTION_KEEP_LOG.
This allowed me to remove some wrong warnings about:
"Some non-transactional changed tables couldn't be rolled back"
- Fixed handling of INSERT .. SELECT and CREATE ... SELECT that wrongly reset
(thd->options & OPTION_STATUS_NO_TRANS_UPDATE) which caused us to loose
some warnings about
"Some non-transactional changed tables couldn't be rolled back")
- Fixed use of uninitialized memory in ha_ndbcluster.cc::delete_table()
which could cause delete_table to report random failures.
- Fixed core dumps for some tests when running with --debug
- Added missing FN_LIBCHAR in mysql_rm_tmp_tables()
(This has probably caused us to not properly remove temporary files after
crash)
- slow_logs was not properly initialized, which could maybe cause
extra/lost entries in slow log.
- If we get an duplicate row on insert, change column map to read and
write all columns while retrying the operation. This is required by
the definition of REPLACE and also ensures that fields that are only
part of UPDATE are properly handled. This fixed a bug in NDB and
REPLACE where REPLACE wrongly copied some column values from the replaced
row.
- For table handler that doesn't support NULL in keys, we would give an error
when creating a primary key with NULL fields, even after the fields has been
automaticly converted to NOT NULL.
- Creating a primary key on a SPATIAL key, would fail if field was not
declared as NOT NULL.
Cleanups:
- Removed not used condition argument to setup_tables
- Removed not needed item function reset_query_id_processor().
- Field->add_index is removed. Now this is instead maintained in
(field->flags & FIELD_IN_ADD_INDEX)
- Field->fieldnr is removed (use field->field_index instead)
- New argument to filesort() to indicate that it should return a set of
row pointers (not used columns). This allowed me to remove some references
to sql_command in filesort and should also enable us to return column
results in some cases where we couldn't before.
- Changed column bitmap handling in opt_range.cc to be aligned with TABLE
bitmap, which allowed me to use bitmap functions instead of looping over
all fields to create some needed bitmaps. (Faster and smaller code)
- Broke up found too long lines
- Moved some variable declaration at start of function for better code
readability.
- Removed some not used arguments from functions.
(setup_fields(), mysql_prepare_insert_check_table())
- setup_fields() now takes an enum instead of an int for marking columns
usage.
- For internal temporary tables, use handler::write_row(),
handler::delete_row() and handler::update_row() instead of
handler::ha_xxxx() for faster execution.
- Changed some constants to enum's and define's.
- Using separate column read and write sets allows for easier checking
of timestamp field was set by statement.
- Remove calls to free_io_cache() as this is now done automaticly in ha_reset()
- Don't build table->normalized_path as this is now identical to table->path
(after bar's fixes to convert filenames)
- Fixed some missed DBUG_PRINT(.."%lx") to use "0x%lx" to make it easier to
do comparision with the 'convert-dbug-for-diff' tool.
Things left to do in 5.1:
- We wrongly log failed CREATE TABLE ... SELECT in some cases when using
row based logging (as shown by testcase binlog_row_mix_innodb_myisam.result)
Mats has promised to look into this.
- Test that my fix for CREATE TABLE ... SELECT is indeed correct.
(I added several test cases for this, but in this case it's better that
someone else also tests this throughly).
Lars has promosed to do this.
2006-06-04 18:52:22 +03:00
|
|
|
if (!replace_db_table(tables[1].table, acl_db->db, *lex_user,
|
|
|
|
~(ulong)0, 1))
|
2004-10-18 20:25:47 -05:00
|
|
|
{
|
2004-10-22 13:41:40 -05:00
|
|
|
/*
|
|
|
|
Don't increment counter as replace_db_table deleted the
|
|
|
|
current element in acl_dbs.
|
|
|
|
*/
|
|
|
|
revoked= 1;
|
2004-10-18 20:25:47 -05:00
|
|
|
continue;
|
|
|
|
}
|
2004-10-22 13:41:40 -05:00
|
|
|
result= -1; // Something went wrong
|
2004-10-18 20:25:47 -05:00
|
|
|
}
|
2004-10-22 13:41:40 -05:00
|
|
|
counter++;
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-10-22 13:41:40 -05:00
|
|
|
} while (revoked);
|
2003-06-06 17:43:23 +05:00
|
|
|
|
|
|
|
/* Remove column access */
|
2004-10-22 13:41:40 -05:00
|
|
|
do
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-10-22 13:41:40 -05:00
|
|
|
for (counter= 0, revoked= 0 ; counter < column_priv_hash.records ; )
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-10-18 20:25:47 -05:00
|
|
|
const char *user,*host;
|
|
|
|
GRANT_TABLE *grant_table= (GRANT_TABLE*)hash_element(&column_priv_hash,
|
|
|
|
counter);
|
|
|
|
if (!(user=grant_table->user))
|
|
|
|
user= "";
|
2005-03-02 16:30:24 -08:00
|
|
|
if (!(host=grant_table->host.hostname))
|
2004-10-18 20:25:47 -05:00
|
|
|
host= "";
|
2004-11-27 23:07:30 +01:00
|
|
|
|
2004-10-18 20:25:47 -05:00
|
|
|
if (!strcmp(lex_user->user.str,user) &&
|
|
|
|
!my_strcasecmp(system_charset_info, lex_user->host.str, host))
|
2003-06-06 17:43:23 +05:00
|
|
|
{
|
2004-10-18 20:25:47 -05:00
|
|
|
if (replace_table_table(thd,grant_table,tables[2].table,*lex_user,
|
|
|
|
grant_table->db,
|
|
|
|
grant_table->tname,
|
2005-06-01 17:34:10 -07:00
|
|
|
~(ulong)0, 0, 1))
|
2004-10-22 13:41:40 -05:00
|
|
|
{
|
2004-10-18 20:25:47 -05:00
|
|
|
result= -1;
|
2004-10-22 13:41:40 -05:00
|
|
|
}
|
2004-10-18 20:25:47 -05:00
|
|
|
else
|
2004-10-22 14:32:23 -05:00
|
|
|
{
|
2004-10-22 13:41:40 -05:00
|
|
|
if (!grant_table->cols)
|
2004-10-18 20:25:47 -05:00
|
|
|
{
|
2004-10-22 13:41:40 -05:00
|
|
|
revoked= 1;
|
|
|
|
continue;
|
2004-10-18 20:25:47 -05:00
|
|
|
}
|
2004-10-22 13:41:40 -05:00
|
|
|
List<LEX_COLUMN> columns;
|
|
|
|
if (!replace_column_table(grant_table,tables[3].table, *lex_user,
|
2004-10-22 14:32:23 -05:00
|
|
|
columns,
|
|
|
|
grant_table->db,
|
|
|
|
grant_table->tname,
|
2005-06-01 17:34:10 -07:00
|
|
|
~(ulong)0, 1))
|
2004-10-18 20:25:47 -05:00
|
|
|
{
|
2004-10-22 13:41:40 -05:00
|
|
|
revoked= 1;
|
2004-10-01 14:48:48 -05:00
|
|
|
continue;
|
2004-10-18 20:25:47 -05:00
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
result= -1;
|
2004-10-01 07:50:26 -05:00
|
|
|
}
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-10-22 13:41:40 -05:00
|
|
|
counter++;
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-10-22 13:41:40 -05:00
|
|
|
} while (revoked);
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
/* Remove procedure access */
|
2005-05-17 19:54:20 +01:00
|
|
|
for (is_proc=0; is_proc<2; is_proc++) do {
|
|
|
|
HASH *hash= is_proc ? &proc_priv_hash : &func_priv_hash;
|
|
|
|
for (counter= 0, revoked= 0 ; counter < hash->records ; )
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
const char *user,*host;
|
2005-05-17 19:54:20 +01:00
|
|
|
GRANT_NAME *grant_proc= (GRANT_NAME*) hash_element(hash, counter);
|
2004-12-23 10:46:24 +00:00
|
|
|
if (!(user=grant_proc->user))
|
|
|
|
user= "";
|
2005-03-03 17:44:28 -08:00
|
|
|
if (!(host=grant_proc->host.hostname))
|
2004-12-23 10:46:24 +00:00
|
|
|
host= "";
|
|
|
|
|
|
|
|
if (!strcmp(lex_user->user.str,user) &&
|
|
|
|
!my_strcasecmp(system_charset_info, lex_user->host.str, host))
|
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
if (!replace_routine_table(thd,grant_proc,tables[4].table,*lex_user,
|
2004-12-23 10:46:24 +00:00
|
|
|
grant_proc->db,
|
|
|
|
grant_proc->tname,
|
2005-05-17 19:54:20 +01:00
|
|
|
is_proc,
|
2005-06-07 18:23:58 -07:00
|
|
|
~(ulong)0, 1))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
revoked= 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
result= -1; // Something went wrong
|
|
|
|
}
|
|
|
|
counter++;
|
|
|
|
}
|
|
|
|
} while (revoked);
|
2003-06-06 17:43:23 +05:00
|
|
|
}
|
2004-11-27 23:07:30 +01:00
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2006-10-03 13:38:25 -04:00
|
|
|
|
|
|
|
if (mysql_bin_log.is_open())
|
|
|
|
{
|
2006-10-12 17:51:56 -04:00
|
|
|
thd->binlog_query(THD::MYSQL_QUERY_TYPE,
|
|
|
|
thd->query, thd->query_length, FALSE, FALSE);
|
2006-10-03 13:38:25 -04:00
|
|
|
}
|
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
close_thread_tables(thd);
|
2004-11-27 23:07:30 +01:00
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
if (result)
|
2004-11-12 14:34:00 +02:00
|
|
|
my_message(ER_REVOKE_GRANTS, ER(ER_REVOKE_GRANTS), MYF(0));
|
2004-11-27 23:07:30 +01:00
|
|
|
|
2003-06-06 17:43:23 +05:00
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
2002-02-01 21:45:23 +02:00
|
|
|
|
2002-05-15 13:50:38 +03:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
/*
|
|
|
|
Revoke privileges for all users on a stored procedure
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
sp_revoke_privileges()
|
|
|
|
thd The current thread.
|
|
|
|
db DB of the stored procedure
|
|
|
|
name Name of the stored procedure
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 OK.
|
|
|
|
< 0 Error. Error message not yet sent.
|
|
|
|
*/
|
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
bool sp_revoke_privileges(THD *thd, const char *sp_db, const char *sp_name,
|
|
|
|
bool is_proc)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
uint counter, revoked;
|
|
|
|
int result;
|
|
|
|
TABLE_LIST tables[GRANT_TABLES];
|
2005-05-17 19:54:20 +01:00
|
|
|
HASH *hash= is_proc ? &proc_priv_hash : &func_priv_hash;
|
2004-12-23 10:46:24 +00:00
|
|
|
DBUG_ENTER("sp_revoke_privileges");
|
|
|
|
|
|
|
|
if ((result= open_grant_tables(thd, tables)))
|
|
|
|
DBUG_RETURN(result != 1);
|
|
|
|
|
|
|
|
rw_wrlock(&LOCK_grant);
|
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
|
|
|
|
BUG#25091 (A DELETE statement to mysql database is not logged in ROW format):
With this patch, statements that change metadata (in the mysql database)
is logged as statements, while normal changes (e.g., using INSERT, DELETE,
and/or UPDATE) is logged according to the format in effect.
The log tables (i.e., general_log and slow_log) are not replicated at all.
With this patch, the following statements are replicated as statements:
GRANT, REVOKE (ALL), CREATE USER, DROP USER, and RENAME USER.
2007-02-26 10:19:08 +01:00
|
|
|
/*
|
|
|
|
This statement will be replicated as a statement, even when using
|
|
|
|
row-based replication. The flag will be reset at the end of the
|
|
|
|
statement.
|
|
|
|
*/
|
|
|
|
thd->clear_current_stmt_binlog_row_based();
|
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
/* Remove procedure access */
|
2005-02-25 16:53:22 +02:00
|
|
|
do
|
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
for (counter= 0, revoked= 0 ; counter < hash->records ; )
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
2005-05-17 19:54:20 +01:00
|
|
|
GRANT_NAME *grant_proc= (GRANT_NAME*) hash_element(hash, counter);
|
2004-12-23 10:46:24 +00:00
|
|
|
if (!my_strcasecmp(system_charset_info, grant_proc->db, sp_db) &&
|
|
|
|
!my_strcasecmp(system_charset_info, grant_proc->tname, sp_name))
|
|
|
|
{
|
|
|
|
LEX_USER lex_user;
|
|
|
|
lex_user.user.str= grant_proc->user;
|
|
|
|
lex_user.user.length= strlen(grant_proc->user);
|
2006-06-06 14:21:07 +03:00
|
|
|
lex_user.host.str= grant_proc->host.hostname ?
|
|
|
|
grant_proc->host.hostname : (char*)"";
|
|
|
|
lex_user.host.length= grant_proc->host.hostname ?
|
|
|
|
strlen(grant_proc->host.hostname) : 0;
|
2005-05-17 19:54:20 +01:00
|
|
|
if (!replace_routine_table(thd,grant_proc,tables[4].table,lex_user,
|
|
|
|
grant_proc->db, grant_proc->tname,
|
2005-06-07 18:23:58 -07:00
|
|
|
is_proc, ~(ulong)0, 1))
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
|
|
|
revoked= 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
result= -1; // Something went wrong
|
|
|
|
}
|
|
|
|
counter++;
|
|
|
|
}
|
|
|
|
} while (revoked);
|
|
|
|
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
close_thread_tables(thd);
|
|
|
|
|
|
|
|
if (result)
|
|
|
|
my_message(ER_REVOKE_GRANTS, ER(ER_REVOKE_GRANTS), MYF(0));
|
|
|
|
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Grant EXECUTE,ALTER privilege for a stored procedure
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
sp_grant_privileges()
|
|
|
|
thd The current thread.
|
|
|
|
db DB of the stored procedure
|
|
|
|
name Name of the stored procedure
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
0 OK.
|
|
|
|
< 0 Error. Error message not yet sent.
|
|
|
|
*/
|
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
bool sp_grant_privileges(THD *thd, const char *sp_db, const char *sp_name,
|
|
|
|
bool is_proc)
|
2004-12-23 10:46:24 +00:00
|
|
|
{
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2004-12-23 10:46:24 +00:00
|
|
|
LEX_USER *combo;
|
|
|
|
TABLE_LIST tables[1];
|
|
|
|
List<LEX_USER> user_list;
|
|
|
|
bool result;
|
2006-06-28 12:40:17 +02:00
|
|
|
ACL_USER *au;
|
|
|
|
char passwd_buff[SCRAMBLED_PASSWORD_CHAR_LENGTH+1];
|
2005-09-15 22:29:07 +03:00
|
|
|
DBUG_ENTER("sp_grant_privileges");
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (!(combo=(LEX_USER*) thd->alloc(sizeof(st_lex_user))))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
combo->user.str= sctx->user;
|
2006-06-28 12:40:17 +02:00
|
|
|
|
2006-05-06 11:25:59 +04:00
|
|
|
VOID(pthread_mutex_lock(&acl_cache->lock));
|
2006-06-28 12:40:17 +02:00
|
|
|
|
|
|
|
if ((au= find_acl_user(combo->host.str=(char*)sctx->host_or_ip,combo->user.str,FALSE)))
|
|
|
|
goto found_acl;
|
|
|
|
if ((au= find_acl_user(combo->host.str=(char*)sctx->host, combo->user.str,FALSE)))
|
|
|
|
goto found_acl;
|
|
|
|
if ((au= find_acl_user(combo->host.str=(char*)sctx->ip, combo->user.str,FALSE)))
|
|
|
|
goto found_acl;
|
|
|
|
if((au= find_acl_user(combo->host.str=(char*)"%", combo->user.str, FALSE)))
|
|
|
|
goto found_acl;
|
|
|
|
|
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
|
|
|
found_acl:
|
2006-05-06 11:25:59 +04:00
|
|
|
VOID(pthread_mutex_unlock(&acl_cache->lock));
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
bzero((char*)tables, sizeof(TABLE_LIST));
|
|
|
|
user_list.empty();
|
|
|
|
|
|
|
|
tables->db= (char*)sp_db;
|
2005-01-06 13:00:13 +02:00
|
|
|
tables->table_name= tables->alias= (char*)sp_name;
|
2006-06-28 12:40:17 +02:00
|
|
|
|
2004-12-23 10:46:24 +00:00
|
|
|
combo->host.length= strlen(combo->host.str);
|
|
|
|
combo->user.length= strlen(combo->user.str);
|
|
|
|
combo->host.str= thd->strmake(combo->host.str,combo->host.length);
|
|
|
|
combo->user.str= thd->strmake(combo->user.str,combo->user.length);
|
2006-06-28 12:40:17 +02:00
|
|
|
|
|
|
|
|
|
|
|
if(au && au->salt_len)
|
|
|
|
{
|
|
|
|
if (au->salt_len == SCRAMBLE_LENGTH)
|
|
|
|
{
|
|
|
|
make_password_from_salt(passwd_buff, au->salt);
|
|
|
|
combo->password.length= SCRAMBLED_PASSWORD_CHAR_LENGTH;
|
|
|
|
}
|
|
|
|
else if (au->salt_len == SCRAMBLE_LENGTH_323)
|
|
|
|
{
|
|
|
|
make_password_from_salt_323(passwd_buff, (ulong *) au->salt);
|
|
|
|
combo->password.length= SCRAMBLED_PASSWORD_CHAR_LENGTH_323;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
my_error(ER_PASSWD_LENGTH, MYF(0), SCRAMBLED_PASSWORD_CHAR_LENGTH);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
combo->password.str= passwd_buff;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
combo->password.str= (char*)"";
|
|
|
|
combo->password.length= 0;
|
|
|
|
}
|
2004-12-23 10:46:24 +00:00
|
|
|
|
|
|
|
if (user_list.push_back(combo))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
|
|
|
thd->lex->ssl_type= SSL_TYPE_NOT_SPECIFIED;
|
2005-01-10 16:24:05 +00:00
|
|
|
bzero((char*) &thd->lex->mqh, sizeof(thd->lex->mqh));
|
2004-12-23 10:46:24 +00:00
|
|
|
|
2005-05-17 19:54:20 +01:00
|
|
|
result= mysql_routine_grant(thd, tables, is_proc, user_list,
|
2004-12-23 10:46:24 +00:00
|
|
|
DEFAULT_CREATE_PROC_ACLS, 0, 1);
|
|
|
|
DBUG_RETURN(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
/*****************************************************************************
|
2002-06-12 15:04:18 +03:00
|
|
|
Instantiate used templates
|
2000-07-31 21:29:14 +02:00
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-06-22 14:08:28 +05:00
|
|
|
#ifdef HAVE_EXPLICIT_TEMPLATE_INSTANTIATION
|
2000-07-31 21:29:14 +02:00
|
|
|
template class List_iterator<LEX_COLUMN>;
|
|
|
|
template class List_iterator<LEX_USER>;
|
|
|
|
template class List<LEX_COLUMN>;
|
|
|
|
template class List<LEX_USER>;
|
|
|
|
#endif
|
2003-09-26 15:33:13 +05:00
|
|
|
|
|
|
|
#endif /*NO_EMBEDDED_ACCESS_CHECKS */
|
|
|
|
|
|
|
|
|
|
|
|
int wild_case_compare(CHARSET_INFO *cs, const char *str,const char *wildstr)
|
|
|
|
{
|
|
|
|
reg3 int flag;
|
|
|
|
DBUG_ENTER("wild_case_compare");
|
|
|
|
DBUG_PRINT("enter",("str: '%s' wildstr: '%s'",str,wildstr));
|
|
|
|
while (*wildstr)
|
|
|
|
{
|
|
|
|
while (*wildstr && *wildstr != wild_many && *wildstr != wild_one)
|
|
|
|
{
|
|
|
|
if (*wildstr == wild_prefix && wildstr[1])
|
|
|
|
wildstr++;
|
|
|
|
if (my_toupper(cs, *wildstr++) !=
|
|
|
|
my_toupper(cs, *str++)) DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
if (! *wildstr ) DBUG_RETURN (*str != 0);
|
|
|
|
if (*wildstr++ == wild_one)
|
|
|
|
{
|
|
|
|
if (! *str++) DBUG_RETURN (1); /* One char; skip */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ /* Found '*' */
|
|
|
|
if (!*wildstr) DBUG_RETURN(0); /* '*' as last char: OK */
|
|
|
|
flag=(*wildstr != wild_many && *wildstr != wild_one);
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (flag)
|
|
|
|
{
|
|
|
|
char cmp;
|
|
|
|
if ((cmp= *wildstr) == wild_prefix && wildstr[1])
|
|
|
|
cmp=wildstr[1];
|
|
|
|
cmp=my_toupper(cs, cmp);
|
|
|
|
while (*str && my_toupper(cs, *str) != cmp)
|
|
|
|
str++;
|
|
|
|
if (!*str) DBUG_RETURN (1);
|
|
|
|
}
|
|
|
|
if (wild_case_compare(cs, str,wildstr) == 0) DBUG_RETURN (0);
|
|
|
|
} while (*str++);
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_RETURN (*str != '\0');
|
|
|
|
}
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
|
|
|
|
void update_schema_privilege(TABLE *table, char *buff, const char* db,
|
|
|
|
const char* t_name, const char* column,
|
|
|
|
uint col_length, const char *priv,
|
|
|
|
uint priv_length, const char* is_grantable)
|
|
|
|
{
|
|
|
|
int i= 2;
|
|
|
|
CHARSET_INFO *cs= system_charset_info;
|
2005-01-06 13:00:13 +02:00
|
|
|
restore_record(table, s->default_values);
|
2004-11-13 13:56:39 +03:00
|
|
|
table->field[0]->store(buff, strlen(buff), cs);
|
|
|
|
if (db)
|
|
|
|
table->field[i++]->store(db, strlen(db), cs);
|
|
|
|
if (t_name)
|
|
|
|
table->field[i++]->store(t_name, strlen(t_name), cs);
|
|
|
|
if (column)
|
|
|
|
table->field[i++]->store(column, col_length, cs);
|
|
|
|
table->field[i++]->store(priv, priv_length, cs);
|
|
|
|
table->field[i]->store(is_grantable, strlen(is_grantable), cs);
|
2005-12-22 06:39:02 +01:00
|
|
|
table->file->ha_write_row(table->record[0]);
|
2004-11-13 13:56:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int fill_schema_user_privileges(THD *thd, TABLE_LIST *tables, COND *cond)
|
|
|
|
{
|
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
uint counter;
|
|
|
|
ACL_USER *acl_user;
|
|
|
|
ulong want_access;
|
|
|
|
char buff[100];
|
|
|
|
TABLE *table= tables->table;
|
2005-09-13 16:07:38 +05:00
|
|
|
bool no_global_access= check_access(thd, SELECT_ACL, "mysql",0,1,1,0);
|
2005-09-20 21:20:38 +03:00
|
|
|
char *curr_host= thd->security_ctx->priv_host_name();
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_ENTER("fill_schema_user_privileges");
|
2005-06-02 15:33:53 +05:00
|
|
|
|
2007-03-16 12:15:51 +04:00
|
|
|
if (!initialized)
|
|
|
|
DBUG_RETURN(0);
|
2006-05-06 11:25:59 +04:00
|
|
|
pthread_mutex_lock(&acl_cache->lock);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
for (counter=0 ; counter < acl_users.elements ; counter++)
|
|
|
|
{
|
|
|
|
const char *user,*host, *is_grantable="YES";
|
|
|
|
acl_user=dynamic_element(&acl_users,counter,ACL_USER*);
|
|
|
|
if (!(user=acl_user->user))
|
|
|
|
user= "";
|
|
|
|
if (!(host=acl_user->host.hostname))
|
|
|
|
host= "";
|
2005-06-02 15:33:53 +05:00
|
|
|
|
|
|
|
if (no_global_access &&
|
2005-09-15 22:29:07 +03:00
|
|
|
(strcmp(thd->security_ctx->priv_user, user) ||
|
2005-06-02 15:33:53 +05:00
|
|
|
my_strcasecmp(system_charset_info, curr_host, host)))
|
|
|
|
continue;
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
want_access= acl_user->access;
|
|
|
|
if (!(want_access & GRANT_ACL))
|
|
|
|
is_grantable= "NO";
|
|
|
|
|
|
|
|
strxmov(buff,"'",user,"'@'",host,"'",NullS);
|
|
|
|
if (!(want_access & ~GRANT_ACL))
|
2005-11-20 20:47:07 +02:00
|
|
|
update_schema_privilege(table, buff, 0, 0, 0, 0,
|
|
|
|
STRING_WITH_LEN("USAGE"), is_grantable);
|
2004-11-13 13:56:39 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
uint priv_id;
|
|
|
|
ulong j,test_access= want_access & ~GRANT_ACL;
|
|
|
|
for (priv_id=0, j = SELECT_ACL;j <= GLOBAL_ACLS; priv_id++,j <<= 1)
|
|
|
|
{
|
|
|
|
if (test_access & j)
|
|
|
|
update_schema_privilege(table, buff, 0, 0, 0, 0,
|
|
|
|
command_array[priv_id],
|
|
|
|
command_lengths[priv_id], is_grantable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
pthread_mutex_unlock(&acl_cache->lock);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_RETURN(0);
|
2004-11-14 00:28:44 +02:00
|
|
|
#else
|
|
|
|
return(0);
|
|
|
|
#endif
|
2004-11-13 13:56:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int fill_schema_schema_privileges(THD *thd, TABLE_LIST *tables, COND *cond)
|
|
|
|
{
|
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
uint counter;
|
|
|
|
ACL_DB *acl_db;
|
|
|
|
ulong want_access;
|
|
|
|
char buff[100];
|
|
|
|
TABLE *table= tables->table;
|
2005-09-13 16:07:38 +05:00
|
|
|
bool no_global_access= check_access(thd, SELECT_ACL, "mysql",0,1,1,0);
|
2005-09-20 21:20:38 +03:00
|
|
|
char *curr_host= thd->security_ctx->priv_host_name();
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_ENTER("fill_schema_schema_privileges");
|
|
|
|
|
2007-03-16 12:15:51 +04:00
|
|
|
if (!initialized)
|
|
|
|
DBUG_RETURN(0);
|
2006-05-06 11:25:59 +04:00
|
|
|
pthread_mutex_lock(&acl_cache->lock);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
for (counter=0 ; counter < acl_dbs.elements ; counter++)
|
|
|
|
{
|
|
|
|
const char *user, *host, *is_grantable="YES";
|
|
|
|
|
|
|
|
acl_db=dynamic_element(&acl_dbs,counter,ACL_DB*);
|
|
|
|
if (!(user=acl_db->user))
|
|
|
|
user= "";
|
|
|
|
if (!(host=acl_db->host.hostname))
|
|
|
|
host= "";
|
|
|
|
|
2005-06-02 15:33:53 +05:00
|
|
|
if (no_global_access &&
|
2005-09-15 22:29:07 +03:00
|
|
|
(strcmp(thd->security_ctx->priv_user, user) ||
|
2005-06-02 15:33:53 +05:00
|
|
|
my_strcasecmp(system_charset_info, curr_host, host)))
|
|
|
|
continue;
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
want_access=acl_db->access;
|
|
|
|
if (want_access)
|
|
|
|
{
|
|
|
|
if (!(want_access & GRANT_ACL))
|
|
|
|
{
|
|
|
|
is_grantable= "NO";
|
|
|
|
}
|
|
|
|
strxmov(buff,"'",user,"'@'",host,"'",NullS);
|
|
|
|
if (!(want_access & ~GRANT_ACL))
|
|
|
|
update_schema_privilege(table, buff, acl_db->db, 0, 0,
|
2005-11-20 20:47:07 +02:00
|
|
|
0, STRING_WITH_LEN("USAGE"), is_grantable);
|
2004-11-13 13:56:39 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
int cnt;
|
|
|
|
ulong j,test_access= want_access & ~GRANT_ACL;
|
|
|
|
for (cnt=0, j = SELECT_ACL; j <= DB_ACLS; cnt++,j <<= 1)
|
|
|
|
if (test_access & j)
|
|
|
|
update_schema_privilege(table, buff, acl_db->db, 0, 0, 0,
|
|
|
|
command_array[cnt], command_lengths[cnt],
|
|
|
|
is_grantable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
pthread_mutex_unlock(&acl_cache->lock);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_RETURN(0);
|
2004-11-14 00:28:44 +02:00
|
|
|
#else
|
|
|
|
return (0);
|
|
|
|
#endif
|
2004-11-13 13:56:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int fill_schema_table_privileges(THD *thd, TABLE_LIST *tables, COND *cond)
|
|
|
|
{
|
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
uint index;
|
|
|
|
char buff[100];
|
|
|
|
TABLE *table= tables->table;
|
2005-09-13 16:07:38 +05:00
|
|
|
bool no_global_access= check_access(thd, SELECT_ACL, "mysql",0,1,1,0);
|
2005-09-20 21:20:38 +03:00
|
|
|
char *curr_host= thd->security_ctx->priv_host_name();
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_ENTER("fill_schema_table_privileges");
|
|
|
|
|
2006-05-06 11:25:59 +04:00
|
|
|
rw_rdlock(&LOCK_grant);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
for (index=0 ; index < column_priv_hash.records ; index++)
|
|
|
|
{
|
2006-06-06 14:21:07 +03:00
|
|
|
const char *user, *host, *is_grantable= "YES";
|
2004-11-13 13:56:39 +03:00
|
|
|
GRANT_TABLE *grant_table= (GRANT_TABLE*) hash_element(&column_priv_hash,
|
|
|
|
index);
|
|
|
|
if (!(user=grant_table->user))
|
|
|
|
user= "";
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!(host= grant_table->host.hostname))
|
|
|
|
host= "";
|
2005-06-02 15:33:53 +05:00
|
|
|
|
|
|
|
if (no_global_access &&
|
2005-09-15 22:29:07 +03:00
|
|
|
(strcmp(thd->security_ctx->priv_user, user) ||
|
2006-06-06 14:21:07 +03:00
|
|
|
my_strcasecmp(system_charset_info, curr_host, host)))
|
2005-06-02 15:33:53 +05:00
|
|
|
continue;
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
ulong table_access= grant_table->privs;
|
2004-12-10 12:07:11 +03:00
|
|
|
if (table_access)
|
2004-11-13 13:56:39 +03:00
|
|
|
{
|
|
|
|
ulong test_access= table_access & ~GRANT_ACL;
|
2004-12-10 18:25:12 +03:00
|
|
|
/*
|
|
|
|
We should skip 'usage' privilege on table if
|
|
|
|
we have any privileges on column(s) of this table
|
|
|
|
*/
|
2004-12-10 12:07:11 +03:00
|
|
|
if (!test_access && grant_table->cols)
|
|
|
|
continue;
|
2004-11-13 13:56:39 +03:00
|
|
|
if (!(table_access & GRANT_ACL))
|
|
|
|
is_grantable= "NO";
|
|
|
|
|
2006-06-06 14:21:07 +03:00
|
|
|
strxmov(buff, "'", user, "'@'", host, "'", NullS);
|
2004-11-13 13:56:39 +03:00
|
|
|
if (!test_access)
|
|
|
|
update_schema_privilege(table, buff, grant_table->db, grant_table->tname,
|
2005-11-20 20:47:07 +02:00
|
|
|
0, 0, STRING_WITH_LEN("USAGE"), is_grantable);
|
2004-11-13 13:56:39 +03:00
|
|
|
else
|
|
|
|
{
|
|
|
|
ulong j;
|
|
|
|
int cnt;
|
|
|
|
for (cnt= 0, j= SELECT_ACL; j <= TABLE_ACLS; cnt++, j<<= 1)
|
|
|
|
{
|
|
|
|
if (test_access & j)
|
|
|
|
update_schema_privilege(table, buff, grant_table->db,
|
|
|
|
grant_table->tname, 0, 0, command_array[cnt],
|
|
|
|
command_lengths[cnt], is_grantable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_RETURN(0);
|
2004-11-14 00:28:44 +02:00
|
|
|
#else
|
|
|
|
return (0);
|
|
|
|
#endif
|
2004-11-13 13:56:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int fill_schema_column_privileges(THD *thd, TABLE_LIST *tables, COND *cond)
|
|
|
|
{
|
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
uint index;
|
|
|
|
char buff[100];
|
|
|
|
TABLE *table= tables->table;
|
2005-09-13 16:07:38 +05:00
|
|
|
bool no_global_access= check_access(thd, SELECT_ACL, "mysql",0,1,1,0);
|
2005-09-20 21:20:38 +03:00
|
|
|
char *curr_host= thd->security_ctx->priv_host_name();
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_ENTER("fill_schema_table_privileges");
|
|
|
|
|
2006-05-06 11:25:59 +04:00
|
|
|
rw_rdlock(&LOCK_grant);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
for (index=0 ; index < column_priv_hash.records ; index++)
|
|
|
|
{
|
2006-06-06 14:21:07 +03:00
|
|
|
const char *user, *host, *is_grantable= "YES";
|
2004-11-13 13:56:39 +03:00
|
|
|
GRANT_TABLE *grant_table= (GRANT_TABLE*) hash_element(&column_priv_hash,
|
|
|
|
index);
|
|
|
|
if (!(user=grant_table->user))
|
|
|
|
user= "";
|
2006-06-06 14:21:07 +03:00
|
|
|
if (!(host= grant_table->host.hostname))
|
|
|
|
host= "";
|
2005-06-02 15:33:53 +05:00
|
|
|
|
|
|
|
if (no_global_access &&
|
2005-09-15 22:29:07 +03:00
|
|
|
(strcmp(thd->security_ctx->priv_user, user) ||
|
2006-06-06 14:21:07 +03:00
|
|
|
my_strcasecmp(system_charset_info, curr_host, host)))
|
2005-06-02 15:33:53 +05:00
|
|
|
continue;
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
ulong table_access= grant_table->cols;
|
|
|
|
if (table_access != 0)
|
|
|
|
{
|
2004-11-15 19:20:45 +03:00
|
|
|
if (!(grant_table->privs & GRANT_ACL))
|
2004-11-13 13:56:39 +03:00
|
|
|
is_grantable= "NO";
|
|
|
|
|
2004-11-15 19:20:45 +03:00
|
|
|
ulong test_access= table_access & ~GRANT_ACL;
|
2006-06-06 14:21:07 +03:00
|
|
|
strxmov(buff, "'", user, "'@'", host, "'", NullS);
|
2004-11-13 13:56:39 +03:00
|
|
|
if (!test_access)
|
|
|
|
continue;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ulong j;
|
|
|
|
int cnt;
|
|
|
|
for (cnt= 0, j= SELECT_ACL; j <= TABLE_ACLS; cnt++, j<<= 1)
|
|
|
|
{
|
|
|
|
if (test_access & j)
|
|
|
|
{
|
|
|
|
for (uint col_index=0 ;
|
|
|
|
col_index < grant_table->hash_columns.records ;
|
|
|
|
col_index++)
|
|
|
|
{
|
|
|
|
GRANT_COLUMN *grant_column = (GRANT_COLUMN*)
|
|
|
|
hash_element(&grant_table->hash_columns,col_index);
|
|
|
|
if ((grant_column->rights & j) && (table_access & j))
|
|
|
|
update_schema_privilege(table, buff, grant_table->db,
|
|
|
|
grant_table->tname,
|
|
|
|
grant_column->column,
|
|
|
|
grant_column->key_length,
|
|
|
|
command_array[cnt],
|
|
|
|
command_lengths[cnt], is_grantable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-05-06 11:25:59 +04:00
|
|
|
|
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
|
2004-11-13 13:56:39 +03:00
|
|
|
DBUG_RETURN(0);
|
2004-11-14 00:28:44 +02:00
|
|
|
#else
|
|
|
|
return (0);
|
|
|
|
#endif
|
2004-11-13 13:56:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
#ifndef NO_EMBEDDED_ACCESS_CHECKS
|
|
|
|
/*
|
|
|
|
fill effective privileges for table
|
|
|
|
|
|
|
|
SYNOPSIS
|
2004-10-07 01:45:06 +03:00
|
|
|
fill_effective_table_privileges()
|
|
|
|
thd thread handler
|
2004-07-16 01:15:55 +03:00
|
|
|
grant grants table descriptor
|
|
|
|
db db name
|
|
|
|
table table name
|
|
|
|
*/
|
|
|
|
|
|
|
|
void fill_effective_table_privileges(THD *thd, GRANT_INFO *grant,
|
|
|
|
const char *db, const char *table)
|
|
|
|
{
|
2005-09-20 21:20:38 +03:00
|
|
|
Security_context *sctx= thd->security_ctx;
|
2005-10-28 00:18:23 +03:00
|
|
|
DBUG_ENTER("fill_effective_table_privileges");
|
|
|
|
DBUG_PRINT("enter", ("Host: '%s', Ip: '%s', User: '%s', table: `%s`.`%s`",
|
|
|
|
sctx->priv_host, (sctx->ip ? sctx->ip : "(NULL)"),
|
|
|
|
(sctx->priv_user ? sctx->priv_user : "(NULL)"),
|
|
|
|
db, table));
|
2004-10-25 17:32:28 +03:00
|
|
|
/* --skip-grants */
|
|
|
|
if (!initialized)
|
|
|
|
{
|
2005-10-28 00:18:23 +03:00
|
|
|
DBUG_PRINT("info", ("skip grants"));
|
2004-10-25 17:32:28 +03:00
|
|
|
grant->privilege= ~NO_ACCESS; // everything is allowed
|
2005-10-28 00:18:23 +03:00
|
|
|
DBUG_PRINT("info", ("privilege 0x%lx", grant->privilege));
|
|
|
|
DBUG_VOID_RETURN;
|
2004-10-25 17:32:28 +03:00
|
|
|
}
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
/* global privileges */
|
2005-09-15 22:29:07 +03:00
|
|
|
grant->privilege= sctx->master_access;
|
2004-07-20 01:01:02 +03:00
|
|
|
|
2005-09-15 22:29:07 +03:00
|
|
|
if (!sctx->priv_user)
|
2005-10-28 00:18:23 +03:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("privilege 0x%lx", grant->privilege));
|
|
|
|
DBUG_VOID_RETURN; // it is slave
|
|
|
|
}
|
2005-01-17 01:18:08 +02:00
|
|
|
|
2004-10-22 15:43:22 +03:00
|
|
|
/* db privileges */
|
2005-09-15 22:29:07 +03:00
|
|
|
grant->privilege|= acl_get(sctx->host, sctx->ip, sctx->priv_user, db, 0);
|
2004-10-22 15:43:22 +03:00
|
|
|
|
2004-07-20 01:01:02 +03:00
|
|
|
if (!grant_option)
|
2005-10-28 00:18:23 +03:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("privilege 0x%lx", grant->privilege));
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
2004-07-20 01:01:02 +03:00
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
/* table privileges */
|
2006-05-15 11:56:02 +04:00
|
|
|
rw_rdlock(&LOCK_grant);
|
2004-07-16 01:15:55 +03:00
|
|
|
if (grant->version != grant_version)
|
|
|
|
{
|
|
|
|
grant->grant_table=
|
2005-09-15 22:29:07 +03:00
|
|
|
table_hash_search(sctx->host, sctx->ip, db,
|
|
|
|
sctx->priv_user,
|
2004-07-16 01:15:55 +03:00
|
|
|
table, 0); /* purecov: inspected */
|
|
|
|
grant->version= grant_version; /* purecov: inspected */
|
|
|
|
}
|
|
|
|
if (grant->grant_table != 0)
|
|
|
|
{
|
|
|
|
grant->privilege|= grant->grant_table->privs;
|
|
|
|
}
|
2006-05-15 11:56:02 +04:00
|
|
|
rw_unlock(&LOCK_grant);
|
|
|
|
|
2005-10-28 00:18:23 +03:00
|
|
|
DBUG_PRINT("info", ("privilege 0x%lx", grant->privilege));
|
|
|
|
DBUG_VOID_RETURN;
|
2004-07-16 01:15:55 +03:00
|
|
|
}
|
2005-03-15 16:07:28 +02:00
|
|
|
|
|
|
|
#else /* NO_EMBEDDED_ACCESS_CHECKS */
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
Dummy wrappers when we don't have any access checks
|
|
|
|
****************************************************************************/
|
|
|
|
|
2005-05-17 23:49:01 +01:00
|
|
|
bool check_routine_level_acl(THD *thd, const char *db, const char *name,
|
|
|
|
bool is_proc)
|
2005-03-15 16:07:28 +02:00
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2004-07-16 01:15:55 +03:00
|
|
|
#endif
|