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
559a2436
Commit
559a2436
authored
Jan 16, 2006
by
pem@mysql.com
Browse files
Options
Browse Files
Download
Plain Diff
Merge mysql.com:/extern/mysql/bk/mysql-5.0
into mysql.com:/extern/mysql/work/bug14498/mysql-5.0
parents
e5598d2f
a1b2960a
Changes
6
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
319 additions
and
113 deletions
+319
-113
mysql-test/r/sp.result
mysql-test/r/sp.result
+84
-0
mysql-test/t/sp.test
mysql-test/t/sp.test
+87
-1
sql/sp_head.cc
sql/sp_head.cc
+75
-68
sql/sp_head.h
sql/sp_head.h
+47
-41
sql/sp_pcontext.h
sql/sp_pcontext.h
+9
-0
sql/sql_yacc.yy
sql/sql_yacc.yy
+17
-3
No files found.
mysql-test/r/sp.result
View file @
559a2436
...
@@ -4387,4 +4387,88 @@ id county
...
@@ -4387,4 +4387,88 @@ id county
2 NULL
2 NULL
drop table t3|
drop table t3|
drop procedure bug15441|
drop procedure bug15441|
drop procedure if exists bug14498_1|
drop procedure if exists bug14498_2|
drop procedure if exists bug14498_3|
drop procedure if exists bug14498_4|
drop procedure if exists bug14498_5|
create procedure bug14498_1()
begin
declare continue handler for sqlexception select 'error' as 'Handler';
if v then
select 'yes' as 'v';
else
select 'no' as 'v';
end if;
select 'done' as 'End';
end|
create procedure bug14498_2()
begin
declare continue handler for sqlexception select 'error' as 'Handler';
while v do
select 'yes' as 'v';
end while;
select 'done' as 'End';
end|
create procedure bug14498_3()
begin
declare continue handler for sqlexception select 'error' as 'Handler';
repeat
select 'maybe' as 'v';
until v end repeat;
select 'done' as 'End';
end|
create procedure bug14498_4()
begin
declare continue handler for sqlexception select 'error' as 'Handler';
case v
when 1 then
select '1' as 'v';
when 2 then
select '2' as 'v';
else
select '?' as 'v';
end case;
select 'done' as 'End';
end|
create procedure bug14498_5()
begin
declare continue handler for sqlexception select 'error' as 'Handler';
case
when v = 1 then
select '1' as 'v';
when v = 2 then
select '2' as 'v';
else
select '?' as 'v';
end case;
select 'done' as 'End';
end|
call bug14498_1()|
Handler
error
End
done
call bug14498_2()|
Handler
error
End
done
call bug14498_3()|
v
maybe
Handler
error
End
done
call bug14498_5()|
Handler
error
End
done
drop procedure bug14498_1|
drop procedure bug14498_2|
drop procedure bug14498_3|
drop procedure bug14498_4|
drop procedure bug14498_5|
drop table t1,t2;
drop table t1,t2;
mysql-test/t/sp.test
View file @
559a2436
...
@@ -3824,7 +3824,7 @@ drop procedure if exists bug7088_2|
...
@@ -3824,7 +3824,7 @@ drop procedure if exists bug7088_2|
--disable_parsing # temporarily disabled until Bar fixes BUG#11986
--disable_parsing # temporarily disabled until Bar fixes BUG#11986
create procedure bug6063()
create procedure bug6063()
lbel: begin end|
l
â
bel: begin end|
call bug6063()|
call bug6063()|
# QQ Known bug: this will not show the label correctly.
# QQ Known bug: this will not show the label correctly.
show create procedure bug6063|
show create procedure bug6063|
...
@@ -5154,6 +5154,92 @@ call bug15441('Yale')|
...
@@ -5154,6 +5154,92 @@ call bug15441('Yale')|
drop
table
t3
|
drop
table
t3
|
drop
procedure
bug15441
|
drop
procedure
bug15441
|
#
# BUG#14498: Stored procedures: hang if undefined variable and exception
#
--
disable_warnings
drop
procedure
if
exists
bug14498_1
|
drop
procedure
if
exists
bug14498_2
|
drop
procedure
if
exists
bug14498_3
|
drop
procedure
if
exists
bug14498_4
|
drop
procedure
if
exists
bug14498_5
|
--
enable_warnings
create
procedure
bug14498_1
()
begin
declare
continue
handler
for
sqlexception
select
'error'
as
'Handler'
;
if
v
then
select
'yes'
as
'v'
;
else
select
'no'
as
'v'
;
end
if
;
select
'done'
as
'End'
;
end
|
create
procedure
bug14498_2
()
begin
declare
continue
handler
for
sqlexception
select
'error'
as
'Handler'
;
while
v
do
select
'yes'
as
'v'
;
end
while
;
select
'done'
as
'End'
;
end
|
create
procedure
bug14498_3
()
begin
declare
continue
handler
for
sqlexception
select
'error'
as
'Handler'
;
repeat
select
'maybe'
as
'v'
;
until
v
end
repeat
;
select
'done'
as
'End'
;
end
|
create
procedure
bug14498_4
()
begin
declare
continue
handler
for
sqlexception
select
'error'
as
'Handler'
;
case
v
when
1
then
select
'1'
as
'v'
;
when
2
then
select
'2'
as
'v'
;
else
select
'?'
as
'v'
;
end
case
;
select
'done'
as
'End'
;
end
|
create
procedure
bug14498_5
()
begin
declare
continue
handler
for
sqlexception
select
'error'
as
'Handler'
;
case
when
v
=
1
then
select
'1'
as
'v'
;
when
v
=
2
then
select
'2'
as
'v'
;
else
select
'?'
as
'v'
;
end
case
;
select
'done'
as
'End'
;
end
|
call
bug14498_1
()
|
call
bug14498_2
()
|
call
bug14498_3
()
|
# QQ We can't call this at the moment, due to a known bug (BUG#14643)
#call bug14498_4()|
call
bug14498_5
()
|
drop
procedure
bug14498_1
|
drop
procedure
bug14498_2
|
drop
procedure
bug14498_3
|
drop
procedure
bug14498_4
|
drop
procedure
bug14498_5
|
#
#
# BUG#NNNN: New bug synopsis
# BUG#NNNN: New bug synopsis
#
#
...
...
sql/sp_head.cc
View file @
559a2436
...
@@ -430,7 +430,8 @@ sp_head::operator delete(void *ptr, size_t size)
...
@@ -430,7 +430,8 @@ sp_head::operator delete(void *ptr, size_t size)
sp_head
::
sp_head
()
sp_head
::
sp_head
()
:
Query_arena
(
&
main_mem_root
,
INITIALIZED_FOR_SP
),
:
Query_arena
(
&
main_mem_root
,
INITIALIZED_FOR_SP
),
m_flags
(
0
),
m_recursion_level
(
0
),
m_next_cached_sp
(
0
),
m_flags
(
0
),
m_recursion_level
(
0
),
m_next_cached_sp
(
0
),
m_first_instance
(
this
),
m_first_free_instance
(
this
),
m_last_cached_sp
(
this
)
m_first_instance
(
this
),
m_first_free_instance
(
this
),
m_last_cached_sp
(
this
),
m_cont_level
(
0
)
{
{
m_return_field_def
.
charset
=
NULL
;
m_return_field_def
.
charset
=
NULL
;
...
@@ -439,6 +440,7 @@ sp_head::sp_head()
...
@@ -439,6 +440,7 @@ sp_head::sp_head()
DBUG_ENTER
(
"sp_head::sp_head"
);
DBUG_ENTER
(
"sp_head::sp_head"
);
m_backpatch
.
empty
();
m_backpatch
.
empty
();
m_cont_backpatch
.
empty
();
m_lex
.
empty
();
m_lex
.
empty
();
hash_init
(
&
m_sptabs
,
system_charset_info
,
0
,
0
,
0
,
sp_table_key
,
0
,
0
);
hash_init
(
&
m_sptabs
,
system_charset_info
,
0
,
0
,
0
,
sp_table_key
,
0
,
0
);
hash_init
(
&
m_sroutines
,
system_charset_info
,
0
,
0
,
0
,
sp_sroutine_key
,
0
,
0
);
hash_init
(
&
m_sroutines
,
system_charset_info
,
0
,
0
,
0
,
sp_sroutine_key
,
0
,
0
);
...
@@ -1735,6 +1737,39 @@ sp_head::fill_field_definition(THD *thd, LEX *lex,
...
@@ -1735,6 +1737,39 @@ sp_head::fill_field_definition(THD *thd, LEX *lex,
}
}
void
sp_head
::
new_cont_backpatch
(
sp_instr_jump_if_not
*
i
)
{
m_cont_level
+=
1
;
if
(
i
)
{
/* Use the cont. destination slot to store the level */
i
->
m_cont_dest
=
m_cont_level
;
(
void
)
m_cont_backpatch
.
push_front
(
i
);
}
}
void
sp_head
::
add_cont_backpatch
(
sp_instr_jump_if_not
*
i
)
{
i
->
m_cont_dest
=
m_cont_level
;
(
void
)
m_cont_backpatch
.
push_front
(
i
);
}
void
sp_head
::
do_cont_backpatch
()
{
uint
dest
=
instructions
();
uint
lev
=
m_cont_level
--
;
sp_instr_jump_if_not
*
i
;
while
((
i
=
m_cont_backpatch
.
head
())
&&
i
->
m_cont_dest
==
lev
)
{
i
->
m_cont_dest
=
dest
;
(
void
)
m_cont_backpatch
.
pop
();
}
}
void
void
sp_head
::
set_info
(
longlong
created
,
longlong
modified
,
sp_head
::
set_info
(
longlong
created
,
longlong
modified
,
st_sp_chistics
*
chistics
,
ulong
sql_mode
)
st_sp_chistics
*
chistics
,
ulong
sql_mode
)
...
@@ -1949,7 +1984,10 @@ sp_head::show_create_function(THD *thd)
...
@@ -1949,7 +1984,10 @@ sp_head::show_create_function(THD *thd)
/*
/*
TODO: what does this do??
Do some minimal optimization of the code:
1) Mark used instructions
1.1) While doing this, shortcut jumps to jump instructions
2) Compact the code, removing unused instructions
*/
*/
void
sp_head
::
optimize
()
void
sp_head
::
optimize
()
...
@@ -1972,7 +2010,7 @@ void sp_head::optimize()
...
@@ -1972,7 +2010,7 @@ void sp_head::optimize()
else
else
{
{
if
(
src
!=
dst
)
if
(
src
!=
dst
)
{
{
// Move the instruction and update prev. jumps
sp_instr
*
ibp
;
sp_instr
*
ibp
;
List_iterator_fast
<
sp_instr
>
li
(
bp
);
List_iterator_fast
<
sp_instr
>
li
(
bp
);
...
@@ -1980,8 +2018,7 @@ void sp_head::optimize()
...
@@ -1980,8 +2018,7 @@ void sp_head::optimize()
while
((
ibp
=
li
++
))
while
((
ibp
=
li
++
))
{
{
sp_instr_jump
*
ji
=
static_cast
<
sp_instr_jump
*>
(
ibp
);
sp_instr_jump
*
ji
=
static_cast
<
sp_instr_jump
*>
(
ibp
);
if
(
ji
->
m_dest
==
src
)
ji
->
set_destination
(
src
,
dst
);
ji
->
m_dest
=
dst
;
}
}
}
}
i
->
opt_move
(
dst
,
&
bp
);
i
->
opt_move
(
dst
,
&
bp
);
...
@@ -2414,67 +2451,6 @@ sp_instr_jump::opt_move(uint dst, List<sp_instr> *bp)
...
@@ -2414,67 +2451,6 @@ sp_instr_jump::opt_move(uint dst, List<sp_instr> *bp)
}
}
/*
sp_instr_jump_if class functions
*/
int
sp_instr_jump_if
::
execute
(
THD
*
thd
,
uint
*
nextp
)
{
DBUG_ENTER
(
"sp_instr_jump_if::execute"
);
DBUG_PRINT
(
"info"
,
(
"destination: %u"
,
m_dest
));
DBUG_RETURN
(
m_lex_keeper
.
reset_lex_and_exec_core
(
thd
,
nextp
,
TRUE
,
this
));
}
int
sp_instr_jump_if
::
exec_core
(
THD
*
thd
,
uint
*
nextp
)
{
Item
*
it
;
int
res
;
it
=
sp_prepare_func_item
(
thd
,
&
m_expr
);
if
(
!
it
)
res
=
-
1
;
else
{
res
=
0
;
if
(
it
->
val_bool
())
*
nextp
=
m_dest
;
else
*
nextp
=
m_ip
+
1
;
}
return
res
;
}
void
sp_instr_jump_if
::
print
(
String
*
str
)
{
/* jump_if dest ... */
if
(
str
->
reserve
(
SP_INSTR_UINT_MAXLEN
+
8
+
32
))
// Add some for the expr. too
return
;
str
->
qs_append
(
STRING_WITH_LEN
(
"jump_if "
));
str
->
qs_append
(
m_dest
);
str
->
qs_append
(
' '
);
m_expr
->
print
(
str
);
}
uint
sp_instr_jump_if
::
opt_mark
(
sp_head
*
sp
)
{
sp_instr
*
i
;
marked
=
1
;
if
((
i
=
sp
->
get_instr
(
m_dest
)))
{
m_dest
=
i
->
opt_shortcut_jump
(
sp
,
this
);
m_optdest
=
sp
->
get_instr
(
m_dest
);
}
sp
->
opt_mark
(
m_dest
);
return
m_ip
+
1
;
}
/*
/*
sp_instr_jump_if_not class functions
sp_instr_jump_if_not class functions
*/
*/
...
@@ -2496,7 +2472,10 @@ sp_instr_jump_if_not::exec_core(THD *thd, uint *nextp)
...
@@ -2496,7 +2472,10 @@ sp_instr_jump_if_not::exec_core(THD *thd, uint *nextp)
it
=
sp_prepare_func_item
(
thd
,
&
m_expr
);
it
=
sp_prepare_func_item
(
thd
,
&
m_expr
);
if
(
!
it
)
if
(
!
it
)
{
res
=
-
1
;
res
=
-
1
;
*
nextp
=
m_cont_dest
;
}
else
else
{
{
res
=
0
;
res
=
0
;
...
@@ -2514,11 +2493,13 @@ void
...
@@ -2514,11 +2493,13 @@ void
sp_instr_jump_if_not
::
print
(
String
*
str
)
sp_instr_jump_if_not
::
print
(
String
*
str
)
{
{
/* jump_if_not dest ... */
/* jump_if_not dest ... */
if
(
str
->
reserve
(
SP_INSTR_UINT_MAXLEN
+
12
+
32
))
// Add some for the expr. too
if
(
str
->
reserve
(
2
*
SP_INSTR_UINT_MAXLEN
+
14
+
32
))
// Add some for the expr. too
return
;
return
;
str
->
qs_append
(
STRING_WITH_LEN
(
"jump_if_not "
));
str
->
qs_append
(
STRING_WITH_LEN
(
"jump_if_not "
));
str
->
qs_append
(
m_dest
);
str
->
qs_append
(
m_dest
);
str
->
qs_append
(
' '
);
str
->
append
(
'('
);
str
->
qs_append
(
m_cont_dest
);
str
->
append
(
") "
);
m_expr
->
print
(
str
);
m_expr
->
print
(
str
);
}
}
...
@@ -2535,9 +2516,35 @@ sp_instr_jump_if_not::opt_mark(sp_head *sp)
...
@@ -2535,9 +2516,35 @@ sp_instr_jump_if_not::opt_mark(sp_head *sp)
m_optdest
=
sp
->
get_instr
(
m_dest
);
m_optdest
=
sp
->
get_instr
(
m_dest
);
}
}
sp
->
opt_mark
(
m_dest
);
sp
->
opt_mark
(
m_dest
);
if
((
i
=
sp
->
get_instr
(
m_cont_dest
)))
{
m_cont_dest
=
i
->
opt_shortcut_jump
(
sp
,
this
);
m_cont_optdest
=
sp
->
get_instr
(
m_cont_dest
);
}
sp
->
opt_mark
(
m_cont_dest
);
return
m_ip
+
1
;
return
m_ip
+
1
;
}
}
void
sp_instr_jump_if_not
::
opt_move
(
uint
dst
,
List
<
sp_instr
>
*
bp
)
{
/*
cont. destinations may point backwards after shortcutting jumps
during the mark phase. If it's still pointing forwards, only
push this for backpatching if sp_instr_jump::opt_move() will not
do it (i.e. if the m_dest points backwards).
*/
if
(
m_cont_dest
>
m_ip
)
{
// Forward
if
(
m_dest
<
m_ip
)
bp
->
push_back
(
this
);
}
else
if
(
m_cont_optdest
)
m_cont_dest
=
m_cont_optdest
->
m_ip
;
// Backward
/* This will take care of m_dest and m_ip */
sp_instr_jump
::
opt_move
(
dst
,
bp
);
}
/*
/*
sp_instr_freturn class functions
sp_instr_freturn class functions
...
...
sql/sp_head.h
View file @
559a2436
...
@@ -41,6 +41,7 @@ sp_get_flags_for_command(LEX *lex);
...
@@ -41,6 +41,7 @@ sp_get_flags_for_command(LEX *lex);
struct
sp_label
;
struct
sp_label
;
class
sp_instr
;
class
sp_instr
;
class
sp_instr_jump_if_not
;
struct
sp_cond_type
;
struct
sp_cond_type
;
struct
sp_pvar
;
struct
sp_pvar
;
...
@@ -266,6 +267,18 @@ class sp_head :private Query_arena
...
@@ -266,6 +267,18 @@ class sp_head :private Query_arena
int
int
check_backpatch
(
THD
*
thd
);
check_backpatch
(
THD
*
thd
);
// Start a new cont. backpatch level. If 'i' is NULL, the level is just incr.
void
new_cont_backpatch
(
sp_instr_jump_if_not
*
i
);
// Add an instruction to the current level
void
add_cont_backpatch
(
sp_instr_jump_if_not
*
i
);
// Backpatch (and pop) the current level to the current position.
void
do_cont_backpatch
();
char
*
name
(
uint
*
lenp
=
0
)
const
char
*
name
(
uint
*
lenp
=
0
)
const
{
{
if
(
lenp
)
if
(
lenp
)
...
@@ -356,6 +369,18 @@ class sp_head :private Query_arena
...
@@ -356,6 +369,18 @@ class sp_head :private Query_arena
sp_instr
*
instr
;
sp_instr
*
instr
;
}
bp_t
;
}
bp_t
;
List
<
bp_t
>
m_backpatch
;
// Instructions needing backpatching
List
<
bp_t
>
m_backpatch
;
// Instructions needing backpatching
/*
We need a special list for backpatching of conditional jump's continue
destination (in the case of a continue handler catching an error in
the test), since it would otherwise interfere with the normal backpatch
mechanism - jump_if_not instructions have two different destination
which are to be patched differently.
Since these occur in a more restricted way (always the same "level" in
the code), we don't need the label.
*/
List
<
sp_instr_jump_if_not
>
m_cont_backpatch
;
uint
m_cont_level
;
// The current cont. backpatch level
/*
/*
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.
...
@@ -669,50 +694,17 @@ class sp_instr_jump : public sp_instr
...
@@ -669,50 +694,17 @@ class sp_instr_jump : public sp_instr
m_dest
=
dest
;
m_dest
=
dest
;
}
}
protected:
virtual
void
set_destination
(
uint
old_dest
,
uint
new_dest
)
sp_instr
*
m_optdest
;
// Used during optimization
};
// class sp_instr_jump : public sp_instr
class
sp_instr_jump_if
:
public
sp_instr_jump
{
sp_instr_jump_if
(
const
sp_instr_jump_if
&
);
/* Prevent use of these */
void
operator
=
(
sp_instr_jump_if
&
);
public:
sp_instr_jump_if
(
uint
ip
,
sp_pcontext
*
ctx
,
Item
*
i
,
LEX
*
lex
)
:
sp_instr_jump
(
ip
,
ctx
),
m_expr
(
i
),
m_lex_keeper
(
lex
,
TRUE
)
{}
sp_instr_jump_if
(
uint
ip
,
sp_pcontext
*
ctx
,
Item
*
i
,
uint
dest
,
LEX
*
lex
)
:
sp_instr_jump
(
ip
,
ctx
,
dest
),
m_expr
(
i
),
m_lex_keeper
(
lex
,
TRUE
)
{}
virtual
~
sp_instr_jump_if
()
{}
virtual
int
execute
(
THD
*
thd
,
uint
*
nextp
);
virtual
int
exec_core
(
THD
*
thd
,
uint
*
nextp
);
virtual
void
print
(
String
*
str
);
virtual
uint
opt_mark
(
sp_head
*
sp
);
virtual
uint
opt_shortcut_jump
(
sp_head
*
sp
,
sp_instr
*
start
)
{
{
return
m_ip
;
if
(
m_dest
==
old_dest
)
m_dest
=
new_dest
;
}
}
pr
ivate
:
pr
otected
:
Item
*
m_expr
;
// The condition
sp_instr
*
m_optdest
;
// Used during optimization
sp_lex_keeper
m_lex_keeper
;
};
// class sp_instr_jump
_if : public sp_instr_jump
};
// class sp_instr_jump
: public sp_instr
class
sp_instr_jump_if_not
:
public
sp_instr_jump
class
sp_instr_jump_if_not
:
public
sp_instr_jump
...
@@ -722,12 +714,16 @@ class sp_instr_jump_if_not : public sp_instr_jump
...
@@ -722,12 +714,16 @@ class sp_instr_jump_if_not : public sp_instr_jump
public:
public:
uint
m_cont_dest
;
// Where continue handlers will go
sp_instr_jump_if_not
(
uint
ip
,
sp_pcontext
*
ctx
,
Item
*
i
,
LEX
*
lex
)
sp_instr_jump_if_not
(
uint
ip
,
sp_pcontext
*
ctx
,
Item
*
i
,
LEX
*
lex
)
:
sp_instr_jump
(
ip
,
ctx
),
m_expr
(
i
),
m_lex_keeper
(
lex
,
TRUE
)
:
sp_instr_jump
(
ip
,
ctx
),
m_cont_dest
(
0
),
m_expr
(
i
),
m_lex_keeper
(
lex
,
TRUE
),
m_cont_optdest
(
0
)
{}
{}
sp_instr_jump_if_not
(
uint
ip
,
sp_pcontext
*
ctx
,
Item
*
i
,
uint
dest
,
LEX
*
lex
)
sp_instr_jump_if_not
(
uint
ip
,
sp_pcontext
*
ctx
,
Item
*
i
,
uint
dest
,
LEX
*
lex
)
:
sp_instr_jump
(
ip
,
ctx
,
dest
),
m_expr
(
i
),
m_lex_keeper
(
lex
,
TRUE
)
:
sp_instr_jump
(
ip
,
ctx
,
dest
),
m_cont_dest
(
0
),
m_expr
(
i
),
m_lex_keeper
(
lex
,
TRUE
),
m_cont_optdest
(
0
)
{}
{}
virtual
~
sp_instr_jump_if_not
()
virtual
~
sp_instr_jump_if_not
()
...
@@ -746,10 +742,20 @@ class sp_instr_jump_if_not : public sp_instr_jump
...
@@ -746,10 +742,20 @@ class sp_instr_jump_if_not : public sp_instr_jump
return
m_ip
;
return
m_ip
;
}
}
virtual
void
opt_move
(
uint
dst
,
List
<
sp_instr
>
*
ibp
);
virtual
void
set_destination
(
uint
old_dest
,
uint
new_dest
)
{
sp_instr_jump
::
set_destination
(
old_dest
,
new_dest
);
if
(
m_cont_dest
==
old_dest
)
m_cont_dest
=
new_dest
;
}
private:
private:
Item
*
m_expr
;
// The condition
Item
*
m_expr
;
// The condition
sp_lex_keeper
m_lex_keeper
;
sp_lex_keeper
m_lex_keeper
;
sp_instr
*
m_cont_optdest
;
// Used during optimization
};
// class sp_instr_jump_if_not : public sp_instr_jump
};
// class sp_instr_jump_if_not : public sp_instr_jump
...
...
sql/sp_pcontext.h
View file @
559a2436
...
@@ -52,6 +52,15 @@ typedef struct sp_pvar
...
@@ -52,6 +52,15 @@ typedef struct sp_pvar
#define SP_LAB_BEGIN 2 // Label at BEGIN
#define SP_LAB_BEGIN 2 // Label at BEGIN
#define SP_LAB_ITER 3 // Label at iteration control
#define SP_LAB_ITER 3 // Label at iteration control
/*
An SQL/PSM label. Can refer to the identifier used with the
"label_name:" construct which may precede some SQL/PSM statements, or
to an implicit implementation-dependent identifier which the parser
inserts before a high-level flow control statement such as
IF/WHILE/REPEAT/LOOP, when such statement is rewritten into
a combination of low-level jump/jump_if instructions and labels.
*/
typedef
struct
sp_label
typedef
struct
sp_label
{
{
char
*
name
;
char
*
name
;
...
...
sql/sql_yacc.yy
View file @
559a2436
...
@@ -1982,14 +1982,21 @@ sp_proc_stmt:
...
@@ -1982,14 +1982,21 @@ sp_proc_stmt:
}
}
sp->restore_lex(YYTHD);
sp->restore_lex(YYTHD);
}
}
| IF sp_if END IF {}
| IF
{ Lex->sphead->new_cont_backpatch(NULL); }
sp_if END IF
{ Lex->sphead->do_cont_backpatch(); }
| CASE_SYM WHEN_SYM
| CASE_SYM WHEN_SYM
{
{
Lex->sphead->m_flags&= ~sp_head::IN_SIMPLE_CASE;
Lex->sphead->m_flags&= ~sp_head::IN_SIMPLE_CASE;
Lex->sphead->new_cont_backpatch(NULL);
}
}
sp_case END CASE_SYM {
}
sp_case END CASE_SYM { Lex->sphead->do_cont_backpatch();
}
| CASE_SYM
| CASE_SYM
{ Lex->sphead->reset_lex(YYTHD); }
{
Lex->sphead->reset_lex(YYTHD);
Lex->sphead->new_cont_backpatch(NULL);
}
expr WHEN_SYM
expr WHEN_SYM
{
{
LEX *lex= Lex;
LEX *lex= Lex;
...
@@ -2013,6 +2020,7 @@ sp_proc_stmt:
...
@@ -2013,6 +2020,7 @@ sp_proc_stmt:
sp_case END CASE_SYM
sp_case END CASE_SYM
{
{
Lex->spcont->pop_case_expr_id();
Lex->spcont->pop_case_expr_id();
Lex->sphead->do_cont_backpatch();
}
}
| sp_labeled_control
| sp_labeled_control
{}
{}
...
@@ -2281,6 +2289,7 @@ sp_if:
...
@@ -2281,6 +2289,7 @@ sp_if:
$2, lex);
$2, lex);
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
sp->add_cont_backpatch(i);
sp->add_instr(i);
sp->add_instr(i);
sp->restore_lex(YYTHD);
sp->restore_lex(YYTHD);
}
}
...
@@ -2339,6 +2348,7 @@ sp_case:
...
@@ -2339,6 +2348,7 @@ sp_case:
i= new sp_instr_jump_if_not(ip, ctx, expr, lex);
i= new sp_instr_jump_if_not(ip, ctx, expr, lex);
}
}
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
sp->push_backpatch(i, ctx->push_label((char *)"", 0));
sp->add_cont_backpatch(i);
sp->add_instr(i);
sp->add_instr(i);
sp->restore_lex(YYTHD);
sp->restore_lex(YYTHD);
}
}
...
@@ -2468,6 +2478,7 @@ sp_unlabeled_control:
...
@@ -2468,6 +2478,7 @@ sp_unlabeled_control:
/* Jumping forward */
/* Jumping forward */
sp->push_backpatch(i, lex->spcont->last_label());
sp->push_backpatch(i, lex->spcont->last_label());
sp->new_cont_backpatch(i);
sp->add_instr(i);
sp->add_instr(i);
sp->restore_lex(YYTHD);
sp->restore_lex(YYTHD);
}
}
...
@@ -2479,6 +2490,7 @@ sp_unlabeled_control:
...
@@ -2479,6 +2490,7 @@ sp_unlabeled_control:
sp_instr_jump *i = new sp_instr_jump(ip, lex->spcont, lab->ip);
sp_instr_jump *i = new sp_instr_jump(ip, lex->spcont, lab->ip);
lex->sphead->add_instr(i);
lex->sphead->add_instr(i);
lex->sphead->do_cont_backpatch();
}
}
| REPEAT_SYM sp_proc_stmts1 UNTIL_SYM
| REPEAT_SYM sp_proc_stmts1 UNTIL_SYM
{ Lex->sphead->reset_lex(YYTHD); }
{ Lex->sphead->reset_lex(YYTHD); }
...
@@ -2492,6 +2504,8 @@ sp_unlabeled_control:
...
@@ -2492,6 +2504,8 @@ sp_unlabeled_control:
lex);
lex);
lex->sphead->add_instr(i);
lex->sphead->add_instr(i);
lex->sphead->restore_lex(YYTHD);
lex->sphead->restore_lex(YYTHD);
/* We can shortcut the cont_backpatch here */
i->m_cont_dest= ip+1;
}
}
;
;
...
...
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