Summary: Fix flush not being commit while writing manifest, which is a recent bug introduced by D60075. The issue: # Options.max_background_flushes > 1 # Background thread A pick up a flush job, flush, then commit to manifest. (Note that mutex is released before writing manifest.) # Background thread B pick up another flush job, flush. When it gets to `MemTableList::InstallMemtableFlushResults`, it notices another thread is commiting, so it quit. # After the first commit, thread A doesn't double check if there are more flush result need to commit, leaving the second flush uncommitted. Test Plan: run the test. Also verify the new test hit deadlock without the fix. Reviewers: sdong, igor, lightmark Reviewed By: lightmark Subscribers: andrewkr, omegaga, dhruba, leveldb Differential Revision: https://reviews.facebook.net/D60969main
parent
9ab38c45ad
commit
32604e6601
@ -0,0 +1,55 @@ |
||||
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
||||
// This source code is licensed under the BSD-style license found in the
|
||||
// LICENSE file in the root directory of this source tree. An additional grant
|
||||
// of patent rights can be found in the PATENTS file in the same directory.
|
||||
//
|
||||
// Copyright (c) 2011 The LevelDB Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style license that can be
|
||||
// found in the LICENSE file. See the AUTHORS file for names of contributors.
|
||||
|
||||
#include "db/db_test_util.h" |
||||
#include "port/stack_trace.h" |
||||
#include "util/sync_point.h" |
||||
|
||||
namespace rocksdb { |
||||
|
||||
class DBFlushTest : public DBTestBase { |
||||
public: |
||||
DBFlushTest() : DBTestBase("/db_flush_test") {} |
||||
}; |
||||
|
||||
// We had issue when two background threads trying to flush at the same time,
|
||||
// only one of them get committed. The test verifies the issue is fixed.
|
||||
TEST_F(DBFlushTest, FlushWhileWritingManifest) { |
||||
Options options; |
||||
options.disable_auto_compactions = true; |
||||
options.max_background_flushes = 2; |
||||
Reopen(options); |
||||
FlushOptions no_wait; |
||||
no_wait.wait = false; |
||||
|
||||
SyncPoint::GetInstance()->LoadDependency( |
||||
{{"VersionSet::LogAndApply:WriteManifest", |
||||
"DBFlushTest::FlushWhileWritingManifest:1"}, |
||||
{"MemTableList::InstallMemtableFlushResults:InProgress", |
||||
"VersionSet::LogAndApply:WriteManifestDone"}}); |
||||
SyncPoint::GetInstance()->EnableProcessing(); |
||||
|
||||
ASSERT_OK(Put("foo", "v")); |
||||
ASSERT_OK(dbfull()->Flush(no_wait)); |
||||
TEST_SYNC_POINT("DBFlushTest::FlushWhileWritingManifest:1"); |
||||
ASSERT_OK(Put("bar", "v")); |
||||
ASSERT_OK(dbfull()->Flush(no_wait)); |
||||
// If the issue is hit we will wait here forever.
|
||||
dbfull()->TEST_WaitForFlushMemTable(); |
||||
|
||||
ASSERT_EQ(2, TotalTableFiles()); |
||||
} |
||||
|
||||
} // namespace rocksdb
|
||||
|
||||
int main(int argc, char** argv) { |
||||
rocksdb::port::InstallStackTraceHandler(); |
||||
::testing::InitGoogleTest(&argc, argv); |
||||
return RUN_ALL_TESTS(); |
||||
} |
Loading…
Reference in new issue