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
62296eab
Commit
62296eab
authored
Dec 04, 2007
by
Bradley C. Kuszmaul
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Log some pma movements. Addresses #27.
git-svn-id:
file:///svn/tokudb@911
c7de825b-a66e-492c-adef-691d508d4ae1
parent
b7933628
Changes
14
Show whitespace changes
Inline
Side-by-side
Showing
14 changed files
with
187 additions
and
70 deletions
+187
-70
newbrt/brt-serialize.c
newbrt/brt-serialize.c
+1
-1
newbrt/brt.c
newbrt/brt.c
+7
-6
newbrt/brttypes.h
newbrt/brttypes.h
+7
-0
newbrt/log-internal.h
newbrt/log-internal.h
+4
-0
newbrt/log.c
newbrt/log.c
+28
-1
newbrt/log.h
newbrt/log.h
+2
-0
newbrt/logformat.c
newbrt/logformat.c
+11
-0
newbrt/pma-internal.h
newbrt/pma-internal.h
+2
-2
newbrt/pma-test.c
newbrt/pma-test.c
+34
-18
newbrt/pma.c
newbrt/pma.c
+60
-31
newbrt/pma.h
newbrt/pma.h
+10
-9
newbrt/recover.c
newbrt/recover.c
+11
-1
newbrt/wbuf.h
newbrt/wbuf.h
+9
-0
src/tests/test_log4.c
src/tests/test_log4.c
+1
-1
No files found.
newbrt/brt-serialize.c
View file @
62296eab
...
...
@@ -390,7 +390,7 @@ int toku_deserialize_brtnode_from (int fd, DISKOFF off, BRTNODE *brtnode, int fl
}
if
(
n_in_buf
>
0
)
{
u_int32_t
actual_sum
=
0
;
r
=
toku_pma_bulk_insert
(
result
->
u
.
l
.
buffer
,
keys
,
vals
,
n_in_buf
,
result
->
rand4fingerprint
,
&
actual_sum
);
r
=
toku_pma_bulk_insert
(
(
TOKUTXN
)
0
,
(
FILENUM
){
0
},
(
DISKOFF
)
0
,
result
->
u
.
l
.
buffer
,
keys
,
vals
,
n_in_buf
,
result
->
rand4fingerprint
,
&
actual_sum
);
if
(
r
!=
0
)
goto
died_21
;
if
(
actual_sum
!=
result
->
local_fingerprint
)
{
//fprintf(stderr, "%s:%d Corrupted checksum stored=%08x rand=%08x actual=%08x height=%d n_keys=%d\n", __FILE__, __LINE__, result->rand4fingerprint, result->local_fingerprint, actual_sum, result->height, n_in_buf);
...
...
newbrt/brt.c
View file @
62296eab
...
...
@@ -393,7 +393,7 @@ static int insert_to_hash_in_nonleaf (BRTNODE node, int childnum, DBT *k, DBT *v
}
static
int
brtleaf_split
(
BRT
t
,
BRTNODE
node
,
BRTNODE
*
nodea
,
BRTNODE
*
nodeb
,
DBT
*
splitk
)
{
static
int
brtleaf_split
(
TOKUTXN
txn
,
FILENUM
filenum
,
BRT
t
,
BRTNODE
node
,
BRTNODE
*
nodea
,
BRTNODE
*
nodeb
,
DBT
*
splitk
)
{
BRTNODE
A
,
B
;
assert
(
node
->
height
==
0
);
assert
(
t
->
h
->
nodesize
>=
node
->
nodesize
);
/* otherwise we might be in trouble because the nodesize shrank. */
...
...
@@ -408,9 +408,9 @@ static int brtleaf_split (BRT t, BRTNODE node, BRTNODE *nodea, BRTNODE *nodeb, D
//printf("%s:%d B is at %lld nodesize=%d\n", __FILE__, __LINE__, B->thisnodename, B->nodesize);
assert
(
node
->
height
>
0
||
node
->
u
.
l
.
buffer
!=
0
);
int
r
;
r
=
toku_pma_split
(
node
->
u
.
l
.
buffer
,
&
node
->
u
.
l
.
n_bytes_in_buffer
,
splitk
,
A
->
u
.
l
.
buffer
,
&
A
->
u
.
l
.
n_bytes_in_buffer
,
A
->
rand4fingerprint
,
&
A
->
local_fingerprint
,
B
->
u
.
l
.
buffer
,
&
B
->
u
.
l
.
n_bytes_in_buffer
,
B
->
rand4fingerprint
,
&
B
->
local_fingerprint
);
r
=
toku_pma_split
(
txn
,
filenum
,
node
->
u
.
l
.
buffer
,
&
node
->
u
.
l
.
n_bytes_in_buffer
,
splitk
,
A
->
thisnodename
,
A
->
u
.
l
.
buffer
,
&
A
->
u
.
l
.
n_bytes_in_buffer
,
A
->
rand4fingerprint
,
&
A
->
local_fingerprint
,
A
->
thisnodename
,
B
->
u
.
l
.
buffer
,
&
B
->
u
.
l
.
n_bytes_in_buffer
,
B
->
rand4fingerprint
,
&
B
->
local_fingerprint
);
assert
(
r
==
0
);
assert
(
node
->
height
>
0
||
node
->
u
.
l
.
buffer
!=
0
);
/* Remove it from the cache table, and free its storage. */
...
...
@@ -933,11 +933,12 @@ static int brt_leaf_put_cmd (BRT t, BRTNODE node, BRT_CMD *cmd,
int
debug
,
TOKUTXN
txn
)
{
// toku_pma_verify_fingerprint(node->u.l.buffer, node->rand4fingerprint, node->subtree_fingerprint);
FILENUM
filenum
=
toku_cachefile_filenum
(
t
->
cf
);
if
(
cmd
->
type
==
BRT_INSERT
)
{
DBT
*
k
=
cmd
->
u
.
id
.
key
;
DBT
*
v
=
cmd
->
u
.
id
.
val
;
int
replaced_v_size
;
enum
pma_errors
pma_status
=
toku_pma_insert_or_replace
(
node
->
u
.
l
.
buffer
,
k
,
v
,
&
replaced_v_size
,
txn
,
node
->
thisnodename
,
node
->
rand4fingerprint
,
&
node
->
local_fingerprint
);
enum
pma_errors
pma_status
=
toku_pma_insert_or_replace
(
node
->
u
.
l
.
buffer
,
k
,
v
,
&
replaced_v_size
,
txn
,
filenum
,
node
->
thisnodename
,
node
->
rand4fingerprint
,
&
node
->
local_fingerprint
);
assert
(
pma_status
==
BRT_OK
);
//printf("replaced_v_size=%d\n", replaced_v_size);
if
(
replaced_v_size
>=
0
)
{
...
...
@@ -951,7 +952,7 @@ static int brt_leaf_put_cmd (BRT t, BRTNODE node, BRT_CMD *cmd,
// If it doesn't fit, then split the leaf.
if
(
toku_serialize_brtnode_size
(
node
)
>
node
->
nodesize
)
{
int
r
=
brtleaf_split
(
t
,
node
,
nodea
,
nodeb
,
splitk
);
int
r
=
brtleaf_split
(
t
xn
,
filenum
,
t
,
node
,
nodea
,
nodeb
,
splitk
);
if
(
r
!=
0
)
return
r
;
//printf("%s:%d splitkey=%s\n", __FILE__, __LINE__, (char*)*splitkey);
split_count
++
;
...
...
newbrt/brttypes.h
View file @
62296eab
...
...
@@ -56,4 +56,11 @@ typedef struct loggedbrtheader {
}
u
;
}
LOGGEDBRTHEADER
;
typedef
struct
intpairarray
{
u_int32_t
size
;
struct
intpair
{
u_int32_t
a
,
b
;
}
*
array
;
}
INTPAIRARRAY
;
#endif
newbrt/log-internal.h
View file @
62296eab
...
...
@@ -72,3 +72,7 @@ static inline int toku_logsizeof_LOGGEDBRTHEADER (LOGGEDBRTHEADER bs) {
assert
(
bs
.
n_named_roots
=
0
);
return
4
+
4
+
4
+
8
+
8
+
4
+
8
;
}
static
inline
int
toku_logsizeof_INTPAIRARRAY
(
INTPAIRARRAY
pa
)
{
return
4
+
(
4
+
4
)
*
pa
.
size
;
}
newbrt/log.c
View file @
62296eab
...
...
@@ -469,7 +469,7 @@ int toku_fread_BYTESTRING (FILE *f, BYTESTRING *bs, u_int32_t *crc, u_int32_t *l
return
0
;
}
int
toku_fread_LOGGEDBRTHEADER
(
FILE
*
f
,
LOGGEDBRTHEADER
*
v
,
u_int32_t
*
crc
,
u_int32_t
*
len
)
{
int
toku_fread_LOGGEDBRTHEADER
(
FILE
*
f
,
LOGGEDBRTHEADER
*
v
,
u_int32_t
*
crc
,
u_int32_t
*
len
)
{
int
r
;
r
=
toku_fread_u_int32_t
(
f
,
&
v
->
size
,
crc
,
len
);
if
(
r
!=
0
)
return
r
;
r
=
toku_fread_u_int32_t
(
f
,
&
v
->
flags
,
crc
,
len
);
if
(
r
!=
0
)
return
r
;
...
...
@@ -482,6 +482,19 @@ int toku_fread_LOGGEDBRTHEADER(FILE *f, LOGGEDBRTHEADER *v, u_int32_t *crc, u_in
return
0
;
}
int
toku_fread_INTPAIRARRAY
(
FILE
*
f
,
INTPAIRARRAY
*
v
,
u_int32_t
*
crc
,
u_int32_t
*
len
)
{
int
r
;
u_int32_t
i
;
r
=
toku_fread_u_int32_t
(
f
,
&
v
->
size
,
crc
,
len
);
if
(
r
!=
0
)
return
r
;
MALLOC_N
(
v
->
size
,
v
->
array
);
if
(
v
->
array
!=
0
)
return
errno
;
for
(
i
=
0
;
i
<
v
->
size
;
i
++
)
{
r
=
toku_fread_u_int32_t
(
f
,
&
v
->
array
[
i
].
a
,
crc
,
len
);
if
(
r
!=
0
)
return
r
;
r
=
toku_fread_u_int32_t
(
f
,
&
v
->
array
[
i
].
b
,
crc
,
len
);
if
(
r
!=
0
)
return
r
;
}
return
0
;
}
int
toku_logprint_LSN
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
format
__attribute__
((
__unused__
)))
{
LSN
v
;
int
r
=
toku_fread_LSN
(
inf
,
&
v
,
crc
,
len
);
...
...
@@ -559,6 +572,20 @@ int toku_logprint_LOGGEDBRTHEADER (FILE *outf, FILE *inf, const char *fieldname,
}
int
toku_logprint_INTPAIRARRAY
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
format
__attribute__
((
__unused__
)))
{
INTPAIRARRAY
v
;
u_int32_t
i
;
int
r
=
toku_fread_INTPAIRARRAY
(
inf
,
&
v
,
crc
,
len
);
if
(
r
!=
0
)
return
r
;
fprintf
(
outf
,
" %s={size=%d array={"
,
fieldname
,
v
.
size
);
for
(
i
=
0
;
i
<
v
.
size
;
i
++
)
{
if
(
i
!=
0
)
fprintf
(
outf
,
" "
);
fprintf
(
outf
,
"{%d %d}"
,
v
.
array
[
i
].
a
,
v
.
array
[
i
].
b
);
}
toku_free
(
v
.
array
);
return
0
;
}
int
toku_read_and_print_logmagic
(
FILE
*
f
,
u_int32_t
*
versionp
)
{
{
char
magic
[
8
];
...
...
newbrt/log.h
View file @
62296eab
...
...
@@ -42,6 +42,7 @@ int toku_fread_TXNID (FILE *f, TXNID *txnid, u_int32_t *crc, u_int32_t *len);
// fills in the bs with malloced data.
int
toku_fread_BYTESTRING
(
FILE
*
f
,
BYTESTRING
*
bs
,
u_int32_t
*
crc
,
u_int32_t
*
len
);
int
toku_fread_LOGGEDBRTHEADER
(
FILE
*
f
,
LOGGEDBRTHEADER
*
v
,
u_int32_t
*
crc
,
u_int32_t
*
len
);
int
toku_fread_INTPAIRARRAY
(
FILE
*
f
,
INTPAIRARRAY
*
v
,
u_int32_t
*
crc
,
u_int32_t
*
len
);
int
toku_logprint_LSN
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
);
int
toku_logprint_TXNID
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
);
...
...
@@ -51,6 +52,7 @@ int toku_logprint_DISKOFF (FILE *outf, FILE *inf, const char *fieldname,
int
toku_logprint_u_int8_t
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
);
int
toku_logprint_u_int32_t
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
);
int
toku_logprint_LOGGEDBRTHEADER
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
);
int
toku_logprint_INTPAIRARRAY
(
FILE
*
outf
,
FILE
*
inf
,
const
char
*
fieldname
,
u_int32_t
*
crc
,
u_int32_t
*
len
,
const
char
*
);
int
toku_read_and_print_logmagic
(
FILE
*
f
,
u_int32_t
*
version
);
...
...
newbrt/logformat.c
View file @
62296eab
...
...
@@ -71,6 +71,17 @@ const struct logtype logtypes[] = {
{
"BYTESTRING"
,
"key"
,
0
},
{
"BYTESTRING"
,
"data"
,
0
},
NULLFIELD
}},
{
"resizepma"
,
'R'
,
FA
{{
"TXNID"
,
"txnid"
,
0
},
{
"FILENUM"
,
"filenum"
,
0
},
{
"DISKOFF"
,
"diskoff"
,
0
},
{
"u_int32_t"
,
"oldsize"
,
0
},
{
"u_int32_t"
,
"newsize"
,
0
},
NULLFIELD
}},
{
"pmadistribute"
,
'M'
,
FA
{{
"TXNID"
,
"txnid"
,
0
},
{
"FILENUM"
,
"filenum"
,
0
},
{
"DISKOFF"
,
"diskoff"
,
0
},
{
"INTPAIRARRAY"
,
"fromto"
,
0
},
NULLFIELD
}},
{
0
,
0
,
FA
{
NULLFIELD
}}
};
...
...
newbrt/pma-internal.h
View file @
62296eab
...
...
@@ -36,9 +36,9 @@ struct pma {
int
toku_pmainternal_count_region
(
struct
kv_pair
*
pairs
[],
int
lo
,
int
hi
);
void
toku_pmainternal_calculate_parameters
(
PMA
pma
);
int
toku_pmainternal_smooth_region
(
struct
kv_pair
*
pairs
[],
int
n
,
int
idx
,
int
base
,
PMA
pma
);
int
toku_pmainternal_smooth_region
(
TOKUTXN
,
FILENUM
,
DISKOFF
,
struct
kv_pair
*/
*
pairs
*/
[],
int
/*n*/
,
int
/*idx*/
,
int
/*base*/
,
PMA
/*pma*/
,
int
*/
*
new_idx
*/
);
int
toku_pmainternal_printpairs
(
struct
kv_pair
*
pairs
[],
int
N
);
int
toku_pmainternal_make_space_at
(
PMA
pma
,
int
id
x
);
int
toku_pmainternal_make_space_at
(
TOKUTXN
,
FILENUM
,
DISKOFF
,
PMA
pma
,
int
idx
,
int
*
new_inde
x
);
int
toku_pmainternal_find
(
PMA
pma
,
DBT
*
);
// The DB is so the comparison fuction can be called.
void
toku_print_pma
(
PMA
pma
);
/* useful for debugging, so keep the name short. I.e., not pmainternal_print_pma() */
...
...
newbrt/pma-test.c
View file @
62296eab
...
...
@@ -16,7 +16,7 @@ static DB * const null_db = 0;
static
const
DISKOFF
null_diskoff
=
-
1
;
static
const
FILENUM
null_filenum
=
{
0
};
#define NULL_ARGS null_txn, null_diskoff
#define NULL_ARGS null_txn, null_
filenum, null_
diskoff
void
*
skey
=
0
,
*
sval
=
0
;
...
...
@@ -30,7 +30,7 @@ void local_memory_check_all_free(void) {
static
void
test_make_space_at
(
void
)
{
PMA
pma
;
char
*
key
;
int
r
;
int
r
,
newi
;
struct
kv_pair
*
key_A
,
*
key_B
;
key
=
"A"
;
...
...
@@ -41,17 +41,19 @@ static void test_make_space_at (void) {
r
=
toku_pma_create
(
&
pma
,
toku_default_compare_fun
,
null_db
,
null_filenum
,
0
);
assert
(
r
==
0
);
assert
(
toku_pma_n_entries
(
pma
)
==
0
);
r
=
toku_pmainternal_make_space_at
(
pma
,
2
);
r
=
toku_pmainternal_make_space_at
(
null_txn
,
null_filenum
,
null_diskoff
,
pma
,
2
,
&
newi
);
assert
(
r
==
0
);
assert
(
toku_pma_index_limit
(
pma
)
==
4
);
assert
((
unsigned
long
)
pma
->
pairs
[
toku_pma_index_limit
(
pma
)]
==
0xdeadbeefL
);
toku_print_pma
(
pma
);
pma
->
pairs
[
2
]
=
key_A
;
pma
->
n_pairs_present
++
;
r
=
toku_pmainternal_make_space_at
(
pma
,
2
);
printf
(
"Requested space at 2, got space at %d
\n
"
,
r
);
r
=
toku_pmainternal_make_space_at
(
null_txn
,
null_filenum
,
null_diskoff
,
pma
,
2
,
&
newi
);
assert
(
r
==
0
);
printf
(
"Requested space at 2, got space at %d
\n
"
,
newi
);
toku_print_pma
(
pma
);
assert
(
pma
->
pairs
[
r
]
==
0
);
assert
(
pma
->
pairs
[
newi
]
==
0
);
assert
((
unsigned
long
)
pma
->
pairs
[
toku_pma_index_limit
(
pma
)]
==
0xdeadbeefL
);
assert
(
toku_pma_index_limit
(
pma
)
==
4
);
...
...
@@ -61,8 +63,9 @@ static void test_make_space_at (void) {
pma
->
pairs
[
3
]
=
0
;
pma
->
n_pairs_present
=
2
;
toku_print_pma
(
pma
);
r
=
toku_pmainternal_make_space_at
(
pma
,
0
);
printf
(
"Requested space at 0, got space at %d
\n
"
,
r
);
toku_pmainternal_make_space_at
(
null_txn
,
null_filenum
,
null_diskoff
,
pma
,
0
,
&
newi
);
assert
(
r
==
0
);
printf
(
"Requested space at 0, got space at %d
\n
"
,
newi
);
toku_print_pma
(
pma
);
assert
((
unsigned
long
)
pma
->
pairs
[
toku_pma_index_limit
(
pma
)]
==
0xdeadbeefL
);
// make sure it doesn't go off the end.
...
...
@@ -77,14 +80,15 @@ static void test_make_space_at (void) {
pma
->
pairs
[
7
]
=
0
;
pma
->
n_pairs_present
=
2
;
toku_print_pma
(
pma
);
r
=
toku_pmainternal_make_space_at
(
pma
,
5
);
r
=
toku_pmainternal_make_space_at
(
null_txn
,
null_filenum
,
null_diskoff
,
pma
,
5
,
&
newi
);
assert
(
r
==
0
);
toku_print_pma
(
pma
);
printf
(
"r=%d
\n
"
,
r
);
printf
(
"r=%d
\n
"
,
newi
);
{
int
i
;
for
(
i
=
0
;
i
<
toku_pma_index_limit
(
pma
);
i
++
)
{
if
(
pma
->
pairs
[
i
])
{
assert
(
i
<
r
);
assert
(
i
<
newi
);
pma
->
pairs
[
i
]
=
0
;
}
}
...
...
@@ -170,7 +174,7 @@ static void test_smooth_region_N (int N) {
}
}
toku_pmainternal_printpairs
(
pairs
,
N
);
printf
(
" at %d becomes f"
,
insertat
);
r
=
toku_pmainternal_smooth_region
(
pairs
,
N
,
insertat
,
0
,
0
);
toku_pmainternal_smooth_region
(
null_txn
,
null_filenum
,
null_diskoff
,
pairs
,
N
,
insertat
,
0
,
0
,
&
r
);
toku_pmainternal_printpairs
(
pairs
,
N
);
printf
(
" at %d
\n
"
,
r
);
assert
(
0
<=
r
);
assert
(
r
<
N
);
assert
(
pairs
[
r
]
==
0
);
...
...
@@ -211,7 +215,8 @@ static void test_smooth_region6 (void) {
key
=
"B"
;
pairs
[
1
]
=
kv_pair_malloc
(
key
,
strlen
(
key
)
+
1
,
0
,
0
);
int
r
=
toku_pmainternal_smooth_region
(
pairs
,
N
,
2
,
0
,
0
);
int
r
;
toku_pmainternal_smooth_region
(
null_txn
,
null_filenum
,
null_diskoff
,
pairs
,
N
,
2
,
0
,
0
,
&
r
);
printf
(
"{ "
);
for
(
i
=
0
;
i
<
N
;
i
++
)
printf
(
"%s "
,
pairs
[
i
]
?
pairs
[
i
]
->
key
:
"?"
);
...
...
@@ -859,7 +864,10 @@ static void test_pma_split_n(int n) {
printf
(
"a:"
);
toku_print_pma
(
pmaa
);
error
=
toku_pma_split
(
pmaa
,
0
,
0
,
pmab
,
0
,
brand
,
&
bsum
,
pmac
,
0
,
crand
,
&
csum
);
error
=
toku_pma_split
(
null_txn
,
null_filenum
,
pmaa
,
0
,
0
,
null_diskoff
,
pmab
,
0
,
brand
,
&
bsum
,
null_diskoff
,
pmac
,
0
,
crand
,
&
csum
);
assert
(
error
==
0
);
toku_pma_verify
(
pmaa
);
toku_pma_verify
(
pmab
);
...
...
@@ -925,7 +933,10 @@ static void test_pma_dup_split_n(int n, int dup_mode) {
DBT
splitk
;
error
=
toku_pma_split
(
pmaa
,
0
,
&
splitk
,
pmab
,
0
,
brand
,
&
bsum
,
pmac
,
0
,
crand
,
&
csum
);
error
=
toku_pma_split
(
null_txn
,
null_filenum
,
pmaa
,
0
,
&
splitk
,
(
DISKOFF
)
0
,
pmab
,
0
,
brand
,
&
bsum
,
(
DISKOFF
)
0
,
pmac
,
0
,
crand
,
&
csum
);
assert
(
error
==
0
);
toku_pma_verify
(
pmaa
);
toku_pma_verify
(
pmab
);
...
...
@@ -997,7 +1008,10 @@ static void test_pma_split_varkey(void) {
printf
(
"a:"
);
toku_print_pma
(
pmaa
);
error
=
toku_pma_split
(
pmaa
,
0
,
0
,
pmab
,
0
,
brand
,
&
bsum
,
pmac
,
0
,
crand
,
&
csum
);
error
=
toku_pma_split
(
null_txn
,
null_filenum
,
pmaa
,
0
,
0
,
(
DISKOFF
)
0
,
pmab
,
0
,
brand
,
&
bsum
,
(
DISKOFF
)
0
,
pmac
,
0
,
crand
,
&
csum
);
assert
(
error
==
0
);
toku_pma_verify
(
pmaa
);
toku_pma_verify
(
pmab
);
...
...
@@ -1143,7 +1157,9 @@ static void test_pma_split_cursor(void) {
// print_cursor("cursorc", cursorc);
assert_cursor_val
(
cursorc
,
16
);
error
=
toku_pma_split
(
pmaa
,
0
,
0
,
pmab
,
0
,
brand
,
&
bsum
,
pmac
,
0
,
crand
,
&
csum
);
error
=
toku_pma_split
(
null_txn
,
null_filenum
,
pmaa
,
0
,
0
,
(
DISKOFF
)
0
,
pmab
,
0
,
brand
,
&
bsum
,
(
DISKOFF
)
0
,
pmac
,
0
,
crand
,
&
csum
);
assert
(
error
==
0
);
toku_pma_verify_fingerprint
(
pmab
,
brand
,
bsum
);
...
...
@@ -1253,7 +1269,7 @@ static void test_pma_bulk_insert_n(int n) {
}
/* bulk insert n kv pairs */
error
=
toku_pma_bulk_insert
(
pma
,
keys
,
vals
,
n
,
rand4fingerprint
,
&
sum
);
error
=
toku_pma_bulk_insert
(
null_txn
,
null_filenum
,
(
DISKOFF
)
0
,
pma
,
keys
,
vals
,
n
,
rand4fingerprint
,
&
sum
);
assert
(
error
==
0
);
assert
(
sum
==
expect_fingerprint
);
toku_pma_verify
(
pma
);
...
...
newbrt/pma.c
View file @
62296eab
...
...
@@ -33,7 +33,10 @@ static void __pma_delete_finish(PMA pma, int here);
/*
* resize the pma array to asksize. zero all array entries starting from startx.
*/
static
int
__pma_resize_array
(
PMA
pma
,
int
asksize
,
int
startx
);
static
int
pma_resize_array
(
TOKUTXN
,
FILENUM
,
DISKOFF
,
PMA
pma
,
int
asksize
,
int
startx
);
static
int
old_pma_resize_array
(
PMA
pma
,
int
asksize
,
int
startx
)
{
return
pma_resize_array
((
TOKUTXN
)
0
,
(
FILENUM
){
0
},
(
DISKOFF
)
0
,
pma
,
asksize
,
startx
);
}
/*
* extract pairs from the pma in the window delimited by lo and hi.
...
...
@@ -481,7 +484,8 @@ void toku_print_pma (PMA pma) {
/* Smooth the data, and return the location of the null. */
static
int
distribute_data
(
struct
kv_pair
*
destpairs
[],
int
dcount
,
struct
kv_pair_tag
sourcepairs
[],
int
scount
,
PMA
pma
)
{
struct
kv_pair_tag
sourcepairs
[],
int
scount
,
PMA
pma
)
{
assert
(
scount
<=
dcount
);
if
(
scount
==
0
)
{
return
-
1
;
...
...
@@ -506,7 +510,7 @@ static int distribute_data (struct kv_pair *destpairs[], int dcount,
/* spread the non-empty pairs around. There are n of them. Create an empty slot just before the IDXth
element, and return that slot's index in the smoothed array. */
int
toku_pmainternal_smooth_region
(
struct
kv_pair
*
pairs
[],
int
n
,
int
idx
,
int
base
,
PMA
pma
)
{
int
toku_pmainternal_smooth_region
(
TOKUTXN
txn
,
FILENUM
filenum
,
DISKOFF
diskoff
,
struct
kv_pair
*
pairs
[],
int
n
,
int
idx
,
int
base
,
PMA
pma
,
int
*
new_idx
)
{
int
i
;
int
n_present
=
0
;
for
(
i
=
0
;
i
<
n
;
i
++
)
{
...
...
@@ -521,7 +525,7 @@ int toku_pmainternal_smooth_region (struct kv_pair *pairs[], int n, int idx, int
struct
kv_pair_tag
tmppairs
[
n_present
];
#endif
int
n_saved
=
0
;
int
r
;
int
newidx
;
for
(
i
=
0
;
i
<
n
;
i
++
)
{
if
(
i
==
idx
)
{
...
...
@@ -539,15 +543,29 @@ int toku_pmainternal_smooth_region (struct kv_pair *pairs[], int n, int idx, int
//printf(" temp="); printpairs(tmppairs, n_saved);
assert
(
n_saved
==
n_present
);
/* Now the tricky part. Distribute the data. */
r
=
distribute_data
(
pairs
,
n
,
newidx
=
distribute_data
(
pairs
,
n
,
tmppairs
,
n_saved
,
pma
);
{
INTPAIRARRAY
ipa
;
ipa
.
size
=
n_saved
;
MALLOC_N
(
n_saved
,
ipa
.
array
);
if
(
ipa
.
array
==
0
)
return
errno
;
for
(
i
=
0
;
i
<
n_saved
;
i
++
)
{
ipa
.
array
[
i
].
a
=
tmppairs
[
i
].
oldtag
;
ipa
.
array
[
i
].
b
=
tmppairs
[
i
].
newtag
;
}
int
r
=
toku_log_pmadistribute
(
txn
,
toku_txn_get_txnid
(
txn
),
filenum
,
diskoff
,
ipa
);
toku_free
(
ipa
.
array
);
if
(
r
!=
0
)
return
r
;
}
if
(
pma
&&
!
list_empty
(
&
pma
->
cursors
))
__pma_update_my_cursors
(
pma
,
tmppairs
,
n_present
);
#ifdef USE_MALLOC_IN_SMOOTH
toku_free
(
tmppairs
);
#endif
return
r
;
*
new_idx
=
newidx
;
return
0
;
}
}
...
...
@@ -602,7 +620,7 @@ int toku_pma_create(PMA *pma, pma_compare_fun_t compare_fun, DB *db, FILENUM fil
result
->
sval
=
0
;
result
->
N
=
PMA_MIN_ARRAY_SIZE
;
result
->
pairs
=
0
;
error
=
_
_pma_resize_array
(
result
,
result
->
N
,
0
);
error
=
old
_pma_resize_array
(
result
,
result
->
N
,
0
);
if
(
error
)
{
toku_free
(
result
);
return
-
1
;
...
...
@@ -627,9 +645,10 @@ static int __pma_array_size(PMA pma __attribute__((unused)), int asksize) {
return
n
;
}
static
int
__pma_resize_array
(
PMA
pma
,
int
asksize
,
int
startz
)
{
static
int
pma_resize_array
(
TOKUTXN
txn
,
FILENUM
filenum
,
DISKOFF
offset
,
PMA
pma
,
int
asksize
,
int
startz
)
{
int
i
;
int
n
;
int
oldN
=
pma
->
N
;
n
=
__pma_array_size
(
pma
,
asksize
);
pma
->
N
=
n
;
...
...
@@ -645,6 +664,7 @@ static int __pma_resize_array(PMA pma, int asksize, int startz) {
pma
->
pairs
[
i
]
=
0
;
}
toku_pmainternal_calculate_parameters
(
pma
);
toku_log_resizepma
(
txn
,
toku_txn_get_txnid
(
txn
),
filenum
,
offset
,
oldN
,
n
);
return
0
;
}
...
...
@@ -870,8 +890,8 @@ int toku_pma_cursor_free (PMA_CURSOR *cursp) {
}
/* Make some space for a key to go at idx (the thing currently at idx should end up at to the right.) */
/*
Return the new index.
(Making space may involve moving things around, including the hole at index.) */
int
toku_pmainternal_make_space_at
(
PMA
pma
,
int
id
x
)
{
/* (Making space may involve moving things around, including the hole at index.) */
int
toku_pmainternal_make_space_at
(
TOKUTXN
txn
,
FILENUM
filenum
,
DISKOFF
offset
,
PMA
pma
,
int
idx
,
int
*
new_inde
x
)
{
/* Within a range LO to HI we have a limit of how much packing we will tolerate.
* We allow the entire array to be 50% full.
* We allow a region of size lgN to be full.
...
...
@@ -907,7 +927,7 @@ int toku_pmainternal_make_space_at (PMA pma, int idx) {
size
*=
2
;
// printf("pma_make_space_realloc %d to %d hi %d\n", pma->N, size, hi);
__pma_resize_array
(
pma
,
size
,
hi
);
pma_resize_array
(
txn
,
filenum
,
offset
,
pma
,
size
,
hi
);
hi
=
size
;
//printf("doubled N\n");
...
...
@@ -919,9 +939,11 @@ int toku_pmainternal_make_space_at (PMA pma, int idx) {
}
//printf("%s:%d Smoothing from %d to %d to density %f\n", __FILE__, __LINE__, lo, hi, density);
{
int
new_index
=
toku_pmainternal_smooth_region
(
pma
->
pairs
+
lo
,
hi
-
lo
,
idx
-
lo
,
lo
,
pma
);
return
new_index
+
lo
;
int
sub_new_index
;
int
r
=
toku_pmainternal_smooth_region
(
txn
,
filenum
,
offset
,
pma
->
pairs
+
lo
,
hi
-
lo
,
idx
-
lo
,
lo
,
pma
,
&
sub_new_index
);
if
(
r
!=
0
)
return
r
;
*
new_index
=
sub_new_index
+
lo
;
return
0
;
}
}
...
...
@@ -977,7 +999,7 @@ int toku_pma_free (PMA *pmap) {
/* Copies keylen and datalen */
/* returns an error if the key is already present. */
int
toku_pma_insert
(
PMA
pma
,
DBT
*
k
,
DBT
*
v
,
TOKUTXN
txn
,
DISKOFF
diskoff
,
u_int32_t
rand4fingerprint
,
u_int32_t
*
fingerprint
)
{
int
toku_pma_insert
(
PMA
pma
,
DBT
*
k
,
DBT
*
v
,
TOKUTXN
txn
,
FILENUM
filenum
,
DISKOFF
diskoff
,
u_int32_t
rand4fingerprint
,
u_int32_t
*
fingerprint
)
{
int
found
,
idx
;
if
(
pma
->
dup_mode
&
TOKU_DB_DUPSORT
)
{
...
...
@@ -1007,7 +1029,10 @@ int toku_pma_insert (PMA pma, DBT *k, DBT *v, TOKUTXN txn, DISKOFF diskoff, u_in
}
}
if
(
kv_pair_inuse
(
pma
->
pairs
[
idx
]))
{
idx
=
toku_pmainternal_make_space_at
(
pma
,
idx
);
/* returns the new idx. */
int
newidx
;
int
r
=
toku_pmainternal_make_space_at
(
txn
,
filenum
,
diskoff
,
pma
,
idx
,
&
newidx
);
/* returns the new idx. */
if
(
r
!=
0
)
return
r
;
idx
=
newidx
;
}
assert
(
0
<=
idx
&&
idx
<
pma
->
N
);
assert
(
!
kv_pair_inuse
(
pma
->
pairs
[
idx
]));
...
...
@@ -1169,7 +1194,7 @@ static void __pma_delete_at(PMA pma, int here) {
if
(
0
)
printf
(
"shrink %d from %d to %d
\n
"
,
count
,
pma
->
N
,
size
);
newpairs
=
__pma_extract_pairs
(
pma
,
count
,
0
,
pma
->
N
);
assert
(
newpairs
);
_
_pma_resize_array
(
pma
,
size
,
0
);
old
_pma_resize_array
(
pma
,
size
,
0
);
distribute_data
(
pma
->
pairs
,
pma
->
N
,
newpairs
,
count
,
pma
);
/* update the cursors */
__pma_update_my_cursors
(
pma
,
newpairs
,
count
);
...
...
@@ -1178,7 +1203,7 @@ static void __pma_delete_at(PMA pma, int here) {
int
toku_pma_insert_or_replace
(
PMA
pma
,
DBT
*
k
,
DBT
*
v
,
int
*
replaced_v_size
,
/* If it is a replacement, set to the size of the old value, otherwise set to -1. */
TOKUTXN
txn
,
DISKOFF
diskoff
,
TOKUTXN
txn
,
FILENUM
filenum
,
DISKOFF
diskoff
,
u_int32_t
rand4fingerprint
,
u_int32_t
*
fingerprint
)
{
//printf("%s:%d v->size=%d\n", __FILE__, __LINE__, v->size);
int
r
;
...
...
@@ -1219,7 +1244,10 @@ int toku_pma_insert_or_replace (PMA pma, DBT *k, DBT *v,
}
}
if
(
kv_pair_inuse
(
pma
->
pairs
[
idx
]))
{
idx
=
toku_pmainternal_make_space_at
(
pma
,
idx
);
/* returns the new idx. */
int
newidx
;
r
=
toku_pmainternal_make_space_at
(
txn
,
filenum
,
diskoff
,
pma
,
idx
,
&
newidx
);
/* returns the new idx. */
if
(
r
!=
0
)
return
r
;
idx
=
newidx
;
}
assert
(
!
kv_pair_inuse
(
pma
->
pairs
[
idx
]));
//printf("%s:%d v->size=%d\n", __FILE__, __LINE__, v->size);
...
...
@@ -1359,9 +1387,10 @@ static int __pma_compare_kv(PMA pma, struct kv_pair *a, struct kv_pair *b) {
return
cmp
;
}
int
toku_pma_split
(
PMA
origpma
,
unsigned
int
*
origpma_size
,
DBT
*
splitk
,
PMA
leftpma
,
unsigned
int
*
leftpma_size
,
u_int32_t
leftrand4fp
,
u_int32_t
*
leftfingerprint
,
PMA
rightpma
,
unsigned
int
*
rightpma_size
,
u_int32_t
rightrand4fp
,
u_int32_t
*
rightfingerprint
)
{
int
toku_pma_split
(
TOKUTXN
txn
,
FILENUM
filenum
,
PMA
origpma
,
unsigned
int
*
origpma_size
,
DBT
*
splitk
,
DISKOFF
leftdiskoff
,
PMA
leftpma
,
unsigned
int
*
leftpma_size
,
u_int32_t
leftrand4fp
,
u_int32_t
*
leftfingerprint
,
DISKOFF
rightdiskoff
,
PMA
rightpma
,
unsigned
int
*
rightpma_size
,
u_int32_t
rightrand4fp
,
u_int32_t
*
rightfingerprint
)
{
int
error
;
int
npairs
;
struct
kv_pair_tag
*
pairs
;
...
...
@@ -1447,7 +1476,7 @@ int toku_pma_split(PMA origpma, unsigned int *origpma_size, DBT *splitk,
/* put the first half of pairs into the left pma */
n
=
spliti
;
error
=
__pma_resize_array
(
leftpma
,
n
+
n
/
4
,
0
);
error
=
pma_resize_array
(
txn
,
filenum
,
leftdiskoff
,
leftpma
,
n
+
n
/
4
,
0
);
assert
(
error
==
0
);
distribute_data
(
leftpma
->
pairs
,
toku_pma_index_limit
(
leftpma
),
&
pairs
[
0
],
n
,
leftpma
);
#if PMA_USE_MEMPOOL
...
...
@@ -1458,7 +1487,7 @@ int toku_pma_split(PMA origpma, unsigned int *origpma_size, DBT *splitk,
/* put the second half of pairs into the right pma */
n
=
npairs
-
spliti
;
error
=
__pma_resize_array
(
rightpma
,
n
+
n
/
4
,
0
);
error
=
pma_resize_array
(
txn
,
filenum
,
rightdiskoff
,
rightpma
,
n
+
n
/
4
,
0
);
assert
(
error
==
0
);
distribute_data
(
rightpma
->
pairs
,
toku_pma_index_limit
(
rightpma
),
&
pairs
[
spliti
],
n
,
rightpma
);
#if PMA_USE_MEMPOOL
...
...
@@ -1487,7 +1516,7 @@ static void __pma_bulk_cleanup(struct pma *pma, struct kv_pair_tag *pairs, int n
pma_mfree_kv_pair
(
pma
,
pairs
[
i
].
pair
);
}
int
toku_pma_bulk_insert
(
PMA
pma
,
DBT
*
keys
,
DBT
*
vals
,
int
n_newpairs
,
u_int32_t
rand4fp
,
u_int32_t
*
sum
)
{
int
toku_pma_bulk_insert
(
TOKUTXN
txn
,
FILENUM
filenum
,
DISKOFF
diskoff
,
PMA
pma
,
DBT
*
keys
,
DBT
*
vals
,
int
n_newpairs
,
u_int32_t
rand4fp
,
u_int32_t
*
sum
)
{
struct
kv_pair_tag
*
newpairs
;
int
i
;
int
error
;
...
...
@@ -1521,7 +1550,7 @@ int toku_pma_bulk_insert(PMA pma, DBT *keys, DBT *vals, int n_newpairs, u_int32_
}
}
error
=
__pma_resize_array
(
pma
,
n_newpairs
+
n_newpairs
/
4
,
0
);
error
=
pma_resize_array
(
txn
,
filenum
,
diskoff
,
pma
,
n_newpairs
+
n_newpairs
/
4
,
0
);
if
(
error
)
{
__pma_bulk_cleanup
(
pma
,
newpairs
,
n_newpairs
);
toku_free
(
newpairs
);
...
...
newbrt/pma.h
View file @
62296eab
...
...
@@ -47,15 +47,15 @@ int toku_pma_n_entries (PMA);
/* Duplicates the key and keylen. */
//enum pma_errors toku_pma_insert (PMA, bytevec key, ITEMLEN keylen, bytevec data, ITEMLEN datalen);
// The DB pointer is there so that the comparison function can be called.
enum
pma_errors
toku_pma_insert
(
PMA
,
DBT
*
,
DBT
*
,
TOKUTXN
txn
,
DISKOFF
,
u_int32_t
/*random for fingerprint */
,
u_int32_t
*/
*
fingerprint
*/
);
enum
pma_errors
toku_pma_insert
(
PMA
,
DBT
*
,
DBT
*
,
TOKUTXN
,
FILENUM
,
DISKOFF
,
u_int32_t
/*random for fingerprint */
,
u_int32_t
*/
*
fingerprint
*/
);
/* This returns an error if the key is NOT present. */
int
pma_replace
(
PMA
,
bytevec
key
,
ITEMLEN
keylen
,
bytevec
data
,
ITEMLEN
datalen
);
/* This returns an error if the key is NOT present. */
int
toku_pma_delete
(
PMA
,
DBT
*
,
u_int32_t
/*random for fingerprint*/
,
u_int32_t
*/
*
fingerprint
*/
,
u_int32_t
*
deleted_size
);
int
toku_pma_insert_or_replace
(
PMA
pma
,
DBT
*
k
,
DBT
*
v
,
int
*
replaced_v_size
,
/* If it is a replacement, set to the size of the old value, otherwise set to -1. */
TOKUTXN
txn
,
DISKOFF
,
int
toku_pma_insert_or_replace
(
PMA
/*pma*/
,
DBT
*/
*
k
*/
,
DBT
*/
*
v
*/
,
int
*/
*
replaced_v_size
*/
,
/* If it is a replacement, set to the size of the old value, otherwise set to -1. */
TOKUTXN
/*txn*/
,
FILENUM
,
DISKOFF
,
u_int32_t
/*random for fingerprint*/
,
u_int32_t
*/
*
fingerprint
*/
);
...
...
@@ -73,9 +73,10 @@ enum pma_errors toku_pma_lookup (PMA, DBT*, DBT*);
* leftpma - the pma assigned keys <= pivot key
* rightpma - the pma assigned keys > pivot key
*/
int
toku_pma_split
(
PMA
origpma
,
unsigned
int
*
origpma_size
,
DBT
*
splitk
,
PMA
leftpma
,
unsigned
int
*
leftpma_size
,
u_int32_t
leftrand4sum
,
u_int32_t
*
leftfingerprint
,
PMA
rightpma
,
unsigned
int
*
rightpma_size
,
u_int32_t
rightrand4sum
,
u_int32_t
*
rightfingerprint
);
int
toku_pma_split
(
TOKUTXN
,
FILENUM
,
PMA
/*origpma*/
,
unsigned
int
*/
*
origpma_size
*/
,
DBT
*/
*
splitk
*/
,
DISKOFF
/*leftdiskoff*/
,
PMA
/*leftpma*/
,
unsigned
int
*/
*
leftpma_size
*/
,
u_int32_t
/*leftrand4sum*/
,
u_int32_t
*/
*
leftfingerprint
*/
,
DISKOFF
/*rightdiskoff*/
,
PMA
/*rightpma*/
,
unsigned
int
*/
*
rightpma_size
*/
,
u_int32_t
/*rightrand4sum*/
,
u_int32_t
*/
*
rightfingerprint
*/
);
/*
* Insert several key value pairs into an empty pma.
...
...
@@ -88,7 +89,7 @@ int toku_pma_split(PMA origpma, unsigned int *origpma_size, DBT *splitk,
* vals - an array of values
* n_newpairs - the number of key value pairs
*/
int
toku_pma_bulk_insert
(
PMA
pma
,
DBT
*
keys
,
DBT
*
vals
,
int
n_newpairs
,
u_int32_t
rand4sem
,
u_int32_t
*
fingerprint
);
int
toku_pma_bulk_insert
(
TOKUTXN
,
FILENUM
,
DISKOFF
,
PMA
pma
,
DBT
*
keys
,
DBT
*
vals
,
int
n_newpairs
,
u_int32_t
rand4sem
,
u_int32_t
*
fingerprint
);
/* Move the cursor to the beginning or the end or to a key */
int
toku_pma_cursor
(
PMA
,
PMA_CURSOR
*
,
void
**
/*sskey*/
,
void
**
/*ssval*/
);
// the sskey and ssval point to variables that hold blocks that can be used to return values for zero'd DBTS.
...
...
newbrt/recover.c
View file @
62296eab
...
...
@@ -145,9 +145,19 @@ static void toku_recover_insertinleaf (struct logtype_insertinleaf *c) {
assert
(
node
->
height
==
0
);
DBT
key
,
data
;
r
=
toku_pma_set_at_index
(
node
->
u
.
l
.
buffer
,
c
->
pmaidx
,
toku_fill_dbt
(
&
key
,
c
->
key
.
data
,
c
->
key
.
len
),
toku_fill_dbt
(
&
data
,
c
->
data
.
data
,
c
->
data
.
len
));
assert
(
r
==
0
);
node
->
local_fingerprint
+=
node
->
rand4fingerprint
*
toku_calccrc32_kvpair
(
c
->
key
.
data
,
c
->
key
.
len
,
c
->
data
.
data
,
c
->
data
.
len
);
node
->
u
.
l
.
n_bytes_in_buffer
+=
KEY_VALUE_OVERHEAD
+
c
->
key
.
len
+
c
->
data
.
len
;
assert
(
r
==
0
);
}
static
void
toku_recover_resizepma
(
struct
logtype_resizepma
*
c
)
{
c
=
c
;
abort
();
}
static
void
toku_recover_pmadistribute
(
struct
logtype_pmadistribute
*
c
)
{
c
=
c
;
abort
();
}
int
main
(
int
argc
,
char
*
argv
[])
{
...
...
newbrt/wbuf.h
View file @
62296eab
...
...
@@ -137,4 +137,13 @@ static inline void wbuf_LOGGEDBRTHEADER (struct wbuf *w, LOGGEDBRTHEADER h) {
}
}
static
inline
void
wbuf_INTPAIRARRAY
(
struct
wbuf
*
w
,
INTPAIRARRAY
h
)
{
u_int32_t
i
;
wbuf_int
(
w
,
h
.
size
);
for
(
i
=
0
;
i
<
h
.
size
;
i
++
)
{
wbuf_int
(
w
,
h
.
array
[
i
].
a
);
wbuf_int
(
w
,
h
.
array
[
i
].
b
);
}
}
#endif
src/tests/test_log4.c
View file @
62296eab
...
...
@@ -32,7 +32,7 @@ static void make_db (void) {
r
=
tid
->
commit
(
tid
,
0
);
assert
(
r
==
0
);
r
=
env
->
txn_begin
(
env
,
0
,
&
tid
,
0
);
assert
(
r
==
0
);
for
(
i
=
0
;
i
<
10
;
i
++
)
{
for
(
i
=
0
;
i
<
2
;
i
++
)
{
char
hello
[
30
],
there
[
30
];
DBT
key
,
data
;
snprintf
(
hello
,
sizeof
(
hello
),
"hello%ld.%d"
,
random
(),
i
);
...
...
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