Commit 55da2916 authored by Dave Wells's avatar Dave Wells Committed by Yoni Fogel

fix memory leak in logcursor, rearrange code to share common code, improve...

fix memory leak in logcursor, rearrange code to share common code, improve error messages refs #2446 [t:2446]

git-svn-id: file:///svn/toku/tokudb@18896 c7de825b-a66e-492c-adef-691d508d4ae1
parent e889f37c
...@@ -221,6 +221,7 @@ fail: ...@@ -221,6 +221,7 @@ fail:
int toku_logcursor_destroy(TOKULOGCURSOR *lc) { int toku_logcursor_destroy(TOKULOGCURSOR *lc) {
int r=0; int r=0;
if ( *lc ) {
if ( (*lc)->entry_valid ) { if ( (*lc)->entry_valid ) {
toku_log_free_log_entry_resources(&((*lc)->entry)); toku_log_free_log_entry_resources(&((*lc)->entry));
(*lc)->entry_valid = FALSE; (*lc)->entry_valid = FALSE;
...@@ -228,58 +229,95 @@ int toku_logcursor_destroy(TOKULOGCURSOR *lc) { ...@@ -228,58 +229,95 @@ int toku_logcursor_destroy(TOKULOGCURSOR *lc) {
r = lc_close_cur_logfile(*lc); r = lc_close_cur_logfile(*lc);
int lf; int lf;
for(lf=0;lf<(*lc)->n_logfiles;lf++) { for(lf=0;lf<(*lc)->n_logfiles;lf++) {
toku_free((*lc)->logfiles[lf]); if ( (*lc)->logfiles[lf] ) toku_free((*lc)->logfiles[lf]);
} }
toku_free((*lc)->logfiles); if ( (*lc)->logfiles ) toku_free((*lc)->logfiles);
toku_free((*lc)->logdir); if ( (*lc)->logdir ) toku_free((*lc)->logdir);
if ((*lc)->buffer) if ( (*lc)->buffer ) toku_free((*lc)->buffer);
toku_free((*lc)->buffer);
toku_free(*lc); toku_free(*lc);
*lc = NULL; *lc = NULL;
}
return r; return r;
} }
int toku_logcursor_next(TOKULOGCURSOR lc, struct log_entry **le) { static int lc_log_read(TOKULOGCURSOR lc)
int r=0; {
if ( lc->entry_valid ) { int r = toku_log_fread(lc->cur_fp, &(lc->entry));
while ( r == EOF ) {
// move to next file
r = lc_close_cur_logfile(lc);
if (r!=0) return r;
if ( lc->cur_logfiles_index == lc->n_logfiles-1) return DB_NOTFOUND;
lc->cur_logfiles_index++;
r = lc_open_logfile(lc, lc->cur_logfiles_index);
if (r!=0) return r;
r = toku_log_fread(lc->cur_fp, &(lc->entry));
}
if (r!=0) {
toku_log_free_log_entry_resources(&(lc->entry)); toku_log_free_log_entry_resources(&(lc->entry));
lc->entry_valid = FALSE; time_t tnow = time(NULL);
if (lc->last_direction == LC_BACKWARD) { if (r==DB_BADFORMAT) {
struct log_entry junk; fprintf(stderr, "%.24s Tokudb bad log format in %s\n", ctime(&tnow), lc->logfiles[lc->cur_logfiles_index]);
r = toku_log_fread(lc->cur_fp, &junk);
assert(r == 0);
toku_log_free_log_entry_resources(&junk);
} }
} else { else {
r = toku_logcursor_first(lc, le); fprintf(stderr, "%.24s Tokudb unexpected log format error '%s' in %s\n", ctime(&tnow), strerror(r), lc->logfiles[lc->cur_logfiles_index]);
return r;
} }
r = toku_log_fread(lc->cur_fp, &(lc->entry)); }
while ( EOF == r ) { return r;
// move to next file }
static int lc_log_read_backward(TOKULOGCURSOR lc)
{
int r = toku_log_fread_backward(lc->cur_fp, &(lc->entry));
while ( -1 == r) { // if within header length of top of file
// move to previous file
r = lc_close_cur_logfile(lc); r = lc_close_cur_logfile(lc);
if (r!=0) if (r!=0)
return r; return r;
if ( lc->cur_logfiles_index == lc->n_logfiles-1) if ( lc->cur_logfiles_index == 0 )
return DB_NOTFOUND; return DB_NOTFOUND;
lc->cur_logfiles_index++; lc->cur_logfiles_index--;
r = lc_open_logfile(lc, lc->cur_logfiles_index); r = lc_open_logfile(lc, lc->cur_logfiles_index);
if (r!= 0) if (r!=0)
return r; return r;
r = toku_log_fread(lc->cur_fp, &(lc->entry)); // seek to end
r = fseek(lc->cur_fp, 0, SEEK_END);
assert(0==r);
r = toku_log_fread_backward(lc->cur_fp, &(lc->entry));
} }
if (r!=0) { if (r!=0) {
toku_log_free_log_entry_resources(&(lc->entry));
time_t tnow = time(NULL);
if (r==DB_BADFORMAT) { if (r==DB_BADFORMAT) {
fprintf(stderr, "Bad log format in %s\n", lc->logfiles[lc->cur_logfiles_index]); fprintf(stderr, "%.24s Tokudb bad log format in %s\n", ctime(&tnow), lc->logfiles[lc->cur_logfiles_index]);
}
else {
fprintf(stderr, "%.24s Tokudb uUnexpected log format error '%s' in %s\n", ctime(&tnow), strerror(r), lc->logfiles[lc->cur_logfiles_index]);
}
}
return r; return r;
}
int toku_logcursor_next(TOKULOGCURSOR lc, struct log_entry **le) {
int r=0;
if ( lc->entry_valid ) {
toku_log_free_log_entry_resources(&(lc->entry));
lc->entry_valid = FALSE;
if (lc->last_direction == LC_BACKWARD) {
struct log_entry junk;
r = toku_log_fread(lc->cur_fp, &junk);
assert(r == 0);
toku_log_free_log_entry_resources(&junk);
}
} else { } else {
fprintf(stderr, "Unexpected log format error '%s' in %s\n", strerror(r), lc->logfiles[lc->cur_logfiles_index]); r = toku_logcursor_first(lc, le);
return r; return r;
} }
} // read the entry
r = lc_log_read(lc);
if (r!=0) return r;
r = lc_check_lsn(lc, LC_FORWARD); r = lc_check_lsn(lc, LC_FORWARD);
if (r!=0) if (r!=0) return r;
return r;
lc->last_direction = LC_FORWARD; lc->last_direction = LC_FORWARD;
lc->entry_valid = TRUE; lc->entry_valid = TRUE;
*le = &(lc->entry); *le = &(lc->entry);
...@@ -301,35 +339,11 @@ int toku_logcursor_prev(TOKULOGCURSOR lc, struct log_entry **le) { ...@@ -301,35 +339,11 @@ int toku_logcursor_prev(TOKULOGCURSOR lc, struct log_entry **le) {
r = toku_logcursor_last(lc, le); r = toku_logcursor_last(lc, le);
return r; return r;
} }
r = toku_log_fread_backward(lc->cur_fp, &(lc->entry)); // read the entry
while ( -1 == r) { // if within header length of top of file r = lc_log_read_backward(lc);
// move to previous file if (r!=0) return r;
r = lc_close_cur_logfile(lc);
if (r!=0)
return r;
if ( lc->cur_logfiles_index == 0 )
return DB_NOTFOUND;
lc->cur_logfiles_index--;
r = lc_open_logfile(lc, lc->cur_logfiles_index);
if (r!=0)
return r;
// seek to end
r = fseek(lc->cur_fp, 0, SEEK_END);
assert(0==r);
r = toku_log_fread_backward(lc->cur_fp, &(lc->entry));
}
if (r!=0) {
if (r==DB_BADFORMAT) {
fprintf(stderr, "Bad log format in %s\n", lc->logfiles[lc->cur_logfiles_index]);
return r;
} else {
fprintf(stderr, "Unexpected log format error '%s' in %s\n", strerror(r), lc->logfiles[lc->cur_logfiles_index]);
return r;
}
}
r = lc_check_lsn(lc, LC_BACKWARD); r = lc_check_lsn(lc, LC_BACKWARD);
if (r!=0) if (r!=0) return r;
return r;
lc->last_direction = LC_BACKWARD; lc->last_direction = LC_BACKWARD;
lc->entry_valid = TRUE; lc->entry_valid = TRUE;
*le = &(lc->entry); *le = &(lc->entry);
...@@ -353,24 +367,12 @@ int toku_logcursor_first(TOKULOGCURSOR lc, struct log_entry **le) { ...@@ -353,24 +367,12 @@ int toku_logcursor_first(TOKULOGCURSOR lc, struct log_entry **le) {
return r; return r;
lc->cur_logfiles_index = 0; lc->cur_logfiles_index = 0;
} }
while (1) { // read the entry
r = toku_log_fread(lc->cur_fp, &(lc->entry)); r = lc_log_read(lc);
if (r==0) if (r!=0) return r;
break;
// move to next file
r = lc_close_cur_logfile(lc);
if (r!=0)
return r;
if ( lc->cur_logfiles_index == lc->n_logfiles-1)
return DB_NOTFOUND;
lc->cur_logfiles_index++;
r = lc_open_logfile(lc, lc->cur_logfiles_index);
if (r!= 0)
return r;
}
r = lc_check_lsn(lc, LC_FIRST); r = lc_check_lsn(lc, LC_FIRST);
if (r!=0) if (r!=0) return r;
return r;
lc->last_direction = LC_FIRST; lc->last_direction = LC_FIRST;
lc->entry_valid = TRUE; lc->entry_valid = TRUE;
*le = &(lc->entry); *le = &(lc->entry);
...@@ -402,6 +404,7 @@ int toku_logcursor_last(TOKULOGCURSOR lc, struct log_entry **le) { ...@@ -402,6 +404,7 @@ int toku_logcursor_last(TOKULOGCURSOR lc, struct log_entry **le) {
if (r==0) // got a good entry if (r==0) // got a good entry
break; break;
if (r>0) { if (r>0) {
toku_log_free_log_entry_resources(&(lc->entry));
// got an error, // got an error,
// probably a corrupted last log entry due to a crash // probably a corrupted last log entry due to a crash
// try scanning forward from the beginning to find the last good entry // try scanning forward from the beginning to find the last good entry
......
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