Make InMemoryEnv more consistent with filesystem based Env's.

Env's (like the POSIX Env) which use an actual filesystem behave
differently than InMemoryEnv with regards to writing data to a currently
open file.

InMemoryEnv::NewWritableFile would previously delete that file,
if it was open, before creating a new file so any previously
open file would be unlinked. This change truncates an open file
so that subsequent reads will read that new data.

This should have no impact on leveldb as it never has the same
file open for both read and write access. This change is only
being made for tests (specifically a future change to corruption_test)
to allow them to be decoupled from the underlying platform and
allow them to use an Env.

-------------
Created by MOE: https://github.com/google/moe
MOE_MIGRATED_REVID=237858231
This commit is contained in:
cmumford 2019-03-11 12:32:50 -07:00 committed by Chris Mumford
parent cf1d1ab255
commit dd906262fd
2 changed files with 53 additions and 16 deletions

View File

@ -51,9 +51,22 @@ class FileState {
} }
} }
uint64_t Size() const { return size_; } uint64_t Size() const {
MutexLock lock(&blocks_mutex_);
return size_;
}
void Truncate() {
MutexLock lock(&blocks_mutex_);
for (char*& block : blocks_) {
delete[] block;
}
blocks_.clear();
size_ = 0;
}
Status Read(uint64_t offset, size_t n, Slice* result, char* scratch) const { Status Read(uint64_t offset, size_t n, Slice* result, char* scratch) const {
MutexLock lock(&blocks_mutex_);
if (offset > size_) { if (offset > size_) {
return Status::IOError("Offset greater than file size."); return Status::IOError("Offset greater than file size.");
} }
@ -100,6 +113,7 @@ class FileState {
const char* src = data.data(); const char* src = data.data();
size_t src_len = data.size(); size_t src_len = data.size();
MutexLock lock(&blocks_mutex_);
while (src_len > 0) { while (src_len > 0) {
size_t avail; size_t avail;
size_t offset = size_ % kBlockSize; size_t offset = size_ % kBlockSize;
@ -128,10 +142,7 @@ class FileState {
private: private:
// Private since only Unref() should be used to delete it. // Private since only Unref() should be used to delete it.
~FileState() { ~FileState() {
for (std::vector<char*>::iterator i = blocks_.begin(); i != blocks_.end(); Truncate();
++i) {
delete [] *i;
}
} }
// No copying allowed. // No copying allowed.
@ -141,11 +152,9 @@ class FileState {
port::Mutex refs_mutex_; port::Mutex refs_mutex_;
int refs_ GUARDED_BY(refs_mutex_); int refs_ GUARDED_BY(refs_mutex_);
// The following fields are not protected by any mutex. They are only mutable mutable port::Mutex blocks_mutex_;
// while the file is being written, and concurrent access is not allowed std::vector<char*> blocks_ GUARDED_BY(blocks_mutex_);
// to writable files. uint64_t size_ GUARDED_BY(blocks_mutex_);
std::vector<char*> blocks_;
uint64_t size_;
enum { kBlockSize = 8 * 1024 }; enum { kBlockSize = 8 * 1024 };
}; };
@ -269,13 +278,18 @@ class InMemoryEnv : public EnvWrapper {
virtual Status NewWritableFile(const std::string& fname, virtual Status NewWritableFile(const std::string& fname,
WritableFile** result) { WritableFile** result) {
MutexLock lock(&mutex_); MutexLock lock(&mutex_);
if (file_map_.find(fname) != file_map_.end()) { FileSystem::iterator it = file_map_.find(fname);
DeleteFileInternal(fname);
}
FileState* file = new FileState(); FileState* file;
if (it == file_map_.end()) {
// File is not currently open.
file = new FileState();
file->Ref(); file->Ref();
file_map_[fname] = file; file_map_[fname] = file;
} else {
file = it->second;
file->Truncate();
}
*result = new WritableFileImpl(file); *result = new WritableFileImpl(file);
return Status::OK(); return Status::OK();

View File

@ -191,6 +191,29 @@ TEST(MemEnvTest, LargeWrite) {
delete [] scratch; delete [] scratch;
} }
TEST(MemEnvTest, OverwriteOpenFile) {
const char kWrite1Data[] = "Write #1 data";
const size_t kFileDataLen = sizeof(kWrite1Data) - 1;
const std::string kTestFileName = test::TmpDir() + "/leveldb-TestFile.dat";
ASSERT_OK(WriteStringToFile(env_, kWrite1Data, kTestFileName));
RandomAccessFile* rand_file;
ASSERT_OK(env_->NewRandomAccessFile(kTestFileName, &rand_file));
const char kWrite2Data[] = "Write #2 data";
ASSERT_OK(WriteStringToFile(env_, kWrite2Data, kTestFileName));
// Verify that overwriting an open file will result in the new file data
// being read from files opened before the write.
Slice result;
char scratch[kFileDataLen];
ASSERT_OK(rand_file->Read(0, kFileDataLen, &result, scratch));
ASSERT_EQ(0, result.compare(kWrite2Data));
delete rand_file;
}
TEST(MemEnvTest, DBTest) { TEST(MemEnvTest, DBTest) {
Options options; Options options;
options.create_if_missing = true; options.create_if_missing = true;