Commit de02193b authored by sergefp@mysql.com's avatar sergefp@mysql.com

Added Non-prelocked SP execution: Now a PROCEDURE doesn't enter/leave prelocked mode for

its body, but lets each statement to get/release its own locks. This allows a broader set
of statements to be executed inside PROCEDUREs (but breaks replication)
This patch should fix BUG#8072, BUG#8766, BUG#9563, BUG#11126
parent c1c504ab
drop database if exists testdb;
drop table if exists t1, t2, t3, t4;
drop procedure if exists sp1;
drop procedure if exists sp2;
drop procedure if exists sp3;
drop procedure if exists sp4;
drop function if exists f1;
drop function if exists f2;
drop function if exists f3;
create database testdb;
use testdb//
create procedure sp1 ()
begin
drop table if exists t1;
select 1 as "my-col";
end;
//
select database();
database()
testdb
call sp1();
my-col
1
Warnings:
Note 1051 Unknown table 't1'
select database();
database()
testdb
use test;
select database();
database()
test
call testdb.sp1();
my-col
1
Warnings:
Note 1051 Unknown table 't1'
select database();
database()
test
drop procedure testdb.sp1;
drop database testdb;
create procedure sp1()
begin
create table t1 (a int);
insert into t1 values (10);
end//
create procedure sp2()
begin
create table t2(a int);
insert into t2 values(1);
call sp1();
end//
create function f1() returns int
begin
return (select max(a) from t1);
end//
create procedure sp3()
begin
call sp1();
select 'func', f1();
end//
call sp1();
select 't1',a from t1;
t1 a
t1 10
drop table t1;
call sp2();
select 't1',a from t1;
t1 a
t1 10
select 't2',a from t2;
t2 a
t2 1
drop table t1, t2;
call sp3();
func f1()
func 10
select 't1',a from t1;
t1 a
t1 10
drop table t1;
drop procedure sp1;
drop procedure sp2;
drop procedure sp3;
drop function f1;
create procedure sp1()
begin
create temporary table t2(a int);
insert into t2 select * from t1;
end//
create procedure sp2()
begin
create temporary table t1 (a int);
insert into t1 values(1);
call sp1();
select 't1', a from t1;
select 't2', b from t2;
drop table t1;
drop table t2;
end//
call sp2();
t1 a
t1 1
drop procedure sp1;
drop procedure sp2;
create table t1 (a int);
insert into t1 values(1),(2);
create table t2 as select * from t1;
create table t3 as select * from t1;
create table t4 as select * from t1;
create procedure sp1(a int)
begin
select a;
end //
create function f1() returns int
begin
return (select max(a) from t1);
end //
CALL sp1(f1());
a
2
create procedure sp2(a int)
begin
select * from t3;
select a;
end //
create procedure sp3()
begin
select * from t1;
call sp2(5);
end //
create procedure sp4()
begin
select * from t2;
call sp3();
end //
call sp4();
a
1
1
1
2
a
1
1
2
a
1
1
2
a
5
drop temporary table t1;
drop temporary table t2;
drop procedure sp1;
drop procedure sp2;
drop procedure sp3;
drop procedure sp4;
drop function f1;
drop view if exists v1;
create function f1(ab int) returns int
begin
declare i int;
set i= (select max(a) from t1 where a < ab) ;
return i;
end //
create function f2(ab int) returns int
begin
declare i int;
set i= (select max(a) from t2 where a < ab) ;
return i;
end //
create view v1 as
select t3.a as x, t4.a as y, f2(3) as z
from t3, t4 where t3.a = t4.a //
create procedure sp1()
begin
declare a int;
set a= (select f1(4) + count(*) A from t1, v1);
end //
create function f3() returns int
begin
call sp1();
return 1;
end //
call sp1() //
select f3() //
f3()
1
select f3() //
f3()
1
call sp1() //
drop procedure sp1//
drop function f3//
create procedure sp1()
begin
declare x int;
declare c cursor for select f1(3) + count(*) from v1;
open c;
fetch c into x;
end;//
create function f3() returns int
begin
call sp1();
return 1;
end //
call sp1() //
call sp1() //
select f3() //
f3()
1
call sp1() //
drop table t1,t2,t3;
drop function f1;
drop function f2;
drop function f3;
drop procedure sp1;
use test; use test;
grant usage on *.* to user1@localhost; grant usage on *.* to user1@localhost;
flush privileges; flush privileges;
drop table if exists t1; drop table if exists t1,t2;
drop database if exists db1_secret; drop database if exists db1_secret;
create database db1_secret; create database db1_secret;
create procedure db1_secret.dummy() begin end; create procedure db1_secret.dummy() begin end;
......
...@@ -35,7 +35,7 @@ lock tables t2 write; ...@@ -35,7 +35,7 @@ lock tables t2 write;
show processlist; show processlist;
Id User Host db Command Time State Info Id User Host db Command Time State Info
# root localhost test Sleep # NULL # root localhost test Sleep # NULL
# root localhost test Query # Locked call bug9486() # root localhost test Query # Locked update t1, t2 set val= 1 where id1=id2
# root localhost test Query # NULL show processlist # root localhost test Query # NULL show processlist
unlock tables; unlock tables;
drop procedure bug9486; drop procedure bug9486;
......
use test; use test;
drop table if exists t1; drop table if exists t1,t2,t3,t4;
create table t1 ( create table t1 (
id char(16) not null default '', id char(16) not null default '',
data int not null data int not null
); );
drop table if exists t2;
create table t2 ( create table t2 (
s char(16), s char(16),
i int, i int,
...@@ -3042,32 +3041,6 @@ drop procedure bug11529| ...@@ -3042,32 +3041,6 @@ drop procedure bug11529|
drop procedure if exists bug6063| drop procedure if exists bug6063|
drop procedure if exists bug7088_1| drop procedure if exists bug7088_1|
drop procedure if exists bug7088_2| drop procedure if exists bug7088_2|
create procedure bug6063()
lbel: begin end|
call bug6063()|
show create procedure bug6063|
Procedure sql_mode Create Procedure
bug6063 CREATE PROCEDURE `test`.`bug6063`()
l?bel: begin end
set character set utf8|
create procedure bug7088_1()
label1: begin end label1|
create procedure bug7088_2()
läbel1: begin end|
call bug7088_1()|
call bug7088_2()|
set character set default|
show create procedure bug7088_1|
Procedure sql_mode Create Procedure
bug7088_1 CREATE PROCEDURE `test`.`bug7088_1`()
label1: begin end label1
show create procedure bug7088_2|
Procedure sql_mode Create Procedure
bug7088_2 CREATE PROCEDURE `test`.`bug7088_2`()
lbel1: begin end
drop procedure bug6063|
drop procedure bug7088_1|
drop procedure bug7088_2|
drop procedure if exists bug9565_sub| drop procedure if exists bug9565_sub|
drop procedure if exists bug9565| drop procedure if exists bug9565|
create procedure bug9565_sub() create procedure bug9565_sub()
......
...@@ -581,6 +581,11 @@ ERROR HY000: View 'test.v1' references invalid table(s) or column(s) or function ...@@ -581,6 +581,11 @@ ERROR HY000: View 'test.v1' references invalid table(s) or column(s) or function
drop view v1; drop view v1;
create view v1 (a,a) as select 'a','a'; create view v1 (a,a) as select 'a','a';
ERROR 42S21: Duplicate column name 'a' ERROR 42S21: Duplicate column name 'a'
drop procedure if exists p1;
create procedure p1 () begin declare v int; create view v1 as select v; end;//
call p1();
ERROR HY000: View's SELECT contains a variable or parameter
drop procedure p1;
create table t1 (col1 int,col2 char(22)); create table t1 (col1 int,col2 char(22));
insert into t1 values(5,'Hello, world of views'); insert into t1 values(5,'Hello, world of views');
create view v1 as select * from t1; create view v1 as select * from t1;
......
--disable_warnings
drop database if exists testdb;
drop table if exists t1, t2, t3, t4;
drop procedure if exists sp1;
drop procedure if exists sp2;
drop procedure if exists sp3;
drop procedure if exists sp4;
drop function if exists f1;
drop function if exists f2;
drop function if exists f3;
--enable_warnings
# BUG#8072
create database testdb;
delimiter //;
use testdb//
create procedure sp1 ()
begin
drop table if exists t1;
select 1 as "my-col";
end;
//
delimiter ;//
select database();
call sp1();
select database();
use test;
select database();
call testdb.sp1();
select database();
drop procedure testdb.sp1;
drop database testdb;
# BUG#8766
delimiter //;
create procedure sp1()
begin
create table t1 (a int);
insert into t1 values (10);
end//
create procedure sp2()
begin
create table t2(a int);
insert into t2 values(1);
call sp1();
end//
create function f1() returns int
begin
return (select max(a) from t1);
end//
create procedure sp3()
begin
call sp1();
select 'func', f1();
end//
delimiter ;//
call sp1();
select 't1',a from t1;
drop table t1;
call sp2();
select 't1',a from t1;
select 't2',a from t2;
drop table t1, t2;
call sp3();
select 't1',a from t1;
drop table t1;
drop procedure sp1;
drop procedure sp2;
drop procedure sp3;
drop function f1;
delimiter //;
create procedure sp1()
begin
create temporary table t2(a int);
insert into t2 select * from t1;
end//
create procedure sp2()
begin
create temporary table t1 (a int);
insert into t1 values(1);
call sp1();
select 't1', a from t1;
select 't2', b from t2;
drop table t1;
drop table t2;
end//
delimiter ;//
call sp2();
drop procedure sp1;
drop procedure sp2;
# Miscelaneous tests
create table t1 (a int);
insert into t1 values(1),(2);
create table t2 as select * from t1;
create table t3 as select * from t1;
create table t4 as select * from t1;
delimiter //;
create procedure sp1(a int)
begin
select a;
end //
create function f1() returns int
begin
return (select max(a) from t1);
end //
delimiter ;//
CALL sp1(f1());
#############
delimiter //;
create procedure sp2(a int)
begin
select * from t3;
select a;
end //
create procedure sp3()
begin
select * from t1;
call sp2(5);
end //
create procedure sp4()
begin
select * from t2;
call sp3();
end //
delimiter ;//
call sp4();
drop temporary table t1;
drop temporary table t2;
drop procedure sp1;
drop procedure sp2;
drop procedure sp3;
drop procedure sp4;
drop function f1;
# Test that prelocking state restoration works with cursors
--disable_warnings
drop view if exists v1;
--enable_warnings
delimiter //;
create function f1(ab int) returns int
begin
declare i int;
set i= (select max(a) from t1 where a < ab) ;
return i;
end //
create function f2(ab int) returns int
begin
declare i int;
set i= (select max(a) from t2 where a < ab) ;
return i;
end //
create view v1 as
select t3.a as x, t4.a as y, f2(3) as z
from t3, t4 where t3.a = t4.a //
create procedure sp1()
begin
declare a int;
set a= (select f1(4) + count(*) A from t1, v1);
end //
create function f3() returns int
begin
call sp1();
return 1;
end //
call sp1() //
select f3() //
select f3() //
call sp1() //
---------------
drop procedure sp1//
drop function f3//
create procedure sp1()
begin
declare x int;
declare c cursor for select f1(3) + count(*) from v1;
open c;
fetch c into x;
end;//
create function f3() returns int
begin
call sp1();
return 1;
end //
call sp1() //
call sp1() //
select f3() //
call sp1() //
delimiter ;//
drop table t1,t2,t3;
drop function f1;
drop function f2;
drop function f3;
drop procedure sp1;
...@@ -15,7 +15,7 @@ grant usage on *.* to user1@localhost; ...@@ -15,7 +15,7 @@ grant usage on *.* to user1@localhost;
flush privileges; flush privileges;
--disable_warnings --disable_warnings
drop table if exists t1; drop table if exists t1,t2;
drop database if exists db1_secret; drop database if exists db1_secret;
--enable_warnings --enable_warnings
# Create our secret database # Create our secret database
......
...@@ -22,15 +22,12 @@ use test; ...@@ -22,15 +22,12 @@ use test;
# t3 and up are created and dropped when needed. # t3 and up are created and dropped when needed.
# #
--disable_warnings --disable_warnings
drop table if exists t1; drop table if exists t1,t2,t3,t4;
--enable_warnings --enable_warnings
create table t1 ( create table t1 (
id char(16) not null default '', id char(16) not null default '',
data int not null data int not null
); );
--disable_warnings
drop table if exists t2;
--enable_warnings
create table t2 ( create table t2 (
s char(16), s char(16),
i int, i int,
...@@ -3812,26 +3809,27 @@ drop procedure if exists bug7088_1| ...@@ -3812,26 +3809,27 @@ drop procedure if exists bug7088_1|
drop procedure if exists bug7088_2| drop procedure if exists bug7088_2|
--enable_warnings --enable_warnings
create procedure bug6063() # psergey: temporarily disabled until Bar fixes BUG#11986
lbel: begin end| # create procedure bug6063()
call bug6063()| # lbel: begin end|
# QQ Known bug: this will not show the label correctly. # call bug6063()|
show create procedure bug6063| # # QQ Known bug: this will not show the label correctly.
# show create procedure bug6063|
set character set utf8| #
create procedure bug7088_1() # set character set utf8|
label1: begin end label1| # create procedure bug7088_1()
create procedure bug7088_2() # label1: begin end label1|
läbel1: begin end| # create procedure bug7088_2()
call bug7088_1()| # läbel1: begin end|
call bug7088_2()| # call bug7088_1()|
set character set default| # call bug7088_2()|
show create procedure bug7088_1| # set character set default|
show create procedure bug7088_2| # show create procedure bug7088_1|
# show create procedure bug7088_2|
drop procedure bug6063| #
drop procedure bug7088_1| # drop procedure bug6063|
drop procedure bug7088_2| # drop procedure bug7088_1|
# drop procedure bug7088_2|
# #
# BUG#9565: "Wrong locking in stored procedure if a sub-sequent procedure # BUG#9565: "Wrong locking in stored procedure if a sub-sequent procedure
......
...@@ -490,15 +490,15 @@ create view v1 (a,a) as select 'a','a'; ...@@ -490,15 +490,15 @@ create view v1 (a,a) as select 'a','a';
# #
# SP variables inside view test # SP variables inside view test
# #
# QQ This can't be tested with the new table locking for functions, --disable_warnings
# QQ since views created in an SP can't be used within the same SP drop procedure if exists p1;
# QQ (just as for tables). Instead it fails with error 1146. --enable_warnings
#delimiter //; delimiter //;
#create procedure p1 () begin declare v int; create view v1 as select v; end;// create procedure p1 () begin declare v int; create view v1 as select v; end;//
#delimiter ;// delimiter ;//
#-- error 1351 -- error 1351
#call p1(); call p1();
#drop procedure p1; drop procedure p1;
# #
# updatablity should be transitive # updatablity should be transitive
......
...@@ -598,7 +598,7 @@ int ha_commit_trans(THD *thd, bool all) ...@@ -598,7 +598,7 @@ int ha_commit_trans(THD *thd, bool all)
my_xid xid= thd->transaction.xid.get_my_xid(); my_xid xid= thd->transaction.xid.get_my_xid();
DBUG_ENTER("ha_commit_trans"); DBUG_ENTER("ha_commit_trans");
if (thd->transaction.in_sub_stmt) if (thd->in_sub_stmt)
{ {
/* /*
Since we don't support nested statement transactions in 5.0, Since we don't support nested statement transactions in 5.0,
...@@ -717,7 +717,7 @@ int ha_rollback_trans(THD *thd, bool all) ...@@ -717,7 +717,7 @@ int ha_rollback_trans(THD *thd, bool all)
THD_TRANS *trans=all ? &thd->transaction.all : &thd->transaction.stmt; THD_TRANS *trans=all ? &thd->transaction.all : &thd->transaction.stmt;
bool is_real_trans=all || thd->transaction.all.nht == 0; bool is_real_trans=all || thd->transaction.all.nht == 0;
DBUG_ENTER("ha_rollback_trans"); DBUG_ENTER("ha_rollback_trans");
if (thd->transaction.in_sub_stmt) if (thd->in_sub_stmt)
{ {
/* /*
If we are inside stored function or trigger we should not commit or If we are inside stored function or trigger we should not commit or
......
...@@ -4844,7 +4844,7 @@ Item_func_sp::execute(Item **itp) ...@@ -4844,7 +4844,7 @@ Item_func_sp::execute(Item **itp)
THD *thd= current_thd; THD *thd= current_thd;
ulong old_client_capabilites; ulong old_client_capabilites;
int res= -1; int res= -1;
bool save_in_sub_stmt= thd->transaction.in_sub_stmt; bool save_in_sub_stmt= thd->in_sub_stmt;
my_bool save_no_send_ok; my_bool save_no_send_ok;
#ifndef NO_EMBEDDED_ACCESS_CHECKS #ifndef NO_EMBEDDED_ACCESS_CHECKS
st_sp_security_context save_ctx; st_sp_security_context save_ctx;
...@@ -4882,11 +4882,11 @@ Item_func_sp::execute(Item **itp) ...@@ -4882,11 +4882,11 @@ Item_func_sp::execute(Item **itp)
*/ */
tmp_disable_binlog(thd); /* don't binlog the substatements */ tmp_disable_binlog(thd); /* don't binlog the substatements */
thd->transaction.in_sub_stmt= TRUE; thd->in_sub_stmt= TRUE;
res= m_sp->execute_function(thd, args, arg_count, itp); res= m_sp->execute_function(thd, args, arg_count, itp);
thd->transaction.in_sub_stmt= save_in_sub_stmt; thd->in_sub_stmt= save_in_sub_stmt;
reenable_binlog(thd); reenable_binlog(thd);
if (res && mysql_bin_log.is_open() && if (res && mysql_bin_log.is_open() &&
(m_sp->m_chistics->daccess == SP_CONTAINS_SQL || (m_sp->m_chistics->daccess == SP_CONTAINS_SQL ||
......
...@@ -1175,6 +1175,44 @@ extern "C" byte* sp_sroutine_key(const byte *ptr, uint *plen, my_bool first) ...@@ -1175,6 +1175,44 @@ extern "C" byte* sp_sroutine_key(const byte *ptr, uint *plen, my_bool first)
} }
/*
Check if routines in routines_list require sp_cache_routines_and_add_tables
call.
SYNOPSIS
sp_need_cache_routines()
thd
routines
need_skip_first OUT TRUE - don't do prelocking for the 1st element in
routines list.
FALSE- otherwise
NOTES
This function assumes that for any "CALL proc(...)" statement routines_list
will have 'proc' as first element (it may have several, consider e.g.
"proc(sp_func(...)))". This property is currently guaranted by the parser.
RETURN
TRUE Need to sp_cache_routines_and_add_tables call for this statement.
FALSE Otherwise.
*/
bool sp_need_cache_routines(THD *thd, SQL_LIST *routines_list, bool *need_skip_first)
{
Sroutine_hash_entry *routine;
routine= (Sroutine_hash_entry*)routines_list->first;
*need_skip_first= FALSE;
if (!routine)
return FALSE;
if (routine->key.str[0] != TYPE_ENUM_PROCEDURE)
return TRUE;
*need_skip_first= TRUE;
return TRUE;
}
/* /*
Auxilary function that adds new element to the set of stored routines Auxilary function that adds new element to the set of stored routines
used by statement. used by statement.
...@@ -1312,11 +1350,13 @@ static void sp_update_stmt_used_routines(THD *thd, LEX *lex, HASH *src) ...@@ -1312,11 +1350,13 @@ static void sp_update_stmt_used_routines(THD *thd, LEX *lex, HASH *src)
SYNOPSIS SYNOPSIS
sp_cache_routines_and_add_tables_aux() sp_cache_routines_and_add_tables_aux()
thd - thread context thd - thread context
lex - LEX representing statement lex - LEX representing statement
start - first routine from the list of routines to be cached start - first routine from the list of routines to be cached
(this list defines mentioned sub-set). (this list defines mentioned sub-set).
first_no_prelock - If true, don't add tables or cache routines used by
the body of the first routine (i.e. *start)
will be executed in non-prelocked mode.
NOTE NOTE
If some function is missing this won't be reported here. If some function is missing this won't be reported here.
Instead this fact will be discovered during query execution. Instead this fact will be discovered during query execution.
...@@ -1328,10 +1368,11 @@ static void sp_update_stmt_used_routines(THD *thd, LEX *lex, HASH *src) ...@@ -1328,10 +1368,11 @@ static void sp_update_stmt_used_routines(THD *thd, LEX *lex, HASH *src)
static bool static bool
sp_cache_routines_and_add_tables_aux(THD *thd, LEX *lex, sp_cache_routines_and_add_tables_aux(THD *thd, LEX *lex,
Sroutine_hash_entry *start) Sroutine_hash_entry *start,
bool first_no_prelock)
{ {
bool result= FALSE; bool result= FALSE;
bool first= TRUE;
DBUG_ENTER("sp_cache_routines_and_add_tables_aux"); DBUG_ENTER("sp_cache_routines_and_add_tables_aux");
for (Sroutine_hash_entry *rt= start; rt; rt= rt->next) for (Sroutine_hash_entry *rt= start; rt; rt= rt->next)
...@@ -1367,9 +1408,13 @@ sp_cache_routines_and_add_tables_aux(THD *thd, LEX *lex, ...@@ -1367,9 +1408,13 @@ sp_cache_routines_and_add_tables_aux(THD *thd, LEX *lex,
} }
if (sp) if (sp)
{ {
sp_update_stmt_used_routines(thd, lex, &sp->m_sroutines); if (!(first && first_no_prelock))
result|= sp->add_used_tables_to_table_list(thd, &lex->query_tables_last); {
sp_update_stmt_used_routines(thd, lex, &sp->m_sroutines);
result|= sp->add_used_tables_to_table_list(thd, &lex->query_tables_last);
}
} }
first= FALSE;
} }
DBUG_RETURN(result); DBUG_RETURN(result);
} }
...@@ -1382,20 +1427,22 @@ sp_cache_routines_and_add_tables_aux(THD *thd, LEX *lex, ...@@ -1382,20 +1427,22 @@ sp_cache_routines_and_add_tables_aux(THD *thd, LEX *lex,
SYNOPSIS SYNOPSIS
sp_cache_routines_and_add_tables() sp_cache_routines_and_add_tables()
thd - thread context thd - thread context
lex - LEX representing statement lex - LEX representing statement
first_no_prelock - If true, don't add tables or cache routines used by
the body of the first routine (i.e. *start)
RETURN VALUE RETURN VALUE
TRUE - some tables were added TRUE - some tables were added
FALSE - no tables were added. FALSE - no tables were added.
*/ */
bool bool
sp_cache_routines_and_add_tables(THD *thd, LEX *lex) sp_cache_routines_and_add_tables(THD *thd, LEX *lex, bool first_no_prelock)
{ {
return sp_cache_routines_and_add_tables_aux(thd, lex, return sp_cache_routines_and_add_tables_aux(thd, lex,
(Sroutine_hash_entry *)lex->sroutines_list.first); (Sroutine_hash_entry *)lex->sroutines_list.first,
first_no_prelock);
} }
...@@ -1417,8 +1464,8 @@ sp_cache_routines_and_add_tables_for_view(THD *thd, LEX *lex, LEX *aux_lex) ...@@ -1417,8 +1464,8 @@ sp_cache_routines_and_add_tables_for_view(THD *thd, LEX *lex, LEX *aux_lex)
Sroutine_hash_entry **last_cached_routine_ptr= Sroutine_hash_entry **last_cached_routine_ptr=
(Sroutine_hash_entry **)lex->sroutines_list.next; (Sroutine_hash_entry **)lex->sroutines_list.next;
sp_update_stmt_used_routines(thd, lex, &aux_lex->sroutines); sp_update_stmt_used_routines(thd, lex, &aux_lex->sroutines);
(void)sp_cache_routines_and_add_tables_aux(thd, lex, (void)sp_cache_routines_and_add_tables_aux(thd, lex,
*last_cached_routine_ptr); *last_cached_routine_ptr, FALSE);
} }
...@@ -1453,7 +1500,8 @@ sp_cache_routines_and_add_tables_for_triggers(THD *thd, LEX *lex, ...@@ -1453,7 +1500,8 @@ sp_cache_routines_and_add_tables_for_triggers(THD *thd, LEX *lex,
} }
(void)sp_cache_routines_and_add_tables_aux(thd, lex, (void)sp_cache_routines_and_add_tables_aux(thd, lex,
*last_cached_routine_ptr); *last_cached_routine_ptr,
FALSE);
} }
} }
......
...@@ -79,10 +79,13 @@ sp_show_status_function(THD *thd, const char *wild); ...@@ -79,10 +79,13 @@ sp_show_status_function(THD *thd, const char *wild);
Procedures for pre-caching of stored routines and building table list Procedures for pre-caching of stored routines and building table list
for prelocking. for prelocking.
*/ */
bool sp_need_cache_routines(THD *thd, SQL_LIST *routines_list,
bool *need_skip_first);
void sp_add_used_routine(LEX *lex, Query_arena *arena, void sp_add_used_routine(LEX *lex, Query_arena *arena,
sp_name *rt, char rt_type); sp_name *rt, char rt_type);
void sp_update_sp_used_routines(HASH *dst, HASH *src); void sp_update_sp_used_routines(HASH *dst, HASH *src);
bool sp_cache_routines_and_add_tables(THD *thd, LEX *lex); bool sp_cache_routines_and_add_tables(THD *thd, LEX *lex,
bool first_no_prelock);
void sp_cache_routines_and_add_tables_for_view(THD *thd, LEX *lex, void sp_cache_routines_and_add_tables_for_view(THD *thd, LEX *lex,
LEX *aux_lex); LEX *aux_lex);
void sp_cache_routines_and_add_tables_for_triggers(THD *thd, LEX *lex, void sp_cache_routines_and_add_tables_for_triggers(THD *thd, LEX *lex,
......
...@@ -22,6 +22,12 @@ ...@@ -22,6 +22,12 @@
#pragma interface /* gcc class implementation */ #pragma interface /* gcc class implementation */
#endif #endif
/*
Stored procedures/functions cache. This is used as follows:
* Each thread has its own cache.
* When SP is used it is always in some thread's cache.
*/
class sp_head; class sp_head;
class sp_cache; class sp_cache;
...@@ -31,16 +37,20 @@ void sp_cache_init(); ...@@ -31,16 +37,20 @@ void sp_cache_init();
/* Clear the cache *cp and set *cp to NULL */ /* Clear the cache *cp and set *cp to NULL */
void sp_cache_clear(sp_cache **cp); void sp_cache_clear(sp_cache **cp);
/* Insert an SP to cache. If 'cp' points to NULL, it's set to a new cache */ /* Insert an SP into cache. If 'cp' points to NULL, it's set to a new cache */
void sp_cache_insert(sp_cache **cp, sp_head *sp); void sp_cache_insert(sp_cache **cp, sp_head *sp);
/* Lookup an SP in cache */ /* Lookup an SP in cache */
sp_head *sp_cache_lookup(sp_cache **cp, sp_name *name); sp_head *sp_cache_lookup(sp_cache **cp, sp_name *name);
/* Remove an SP from cache. Returns true if something was removed */ /*
Remove an SP from cache, and also bump the Cversion number so all other
caches are invalidated.
Returns true if something was removed.
*/
bool sp_cache_remove(sp_cache **cp, sp_name *name); bool sp_cache_remove(sp_cache **cp, sp_name *name);
/* Invalidate a cache */ /* Invalidate all existing SP caches by bumping Cversion number. */
void sp_cache_invalidate(); void sp_cache_invalidate();
......
...@@ -879,7 +879,10 @@ sp_head::execute_procedure(THD *thd, List<Item> *args) ...@@ -879,7 +879,10 @@ sp_head::execute_procedure(THD *thd, List<Item> *args)
octx= new sp_rcontext(csize, hmax, cmax); octx= new sp_rcontext(csize, hmax, cmax);
tmp_octx= TRUE; tmp_octx= TRUE;
} }
/* Evaluate SP arguments (i.e. get the values passed as parameters) */
// QQ: Should do type checking? // QQ: Should do type checking?
DBUG_PRINT("info",(" %.*s: eval args", m_name.length, m_name.str));
for (i = 0 ; (it= li++) && i < params ; i++) for (i = 0 ; (it= li++) && i < params ; i++)
{ {
sp_pvar_t *pvar= m_pcont->find_pvar(i); sp_pvar_t *pvar= m_pcont->find_pvar(i);
...@@ -916,6 +919,14 @@ sp_head::execute_procedure(THD *thd, List<Item> *args) ...@@ -916,6 +919,14 @@ sp_head::execute_procedure(THD *thd, List<Item> *args)
} }
} }
/*
Okay, got values for all arguments. Close tables that might be used by
arguments evaluation.
*/
if (!thd->in_sub_stmt)
close_thread_tables(thd, 0, 0, 0);
DBUG_PRINT("info",(" %.*s: eval args done", m_name.length, m_name.str));
// The rest of the frame are local variables which are all IN. // The rest of the frame are local variables which are all IN.
// Default all variables to null (those with default clauses will // Default all variables to null (those with default clauses will
// be set by an set instruction). // be set by an set instruction).
...@@ -1480,8 +1491,37 @@ sp_lex_keeper::reset_lex_and_exec_core(THD *thd, uint *nextp, ...@@ -1480,8 +1491,37 @@ sp_lex_keeper::reset_lex_and_exec_core(THD *thd, uint *nextp,
implemented at the same time as ability not to store LEX for implemented at the same time as ability not to store LEX for
instruction if it is not really used. instruction if it is not really used.
*/ */
reinit_stmt_before_use(thd, m_lex);
bool collect_prelocking_tail= FALSE;
if (thd->prelocked_mode == NON_PRELOCKED)
{
/*
This statement will enter/leave prelocked mode on its own.
Entering prelocked mode changes table list and related members
of LEX, so we'll need to restore them.
*/
if (lex_query_tables_own_last)
{
/*
We've already entered/left prelocked mode with this statement.
Attach the list of tables that need to be prelocked and mark m_lex
as having such list attached.
*/
*lex_query_tables_own_last= prelocking_tables;
m_lex->mark_as_requiring_prelocking(lex_query_tables_own_last);
}
else
{
/*
Let open_tables_calculate list of tables that this statement needs
to have prelocked.
*/
collect_prelocking_tail= TRUE;
}
}
reinit_stmt_before_use(thd, m_lex);
/* /*
If requested check whenever we have access to tables in LEX's table list If requested check whenever we have access to tables in LEX's table list
and open and lock them before executing instructtions core function. and open and lock them before executing instructtions core function.
...@@ -1499,6 +1539,35 @@ sp_lex_keeper::reset_lex_and_exec_core(THD *thd, uint *nextp, ...@@ -1499,6 +1539,35 @@ sp_lex_keeper::reset_lex_and_exec_core(THD *thd, uint *nextp,
thd->proc_info="closing tables"; thd->proc_info="closing tables";
close_thread_tables(thd); close_thread_tables(thd);
if (thd->prelocked_mode == NON_PRELOCKED)
{
if (!lex_query_tables_own_last)
lex_query_tables_own_last= thd->lex->query_tables_own_last;
if (lex_query_tables_own_last)
{
if (collect_prelocking_tail)
{
/*
This is the first time this statement has entered/left prelocked
mode on its own. open_tables() has calculated the set of tables this
statement needs to have prelocked and added them to the end of
m_lex->query_tables(->next_global)*.
Save this "tail" for subsequent calls (and restore original list
below)
*/
lex_query_tables_own_last= m_lex->query_tables_own_last;
prelocking_tables= *lex_query_tables_own_last;
}
/*
The table list now has list of tables that need to be prelocked
when this statement executes, chop it off, and mark this statement
as not requiring prelocking.
*/
*lex_query_tables_own_last= NULL;
m_lex->mark_as_requiring_prelocking(NULL);
}
}
thd->rollback_item_tree_changes(); thd->rollback_item_tree_changes();
/* /*
......
...@@ -282,6 +282,10 @@ class sp_head :private Query_arena ...@@ -282,6 +282,10 @@ class sp_head :private Query_arena
/* /*
Multi-set representing optimized list of tables to be locked by this Multi-set representing optimized list of tables to be locked by this
routine. Does not include tables which are used by invoked routines. routine. Does not include tables which are used by invoked routines.
Note: for prelocking-free SPs this multiset is constructed too.
We do so because the same instance of sp_head may be called both
in prelocked mode and in non-prelocked mode.
*/ */
HASH m_sptabs; HASH m_sptabs;
...@@ -383,7 +387,8 @@ class sp_lex_keeper ...@@ -383,7 +387,8 @@ class sp_lex_keeper
public: public:
sp_lex_keeper(LEX *lex, bool lex_resp) sp_lex_keeper(LEX *lex, bool lex_resp)
: m_lex(lex), m_lex_resp(lex_resp) : m_lex(lex), m_lex_resp(lex_resp),
lex_query_tables_own_last(NULL)
{ {
lex->sp_lex_in_use= TRUE; lex->sp_lex_in_use= TRUE;
} }
...@@ -418,6 +423,25 @@ class sp_lex_keeper ...@@ -418,6 +423,25 @@ class sp_lex_keeper
for LEX deletion. for LEX deletion.
*/ */
bool m_lex_resp; bool m_lex_resp;
/*
Support for being able to execute this statement in two modes:
a) inside prelocked mode set by the calling procedure or its ancestor.
b) outside of prelocked mode, when this statement enters/leaves
prelocked mode itself.
*/
/*
List of additional tables this statement needs to lock when it
enters/leaves prelocked mode on its own.
*/
TABLE_LIST *prelocking_tables;
/*
The value m_lex->query_tables_own_last should be set to this when the
statement enters/leaves prelocked mode on its own.
*/
TABLE_LIST **lex_query_tables_own_last;
}; };
......
...@@ -391,6 +391,8 @@ static void mark_used_tables_as_free_for_reuse(THD *thd, TABLE *table) ...@@ -391,6 +391,8 @@ static void mark_used_tables_as_free_for_reuse(THD *thd, TABLE *table)
LOCK_open LOCK_open
skip_derived Set to 1 (0 = default) if we should not free derived skip_derived Set to 1 (0 = default) if we should not free derived
tables. tables.
stopper When closing tables from thd->open_tables(->next)*,
don't close/remove tables starting from stopper.
IMPLEMENTATION IMPLEMENTATION
Unlocks tables and frees derived tables. Unlocks tables and frees derived tables.
...@@ -474,6 +476,7 @@ void close_thread_tables(THD *thd, bool lock_in_use, bool skip_derived, ...@@ -474,6 +476,7 @@ void close_thread_tables(THD *thd, bool lock_in_use, bool skip_derived,
We are in prelocked mode, so we have to leave it now with doing We are in prelocked mode, so we have to leave it now with doing
implicit UNLOCK TABLES if need. implicit UNLOCK TABLES if need.
*/ */
DBUG_PRINT("info",("thd->prelocked_mode= NON_PRELOCKED"));
thd->prelocked_mode= NON_PRELOCKED; thd->prelocked_mode= NON_PRELOCKED;
if (prelocked_mode == PRELOCKED_UNDER_LOCK_TABLES) if (prelocked_mode == PRELOCKED_UNDER_LOCK_TABLES)
...@@ -1792,6 +1795,7 @@ memory to write 'DELETE FROM `%s`.`%s`' to the binary log",db,name); ...@@ -1792,6 +1795,7 @@ memory to write 'DELETE FROM `%s`.`%s`' to the binary log",db,name);
DBUG_RETURN(1); DBUG_RETURN(1);
} }
/* /*
Open all tables in list Open all tables in list
...@@ -1843,10 +1847,6 @@ int open_tables(THD *thd, TABLE_LIST **start, uint *counter) ...@@ -1843,10 +1847,6 @@ int open_tables(THD *thd, TABLE_LIST **start, uint *counter)
statement for which table list for prelocking is already built, let statement for which table list for prelocking is already built, let
us cache routines and try to build such table list. us cache routines and try to build such table list.
NOTE: If we want queries with functions to work under explicit
LOCK TABLES we have to additionaly lock mysql.proc table in it.
At least until Monty will fix SP loading :)
NOTE: We can't delay prelocking until we will met some sub-statement NOTE: We can't delay prelocking until we will met some sub-statement
which really uses tables, since this will imply that we have to restore which really uses tables, since this will imply that we have to restore
its table list to be able execute it in some other context. its table list to be able execute it in some other context.
...@@ -1860,19 +1860,28 @@ int open_tables(THD *thd, TABLE_LIST **start, uint *counter) ...@@ -1860,19 +1860,28 @@ int open_tables(THD *thd, TABLE_LIST **start, uint *counter)
mode we will have some locked tables, because queries which use only mode we will have some locked tables, because queries which use only
derived/information schema tables and views possible. Thus "counter" derived/information schema tables and views possible. Thus "counter"
may be still zero for prelocked statement... may be still zero for prelocked statement...
NOTE: The above notes may be out of date. Please wait for psergey to
document new prelocked behavior.
*/ */
if (!thd->prelocked_mode && !thd->lex->requires_prelocking() &&
thd->lex->sroutines.records) if (!thd->prelocked_mode && !thd->lex->requires_prelocking())
{ {
TABLE_LIST **save_query_tables_last= thd->lex->query_tables_last; bool first_no_prelocking;
if (sp_need_cache_routines(thd, &thd->lex->sroutines_list,
&first_no_prelocking))
{
TABLE_LIST **save_query_tables_last= thd->lex->query_tables_last;
DBUG_ASSERT(thd->lex->query_tables == *start); DBUG_ASSERT(thd->lex->query_tables == *start);
if (sp_cache_routines_and_add_tables(thd, thd->lex) || if (sp_cache_routines_and_add_tables(thd, thd->lex,
*start) first_no_prelocking) ||
{ *start)
query_tables_last_own= save_query_tables_last; {
*start= thd->lex->query_tables; query_tables_last_own= save_query_tables_last;
*start= thd->lex->query_tables;
}
} }
} }
...@@ -1891,14 +1900,31 @@ int open_tables(THD *thd, TABLE_LIST **start, uint *counter) ...@@ -1891,14 +1900,31 @@ int open_tables(THD *thd, TABLE_LIST **start, uint *counter)
DBUG_RETURN(-1); DBUG_RETURN(-1);
} }
(*counter)++; (*counter)++;
if (!tables->table && if (!tables->table &&
!(tables->table= open_table(thd, tables, &new_frm_mem, &refresh, 0))) !(tables->table= open_table(thd, tables, &new_frm_mem, &refresh, 0)))
{ {
free_root(&new_frm_mem, MYF(MY_KEEP_PREALLOC)); free_root(&new_frm_mem, MYF(MY_KEEP_PREALLOC));
if (tables->view) if (tables->view)
{ {
/* VIEW placeholder */ /* VIEW placeholder */
(*counter)--; (*counter)--;
/*
tables->next_global list consists of two parts:
1) Query tables and underlying tables of views.
2) Tables used by all stored routines that this statement invokes on
execution.
We need to know where the bound between these two parts is. If we've
just opened the last table in part #1, and it added tables after
itself, adjust the boundary pointer accordingly.
*/
if (query_tables_last_own &&
query_tables_last_own == &(tables->next_global) &&
tables->view->query_tables)
query_tables_last_own= tables->view->query_tables_last;
/* /*
Again if needed we have to get cache all routines used by this view Again if needed we have to get cache all routines used by this view
and add tables used by them to table list. and add tables used by them to table list.
...@@ -2323,6 +2349,7 @@ int lock_tables(THD *thd, TABLE_LIST *tables, uint count) ...@@ -2323,6 +2349,7 @@ int lock_tables(THD *thd, TABLE_LIST *tables, uint count)
and was marked as occupied during open_tables() as free for reuse. and was marked as occupied during open_tables() as free for reuse.
*/ */
mark_real_tables_as_free_for_reuse(first_not_own); mark_real_tables_as_free_for_reuse(first_not_own);
DBUG_PRINT("info",("prelocked_mode= PRELOCKED"));
thd->prelocked_mode= PRELOCKED; thd->prelocked_mode= PRELOCKED;
} }
} }
...@@ -2346,6 +2373,7 @@ int lock_tables(THD *thd, TABLE_LIST *tables, uint count) ...@@ -2346,6 +2373,7 @@ int lock_tables(THD *thd, TABLE_LIST *tables, uint count)
if (thd->lex->requires_prelocking()) if (thd->lex->requires_prelocking())
{ {
mark_real_tables_as_free_for_reuse(first_not_own); mark_real_tables_as_free_for_reuse(first_not_own);
DBUG_PRINT("info", ("thd->prelocked_mode= PRELOCKED_UNDER_LOCK_TABLES"));
thd->prelocked_mode= PRELOCKED_UNDER_LOCK_TABLES; thd->prelocked_mode= PRELOCKED_UNDER_LOCK_TABLES;
} }
} }
......
...@@ -178,7 +178,7 @@ THD::THD() ...@@ -178,7 +178,7 @@ THD::THD()
rand_used(0), time_zone_used(0), rand_used(0), time_zone_used(0),
last_insert_id_used(0), insert_id_used(0), clear_next_insert_id(0), last_insert_id_used(0), insert_id_used(0), clear_next_insert_id(0),
in_lock_tables(0), bootstrap(0), derived_tables_processing(FALSE), in_lock_tables(0), bootstrap(0), derived_tables_processing(FALSE),
spcont(NULL) spcont(NULL), in_sub_stmt(FALSE)
{ {
current_arena= this; current_arena= this;
host= user= priv_user= db= ip= 0; host= user= priv_user= db= ip= 0;
......
...@@ -1132,6 +1132,10 @@ class THD :public Statement, ...@@ -1132,6 +1132,10 @@ class THD :public Statement,
thr_lock_type update_lock_default; thr_lock_type update_lock_default;
delayed_insert *di; delayed_insert *di;
my_bool tablespace_op; /* This is TRUE in DISCARD/IMPORT TABLESPACE */ my_bool tablespace_op; /* This is TRUE in DISCARD/IMPORT TABLESPACE */
/* TRUE if we are inside of trigger or stored function. */
bool in_sub_stmt;
/* container for handler's private per-connection data */ /* container for handler's private per-connection data */
void *ha_data[MAX_HA]; void *ha_data[MAX_HA];
struct st_transactions { struct st_transactions {
...@@ -1139,8 +1143,6 @@ class THD :public Statement, ...@@ -1139,8 +1143,6 @@ class THD :public Statement,
THD_TRANS all; // Trans since BEGIN WORK THD_TRANS all; // Trans since BEGIN WORK
THD_TRANS stmt; // Trans for current statement THD_TRANS stmt; // Trans for current statement
bool on; // see ha_enable_transaction() bool on; // see ha_enable_transaction()
/* TRUE if we are inside of trigger or stored function. */
bool in_sub_stmt;
XID xid; // transaction identifier XID xid; // transaction identifier
enum xa_states xa_state; // used by external XA only enum xa_states xa_state; // used by external XA only
/* /*
......
...@@ -2008,6 +2008,7 @@ void st_lex::cleanup_after_one_table_open() ...@@ -2008,6 +2008,7 @@ void st_lex::cleanup_after_one_table_open()
time_zone_tables_used= 0; time_zone_tables_used= 0;
if (sroutines.records) if (sroutines.records)
my_hash_reset(&sroutines); my_hash_reset(&sroutines);
sroutines_list.empty();
} }
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include "sp_head.h" #include "sp_head.h"
#include "sp.h" #include "sp.h"
#include "sp_cache.h"
#ifdef HAVE_OPENSSL #ifdef HAVE_OPENSSL
/* /*
...@@ -124,7 +125,7 @@ static bool end_active_trans(THD *thd) ...@@ -124,7 +125,7 @@ static bool end_active_trans(THD *thd)
{ {
int error=0; int error=0;
DBUG_ENTER("end_active_trans"); DBUG_ENTER("end_active_trans");
if (unlikely(thd->transaction.in_sub_stmt)) if (unlikely(thd->in_sub_stmt))
{ {
my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0)); my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0));
DBUG_RETURN(1); DBUG_RETURN(1);
...@@ -147,11 +148,7 @@ static bool end_active_trans(THD *thd) ...@@ -147,11 +148,7 @@ static bool end_active_trans(THD *thd)
static bool begin_trans(THD *thd) static bool begin_trans(THD *thd)
{ {
int error=0; int error=0;
/* if (unlikely(thd->in_sub_stmt))
QQ: May be it is better to simply prohibit COMMIT and ROLLBACK in
stored routines as SQL2003 suggests?
*/
if (unlikely(thd->transaction.in_sub_stmt))
{ {
my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0)); my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0));
return 1; return 1;
...@@ -1340,11 +1337,7 @@ int end_trans(THD *thd, enum enum_mysql_completiontype completion) ...@@ -1340,11 +1337,7 @@ int end_trans(THD *thd, enum enum_mysql_completiontype completion)
int res= 0; int res= 0;
DBUG_ENTER("end_trans"); DBUG_ENTER("end_trans");
/* if (unlikely(thd->in_sub_stmt))
QQ: May be it is better to simply prohibit COMMIT and ROLLBACK in
stored routines as SQL2003 suggests?
*/
if (unlikely(thd->transaction.in_sub_stmt))
{ {
my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0)); my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0));
DBUG_RETURN(1); DBUG_RETURN(1);
...@@ -4128,9 +4121,8 @@ mysql_execute_command(THD *thd) ...@@ -4128,9 +4121,8 @@ mysql_execute_command(THD *thd)
goto error; goto error;
/* /*
By this moment all needed SPs should be in cache so no need By this moment all needed SPs should be in cache so no need to look
to look into DB. Moreover we may be unable to do it becuase into DB.
we may don't have read lock on mysql.proc
*/ */
if (!(sp= sp_find_procedure(thd, lex->spname, TRUE))) if (!(sp= sp_find_procedure(thd, lex->spname, TRUE)))
{ {
...@@ -4195,7 +4187,7 @@ mysql_execute_command(THD *thd) ...@@ -4195,7 +4187,7 @@ mysql_execute_command(THD *thd)
select_limit= thd->variables.select_limit; select_limit= thd->variables.select_limit;
thd->variables.select_limit= HA_POS_ERROR; thd->variables.select_limit= HA_POS_ERROR;
thd->row_count_func= 0; thd->row_count_func= 0;
tmp_disable_binlog(thd); /* don't binlog the substatements */ tmp_disable_binlog(thd); /* don't binlog the substatements */
res= sp->execute_procedure(thd, &lex->value_list); res= sp->execute_procedure(thd, &lex->value_list);
reenable_binlog(thd); reenable_binlog(thd);
......
...@@ -78,7 +78,7 @@ class Table_triggers_list: public Sql_alloc ...@@ -78,7 +78,7 @@ class Table_triggers_list: public Sql_alloc
if (bodies[event][time_type]) if (bodies[event][time_type])
{ {
bool save_in_sub_stmt= thd->transaction.in_sub_stmt; bool save_in_sub_stmt= thd->in_sub_stmt;
#ifndef EMBEDDED_LIBRARY #ifndef EMBEDDED_LIBRARY
/* Surpress OK packets in case if we will execute statements */ /* Surpress OK packets in case if we will execute statements */
my_bool nsok= thd->net.no_send_ok; my_bool nsok= thd->net.no_send_ok;
...@@ -107,11 +107,11 @@ class Table_triggers_list: public Sql_alloc ...@@ -107,11 +107,11 @@ class Table_triggers_list: public Sql_alloc
does NOT go into binlog. does NOT go into binlog.
*/ */
tmp_disable_binlog(thd); tmp_disable_binlog(thd);
thd->transaction.in_sub_stmt= TRUE; thd->in_sub_stmt= TRUE;
res= bodies[event][time_type]->execute_function(thd, 0, 0, 0); res= bodies[event][time_type]->execute_function(thd, 0, 0, 0);
thd->transaction.in_sub_stmt= save_in_sub_stmt; thd->in_sub_stmt= save_in_sub_stmt;
reenable_binlog(thd); reenable_binlog(thd);
#ifndef EMBEDDED_LIBRARY #ifndef EMBEDDED_LIBRARY
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment