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
37d2bf84
Commit
37d2bf84
authored
Feb 24, 2012
by
Jimmy Yang
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix Bug #64432 Port bug fix #54330 from mysql-5.1 to mysql-5.5
parent
61fb45bc
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
62 additions
and
31 deletions
+62
-31
storage/innobase/row/row0merge.c
storage/innobase/row/row0merge.c
+62
-31
No files found.
storage/innobase/row/row0merge.c
View file @
37d2bf84
...
@@ -1599,22 +1599,29 @@ row_merge(
...
@@ -1599,22 +1599,29 @@ row_merge(
const
dict_index_t
*
index
,
/*!< in: index being created */
const
dict_index_t
*
index
,
/*!< in: index being created */
merge_file_t
*
file
,
/*!< in/out: file containing
merge_file_t
*
file
,
/*!< in/out: file containing
index entries */
index entries */
ulint
*
half
,
/*!< in/out: half the file */
row_merge_block_t
*
block
,
/*!< in/out: 3 buffers */
row_merge_block_t
*
block
,
/*!< in/out: 3 buffers */
int
*
tmpfd
,
/*!< in/out: temporary file handle */
int
*
tmpfd
,
/*!< in/out: temporary file handle */
struct
TABLE
*
table
)
/*!< in/out: MySQL table, for
struct
TABLE
*
table
,
/*!< in/out: MySQL table, for
reporting erroneous key value
reporting erroneous key value
if applicable */
if applicable */
ulint
*
num_run
,
/*!< in/out: Number of runs remain
to be merged */
ulint
*
run_offset
)
/*!< in/out: Array contains the
first offset number for each merge
run */
{
{
ulint
foffs0
;
/*!< first input offset */
ulint
foffs0
;
/*!< first input offset */
ulint
foffs1
;
/*!< second input offset */
ulint
foffs1
;
/*!< second input offset */
ulint
error
;
/*!< error code */
ulint
error
;
/*!< error code */
merge_file_t
of
;
/*!< output file */
merge_file_t
of
;
/*!< output file */
const
ulint
ihalf
=
*
half
;
const
ulint
ihalf
=
run_offset
[
*
num_run
/
2
]
;
/*!< half the input file */
/*!< half the input file */
ulint
ohalf
;
/*!< half the output file */
ulint
n_run
=
0
;
/*!< num of runs generated from this merge */
UNIV_MEM_ASSERT_W
(
block
[
0
],
3
*
sizeof
block
[
0
]);
UNIV_MEM_ASSERT_W
(
block
[
0
],
3
*
sizeof
block
[
0
]);
ut_ad
(
ihalf
<
file
->
offset
);
ut_ad
(
ihalf
<
file
->
offset
);
of
.
fd
=
*
tmpfd
;
of
.
fd
=
*
tmpfd
;
...
@@ -1630,17 +1637,20 @@ row_merge(
...
@@ -1630,17 +1637,20 @@ row_merge(
#endif
/* POSIX_FADV_SEQUENTIAL */
#endif
/* POSIX_FADV_SEQUENTIAL */
/* Merge blocks to the output file. */
/* Merge blocks to the output file. */
ohalf
=
0
;
foffs0
=
0
;
foffs0
=
0
;
foffs1
=
ihalf
;
foffs1
=
ihalf
;
UNIV_MEM_INVALID
(
run_offset
,
*
num_run
*
sizeof
*
run_offset
);
for
(;
foffs0
<
ihalf
&&
foffs1
<
file
->
offset
;
foffs0
++
,
foffs1
++
)
{
for
(;
foffs0
<
ihalf
&&
foffs1
<
file
->
offset
;
foffs0
++
,
foffs1
++
)
{
ulint
ahalf
;
/*!< arithmetic half the input file */
if
(
UNIV_UNLIKELY
(
trx_is_interrupted
(
trx
)))
{
if
(
UNIV_UNLIKELY
(
trx_is_interrupted
(
trx
)))
{
return
(
DB_INTERRUPTED
);
return
(
DB_INTERRUPTED
);
}
}
/* Remember the offset number for this run */
run_offset
[
n_run
++
]
=
of
.
offset
;
error
=
row_merge_blocks
(
index
,
file
,
block
,
error
=
row_merge_blocks
(
index
,
file
,
block
,
&
foffs0
,
&
foffs1
,
&
of
,
table
);
&
foffs0
,
&
foffs1
,
&
of
,
table
);
...
@@ -1648,21 +1658,6 @@ row_merge(
...
@@ -1648,21 +1658,6 @@ row_merge(
return
(
error
);
return
(
error
);
}
}
/* Record the offset of the output file when
approximately half the output has been generated. In
this way, the next invocation of row_merge() will
spend most of the time in this loop. The initial
estimate is ohalf==0. */
ahalf
=
file
->
offset
/
2
;
ut_ad
(
ohalf
<=
of
.
offset
);
/* Improve the estimate until reaching half the input
file size, or we can not get any closer to it. All
comparands should be non-negative when !(ohalf < ahalf)
because ohalf <= of.offset. */
if
(
ohalf
<
ahalf
||
of
.
offset
-
ahalf
<
ohalf
-
ahalf
)
{
ohalf
=
of
.
offset
;
}
}
}
/* Copy the last blocks, if there are any. */
/* Copy the last blocks, if there are any. */
...
@@ -1672,6 +1667,9 @@ row_merge(
...
@@ -1672,6 +1667,9 @@ row_merge(
return
(
DB_INTERRUPTED
);
return
(
DB_INTERRUPTED
);
}
}
/* Remember the offset number for this run */
run_offset
[
n_run
++
]
=
of
.
offset
;
if
(
!
row_merge_blocks_copy
(
index
,
file
,
block
,
&
foffs0
,
&
of
))
{
if
(
!
row_merge_blocks_copy
(
index
,
file
,
block
,
&
foffs0
,
&
of
))
{
return
(
DB_CORRUPTION
);
return
(
DB_CORRUPTION
);
}
}
...
@@ -1684,6 +1682,9 @@ row_merge(
...
@@ -1684,6 +1682,9 @@ row_merge(
return
(
DB_INTERRUPTED
);
return
(
DB_INTERRUPTED
);
}
}
/* Remember the offset number for this run */
run_offset
[
n_run
++
]
=
of
.
offset
;
if
(
!
row_merge_blocks_copy
(
index
,
file
,
block
,
&
foffs1
,
&
of
))
{
if
(
!
row_merge_blocks_copy
(
index
,
file
,
block
,
&
foffs1
,
&
of
))
{
return
(
DB_CORRUPTION
);
return
(
DB_CORRUPTION
);
}
}
...
@@ -1695,10 +1696,23 @@ row_merge(
...
@@ -1695,10 +1696,23 @@ row_merge(
return
(
DB_CORRUPTION
);
return
(
DB_CORRUPTION
);
}
}
ut_ad
(
n_run
<=
*
num_run
);
*
num_run
=
n_run
;
/* Each run can contain one or more offsets. As merge goes on,
the number of runs (to merge) will reduce until we have one
single run. So the number of runs will always be smaller than
the number of offsets in file */
ut_ad
((
*
num_run
)
<=
file
->
offset
);
/* The number of offsets in output file is always equal or
smaller than input file */
ut_ad
(
of
.
offset
<=
file
->
offset
);
/* Swap file descriptors for the next pass. */
/* Swap file descriptors for the next pass. */
*
tmpfd
=
file
->
fd
;
*
tmpfd
=
file
->
fd
;
*
file
=
of
;
*
file
=
of
;
*
half
=
ohalf
;
UNIV_MEM_INVALID
(
block
[
0
],
3
*
sizeof
block
[
0
]);
UNIV_MEM_INVALID
(
block
[
0
],
3
*
sizeof
block
[
0
]);
...
@@ -1723,27 +1737,44 @@ row_merge_sort(
...
@@ -1723,27 +1737,44 @@ row_merge_sort(
if applicable */
if applicable */
{
{
ulint
half
=
file
->
offset
/
2
;
ulint
half
=
file
->
offset
/
2
;
ulint
num_runs
;
ulint
*
run_offset
;
ulint
error
=
DB_SUCCESS
;
/* Record the number of merge runs we need to perform */
num_runs
=
file
->
offset
;
/* If num_runs are less than 1, nothing to merge */
if
(
num_runs
<=
1
)
{
return
(
error
);
}
/* "run_offset" records each run's first offset number */
run_offset
=
(
ulint
*
)
mem_alloc
(
file
->
offset
*
sizeof
(
ulint
));
/* This tells row_merge() where to start for the first round
of merge. */
run_offset
[
half
]
=
half
;
/* The file should always contain at least one byte (the end
/* The file should always contain at least one byte (the end
of file marker). Thus, it must be at least one block. */
of file marker). Thus, it must be at least one block. */
ut_ad
(
file
->
offset
>
0
);
ut_ad
(
file
->
offset
>
0
);
/* Merge the runs until we have one big run */
do
{
do
{
ulint
error
;
error
=
row_merge
(
trx
,
index
,
file
,
block
,
tmpfd
,
table
,
&
num_runs
,
run_offset
);
error
=
row_merge
(
trx
,
index
,
file
,
&
half
,
UNIV_MEM_ASSERT_RW
(
run_offset
,
num_runs
*
sizeof
*
run_offset
);
block
,
tmpfd
,
table
);
if
(
error
!=
DB_SUCCESS
)
{
if
(
error
!=
DB_SUCCESS
)
{
return
(
error
)
;
break
;
}
}
}
while
(
num_runs
>
1
);
/* half > 0 should hold except when the file consists
mem_free
(
run_offset
);
of one block. No need to merge further then. */
ut_ad
(
half
>
0
||
file
->
offset
==
1
);
}
while
(
half
<
file
->
offset
&&
half
>
0
);
return
(
DB_SUCCESS
);
return
(
error
);
}
}
/*************************************************************//**
/*************************************************************//**
...
...
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