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
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
eb4fb725
Commit
eb4fb725
authored
Feb 09, 2005
by
bell@sanja.is.com.ua
Browse files
Options
Browse Files
Download
Plain Diff
Merge
parents
2c26ebe3
b196fa2c
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
265 additions
and
79 deletions
+265
-79
mysql-test/r/sp.result
mysql-test/r/sp.result
+13
-0
mysql-test/r/view_query_cache.result
mysql-test/r/view_query_cache.result
+25
-0
mysql-test/t/sp.test
mysql-test/t/sp.test
+21
-0
mysql-test/t/view_query_cache.test
mysql-test/t/view_query_cache.test
+30
-1
sql/sp_head.cc
sql/sp_head.cc
+1
-0
sql/sql_cache.cc
sql/sql_cache.cc
+168
-78
sql/sql_cache.h
sql/sql_cache.h
+4
-0
sql/sql_view.cc
sql/sql_view.cc
+3
-0
No files found.
mysql-test/r/sp.result
View file @
eb4fb725
...
@@ -2381,3 +2381,16 @@ ERROR 42000: Can't find any matching row in the user table
...
@@ -2381,3 +2381,16 @@ ERROR 42000: Can't find any matching row in the user table
drop function bug5278|
drop function bug5278|
drop table t1;
drop table t1;
drop table t2;
drop table t2;
drop procedure if exists p1;
create table t1(id int);
insert into t1 values(1);
create procedure p1()
begin
declare i int;
select max(id)+1 into i from t1;
end
//
call p1()//
call p1()//
drop procedure p1;
drop table t1;
mysql-test/r/view_query_cache.result
View file @
eb4fb725
drop table if exists t1,t2,v1,v2,v3;
drop view if exists t1,t2,v1,v2,v3;
set GLOBAL query_cache_size=1355776;
set GLOBAL query_cache_size=1355776;
flush status;
flush status;
create table t1 (a int, b int);
create table t1 (a int, b int);
...
@@ -98,4 +100,27 @@ Qcache_hits 1
...
@@ -98,4 +100,27 @@ Qcache_hits 1
drop view v1;
drop view v1;
set query_cache_type=default;
set query_cache_type=default;
drop table t1;
drop table t1;
create table t1 (a int);
insert into t1 values (1), (2), (3);
create view v1 as select a from t1 where a > 1;
select * from v1;
a
2
3
alter view v1 as select a from t1 where a > 2;
select * from v1;
a
3
drop view v1;
select * from v1;
ERROR 42S02: Table 'test.v1' doesn't exist
drop table t1;
create table t1 (a int, primary key (a), b int);
create table t2 (a int, primary key (a), b int);
insert into t2 values (1000, 2000);
create view v3 (a,b) as select t1.a as a, t2.a as b from t1, t2;
select * from v3;
a b
drop view v3;
drop table t1, t2;
set GLOBAL query_cache_size=default;
set GLOBAL query_cache_size=default;
mysql-test/t/sp.test
View file @
eb4fb725
...
@@ -2853,3 +2853,24 @@ drop function bug5278|
...
@@ -2853,3 +2853,24 @@ drop function bug5278|
delimiter
;
|
delimiter
;
|
drop
table
t1
;
drop
table
t1
;
drop
table
t2
;
drop
table
t2
;
#
# rolling back temporary Item tree changes in SP
#
--
disable_warnings
drop
procedure
if
exists
p1
;
--
enable_warnings
create
table
t1
(
id
int
);
insert
into
t1
values
(
1
);
delimiter
//;
create
procedure
p1
()
begin
declare
i
int
;
select
max
(
id
)
+
1
into
i
from
t1
;
end
//
call
p1
()
//
call
p1
()
//
delimiter
;
//
drop
procedure
p1
;
drop
table
t1
;
mysql-test/t/view_query_cache.test
View file @
eb4fb725
...
@@ -2,6 +2,11 @@
...
@@ -2,6 +2,11 @@
#
#
# QUERY CACHE options for VIEWs
# QUERY CACHE options for VIEWs
#
#
--
disable_warnings
drop
table
if
exists
t1
,
t2
,
v1
,
v2
,
v3
;
drop
view
if
exists
t1
,
t2
,
v1
,
v2
,
v3
;
--
enable_warnings
set
GLOBAL
query_cache_size
=
1355776
;
set
GLOBAL
query_cache_size
=
1355776
;
flush
status
;
flush
status
;
create
table
t1
(
a
int
,
b
int
);
create
table
t1
(
a
int
,
b
int
);
...
@@ -53,6 +58,30 @@ drop view v1;
...
@@ -53,6 +58,30 @@ drop view v1;
set
query_cache_type
=
default
;
set
query_cache_type
=
default
;
drop
table
t1
;
drop
table
t1
;
set
GLOBAL
query_cache_size
=
default
;
#
# invalidation of view
#
create
table
t1
(
a
int
);
insert
into
t1
values
(
1
),
(
2
),
(
3
);
create
view
v1
as
select
a
from
t1
where
a
>
1
;
select
*
from
v1
;
alter
view
v1
as
select
a
from
t1
where
a
>
2
;
select
*
from
v1
;
drop
view
v1
;
--
error
1146
select
*
from
v1
;
drop
table
t1
;
#
# join view with QC
#
create
table
t1
(
a
int
,
primary
key
(
a
),
b
int
);
create
table
t2
(
a
int
,
primary
key
(
a
),
b
int
);
insert
into
t2
values
(
1000
,
2000
);
create
view
v3
(
a
,
b
)
as
select
t1
.
a
as
a
,
t2
.
a
as
b
from
t1
,
t2
;
select
*
from
v3
;
drop
view
v3
;
drop
table
t1
,
t2
;
set
GLOBAL
query_cache_size
=
default
;
sql/sp_head.cc
View file @
eb4fb725
...
@@ -488,6 +488,7 @@ sp_head::execute(THD *thd)
...
@@ -488,6 +488,7 @@ sp_head::execute(THD *thd)
break
;
break
;
DBUG_PRINT
(
"execute"
,
(
"Instruction %u"
,
ip
));
DBUG_PRINT
(
"execute"
,
(
"Instruction %u"
,
ip
));
ret
=
i
->
execute
(
thd
,
&
ip
);
ret
=
i
->
execute
(
thd
,
&
ip
);
thd
->
rollback_item_tree_changes
();
if
(
i
->
free_list
)
if
(
i
->
free_list
)
cleanup_items
(
i
->
free_list
);
cleanup_items
(
i
->
free_list
);
// Check if an exception has occurred and a handler has been found
// Check if an exception has occurred and a handler has been found
...
...
sql/sql_cache.cc
View file @
eb4fb725
...
@@ -311,7 +311,7 @@ TODO list:
...
@@ -311,7 +311,7 @@ TODO list:
#include "emb_qcache.h"
#include "emb_qcache.h"
#endif
#endif
#if
defined(EXTRA_DEBUG
) && !defined(DBUG_OFF)
#if
!defined(DBUG_OFF
) && !defined(DBUG_OFF)
#define MUTEX_LOCK(M) { DBUG_PRINT("lock", ("mutex lock 0x%lx", (ulong)(M))); \
#define MUTEX_LOCK(M) { DBUG_PRINT("lock", ("mutex lock 0x%lx", (ulong)(M))); \
pthread_mutex_lock(M);}
pthread_mutex_lock(M);}
#define MUTEX_UNLOCK(M) {DBUG_PRINT("lock", ("mutex unlock 0x%lx",\
#define MUTEX_UNLOCK(M) {DBUG_PRINT("lock", ("mutex unlock 0x%lx",\
...
@@ -2103,6 +2103,90 @@ void Query_cache::invalidate_table(Query_cache_block *table_block)
...
@@ -2103,6 +2103,90 @@ void Query_cache::invalidate_table(Query_cache_block *table_block)
}
}
}
}
TABLE_COUNTER_TYPE
Query_cache
::
register_tables_from_list
(
TABLE_LIST
*
tables_used
,
TABLE_COUNTER_TYPE
counter
,
Query_cache_block_table
*
block_table
)
{
TABLE_COUNTER_TYPE
n
;
DBUG_ENTER
(
"Query_cache::register_tables_from_list"
);
for
(
n
=
counter
;
tables_used
;
tables_used
=
tables_used
->
next_global
,
n
++
,
block_table
++
)
{
block_table
->
n
=
n
;
if
(
tables_used
->
view
)
{
char
key
[
MAX_DBKEY_LENGTH
];
uint
key_length
;
DBUG_PRINT
(
"qcache"
,
(
"view %s, db %s"
,
tables_used
->
view_name
.
str
,
tables_used
->
view_db
.
str
));
key_length
=
(
uint
)
(
strmov
(
strmov
(
key
,
tables_used
->
view_db
.
str
)
+
1
,
tables_used
->
view_name
.
str
)
-
key
)
+
1
;
/*
There are not callback function for for VIEWs
*/
if
(
!
insert_table
(
key_length
,
key
,
block_table
,
tables_used
->
view_db
.
length
+
1
,
HA_CACHE_TBL_NONTRANSACT
,
0
,
0
))
DBUG_RETURN
(
0
);
{
TABLE_COUNTER_TYPE
inc
=
register_tables_from_list
(
tables_used
->
ancestor
,
n
+
1
,
block_table
+
1
);
if
(
!
inc
)
DBUG_RETURN
(
0
);
n
+=
inc
;
block_table
+=
inc
;
}
}
else
{
DBUG_PRINT
(
"qcache"
,
(
"table %s, db %s, openinfo at 0x%lx, keylen %u, key at 0x%lx"
,
tables_used
->
table
->
s
->
table_name
,
tables_used
->
table
->
s
->
table_cache_key
,
(
ulong
)
tables_used
->
table
,
tables_used
->
table
->
s
->
key_length
,
(
ulong
)
tables_used
->
table
->
s
->
table_cache_key
));
if
(
!
insert_table
(
tables_used
->
table
->
s
->
key_length
,
tables_used
->
table
->
s
->
table_cache_key
,
block_table
,
tables_used
->
db_length
,
tables_used
->
table
->
file
->
table_cache_type
(),
tables_used
->
callback_func
,
tables_used
->
engine_data
))
DBUG_RETURN
(
0
);
if
(
tables_used
->
table
->
s
->
db_type
==
DB_TYPE_MRG_MYISAM
)
{
ha_myisammrg
*
handler
=
(
ha_myisammrg
*
)
tables_used
->
table
->
file
;
MYRG_INFO
*
file
=
handler
->
myrg_info
();
for
(
MYRG_TABLE
*
table
=
file
->
open_tables
;
table
!=
file
->
end_table
;
table
++
)
{
char
key
[
MAX_DBKEY_LENGTH
];
uint32
db_length
;
uint
key_length
=
filename_2_table_key
(
key
,
table
->
table
->
filename
,
&
db_length
);
(
++
block_table
)
->
n
=
++
n
;
/*
There are not callback function for for MyISAM, and engine data
*/
if
(
!
insert_table
(
key_length
,
key
,
block_table
,
db_length
,
tables_used
->
table
->
file
->
table_cache_type
(),
0
,
0
))
DBUG_RETURN
(
0
);
}
}
}
}
DBUG_RETURN
(
n
-
counter
);
}
/*
/*
Store all used tables
Store all used tables
...
@@ -2124,52 +2208,10 @@ my_bool Query_cache::register_all_tables(Query_cache_block *block,
...
@@ -2124,52 +2208,10 @@ my_bool Query_cache::register_all_tables(Query_cache_block *block,
Query_cache_block_table
*
block_table
=
block
->
table
(
0
);
Query_cache_block_table
*
block_table
=
block
->
table
(
0
);
for
(
n
=
0
;
n
=
register_tables_from_list
(
tables_used
,
0
,
block_table
);
tables_used
;
tables_used
=
tables_used
->
next_global
,
n
++
,
block_table
++
)
{
DBUG_PRINT
(
"qcache"
,
(
"table %s, db %s, openinfo at 0x%lx, keylen %u, key at 0x%lx"
,
tables_used
->
table_name
,
tables_used
->
db
,
(
ulong
)
tables_used
->
table
,
tables_used
->
table
->
s
->
key_length
,
(
ulong
)
tables_used
->
table
->
s
->
table_cache_key
));
block_table
->
n
=
n
;
if
(
!
insert_table
(
tables_used
->
table
->
s
->
key_length
,
tables_used
->
table
->
s
->
table_cache_key
,
block_table
,
tables_used
->
db_length
,
tables_used
->
table
->
file
->
table_cache_type
(),
tables_used
->
callback_func
,
tables_used
->
engine_data
))
break
;
if
(
tables_used
->
table
->
s
->
db_type
==
DB_TYPE_MRG_MYISAM
)
{
ha_myisammrg
*
handler
=
(
ha_myisammrg
*
)
tables_used
->
table
->
file
;
MYRG_INFO
*
file
=
handler
->
myrg_info
();
for
(
MYRG_TABLE
*
table
=
file
->
open_tables
;
table
!=
file
->
end_table
;
table
++
)
{
char
key
[
MAX_DBKEY_LENGTH
];
uint32
db_length
;
uint
key_length
=
filename_2_table_key
(
key
,
table
->
table
->
filename
,
&
db_length
);
(
++
block_table
)
->
n
=
++
n
;
/*
There are not callback function for for MyISAM, and engine data
*/
if
(
!
insert_table
(
key_length
,
key
,
block_table
,
db_length
,
tables_used
->
table
->
file
->
table_cache_type
(),
0
,
0
))
goto
err
;
}
}
}
err:
err:
if
(
tables_used
)
if
(
n
)
{
{
DBUG_PRINT
(
"qcache"
,
(
"failed at table %d"
,
(
int
)
n
));
DBUG_PRINT
(
"qcache"
,
(
"failed at table %d"
,
(
int
)
n
));
/* Unlink the tables we allocated above */
/* Unlink the tables we allocated above */
...
@@ -2178,7 +2220,7 @@ err:
...
@@ -2178,7 +2220,7 @@ err:
tmp
++
)
tmp
++
)
unlink_table
(
tmp
);
unlink_table
(
tmp
);
}
}
return
(
tables_used
==
0
);
return
(
n
);
}
}
/*
/*
...
@@ -2675,6 +2717,80 @@ void Query_cache::double_linked_list_join(Query_cache_block *head_tail,
...
@@ -2675,6 +2717,80 @@ void Query_cache::double_linked_list_join(Query_cache_block *head_tail,
Query
Query
*****************************************************************************/
*****************************************************************************/
/*
Collect information about table types, check that tables are cachable and
count them
SYNOPSIS
process_and_count_tables()
tables_used table list for processing
tables_type pointer to variable for table types collection
RETURN
0 error
>0 number of tables
*/
static
TABLE_COUNTER_TYPE
process_and_count_tables
(
TABLE_LIST
*
tables_used
,
uint8
*
tables_type
)
{
DBUG_ENTER
(
"process_and_count_tables"
);
TABLE_COUNTER_TYPE
table_count
=
0
;
for
(;
tables_used
;
tables_used
=
tables_used
->
next_global
)
{
if
(
tables_used
->
view
)
{
DBUG_PRINT
(
"qcache"
,
(
"view %s, db %s"
,
tables_used
->
view_name
.
str
,
tables_used
->
view_db
.
str
));
table_count
++
;
*
tables_type
|=
HA_CACHE_TBL_NONTRANSACT
;
{
TABLE_COUNTER_TYPE
subcount
;
if
(
!
(
subcount
=
process_and_count_tables
(
tables_used
->
ancestor
,
tables_type
)))
DBUG_RETURN
(
0
);
table_count
+=
subcount
;
}
}
else
{
table_count
++
;
DBUG_PRINT
(
"qcache"
,
(
"table %s, db %s, type %u"
,
tables_used
->
table
->
s
->
table_name
,
tables_used
->
table
->
s
->
table_cache_key
,
tables_used
->
table
->
s
->
db_type
));
*
tables_type
|=
tables_used
->
table
->
file
->
table_cache_type
();
/*
table_alias_charset used here because it depends of
lower_case_table_names variable
*/
if
(
tables_used
->
table
->
s
->
tmp_table
!=
NO_TMP_TABLE
||
(
*
tables_type
&
HA_CACHE_TBL_NOCACHE
)
||
(
tables_used
->
db_length
==
5
&&
my_strnncoll
(
table_alias_charset
,
(
uchar
*
)
tables_used
->
table
->
s
->
table_cache_key
,
6
,
(
uchar
*
)
"mysql"
,
6
)
==
0
))
{
DBUG_PRINT
(
"qcache"
,
(
"select not cacheable: temporary, system or \
other non-cacheable table(s)"
));
DBUG_RETURN
(
0
);
}
if
(
tables_used
->
table
->
s
->
db_type
==
DB_TYPE_MRG_MYISAM
)
{
ha_myisammrg
*
handler
=
(
ha_myisammrg
*
)
tables_used
->
table
->
file
;
MYRG_INFO
*
file
=
handler
->
myrg_info
();
table_count
+=
(
file
->
end_table
-
file
->
open_tables
);
}
}
}
DBUG_RETURN
(
table_count
);
}
/*
/*
If query is cacheable return number tables in query
If query is cacheable return number tables in query
(query without tables are not cached)
(query without tables are not cached)
...
@@ -2686,7 +2802,7 @@ TABLE_COUNTER_TYPE Query_cache::is_cacheable(THD *thd, uint32 query_len,
...
@@ -2686,7 +2802,7 @@ TABLE_COUNTER_TYPE Query_cache::is_cacheable(THD *thd, uint32 query_len,
TABLE_LIST
*
tables_used
,
TABLE_LIST
*
tables_used
,
uint8
*
tables_type
)
uint8
*
tables_type
)
{
{
TABLE_COUNTER_TYPE
table_count
=
0
;
TABLE_COUNTER_TYPE
table_count
;
DBUG_ENTER
(
"Query_cache::is_cacheable"
);
DBUG_ENTER
(
"Query_cache::is_cacheable"
);
if
(
lex
->
sql_command
==
SQLCOM_SELECT
&&
if
(
lex
->
sql_command
==
SQLCOM_SELECT
&&
...
@@ -2700,36 +2816,8 @@ TABLE_COUNTER_TYPE Query_cache::is_cacheable(THD *thd, uint32 query_len,
...
@@ -2700,36 +2816,8 @@ TABLE_COUNTER_TYPE Query_cache::is_cacheable(THD *thd, uint32 query_len,
lex
->
select_lex
.
options
,
lex
->
select_lex
.
options
,
(
int
)
thd
->
variables
.
query_cache_type
));
(
int
)
thd
->
variables
.
query_cache_type
));
for
(;
tables_used
;
tables_used
=
tables_used
->
next_global
)
if
(
!
(
table_count
=
process_and_count_tables
(
tables_used
,
tables_type
)))
{
DBUG_RETURN
(
0
);
table_count
++
;
DBUG_PRINT
(
"qcache"
,
(
"table %s, db %s, type %u"
,
tables_used
->
table_name
,
tables_used
->
db
,
tables_used
->
table
->
s
->
db_type
));
*
tables_type
|=
tables_used
->
table
->
file
->
table_cache_type
();
/*
table_alias_charset used here because it depends of
lower_case_table_names variable
*/
if
(
tables_used
->
table
->
s
->
tmp_table
!=
NO_TMP_TABLE
||
(
*
tables_type
&
HA_CACHE_TBL_NOCACHE
)
||
(
tables_used
->
db_length
==
5
&&
my_strnncoll
(
table_alias_charset
,
(
uchar
*
)
tables_used
->
db
,
6
,
(
uchar
*
)
"mysql"
,
6
)
==
0
))
{
DBUG_PRINT
(
"qcache"
,
(
"select not cacheable: temporary, system or \
other non-cacheable table(s)"
));
DBUG_RETURN
(
0
);
}
if
(
tables_used
->
table
->
s
->
db_type
==
DB_TYPE_MRG_MYISAM
)
{
ha_myisammrg
*
handler
=
(
ha_myisammrg
*
)
tables_used
->
table
->
file
;
MYRG_INFO
*
file
=
handler
->
myrg_info
();
table_count
+=
(
file
->
end_table
-
file
->
open_tables
);
}
}
if
((
thd
->
options
&
(
OPTION_NOT_AUTOCOMMIT
|
OPTION_BEGIN
))
&&
if
((
thd
->
options
&
(
OPTION_NOT_AUTOCOMMIT
|
OPTION_BEGIN
))
&&
((
*
tables_type
)
&
HA_CACHE_TBL_TRANSACT
))
((
*
tables_type
)
&
HA_CACHE_TBL_TRANSACT
))
...
@@ -2769,7 +2857,9 @@ my_bool Query_cache::ask_handler_allowance(THD *thd,
...
@@ -2769,7 +2857,9 @@ my_bool Query_cache::ask_handler_allowance(THD *thd,
for
(;
tables_used
;
tables_used
=
tables_used
->
next_global
)
for
(;
tables_used
;
tables_used
=
tables_used
->
next_global
)
{
{
TABLE
*
table
=
tables_used
->
table
;
TABLE
*
table
;
if
(
!
(
table
=
tables_used
->
table
))
continue
;
handler
*
handler
=
table
->
file
;
handler
*
handler
=
table
->
file
;
if
(
!
handler
->
register_query_cache_table
(
thd
,
table
->
s
->
table_cache_key
,
if
(
!
handler
->
register_query_cache_table
(
thd
,
table
->
s
->
table_cache_key
,
table
->
s
->
key_length
,
table
->
s
->
key_length
,
...
...
sql/sql_cache.h
View file @
eb4fb725
...
@@ -285,6 +285,10 @@ protected:
...
@@ -285,6 +285,10 @@ protected:
void
invalidate_table
(
TABLE
*
table
);
void
invalidate_table
(
TABLE
*
table
);
void
invalidate_table
(
byte
*
key
,
uint32
key_length
);
void
invalidate_table
(
byte
*
key
,
uint32
key_length
);
void
invalidate_table
(
Query_cache_block
*
table_block
);
void
invalidate_table
(
Query_cache_block
*
table_block
);
TABLE_COUNTER_TYPE
register_tables_from_list
(
TABLE_LIST
*
tables_used
,
TABLE_COUNTER_TYPE
counter
,
Query_cache_block_table
*
block_table
);
my_bool
register_all_tables
(
Query_cache_block
*
block
,
my_bool
register_all_tables
(
Query_cache_block
*
block
,
TABLE_LIST
*
tables_used
,
TABLE_LIST
*
tables_used
,
TABLE_COUNTER_TYPE
tables
);
TABLE_COUNTER_TYPE
tables
);
...
...
sql/sql_view.cc
View file @
eb4fb725
...
@@ -305,6 +305,8 @@ bool mysql_create_view(THD *thd,
...
@@ -305,6 +305,8 @@ bool mysql_create_view(THD *thd,
VOID
(
pthread_mutex_lock
(
&
LOCK_open
));
VOID
(
pthread_mutex_lock
(
&
LOCK_open
));
res
=
mysql_register_view
(
thd
,
view
,
mode
);
res
=
mysql_register_view
(
thd
,
view
,
mode
);
VOID
(
pthread_mutex_unlock
(
&
LOCK_open
));
VOID
(
pthread_mutex_unlock
(
&
LOCK_open
));
if
(
view
->
revision
!=
1
)
query_cache_invalidate3
(
thd
,
view
,
0
);
start_waiting_global_read_lock
(
thd
);
start_waiting_global_read_lock
(
thd
);
if
(
res
)
if
(
res
)
goto
err
;
goto
err
;
...
@@ -917,6 +919,7 @@ bool mysql_drop_view(THD *thd, TABLE_LIST *views, enum_drop_mode drop_mode)
...
@@ -917,6 +919,7 @@ bool mysql_drop_view(THD *thd, TABLE_LIST *views, enum_drop_mode drop_mode)
}
}
if
(
my_delete
(
path
,
MYF
(
MY_WME
)))
if
(
my_delete
(
path
,
MYF
(
MY_WME
)))
goto
err
;
goto
err
;
query_cache_invalidate3
(
thd
,
view
,
0
);
VOID
(
pthread_mutex_unlock
(
&
LOCK_open
));
VOID
(
pthread_mutex_unlock
(
&
LOCK_open
));
}
}
send_ok
(
thd
);
send_ok
(
thd
);
...
...
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