diff options
author | Joe Shaw <jshaw@fastly.com> | 2016-10-27 16:44:56 -0700 |
---|---|---|
committer | Joe Shaw <jshaw@fastly.com> | 2016-10-27 16:44:56 -0700 |
commit | bdab31cc98cdf61a93567c55e85e835ba6fb364f (patch) | |
tree | 0fb90e79837e067eaec23b243a9c58b15294271e /backup.go | |
parent | Merge pull request #333 from nyarly/master (diff) | |
download | golite-bdab31cc98cdf61a93567c55e85e835ba6fb364f.tar.gz golite-bdab31cc98cdf61a93567c55e85e835ba6fb364f.tar.xz |
fix double free in SQLiteBackup.Close() on error
The sqlite3_backup_finish() function never fails, it just returns the
error code from previous operations. Previously if it returned an
error, the finalizer wouldn't be unset and sqlite3_backup_finish() would
be run again on an already-finished backup. This results in a
double-free and often segfaults.
The error handling is described in more detail in the "Error handling"
section of https://www.sqlite.org/backup.html.
Diffstat (limited to 'backup.go')
-rw-r--r-- | backup.go | 9 |
1 files changed, 7 insertions, 2 deletions
@@ -65,10 +65,15 @@ func (b *SQLiteBackup) Finish() error { func (b *SQLiteBackup) Close() error { ret := C.sqlite3_backup_finish(b.b) + + // sqlite3_backup_finish() never fails, it just returns the + // error code from previous operations, so clean up before + // checking and returning an error + b.b = nil + runtime.SetFinalizer(b, nil) + if ret != 0 { return Error{Code: ErrNo(ret)} } - b.b = nil - runtime.SetFinalizer(b, nil) return nil } |