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:
parent
cf1d1ab255
commit
dd906262fd
@ -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();
|
||||||
|
@ -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;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user