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
eac4388c
Commit
eac4388c
authored
Sep 09, 2005
by
sergefp@mysql.com
Browse files
Options
Browse Files
Download
Plain Diff
Merge spetrunia@bk-internal.mysql.com:/home/bk/mysql-5.0
into mysql.com:/home/psergey/mysql-5.0-bug12943
parents
871ac087
e5b42524
Changes
10
Hide whitespace changes
Inline
Side-by-side
Showing
10 changed files
with
204 additions
and
38 deletions
+204
-38
mysql-test/r/rpl_sp_effects.result
mysql-test/r/rpl_sp_effects.result
+57
-0
mysql-test/r/sp.result
mysql-test/r/sp.result
+13
-0
mysql-test/t/rpl_sp_effects.test
mysql-test/t/rpl_sp_effects.test
+48
-0
mysql-test/t/sp.test
mysql-test/t/sp.test
+13
-19
sql/item_func.cc
sql/item_func.cc
+11
-4
sql/log.cc
sql/log.cc
+7
-0
sql/sp_head.cc
sql/sp_head.cc
+39
-11
sql/sp_head.h
sql/sp_head.h
+3
-2
sql/sql_class.h
sql/sql_class.h
+10
-2
sql/sql_parse.cc
sql/sql_parse.cc
+3
-0
No files found.
mysql-test/r/rpl_sp_effects.result
View file @
eac4388c
...
...
@@ -156,3 +156,60 @@ slave: 6
drop procedure p1;
drop function f1;
drop table t1,t2;
create table t1 (a int);
create procedure p1()
begin
insert into t1 values(@x);
set @x=@x+1;
insert into t1 values(@x);
if (f2()) then
insert into t1 values(1243);
end if;
end//
create function f2() returns int
begin
insert into t1 values(@z);
set @z=@z+1;
insert into t1 values(@z);
return 0;
end//
create function f1() returns int
begin
insert into t1 values(@y);
call p1();
return 0;
end//
set @x=10;
set @y=20;
set @z=100;
select f1();
f1()
0
set @x=30;
call p1();
select 'master', a from t1;
master a
master 20
master 10
master 11
master 100
master 101
master 30
master 31
master 101
master 102
select 'slave', a from t1;
slave a
slave 20
slave 10
slave 11
slave 100
slave 101
slave 30
slave 31
slave 101
slave 102
drop table t1;
drop function f1;
drop function f2;
drop procedure p1;
mysql-test/r/sp.result
View file @
eac4388c
...
...
@@ -3085,6 +3085,19 @@ column_name bug10055(t.column_name)
id id
data data
drop function bug10055|
drop procedure if exists bug12297|
create procedure bug12297(lim int)
begin
set @x = 0;
repeat
insert into t1(id,data)
values('aa', @x);
set @x = @x + 1;
until @x >= lim
end repeat;
end|
call bug12297(10)|
drop procedure bug12297|
drop function if exists f_bug11247|
drop procedure if exists p_bug11247|
create function f_bug11247(param int)
...
...
mysql-test/t/rpl_sp_effects.test
View file @
eac4388c
...
...
@@ -152,4 +152,52 @@ drop procedure p1;
drop
function
f1
;
drop
table
t1
,
t2
;
# BUG#12637: User variables + SPs replication
create
table
t1
(
a
int
);
delimiter
//;
create
procedure
p1
()
begin
insert
into
t1
values
(
@
x
);
set
@
x
=@
x
+
1
;
insert
into
t1
values
(
@
x
);
if
(
f2
())
then
insert
into
t1
values
(
1243
);
end
if
;
end
//
create
function
f2
()
returns
int
begin
insert
into
t1
values
(
@
z
);
set
@
z
=@
z
+
1
;
insert
into
t1
values
(
@
z
);
return
0
;
end
//
create
function
f1
()
returns
int
begin
insert
into
t1
values
(
@
y
);
call
p1
();
return
0
;
end
//
delimiter
;
//
set
@
x
=
10
;
set
@
y
=
20
;
set
@
z
=
100
;
select
f1
();
set
@
x
=
30
;
call
p1
();
select
'master'
,
a
from
t1
;
sync_slave_with_master
;
connection
slave
;
select
'slave'
,
a
from
t1
;
connection
master
;
drop
table
t1
;
drop
function
f1
;
drop
function
f2
;
drop
procedure
p1
;
sync_slave_with_master
;
mysql-test/t/sp.test
View file @
eac4388c
...
...
@@ -3877,29 +3877,23 @@ drop function bug10055|
# consumption by passing large input parameter.
#
#
# Note: the test is currenly disabled because of the
# Bug #12637: SP crashes the server if it has update query with user var
# & binlog is enabled.
#
--
disable_warnings
#
drop procedure if exists bug12297|
drop
procedure
if
exists
bug12297
|
--
enable_warnings
#
create procedure bug12297(lim int)
#
begin
#
set @x = 0;
#
repeat
#
insert into t1(id,data)
#
values('aa', @x);
#
set @x = @x + 1;
#
until @x >= lim
#
end repeat;
#
end|
create
procedure
bug12297
(
lim
int
)
begin
set
@
x
=
0
;
repeat
insert
into
t1
(
id
,
data
)
values
(
'aa'
,
@
x
);
set
@
x
=
@
x
+
1
;
until
@
x
>=
lim
end
repeat
;
end
|
#
call bug12297(10)|
#
drop procedure bug12297|
call
bug12297
(
10
)
|
drop
procedure
bug12297
|
#
# Bug #11247 "Stored procedures: Function calls in long loops leak memory"
...
...
sql/item_func.cc
View file @
eac4388c
...
...
@@ -3884,7 +3884,8 @@ int get_var_with_binlog(THD *thd, enum_sql_command sql_command,
if
(
!
(
var_entry
=
get_variable
(
&
thd
->
user_vars
,
name
,
0
)))
goto
err
;
}
else
if
(
var_entry
->
used_query_id
==
thd
->
query_id
)
else
if
(
var_entry
->
used_query_id
==
thd
->
query_id
||
mysql_bin_log
.
is_query_in_union
(
thd
,
var_entry
->
used_query_id
))
{
/*
If this variable was already stored in user_var_events by this query
...
...
@@ -3901,10 +3902,16 @@ int get_var_with_binlog(THD *thd, enum_sql_command sql_command,
appears:
> set @a:=1;
> insert into t1 values (@a), (@a:=@a+1), (@a:=@a+1);
We have to write to binlog value @a= 1;
We have to write to binlog value @a= 1.
We allocate the user_var_event on user_var_events_alloc pool, not on
the this-statement-execution pool because in SPs user_var_event objects
may need to be valid after current [SP] statement execution pool is
destroyed.
*/
size
=
ALIGN_SIZE
(
sizeof
(
BINLOG_USER_VAR_EVENT
))
+
var_entry
->
length
;
if
(
!
(
user_var_event
=
(
BINLOG_USER_VAR_EVENT
*
)
thd
->
alloc
(
size
)))
size
=
ALIGN_SIZE
(
sizeof
(
BINLOG_USER_VAR_EVENT
))
+
var_entry
->
length
;
if
(
!
(
user_var_event
=
(
BINLOG_USER_VAR_EVENT
*
)
alloc_root
(
thd
->
user_var_events_alloc
,
size
)))
goto
err
;
user_var_event
->
value
=
(
char
*
)
user_var_event
+
...
...
sql/log.cc
View file @
eac4388c
...
...
@@ -1559,6 +1559,7 @@ void MYSQL_LOG::start_union_events(THD *thd)
thd
->
binlog_evt_union
.
do_union
=
TRUE
;
thd
->
binlog_evt_union
.
unioned_events
=
FALSE
;
thd
->
binlog_evt_union
.
unioned_events_trans
=
FALSE
;
thd
->
binlog_evt_union
.
first_query_id
=
thd
->
query_id
;
}
void
MYSQL_LOG
::
stop_union_events
(
THD
*
thd
)
...
...
@@ -1567,6 +1568,12 @@ void MYSQL_LOG::stop_union_events(THD *thd)
thd
->
binlog_evt_union
.
do_union
=
FALSE
;
}
bool
MYSQL_LOG
::
is_query_in_union
(
THD
*
thd
,
query_id_t
query_id_param
)
{
return
(
thd
->
binlog_evt_union
.
do_union
&&
query_id_param
>=
thd
->
binlog_evt_union
.
first_query_id
);
}
/*
Write an event to the binary log
*/
...
...
sql/sp_head.cc
View file @
eac4388c
...
...
@@ -689,10 +689,35 @@ int cmp_splocal_locations(Item_splocal * const *a, Item_splocal * const *b)
* If this function invocation is done from a statement that is written
into the binary log.
* If there were any attempts to write events to the binary log during
function execution.
function execution (grep for start_union_events and stop_union_events)
If the answers are No and Yes, we write the function call into the binary
log as "DO spfunc(<param1value>, <param2value>, ...)"
4. Miscellaneous issues.
4.1 User variables.
When we call mysql_bin_log.write() for an SP statement, thd->user_var_events
must hold set<{var_name, value}> pairs for all user variables used during
the statement execution.
This set is produced by tracking user variable reads during statement
execution.
Fo SPs, this has the following implications:
1) thd->user_var_events may contain events from several SP statements and
needs to be valid after exection of these statements was finished. In
order to achieve that, we
* Allocate user_var_events array elements on appropriate mem_root (grep
for user_var_events_alloc).
* Use is_query_in_union() to determine if user_var_event is created.
2) We need to empty thd->user_var_events after we have wrote a function
call. This is currently done by making
reset_dynamic(&thd->user_var_events);
calls in several different places. (TODO cosider moving this into
mysql_bin_log.write() function)
*/
...
...
@@ -908,6 +933,7 @@ int sp_head::execute(THD *thd)
/* Don't change NOW() in FUNCTION or TRIGGER */
if
(
!
thd
->
in_sub_stmt
)
thd
->
set_time
();
// Make current_time() et al work
/*
We have to set thd->stmt_arena before executing the instruction
to store in the instruction free_list all new items, created
...
...
@@ -915,6 +941,13 @@ int sp_head::execute(THD *thd)
items made during other permanent subquery transformations).
*/
thd
->
stmt_arena
=
i
;
/* will binlog this separately */
if
(
thd
->
prelocked_mode
==
NON_PRELOCKED
)
//TODO: change to event union?
{
thd
->
user_var_events_alloc
=
thd
->
mem_root
;
}
ret
=
i
->
execute
(
thd
,
&
ip
);
/*
...
...
@@ -929,15 +962,6 @@ int sp_head::execute(THD *thd)
/* we should cleanup free_list and memroot, used by instruction */
thd
->
free_items
();
/*
FIXME: we must free user var events only if the routine is executed
in non-prelocked mode and statement-by-statement replication is used.
But if we don't free them now, the server crashes because user var
events are allocated in execute_mem_root. This is Bug#12637, and when
it's fixed, please add if (thd->options & OPTION_BIN_LOG) here.
*/
if
(
opt_bin_log
)
reset_dynamic
(
&
thd
->
user_var_events
);
free_root
(
&
execute_mem_root
,
MYF
(
0
));
/*
...
...
@@ -1095,7 +1119,10 @@ sp_head::execute_function(THD *thd, Item **argp, uint argcount, Item **resp)
binlog_save_options
=
thd
->
options
;
need_binlog_call
=
mysql_bin_log
.
is_open
()
&&
(
thd
->
options
&
OPTION_BIN_LOG
);
if
(
need_binlog_call
)
{
reset_dynamic
(
&
thd
->
user_var_events
);
mysql_bin_log
.
start_union_events
(
thd
);
}
thd
->
options
&=
~
OPTION_BIN_LOG
;
ret
=
execute
(
thd
);
...
...
@@ -1129,6 +1156,7 @@ sp_head::execute_function(THD *thd, Item **argp, uint argcount, Item **resp)
"Invoked ROUTINE modified a transactional table but MySQL "
"failed to reflect this change in the binary log"
);
}
reset_dynamic
(
&
thd
->
user_var_events
);
}
if
(
m_type
==
TYPE_ENUM_FUNCTION
&&
ret
==
0
)
...
...
sql/sp_head.h
View file @
eac4388c
...
...
@@ -108,13 +108,14 @@ class sp_head :private Query_arena
MEM_ROOT
main_mem_root
;
public:
/* Possible values of m_flags */
const
static
int
enum
{
HAS_RETURN
=
1
,
// For FUNCTIONs only: is set if has RETURN
IN_SIMPLE_CASE
=
2
,
// Is set if parsing a simple CASE
IN_HANDLER
=
4
,
// Is set if the parser is in a handler body
MULTI_RESULTS
=
8
,
// Is set if a procedure with SELECT(s)
CONTAINS_DYNAMIC_SQL
=
16
,
// Is set if a procedure with PREPARE/EXECUTE
IS_INVOKED
=
32
;
// Is set if this sp_head is being used.
IS_INVOKED
=
32
// Is set if this sp_head is being used.
};
int
m_type
;
// TYPE_ENUM_FUNCTION or TYPE_ENUM_PROCEDURE
uint
m_flags
;
// Boolean attributes of a stored routine
...
...
sql/sql_class.h
View file @
eac4388c
...
...
@@ -313,6 +313,7 @@ public:
void
start_union_events
(
THD
*
thd
);
void
stop_union_events
(
THD
*
thd
);
bool
is_query_in_union
(
THD
*
thd
,
query_id_t
query_id_param
);
/*
v stands for vector
...
...
@@ -1303,8 +1304,9 @@ public:
/* variables.transaction_isolation is reset to this after each commit */
enum_tx_isolation
session_tx_isolation
;
enum_check_fields
count_cuted_fields
;
/* for user variables replication*/
DYNAMIC_ARRAY
user_var_events
;
DYNAMIC_ARRAY
user_var_events
;
/* For user variables replication */
MEM_ROOT
*
user_var_events_alloc
;
/* Allocate above array elements here */
enum
killed_state
{
NOT_KILLED
=
0
,
KILL_BAD_DATA
=
1
,
KILL_CONNECTION
=
ER_SERVER_SHUTDOWN
,
KILL_QUERY
=
ER_QUERY_INTERRUPTED
};
killed_state
volatile
killed
;
...
...
@@ -1366,6 +1368,12 @@ public:
mysql_bin_log.start_union_events() call.
*/
bool
unioned_events_trans
;
/*
'queries' (actually SP statements) that run under inside this binlog
union have thd->query_id >= first_query_id.
*/
query_id_t
first_query_id
;
}
binlog_evt_union
;
THD
();
...
...
sql/sql_parse.cc
View file @
eac4388c
...
...
@@ -5162,7 +5162,10 @@ void mysql_reset_thd_for_next_command(THD *thd)
if
(
!
thd
->
in_sub_stmt
)
{
if
(
opt_bin_log
)
{
reset_dynamic
(
&
thd
->
user_var_events
);
thd
->
user_var_events_alloc
=
thd
->
mem_root
;
}
thd
->
clear_error
();
thd
->
total_warn_count
=
0
;
// Warnings for this query
thd
->
rand_used
=
0
;
...
...
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