Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
MariaDB
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
MariaDB
Commits
e5a323e1
Commit
e5a323e1
authored
Apr 09, 2013
by
Sergei Golubchik
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
single table discovery: handlerton::discover_table() method.
fixes for need_full_discover_for_existence mode
parent
f532653c
Changes
12
Show whitespace changes
Inline
Side-by-side
Showing
12 changed files
with
219 additions
and
175 deletions
+219
-175
mysql-test/t/partition_disabled.test
mysql-test/t/partition_disabled.test
+2
-0
mysql-test/t/show_check.test
mysql-test/t/show_check.test
+2
-0
sql/handler.cc
sql/handler.cc
+75
-45
sql/handler.h
sql/handler.h
+16
-6
sql/sql_base.cc
sql/sql_base.cc
+19
-13
sql/sql_db.cc
sql/sql_db.cc
+0
-4
sql/sql_table.cc
sql/sql_table.cc
+24
-19
sql/table.cc
sql/table.cc
+52
-59
sql/table.h
sql/table.h
+2
-2
storage/archive/azio.c
storage/archive/azio.c
+4
-4
storage/archive/azlib.h
storage/archive/azlib.h
+2
-2
storage/archive/ha_archive.cc
storage/archive/ha_archive.cc
+21
-21
No files found.
mysql-test/t/partition_disabled.test
View file @
e5a323e1
...
@@ -29,7 +29,9 @@ ALTER TABLE t1 ANALYZE PARTITION ALL;
...
@@ -29,7 +29,9 @@ ALTER TABLE t1 ANALYZE PARTITION ALL;
ALTER
TABLE
t1
REBUILD
PARTITION
ALL
;
ALTER
TABLE
t1
REBUILD
PARTITION
ALL
;
ALTER
TABLE
t1
ENGINE
Memory
;
ALTER
TABLE
t1
ENGINE
Memory
;
ALTER
TABLE
t1
ADD
(
new
INT
);
ALTER
TABLE
t1
ADD
(
new
INT
);
--
disable_warnings
DROP
TABLE
t1
;
DROP
TABLE
t1
;
--
enable_warnings
--
error
ER_OPTION_PREVENTS_STATEMENT
--
error
ER_OPTION_PREVENTS_STATEMENT
CREATE
TABLE
t1
(
CREATE
TABLE
t1
(
...
...
mysql-test/t/show_check.test
View file @
e5a323e1
...
@@ -429,7 +429,9 @@ system echo "this is a junk file for test" >> $MYSQLD_DATADIR/test/t1.frm ;
...
@@ -429,7 +429,9 @@ system echo "this is a junk file for test" >> $MYSQLD_DATADIR/test/t1.frm ;
SHOW
TABLE
STATUS
like
't1'
;
SHOW
TABLE
STATUS
like
't1'
;
--
error
ER_NOT_FORM_FILE
--
error
ER_NOT_FORM_FILE
show
create
table
t1
;
show
create
table
t1
;
--
disable_warnings
drop
table
if
exists
t1
;
drop
table
if
exists
t1
;
--
enable_warnings
--
error
1
,
0
--
error
1
,
0
--
remove_file
$MYSQLD_DATADIR
/
test
/
t1
.
frm
--
remove_file
$MYSQLD_DATADIR
/
test
/
t1
.
frm
...
...
sql/handler.cc
View file @
e5a323e1
...
@@ -491,11 +491,11 @@ int ha_initialize_handlerton(st_plugin_int *plugin)
...
@@ -491,11 +491,11 @@ int ha_initialize_handlerton(st_plugin_int *plugin)
// if the enfine can discover a single table and it is file-based
// if the enfine can discover a single table and it is file-based
// then it can use a default file-based table names discovery
// then it can use a default file-based table names discovery
if
(
!
hton
->
discover_table_names
&&
if
(
!
hton
->
discover_table_names
&&
hton
->
discover
&&
hton
->
tablefile_extensions
[
0
])
hton
->
discover
_table
&&
hton
->
tablefile_extensions
[
0
])
hton
->
discover_table_names
=
hton_ext_based_table_discovery
;
hton
->
discover_table_names
=
hton_ext_based_table_discovery
;
// default discover_table_existence implementation
// default discover_table_existence implementation
if
(
!
hton
->
discover_table_existence
&&
hton
->
discover
)
if
(
!
hton
->
discover_table_existence
&&
hton
->
discover
_table
)
{
{
if
(
hton
->
tablefile_extensions
[
0
])
if
(
hton
->
tablefile_extensions
[
0
])
hton
->
discover_table_existence
=
ext_based_existence
;
hton
->
discover_table_existence
=
ext_based_existence
;
...
@@ -4279,56 +4279,44 @@ int ha_change_key_cache(KEY_CACHE *old_key_cache,
...
@@ -4279,56 +4279,44 @@ int ha_change_key_cache(KEY_CACHE *old_key_cache,
}
}
/**
Try to discover one table from handler(s).
@retval
-1 Table did not exists
@retval
0 OK. In this case *frmblob and *frmlen are set
@retval
>0 error. frmblob and frmlen may not be set
*/
struct
st_discover_args
{
const
char
*
db
;
const
char
*
name
;
uchar
**
frmblob
;
size_t
*
frmlen
;
};
static
my_bool
discover_handlerton
(
THD
*
thd
,
plugin_ref
plugin
,
static
my_bool
discover_handlerton
(
THD
*
thd
,
plugin_ref
plugin
,
void
*
arg
)
void
*
arg
)
{
{
st_discover_args
*
vargs
=
(
st_discover_args
*
)
arg
;
TABLE_SHARE
*
share
=
(
TABLE_SHARE
*
)
arg
;
handlerton
*
hton
=
plugin_data
(
plugin
,
handlerton
*
);
handlerton
*
hton
=
plugin_data
(
plugin
,
handlerton
*
);
if
(
hton
->
state
==
SHOW_OPTION_YES
&&
hton
->
discover
&&
if
(
hton
->
state
==
SHOW_OPTION_YES
&&
hton
->
discover_table
)
(
!
(
hton
->
discover
(
hton
,
thd
,
vargs
->
db
,
vargs
->
name
,
{
vargs
->
frmblob
,
int
error
=
hton
->
discover_table
(
hton
,
thd
,
share
);
vargs
->
frmlen
))))
if
(
error
!=
HA_ERR_NO_SUCH_TABLE
)
return
TRUE
;
{
if
(
error
)
{
DBUG_ASSERT
(
share
->
error
);
// MUST be always set for get_cached_table_share to work
my_error
(
ER_GET_ERRNO
,
MYF
(
0
),
error
);
}
else
share
->
error
=
OPEN_FRM_OK
;
return
FALSE
;
status_var_increment
(
thd
->
status_var
.
ha_discover_count
);
return
TRUE
;
// abort the search
}
}
DBUG_ASSERT
(
share
->
error
==
OPEN_FRM_OPEN_ERROR
);
return
FALSE
;
// continue with the next engine
}
}
int
ha_discover
(
THD
*
thd
,
const
char
*
db
,
const
char
*
name
,
int
ha_discover_table
(
THD
*
thd
,
TABLE_SHARE
*
share
)
uchar
**
frmblob
,
size_t
*
frmlen
)
{
{
int
error
=
-
1
;
// Table does not exist in any handler
DBUG_ENTER
(
"ha_discover_table"
);
DBUG_ENTER
(
"ha_discover"
);
DBUG_PRINT
(
"enter"
,
(
"db: %s, name: %s"
,
db
,
name
));
st_discover_args
args
=
{
db
,
name
,
frmblob
,
frmlen
};
if
(
is_prefix
(
name
,
tmp_file_prefix
))
/* skip temporary tables */
DBUG_ASSERT
(
share
->
error
==
OPEN_FRM_OPEN_ERROR
);
// share is not OK yet
DBUG_RETURN
(
error
);
if
(
plugin_foreach
(
thd
,
discover_handlerton
,
if
(
!
plugin_foreach
(
thd
,
discover_handlerton
,
MYSQL_STORAGE_ENGINE_PLUGIN
,
&
args
))
MYSQL_STORAGE_ENGINE_PLUGIN
,
share
))
error
=
0
;
open_table_error
(
share
,
OPEN_FRM_OPEN_ERROR
,
ENOENT
);
// not found
if
(
!
error
)
DBUG_RETURN
(
share
->
error
!=
OPEN_FRM_OK
);
status_var_increment
(
thd
->
status_var
.
ha_discover_count
);
DBUG_RETURN
(
error
);
}
}
/**
/**
...
@@ -4363,6 +4351,45 @@ static my_bool discover_existence(THD *thd, plugin_ref plugin,
...
@@ -4363,6 +4351,45 @@ static my_bool discover_existence(THD *thd, plugin_ref plugin,
return
ht
->
discover_table_existence
(
ht
,
args
->
db
,
args
->
table_name
);
return
ht
->
discover_table_existence
(
ht
,
args
->
db
,
args
->
table_name
);
}
}
class
Table_exists_error_handler
:
public
Internal_error_handler
{
public:
Table_exists_error_handler
()
:
m_handled_errors
(
0
),
m_unhandled_errors
(
0
)
{}
bool
handle_condition
(
THD
*
thd
,
uint
sql_errno
,
const
char
*
sqlstate
,
MYSQL_ERROR
::
enum_warning_level
level
,
const
char
*
msg
,
MYSQL_ERROR
**
cond_hdl
)
{
*
cond_hdl
=
NULL
;
if
(
sql_errno
==
ER_NO_SUCH_TABLE
||
sql_errno
==
ER_NO_SUCH_TABLE_IN_ENGINE
||
sql_errno
==
ER_WRONG_OBJECT
||
sql_errno
==
ER_OPTION_PREVENTS_STATEMENT
)
// partition_disabled.test
{
m_handled_errors
++
;
return
TRUE
;
}
if
(
level
==
MYSQL_ERROR
::
WARN_LEVEL_ERROR
)
m_unhandled_errors
++
;
return
FALSE
;
}
bool
safely_trapped_errors
()
{
return
((
m_handled_errors
>
0
)
&&
(
m_unhandled_errors
==
0
));
}
private:
int
m_handled_errors
;
int
m_unhandled_errors
;
};
bool
ha_table_exists
(
THD
*
thd
,
const
char
*
db
,
const
char
*
table_name
)
bool
ha_table_exists
(
THD
*
thd
,
const
char
*
db
,
const
char
*
table_name
)
{
{
DBUG_ENTER
(
"ha_discover_table_existence"
);
DBUG_ENTER
(
"ha_discover_table_existence"
);
...
@@ -4371,10 +4398,13 @@ bool ha_table_exists(THD *thd, const char *db, const char *table_name)
...
@@ -4371,10 +4398,13 @@ bool ha_table_exists(THD *thd, const char *db, const char *table_name)
{
{
TABLE_LIST
table
;
TABLE_LIST
table
;
DBUG_ASSERT
(
0
);
Table_exists_error_handler
no_such_table_handler
;
TABLE_SHARE
*
share
=
get_table_share
(
thd
,
db
,
table_name
,
thd
->
push_internal_handler
(
&
no_such_table_handler
);
GTS_TABLE
|
GTS_NOLOCK
);
get_table_share
(
thd
,
db
,
table_name
,
GTS_TABLE
|
GTS_VIEW
|
GTS_NOLOCK
);
DBUG_RETURN
(
share
!=
0
);
thd
->
pop_internal_handler
();
// the table doesn't exist if we've caught ER_NO_SUCH_TABLE and nothing else
DBUG_RETURN
(
!
no_such_table_handler
.
safely_trapped_errors
());
}
}
mysql_mutex_lock
(
&
LOCK_open
);
mysql_mutex_lock
(
&
LOCK_open
);
...
...
sql/handler.h
View file @
e5a323e1
...
@@ -1075,10 +1075,6 @@ struct handlerton
...
@@ -1075,10 +1075,6 @@ struct handlerton
enum
handler_create_iterator_result
enum
handler_create_iterator_result
(
*
create_iterator
)(
handlerton
*
hton
,
enum
handler_iterator_type
type
,
(
*
create_iterator
)(
handlerton
*
hton
,
enum
handler_iterator_type
type
,
struct
handler_iterator
*
fill_this_in
);
struct
handler_iterator
*
fill_this_in
);
int
(
*
discover
)(
handlerton
*
hton
,
THD
*
thd
,
const
char
*
db
,
const
char
*
name
,
uchar
**
frmblob
,
size_t
*
frmlen
);
/*
/*
Optional clauses in the CREATE/ALTER TABLE
Optional clauses in the CREATE/ALTER TABLE
*/
*/
...
@@ -1110,6 +1106,20 @@ struct handlerton
...
@@ -1110,6 +1106,20 @@ struct handlerton
engine, without user issuing an explicit CREATE TABLE statement.
engine, without user issuing an explicit CREATE TABLE statement.
**********************************************************************/
**********************************************************************/
/*
This method is required for any engine that supports automatic table
discovery, there is no default implementation.
Given a TABLE_SHARE discover_table() fills it in with a correct table
structure using one of the TABLE_SHARE::init_from_* methods.
Returns HA_ERR_NO_SUCH_TABLE if the table did not exist in the engine,
zero if the table was discovered successfully, or any other
HA_ERR_* error code as appropriate if the table existed, but the
discovery failed.
*/
int
(
*
discover_table
)(
handlerton
*
hton
,
THD
*
thd
,
TABLE_SHARE
*
share
);
/*
/*
The discover_table_names method tells the server
The discover_table_names method tells the server
about all tables in the specified database that the engine
about all tables in the specified database that the engine
...
@@ -1157,6 +1167,7 @@ struct handlerton
...
@@ -1157,6 +1167,7 @@ struct handlerton
*/
*/
int
(
*
discover_table_existence
)(
handlerton
*
hton
,
const
char
*
db
,
int
(
*
discover_table_existence
)(
handlerton
*
hton
,
const
char
*
db
,
const
char
*
table_name
);
const
char
*
table_name
);
};
};
...
@@ -3088,8 +3099,7 @@ int ha_delete_table(THD *thd, handlerton *db_type, const char *path,
...
@@ -3088,8 +3099,7 @@ int ha_delete_table(THD *thd, handlerton *db_type, const char *path,
bool
ha_show_status
(
THD
*
thd
,
handlerton
*
db_type
,
enum
ha_stat_type
stat
);
bool
ha_show_status
(
THD
*
thd
,
handlerton
*
db_type
,
enum
ha_stat_type
stat
);
/* discovery */
/* discovery */
int
ha_discover
(
THD
*
thd
,
const
char
*
dbname
,
const
char
*
name
,
int
ha_discover_table
(
THD
*
thd
,
TABLE_SHARE
*
share
);
uchar
**
frmblob
,
size_t
*
frmlen
);
int
ha_discover_table_names
(
THD
*
thd
,
LEX_STRING
*
db
,
MY_DIR
*
dirp
,
int
ha_discover_table_names
(
THD
*
thd
,
LEX_STRING
*
db
,
MY_DIR
*
dirp
,
handlerton
::
discovered_list
*
result
);
handlerton
::
discovered_list
*
result
);
bool
ha_table_exists
(
THD
*
thd
,
const
char
*
db
,
const
char
*
table_name
);
bool
ha_table_exists
(
THD
*
thd
,
const
char
*
db
,
const
char
*
table_name
);
...
...
sql/sql_base.cc
View file @
e5a323e1
...
@@ -585,24 +585,15 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
...
@@ -585,24 +585,15 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
char
*
key
,
uint
key_length
,
uint
flags
,
char
*
key
,
uint
key_length
,
uint
flags
,
my_hash_value_type
hash_value
)
my_hash_value_type
hash_value
)
{
{
bool
open_failed
;
TABLE_SHARE
*
share
;
TABLE_SHARE
*
share
;
DBUG_ENTER
(
"get_table_share"
);
DBUG_ENTER
(
"get_table_share"
);
DBUG_ASSERT
(
!
(
flags
&
GTS_FORCE_DISCOVERY
));
// FIXME not implemented
mysql_mutex_lock
(
&
LOCK_open
);
mysql_mutex_lock
(
&
LOCK_open
);
/*
To be able perform any operation on table we should own
some kind of metadata lock on it.
*/
DBUG_ASSERT
(
thd
->
mdl_context
.
is_lock_owner
(
MDL_key
::
TABLE
,
db
,
table_name
,
MDL_SHARED
));
/* Read table definition from cache */
/* Read table definition from cache */
share
=
(
TABLE_SHARE
*
)
my_hash_search_using_hash_value
(
&
table_def_cache
,
share
=
(
TABLE_SHARE
*
)
my_hash_search_using_hash_value
(
&
table_def_cache
,
hash_value
,
(
uchar
*
)
key
,
key_length
);
hash_value
,
(
uchar
*
)
key
,
key_length
);
if
(
!
share
)
if
(
!
share
)
{
{
if
(
!
(
share
=
alloc_table_share
(
db
,
table_name
,
key
,
key_length
)))
if
(
!
(
share
=
alloc_table_share
(
db
,
table_name
,
key
,
key_length
)))
...
@@ -633,12 +624,15 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
...
@@ -633,12 +624,15 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
mysql_mutex_lock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_lock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_unlock
(
&
LOCK_open
);
mysql_mutex_unlock
(
&
LOCK_open
);
open_failed
=
open_table_def
(
thd
,
share
,
flags
);
if
(
flags
&
GTS_FORCE_DISCOVERY
)
ha_discover_table
(
thd
,
share
);
// don't read the frm at all
else
open_table_def
(
thd
,
share
,
flags
|
GTS_FORCE_DISCOVERY
);
// frm or discover
mysql_mutex_unlock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_unlock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_lock
(
&
LOCK_open
);
mysql_mutex_lock
(
&
LOCK_open
);
if
(
open_failed
)
if
(
share
->
error
)
{
{
share
->
ref_count
--
;
share
->
ref_count
--
;
(
void
)
my_hash_delete
(
&
table_def_cache
,
(
uchar
*
)
share
);
(
void
)
my_hash_delete
(
&
table_def_cache
,
(
uchar
*
)
share
);
...
@@ -650,6 +644,9 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
...
@@ -650,6 +644,9 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
goto
end
;
goto
end
;
}
}
/* cannot force discovery of a cached share */
DBUG_ASSERT
(
!
(
flags
&
GTS_FORCE_DISCOVERY
));
/* make sure that open_table_def() for this share is not running */
/* make sure that open_table_def() for this share is not running */
mysql_mutex_lock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_lock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_unlock
(
&
share
->
LOCK_ha_data
);
mysql_mutex_unlock
(
&
share
->
LOCK_ha_data
);
...
@@ -706,7 +703,7 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
...
@@ -706,7 +703,7 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
end:
end:
if
(
flags
&
GTS_NOLOCK
)
if
(
flags
&
GTS_NOLOCK
)
{
{
share
->
ref_count
--
;
release_table_share
(
share
)
;
/*
/*
if GTS_NOLOCK is requested, the returned share pointer cannot be used,
if GTS_NOLOCK is requested, the returned share pointer cannot be used,
the share it points to may go away any moment.
the share it points to may go away any moment.
...
@@ -716,6 +713,15 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
...
@@ -716,6 +713,15 @@ TABLE_SHARE *get_table_share(THD *thd, const char *db, const char *table_name,
*/
*/
share
=
(
TABLE_SHARE
*
)
1
;
share
=
(
TABLE_SHARE
*
)
1
;
}
}
else
{
/*
To be able perform any operation on table we should own
some kind of metadata lock on it.
*/
DBUG_ASSERT
(
thd
->
mdl_context
.
is_lock_owner
(
MDL_key
::
TABLE
,
db
,
table_name
,
MDL_SHARED
));
}
mysql_mutex_unlock
(
&
LOCK_open
);
mysql_mutex_unlock
(
&
LOCK_open
);
DBUG_RETURN
(
share
);
DBUG_RETURN
(
share
);
...
...
sql/sql_db.cc
View file @
e5a323e1
...
@@ -832,11 +832,7 @@ bool mysql_rm_db(THD *thd,char *db,bool if_exists, bool silent)
...
@@ -832,11 +832,7 @@ bool mysql_rm_db(THD *thd,char *db,bool if_exists, bool silent)
mysql_ha_rm_tables
(
thd
,
tables
);
mysql_ha_rm_tables
(
thd
,
tables
);
for
(
table
=
tables
;
table
;
table
=
table
->
next_local
)
for
(
table
=
tables
;
table
;
table
=
table
->
next_local
)
{
tdc_remove_table
(
thd
,
TDC_RT_REMOVE_ALL
,
table
->
db
,
table
->
table_name
,
false
);
deleted_tables
++
;
deleted_tables
++
;
}
thd
->
push_internal_handler
(
&
err_handler
);
thd
->
push_internal_handler
(
&
err_handler
);
if
(
!
thd
->
killed
&&
if
(
!
thd
->
killed
&&
...
...
sql/sql_table.cc
View file @
e5a323e1
...
@@ -1928,10 +1928,6 @@ bool mysql_rm_table(THD *thd,TABLE_LIST *tables, my_bool if_exists,
...
@@ -1928,10 +1928,6 @@ bool mysql_rm_table(THD *thd,TABLE_LIST *tables, my_bool if_exists,
if
(
lock_table_names
(
thd
,
tables
,
NULL
,
thd
->
variables
.
lock_wait_timeout
,
if
(
lock_table_names
(
thd
,
tables
,
NULL
,
thd
->
variables
.
lock_wait_timeout
,
MYSQL_OPEN_SKIP_TEMPORARY
))
MYSQL_OPEN_SKIP_TEMPORARY
))
DBUG_RETURN
(
true
);
DBUG_RETURN
(
true
);
for
(
table
=
tables
;
table
;
table
=
table
->
next_local
)
tdc_remove_table
(
thd
,
TDC_RT_REMOVE_ALL
,
table
->
db
,
table
->
table_name
,
false
);
}
}
else
else
{
{
...
@@ -2222,23 +2218,9 @@ int mysql_rm_table_no_locks(THD *thd, TABLE_LIST *tables, bool if_exists,
...
@@ -2222,23 +2218,9 @@ int mysql_rm_table_no_locks(THD *thd, TABLE_LIST *tables, bool if_exists,
{
{
non_temp_tables_count
++
;
non_temp_tables_count
++
;
if
(
thd
->
locked_tables_mode
)
{
if
(
wait_while_table_is_used
(
thd
,
table
->
table
,
HA_EXTRA_NOT_USED
,
TDC_RT_REMOVE_NOT_OWN_AND_MARK_NOT_USABLE
))
{
error
=
-
1
;
goto
err
;
}
close_all_tables_for_name
(
thd
,
table
->
table
->
s
,
HA_EXTRA_PREPARE_FOR_DROP
);
table
->
table
=
0
;
}
/* Check that we have an exclusive lock on the table to be dropped. */
DBUG_ASSERT
(
thd
->
mdl_context
.
is_lock_owner
(
MDL_key
::
TABLE
,
table
->
db
,
DBUG_ASSERT
(
thd
->
mdl_context
.
is_lock_owner
(
MDL_key
::
TABLE
,
table
->
db
,
table
->
table_name
,
table
->
table_name
,
MDL_
EXCLUSIVE
));
MDL_
SHARED
));
alias
=
(
lower_case_table_names
==
2
)
?
table
->
alias
:
table
->
table_name
;
alias
=
(
lower_case_table_names
==
2
)
?
table
->
alias
:
table
->
table_name
;
/* remove .frm file and engine files */
/* remove .frm file and engine files */
...
@@ -2307,6 +2289,28 @@ int mysql_rm_table_no_locks(THD *thd, TABLE_LIST *tables, bool if_exists,
...
@@ -2307,6 +2289,28 @@ int mysql_rm_table_no_locks(THD *thd, TABLE_LIST *tables, bool if_exists,
else
else
{
{
char
*
end
;
char
*
end
;
if
(
thd
->
locked_tables_mode
)
{
if
(
wait_while_table_is_used
(
thd
,
table
->
table
,
HA_EXTRA_NOT_USED
,
TDC_RT_REMOVE_NOT_OWN_AND_MARK_NOT_USABLE
))
{
error
=
-
1
;
goto
err
;
}
close_all_tables_for_name
(
thd
,
table
->
table
->
s
,
HA_EXTRA_PREPARE_FOR_DROP
);
table
->
table
=
0
;
}
else
tdc_remove_table
(
thd
,
TDC_RT_REMOVE_ALL
,
table
->
db
,
table
->
table_name
,
false
);
/* Check that we have an exclusive lock on the table to be dropped. */
DBUG_ASSERT
(
thd
->
mdl_context
.
is_lock_owner
(
MDL_key
::
TABLE
,
table
->
db
,
table
->
table_name
,
MDL_EXCLUSIVE
));
/*
/*
Cannot use the db_type from the table, since that might have changed
Cannot use the db_type from the table, since that might have changed
while waiting for the exclusive name lock.
while waiting for the exclusive name lock.
...
@@ -2372,6 +2376,7 @@ int mysql_rm_table_no_locks(THD *thd, TABLE_LIST *tables, bool if_exists,
...
@@ -2372,6 +2376,7 @@ int mysql_rm_table_no_locks(THD *thd, TABLE_LIST *tables, bool if_exists,
err:
err:
if
(
wrong_tables
.
length
())
if
(
wrong_tables
.
length
())
{
{
thd
->
clear_error
();
if
(
!
foreign_key_error
)
if
(
!
foreign_key_error
)
my_printf_error
(
ER_BAD_TABLE_ERROR
,
ER
(
ER_BAD_TABLE_ERROR
),
MYF
(
0
),
my_printf_error
(
ER_BAD_TABLE_ERROR
,
ER
(
ER_BAD_TABLE_ERROR
),
MYF
(
0
),
wrong_tables
.
c_ptr_safe
());
wrong_tables
.
c_ptr_safe
());
...
...
sql/table.cc
View file @
e5a323e1
...
@@ -316,6 +316,7 @@ TABLE_SHARE *alloc_table_share(const char *db, const char *table_name,
...
@@ -316,6 +316,7 @@ TABLE_SHARE *alloc_table_share(const char *db, const char *table_name,
share
->
normalized_path
.
length
=
path_length
;
share
->
normalized_path
.
length
=
path_length
;
share
->
table_category
=
get_table_category
(
&
share
->
db
,
&
share
->
table_name
);
share
->
table_category
=
get_table_category
(
&
share
->
db
,
&
share
->
table_name
);
share
->
set_refresh_version
();
share
->
set_refresh_version
();
share
->
open_errno
=
ENOENT
;
/*
/*
Since alloc_table_share() can be called without any locking (for
Since alloc_table_share() can be called without any locking (for
...
@@ -570,24 +571,21 @@ inline bool is_system_table_name(const char *name, uint length)
...
@@ -570,24 +571,21 @@ inline bool is_system_table_name(const char *name, uint length)
}
}
/**
/*
Check if a string contains path elements
We don't try to open 5.0 unencoded name, if
*/
- non-encoded name contains '@' signs,
because '@' can be misinterpreted.
It is not clear if '@' is escape character in 5.1,
or a normal character in 5.0.
- non-encoded db or table name contain "#mysql50#" prefix.
This kind of tables must have been opened only by the
mysql_file_open() above.
*/
static
bool
has_disabled_path_chars
(
const
char
*
str
)
static
bool
has_disabled_path_chars
(
const
char
*
str
)
{
{
for
(;
*
str
;
str
++
)
return
strpbrk
(
str
,
"/
\\
~@."
)
!=
0
||
{
strncmp
(
str
,
STRING_WITH_LEN
(
MYSQL50_TABLE_NAME_PREFIX
))
==
0
;
switch
(
*
str
)
{
case
FN_EXTCHAR
:
case
'/'
:
case
'\\'
:
case
'~'
:
case
'@'
:
return
TRUE
;
}
}
return
FALSE
;
}
}
...
@@ -626,25 +624,9 @@ enum open_frm_error open_table_def(THD *thd, TABLE_SHARE *share, uint flags)
...
@@ -626,25 +624,9 @@ enum open_frm_error open_table_def(THD *thd, TABLE_SHARE *share, uint flags)
if
((
file
=
mysql_file_open
(
key_file_frm
,
if
((
file
=
mysql_file_open
(
key_file_frm
,
path
,
O_RDONLY
|
O_SHARE
,
MYF
(
0
)))
<
0
)
path
,
O_RDONLY
|
O_SHARE
,
MYF
(
0
)))
<
0
)
{
{
/*
if
(
!
has_disabled_path_chars
(
share
->
table_name
.
str
)
&&
We don't try to open 5.0 unencoded name, if
!
has_disabled_path_chars
(
share
->
db
.
str
))
- non-encoded name contains '@' signs,
{
because '@' can be misinterpreted.
It is not clear if '@' is escape character in 5.1,
or a normal character in 5.0.
- non-encoded db or table name contain "#mysql50#" prefix.
This kind of tables must have been opened only by the
mysql_file_open() above.
*/
if
(
has_disabled_path_chars
(
share
->
table_name
.
str
)
||
has_disabled_path_chars
(
share
->
db
.
str
)
||
!
strncmp
(
share
->
db
.
str
,
MYSQL50_TABLE_NAME_PREFIX
,
MYSQL50_TABLE_NAME_PREFIX_LENGTH
)
||
!
strncmp
(
share
->
table_name
.
str
,
MYSQL50_TABLE_NAME_PREFIX
,
MYSQL50_TABLE_NAME_PREFIX_LENGTH
))
goto
err_not_open
;
/* Try unencoded 5.0 name */
/* Try unencoded 5.0 name */
uint
length
;
uint
length
;
strxnmov
(
path
,
sizeof
(
path
)
-
1
,
strxnmov
(
path
,
sizeof
(
path
)
-
1
,
...
@@ -661,16 +643,27 @@ enum open_frm_error open_table_def(THD *thd, TABLE_SHARE *share, uint flags)
...
@@ -661,16 +643,27 @@ enum open_frm_error open_table_def(THD *thd, TABLE_SHARE *share, uint flags)
then table name does not have tricky characters,
then table name does not have tricky characters,
so no need to check the old file name.
so no need to check the old file name.
*/
*/
if
(
length
==
share
->
normalized_path
.
length
||
if
(
length
!=
share
->
normalized_path
.
length
&&
((
file
=
mysql_file_open
(
key_file_frm
,
(
file
=
mysql_file_open
(
key_file_frm
,
path
,
O_RDONLY
|
O_SHARE
,
MYF
(
0
)))
<
0
))
path
,
O_RDONLY
|
O_SHARE
,
MYF
(
0
)))
>=
0
)
goto
err_not_open
;
{
/* Unencoded 5.0 table name found */
/* Unencoded 5.0 table name found */
path
[
length
]
=
'\0'
;
// Remove .frm extension
path
[
length
]
=
'\0'
;
// Remove .frm extension
strmov
(
share
->
normalized_path
.
str
,
path
);
strmov
(
share
->
normalized_path
.
str
,
path
);
share
->
normalized_path
.
length
=
length
;
share
->
normalized_path
.
length
=
length
;
}
}
}
/* still no luck? try to discover the table */
if
(
file
<
0
)
{
if
(
flags
&
GTS_TABLE
&&
flags
&
GTS_FORCE_DISCOVERY
)
{
ha_discover_table
(
thd
,
share
);
error_given
=
true
;
}
goto
err_not_open
;
}
}
if
(
mysql_file_read
(
file
,
head
,
sizeof
(
head
),
MYF
(
MY_NABP
)))
if
(
mysql_file_read
(
file
,
head
,
sizeof
(
head
),
MYF
(
MY_NABP
)))
{
{
...
...
sql/table.h
View file @
e5a323e1
...
@@ -2450,8 +2450,8 @@ static inline void dbug_tmp_restore_column_maps(MY_BITMAP *read_set,
...
@@ -2450,8 +2450,8 @@ static inline void dbug_tmp_restore_column_maps(MY_BITMAP *read_set,
enum
get_table_share_flags
{
enum
get_table_share_flags
{
GTS_TABLE
=
1
,
GTS_TABLE
=
1
,
GTS_VIEW
=
2
,
GTS_VIEW
=
2
,
GTS_NOLOCK
=
4
,
// don't increase share->ref_count
GTS_NOLOCK
=
4
,
GTS_FORCE_DISCOVERY
=
8
// don't use the .frm file
GTS_FORCE_DISCOVERY
=
8
};
};
size_t
max_row_length
(
TABLE
*
table
,
const
uchar
*
data
);
size_t
max_row_length
(
TABLE
*
table
,
const
uchar
*
data
);
...
...
storage/archive/azio.c
View file @
e5a323e1
...
@@ -855,7 +855,7 @@ int azclose (azio_stream *s)
...
@@ -855,7 +855,7 @@ int azclose (azio_stream *s)
Though this was added to support MySQL's FRM file, anything can be
Though this was added to support MySQL's FRM file, anything can be
stored in this location.
stored in this location.
*/
*/
int
azwrite_frm
(
azio_stream
*
s
,
char
*
blob
,
unsigned
int
length
)
int
azwrite_frm
(
azio_stream
*
s
,
u
char
*
blob
,
unsigned
int
length
)
{
{
if
(
s
->
mode
==
'r'
)
if
(
s
->
mode
==
'r'
)
return
1
;
return
1
;
...
@@ -867,7 +867,7 @@ int azwrite_frm(azio_stream *s, char *blob, unsigned int length)
...
@@ -867,7 +867,7 @@ int azwrite_frm(azio_stream *s, char *blob, unsigned int length)
s
->
frm_length
=
length
;
s
->
frm_length
=
length
;
s
->
start
+=
length
;
s
->
start
+=
length
;
if
(
my_pwrite
(
s
->
file
,
(
uchar
*
)
blob
,
s
->
frm_length
,
if
(
my_pwrite
(
s
->
file
,
blob
,
s
->
frm_length
,
s
->
frm_start_pos
,
MYF
(
MY_NABP
))
||
s
->
frm_start_pos
,
MYF
(
MY_NABP
))
||
write_header
(
s
)
||
write_header
(
s
)
||
(
my_seek
(
s
->
file
,
0
,
MY_SEEK_END
,
MYF
(
0
))
==
MY_FILEPOS_ERROR
))
(
my_seek
(
s
->
file
,
0
,
MY_SEEK_END
,
MYF
(
0
))
==
MY_FILEPOS_ERROR
))
...
@@ -876,9 +876,9 @@ int azwrite_frm(azio_stream *s, char *blob, unsigned int length)
...
@@ -876,9 +876,9 @@ int azwrite_frm(azio_stream *s, char *blob, unsigned int length)
return
0
;
return
0
;
}
}
int
azread_frm
(
azio_stream
*
s
,
char
*
blob
)
int
azread_frm
(
azio_stream
*
s
,
u
char
*
blob
)
{
{
return
my_pread
(
s
->
file
,
(
uchar
*
)
blob
,
s
->
frm_length
,
return
my_pread
(
s
->
file
,
blob
,
s
->
frm_length
,
s
->
frm_start_pos
,
MYF
(
MY_NABP
))
?
1
:
0
;
s
->
frm_start_pos
,
MYF
(
MY_NABP
))
?
1
:
0
;
}
}
...
...
storage/archive/azlib.h
View file @
e5a323e1
...
@@ -331,8 +331,8 @@ extern int azclose(azio_stream *file);
...
@@ -331,8 +331,8 @@ extern int azclose(azio_stream *file);
error number (see function gzerror below).
error number (see function gzerror below).
*/
*/
extern
int
azwrite_frm
(
azio_stream
*
s
,
char
*
blob
,
unsigned
int
length
);
extern
int
azwrite_frm
(
azio_stream
*
s
,
u
char
*
blob
,
unsigned
int
length
);
extern
int
azread_frm
(
azio_stream
*
s
,
char
*
blob
);
extern
int
azread_frm
(
azio_stream
*
s
,
u
char
*
blob
);
extern
int
azwrite_comment
(
azio_stream
*
s
,
char
*
blob
,
unsigned
int
length
);
extern
int
azwrite_comment
(
azio_stream
*
s
,
char
*
blob
,
unsigned
int
length
);
extern
int
azread_comment
(
azio_stream
*
s
,
char
*
blob
);
extern
int
azread_comment
(
azio_stream
*
s
,
char
*
blob
);
...
...
storage/archive/ha_archive.cc
View file @
e5a323e1
...
@@ -26,6 +26,7 @@
...
@@ -26,6 +26,7 @@
#include <myisam.h> // T_EXTEND
#include <myisam.h> // T_EXTEND
#include "ha_archive.h"
#include "ha_archive.h"
#include "discover.h"
#include <my_dir.h>
#include <my_dir.h>
#include <mysql/plugin.h>
#include <mysql/plugin.h>
...
@@ -120,10 +121,7 @@ extern "C" PSI_file_key arch_key_file_data;
...
@@ -120,10 +121,7 @@ extern "C" PSI_file_key arch_key_file_data;
static
handler
*
archive_create_handler
(
handlerton
*
hton
,
static
handler
*
archive_create_handler
(
handlerton
*
hton
,
TABLE_SHARE
*
table
,
TABLE_SHARE
*
table
,
MEM_ROOT
*
mem_root
);
MEM_ROOT
*
mem_root
);
int
archive_discover
(
handlerton
*
hton
,
THD
*
thd
,
const
char
*
db
,
int
archive_discover
(
handlerton
*
hton
,
THD
*
thd
,
TABLE_SHARE
*
share
);
const
char
*
name
,
uchar
**
frmblob
,
size_t
*
frmlen
);
/*
/*
Number of rows that will force a bulk insert.
Number of rows that will force a bulk insert.
...
@@ -220,7 +218,7 @@ int archive_db_init(void *p)
...
@@ -220,7 +218,7 @@ int archive_db_init(void *p)
archive_hton
->
db_type
=
DB_TYPE_ARCHIVE_DB
;
archive_hton
->
db_type
=
DB_TYPE_ARCHIVE_DB
;
archive_hton
->
create
=
archive_create_handler
;
archive_hton
->
create
=
archive_create_handler
;
archive_hton
->
flags
=
HTON_NO_FLAGS
;
archive_hton
->
flags
=
HTON_NO_FLAGS
;
archive_hton
->
discover
=
archive_discover
;
archive_hton
->
discover
_table
=
archive_discover
;
archive_hton
->
tablefile_extensions
=
ha_archive_exts
;
archive_hton
->
tablefile_extensions
=
ha_archive_exts
;
if
(
mysql_mutex_init
(
az_key_mutex_archive_mutex
,
if
(
mysql_mutex_init
(
az_key_mutex_archive_mutex
,
...
@@ -270,19 +268,17 @@ ha_archive::ha_archive(handlerton *hton, TABLE_SHARE *table_arg)
...
@@ -270,19 +268,17 @@ ha_archive::ha_archive(handlerton *hton, TABLE_SHARE *table_arg)
archive_reader_open
=
FALSE
;
archive_reader_open
=
FALSE
;
}
}
int
archive_discover
(
handlerton
*
hton
,
THD
*
thd
,
const
char
*
db
,
int
archive_discover
(
handlerton
*
hton
,
THD
*
thd
,
TABLE_SHARE
*
share
)
const
char
*
name
,
uchar
**
frmblob
,
size_t
*
frmlen
)
{
{
DBUG_ENTER
(
"archive_discover"
);
DBUG_ENTER
(
"archive_discover"
);
DBUG_PRINT
(
"archive_discover"
,
(
"db: %s, name: %s"
,
db
,
name
));
DBUG_PRINT
(
"archive_discover"
,
(
"db: '%s' name: '%s'"
,
share
->
db
.
str
,
share
->
table_name
.
str
));
azio_stream
frm_stream
;
azio_stream
frm_stream
;
char
az_file
[
FN_REFLEN
];
char
az_file
[
FN_REFLEN
];
char
*
frm_ptr
;
u
char
*
frm_ptr
;
MY_STAT
file_stat
;
MY_STAT
file_stat
;
build_table_filename
(
az_file
,
sizeof
(
az_file
)
-
1
,
db
,
name
,
ARZ
,
0
);
strxmov
(
az_file
,
share
->
normalized_path
.
str
,
ARZ
,
NullS
);
if
(
!
(
mysql_file_stat
(
/* arch_key_file_data */
0
,
az_file
,
&
file_stat
,
MYF
(
0
))))
if
(
!
(
mysql_file_stat
(
/* arch_key_file_data */
0
,
az_file
,
&
file_stat
,
MYF
(
0
))))
goto
err
;
goto
err
;
...
@@ -295,19 +291,23 @@ int archive_discover(handlerton *hton, THD* thd, const char *db,
...
@@ -295,19 +291,23 @@ int archive_discover(handlerton *hton, THD* thd, const char *db,
}
}
if
(
frm_stream
.
frm_length
==
0
)
if
(
frm_stream
.
frm_length
==
0
)
goto
err
;
DBUG_RETURN
(
HA_ERR_CRASHED_ON_USAGE
)
;
frm_ptr
=
(
char
*
)
my_malloc
(
sizeof
(
char
)
*
frm_stream
.
frm_length
,
MYF
(
0
));
frm_ptr
=
(
u
char
*
)
my_malloc
(
sizeof
(
char
)
*
frm_stream
.
frm_length
,
MYF
(
0
));
azread_frm
(
&
frm_stream
,
frm_ptr
);
azread_frm
(
&
frm_stream
,
frm_ptr
);
azclose
(
&
frm_stream
);
azclose
(
&
frm_stream
);
*
frmlen
=
frm_stream
.
frm_length
;
// don't go through the discovery again
*
frmblob
=
(
uchar
*
)
frm_ptr
;
if
(
writefrm
(
share
->
normalized_path
.
str
,
frm_ptr
,
frm_stream
.
frm_length
))
DBUG_RETURN
(
my_errno
);
share
->
init_from_binary_frm_image
(
thd
,
frm_ptr
);
my_free
(
frm_ptr
);
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
err:
err:
my_errno
=
0
;
DBUG_RETURN
(
HA_ERR_NO_SUCH_TABLE
);
DBUG_RETURN
(
1
);
}
}
/*
/*
...
@@ -650,9 +650,9 @@ int ha_archive::close(void)
...
@@ -650,9 +650,9 @@ int ha_archive::close(void)
int
ha_archive
::
frm_copy
(
azio_stream
*
src
,
azio_stream
*
dst
)
int
ha_archive
::
frm_copy
(
azio_stream
*
src
,
azio_stream
*
dst
)
{
{
int
rc
=
0
;
int
rc
=
0
;
char
*
frm_ptr
;
u
char
*
frm_ptr
;
if
(
!
(
frm_ptr
=
(
char
*
)
my_malloc
(
src
->
frm_length
,
MYF
(
0
))))
if
(
!
(
frm_ptr
=
(
u
char
*
)
my_malloc
(
src
->
frm_length
,
MYF
(
0
))))
return
HA_ERR_OUT_OF_MEM
;
return
HA_ERR_OUT_OF_MEM
;
/* Write file offset is set to the end of the file. */
/* Write file offset is set to the end of the file. */
...
@@ -758,7 +758,7 @@ int ha_archive::create(const char *name, TABLE *table_arg,
...
@@ -758,7 +758,7 @@ int ha_archive::create(const char *name, TABLE *table_arg,
if
(
frm_ptr
)
if
(
frm_ptr
)
{
{
mysql_file_read
(
frm_file
,
frm_ptr
,
(
size_t
)
file_stat
.
st_size
,
MYF
(
0
));
mysql_file_read
(
frm_file
,
frm_ptr
,
(
size_t
)
file_stat
.
st_size
,
MYF
(
0
));
azwrite_frm
(
&
create_stream
,
(
char
*
)
frm_ptr
,
(
size_t
)
file_stat
.
st_size
);
azwrite_frm
(
&
create_stream
,
frm_ptr
,
(
size_t
)
file_stat
.
st_size
);
my_free
(
frm_ptr
);
my_free
(
frm_ptr
);
}
}
}
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment