blob: a3402a9ef54286459fe26fea909848324e10f100 [file] [log] [blame]
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/services/filesystem/lock_table.h"
#include "build/build_config.h"
#include "components/services/filesystem/file_impl.h"
namespace filesystem {
LockTable::LockTable() {}
LockTable::~LockTable() {}
base::File::Error LockTable::LockFile(FileImpl* file) {
DCHECK(file->IsValid());
DCHECK(file->path().IsAbsolute());
auto it = locked_files_.find(file->path());
if (it != locked_files_.end()) {
// We're already locked; that's an error condition on Windows.
return base::File::FILE_ERROR_FAILED;
}
#if !defined(OS_FUCHSIA)
// Fuchsia doesn't provide a file locking mechanism, so file locks work only
// within a single process. File locking is used only by LevelDB which stores
// all files in the profile directory and normally there shouldn't be more
// than a single chrome process per profile. So in-process locks should be
// sufficient.
// TODO(fuchsia): Investigate if it's necessary to implement cross-process
// file locks. crbug.com/744893 .
base::File::Error lock_err = file->RawLockFile();
if (lock_err != base::File::FILE_OK) {
// Locking failed for some reason.
return lock_err;
}
#endif // !OS_FUCHSIA
locked_files_.insert(file->path());
return base::File::FILE_OK;
}
base::File::Error LockTable::UnlockFile(FileImpl* file) {
auto it = locked_files_.find(file->path());
if (it != locked_files_.end()) {
#if !defined(OS_FUCHSIA)
base::File::Error lock_err = file->RawUnlockFile();
if (lock_err != base::File::FILE_OK) {
// TODO(erg): When can we fail to release a lock?
NOTREACHED();
return lock_err;
}
#endif // !OS_FUCHSIA
locked_files_.erase(it);
}
return base::File::FILE_OK;
}
void LockTable::RemoveFromLockTable(const base::FilePath& path) {
auto it = locked_files_.find(path);
if (it != locked_files_.end())
locked_files_.erase(it);
}
} // namespace filesystem