| // Copyright (c) 2018 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. | |
|  | |
| // Prevent Windows headers from defining min/max macros and instead | |
| // use STL. | |
| #ifndef NOMINMAX | |
| #define NOMINMAX | |
| #endif  // ifndef NOMINMAX | |
| #include <windows.h> | |
|  | |
| #include <algorithm> | |
| #include <atomic> | |
| #include <chrono> | |
| #include <condition_variable> | |
| #include <cstddef> | |
| #include <cstdint> | |
| #include <cstdlib> | |
| #include <cstring> | |
| #include <memory> | |
| #include <mutex> | |
| #include <queue> | |
| #include <sstream> | |
| #include <string> | |
| #include <vector> | |
|  | |
| #include "leveldb/env.h" | |
| #include "leveldb/slice.h" | |
| #include "port/port.h" | |
| #include "port/thread_annotations.h" | |
| #include "util/env_windows_test_helper.h" | |
| #include "util/logging.h" | |
| #include "util/mutexlock.h" | |
| #include "util/windows_logger.h" | |
|  | |
| namespace leveldb { | |
| 
 | |
| namespace { | |
| 
 | |
| constexpr const size_t kWritableFileBufferSize = 65536; | |
| 
 | |
| // Up to 1000 mmaps for 64-bit binaries; none for 32-bit. | |
| constexpr int kDefaultMmapLimit = (sizeof(void*) >= 8) ? 1000 : 0; | |
| 
 | |
| // Can be set by by EnvWindowsTestHelper::SetReadOnlyMMapLimit(). | |
| int g_mmap_limit = kDefaultMmapLimit; | |
| 
 | |
| std::string GetWindowsErrorMessage(DWORD error_code) { | |
|   std::string message; | |
|   char* error_text = nullptr; | |
|   // Use MBCS version of FormatMessage to match return value. | |
|   size_t error_text_size = ::FormatMessageA( | |
|       FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER | | |
|           FORMAT_MESSAGE_IGNORE_INSERTS, | |
|       nullptr, error_code, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), | |
|       reinterpret_cast<char*>(&error_text), 0, nullptr); | |
|   if (!error_text) { | |
|     return message; | |
|   } | |
|   message.assign(error_text, error_text_size); | |
|   ::LocalFree(error_text); | |
|   return message; | |
| } | |
| 
 | |
| Status WindowsError(const std::string& context, DWORD error_code) { | |
|   if (error_code == ERROR_FILE_NOT_FOUND || error_code == ERROR_PATH_NOT_FOUND) | |
|     return Status::NotFound(context, GetWindowsErrorMessage(error_code)); | |
|   return Status::IOError(context, GetWindowsErrorMessage(error_code)); | |
| } | |
| 
 | |
| class ScopedHandle { | |
|  public: | |
|   ScopedHandle(HANDLE handle) : handle_(handle) {} | |
|   ScopedHandle(const ScopedHandle&) = delete; | |
|   ScopedHandle(ScopedHandle&& other) noexcept : handle_(other.Release()) {} | |
|   ~ScopedHandle() { Close(); } | |
| 
 | |
|   ScopedHandle& operator=(const ScopedHandle&) = delete; | |
| 
 | |
|   ScopedHandle& operator=(ScopedHandle&& rhs) noexcept { | |
|     if (this != &rhs) handle_ = rhs.Release(); | |
|     return *this; | |
|   } | |
| 
 | |
|   bool Close() { | |
|     if (!is_valid()) { | |
|       return true; | |
|     } | |
|     HANDLE h = handle_; | |
|     handle_ = INVALID_HANDLE_VALUE; | |
|     return ::CloseHandle(h); | |
|   } | |
| 
 | |
|   bool is_valid() const { | |
|     return handle_ != INVALID_HANDLE_VALUE && handle_ != nullptr; | |
|   } | |
| 
 | |
|   HANDLE get() const { return handle_; } | |
| 
 | |
|   HANDLE Release() { | |
|     HANDLE h = handle_; | |
|     handle_ = INVALID_HANDLE_VALUE; | |
|     return h; | |
|   } | |
| 
 | |
|  private: | |
|   HANDLE handle_; | |
| }; | |
| 
 | |
| // Helper class to limit resource usage to avoid exhaustion. | |
| // Currently used to limit read-only file descriptors and mmap file usage | |
| // so that we do not run out of file descriptors or virtual memory, or run into | |
| // kernel performance problems for very large databases. | |
| class Limiter { | |
|  public: | |
|   // Limit maximum number of resources to |max_acquires|. | |
|   Limiter(int max_acquires) | |
|       : | |
| #if !defined(NDEBUG) | |
|         max_acquires_(max_acquires), | |
| #endif  // !defined(NDEBUG) | |
|         acquires_allowed_(max_acquires) { | |
|     assert(max_acquires >= 0); | |
|   } | |
| 
 | |
|   Limiter(const Limiter&) = delete; | |
|   Limiter operator=(const Limiter&) = delete; | |
| 
 | |
|   // If another resource is available, acquire it and return true. | |
|   // Else return false. | |
|   bool Acquire() { | |
|     int old_acquires_allowed = | |
|         acquires_allowed_.fetch_sub(1, std::memory_order_relaxed); | |
| 
 | |
|     if (old_acquires_allowed > 0) return true; | |
| 
 | |
|     acquires_allowed_.fetch_add(1, std::memory_order_relaxed); | |
|     return false; | |
|   } | |
| 
 | |
|   // Release a resource acquired by a previous call to Acquire() that returned | |
|   // true. | |
|   void Release() { | |
|     int old_acquires_allowed = | |
|         acquires_allowed_.fetch_add(1, std::memory_order_relaxed); | |
| 
 | |
|     // Silence compiler warnings about unused arguments when NDEBUG is defined. | |
|     (void)old_acquires_allowed; | |
|     // If the check below fails, Release() was called more times than acquire. | |
|     assert(old_acquires_allowed < max_acquires_); | |
|   } | |
| 
 | |
|  private: | |
| #if !defined(NDEBUG) | |
|   // Catches an excessive number of Release() calls. | |
|   const int max_acquires_; | |
| #endif  // !defined(NDEBUG) | |
|  | |
|   // The number of available resources. | |
|   // | |
|   // This is a counter and is not tied to the invariants of any other class, so | |
|   // it can be operated on safely using std::memory_order_relaxed. | |
|   std::atomic<int> acquires_allowed_; | |
| }; | |
| 
 | |
| class WindowsSequentialFile : public SequentialFile { | |
|  public: | |
|   WindowsSequentialFile(std::string filename, ScopedHandle handle) | |
|       : handle_(std::move(handle)), filename_(std::move(filename)) {} | |
|   ~WindowsSequentialFile() override {} | |
| 
 | |
|   Status Read(size_t n, Slice* result, char* scratch) override { | |
|     DWORD bytes_read; | |
|     // DWORD is 32-bit, but size_t could technically be larger. However leveldb | |
|     // files are limited to leveldb::Options::max_file_size which is clamped to | |
|     // 1<<30 or 1 GiB. | |
|     assert(n <= std::numeric_limits<DWORD>::max()); | |
|     if (!::ReadFile(handle_.get(), scratch, static_cast<DWORD>(n), &bytes_read, | |
|                     nullptr)) { | |
|       return WindowsError(filename_, ::GetLastError()); | |
|     } | |
| 
 | |
|     *result = Slice(scratch, bytes_read); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status Skip(uint64_t n) override { | |
|     LARGE_INTEGER distance; | |
|     distance.QuadPart = n; | |
|     if (!::SetFilePointerEx(handle_.get(), distance, nullptr, FILE_CURRENT)) { | |
|       return WindowsError(filename_, ::GetLastError()); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|  private: | |
|   const ScopedHandle handle_; | |
|   const std::string filename_; | |
| }; | |
| 
 | |
| class WindowsRandomAccessFile : public RandomAccessFile { | |
|  public: | |
|   WindowsRandomAccessFile(std::string filename, ScopedHandle handle) | |
|       : handle_(std::move(handle)), filename_(std::move(filename)) {} | |
| 
 | |
|   ~WindowsRandomAccessFile() override = default; | |
| 
 | |
|   Status Read(uint64_t offset, size_t n, Slice* result, | |
|               char* scratch) const override { | |
|     DWORD bytes_read = 0; | |
|     OVERLAPPED overlapped = {0}; | |
| 
 | |
|     overlapped.OffsetHigh = static_cast<DWORD>(offset >> 32); | |
|     overlapped.Offset = static_cast<DWORD>(offset); | |
|     if (!::ReadFile(handle_.get(), scratch, static_cast<DWORD>(n), &bytes_read, | |
|                     &overlapped)) { | |
|       DWORD error_code = ::GetLastError(); | |
|       if (error_code != ERROR_HANDLE_EOF) { | |
|         *result = Slice(scratch, 0); | |
|         return Status::IOError(filename_, GetWindowsErrorMessage(error_code)); | |
|       } | |
|     } | |
| 
 | |
|     *result = Slice(scratch, bytes_read); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|  private: | |
|   const ScopedHandle handle_; | |
|   const std::string filename_; | |
| }; | |
| 
 | |
| class WindowsMmapReadableFile : public RandomAccessFile { | |
|  public: | |
|   // base[0,length-1] contains the mmapped contents of the file. | |
|   WindowsMmapReadableFile(std::string filename, char* mmap_base, size_t length, | |
|                           Limiter* mmap_limiter) | |
|       : mmap_base_(mmap_base), | |
|         length_(length), | |
|         mmap_limiter_(mmap_limiter), | |
|         filename_(std::move(filename)) {} | |
| 
 | |
|   ~WindowsMmapReadableFile() override { | |
|     ::UnmapViewOfFile(mmap_base_); | |
|     mmap_limiter_->Release(); | |
|   } | |
| 
 | |
|   Status Read(uint64_t offset, size_t n, Slice* result, | |
|               char* scratch) const override { | |
|     if (offset + n > length_) { | |
|       *result = Slice(); | |
|       return WindowsError(filename_, ERROR_INVALID_PARAMETER); | |
|     } | |
| 
 | |
|     *result = Slice(mmap_base_ + offset, n); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|  private: | |
|   char* const mmap_base_; | |
|   const size_t length_; | |
|   Limiter* const mmap_limiter_; | |
|   const std::string filename_; | |
| }; | |
| 
 | |
| class WindowsWritableFile : public WritableFile { | |
|  public: | |
|   WindowsWritableFile(std::string filename, ScopedHandle handle) | |
|       : pos_(0), handle_(std::move(handle)), filename_(std::move(filename)) {} | |
| 
 | |
|   ~WindowsWritableFile() override = default; | |
| 
 | |
|   Status Append(const Slice& data) override { | |
|     size_t write_size = data.size(); | |
|     const char* write_data = data.data(); | |
| 
 | |
|     // Fit as much as possible into buffer. | |
|     size_t copy_size = std::min(write_size, kWritableFileBufferSize - pos_); | |
|     std::memcpy(buf_ + pos_, write_data, copy_size); | |
|     write_data += copy_size; | |
|     write_size -= copy_size; | |
|     pos_ += copy_size; | |
|     if (write_size == 0) { | |
|       return Status::OK(); | |
|     } | |
| 
 | |
|     // Can't fit in buffer, so need to do at least one write. | |
|     Status status = FlushBuffer(); | |
|     if (!status.ok()) { | |
|       return status; | |
|     } | |
| 
 | |
|     // Small writes go to buffer, large writes are written directly. | |
|     if (write_size < kWritableFileBufferSize) { | |
|       std::memcpy(buf_, write_data, write_size); | |
|       pos_ = write_size; | |
|       return Status::OK(); | |
|     } | |
|     return WriteUnbuffered(write_data, write_size); | |
|   } | |
| 
 | |
|   Status Close() override { | |
|     Status status = FlushBuffer(); | |
|     if (!handle_.Close() && status.ok()) { | |
|       status = WindowsError(filename_, ::GetLastError()); | |
|     } | |
|     return status; | |
|   } | |
| 
 | |
|   Status Flush() override { return FlushBuffer(); } | |
| 
 | |
|   Status Sync() override { | |
|     // On Windows no need to sync parent directory. Its metadata will be updated | |
|     // via the creation of the new file, without an explicit sync. | |
|  | |
|     Status status = FlushBuffer(); | |
|     if (!status.ok()) { | |
|       return status; | |
|     } | |
| 
 | |
|     if (!::FlushFileBuffers(handle_.get())) { | |
|       return Status::IOError(filename_, | |
|                              GetWindowsErrorMessage(::GetLastError())); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|  private: | |
|   Status FlushBuffer() { | |
|     Status status = WriteUnbuffered(buf_, pos_); | |
|     pos_ = 0; | |
|     return status; | |
|   } | |
| 
 | |
|   Status WriteUnbuffered(const char* data, size_t size) { | |
|     DWORD bytes_written; | |
|     if (!::WriteFile(handle_.get(), data, static_cast<DWORD>(size), | |
|                      &bytes_written, nullptr)) { | |
|       return Status::IOError(filename_, | |
|                              GetWindowsErrorMessage(::GetLastError())); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   // buf_[0, pos_-1] contains data to be written to handle_. | |
|   char buf_[kWritableFileBufferSize]; | |
|   size_t pos_; | |
| 
 | |
|   ScopedHandle handle_; | |
|   const std::string filename_; | |
| }; | |
| 
 | |
| // Lock or unlock the entire file as specified by |lock|. Returns true | |
| // when successful, false upon failure. Caller should call ::GetLastError() | |
| // to determine cause of failure | |
| bool LockOrUnlock(HANDLE handle, bool lock) { | |
|   if (lock) { | |
|     return ::LockFile(handle, | |
|                       /*dwFileOffsetLow=*/0, /*dwFileOffsetHigh=*/0, | |
|                       /*nNumberOfBytesToLockLow=*/MAXDWORD, | |
|                       /*nNumberOfBytesToLockHigh=*/MAXDWORD); | |
|   } else { | |
|     return ::UnlockFile(handle, | |
|                         /*dwFileOffsetLow=*/0, /*dwFileOffsetHigh=*/0, | |
|                         /*nNumberOfBytesToLockLow=*/MAXDWORD, | |
|                         /*nNumberOfBytesToLockHigh=*/MAXDWORD); | |
|   } | |
| } | |
| 
 | |
| class WindowsFileLock : public FileLock { | |
|  public: | |
|   WindowsFileLock(ScopedHandle handle, std::string filename) | |
|       : handle_(std::move(handle)), filename_(std::move(filename)) {} | |
| 
 | |
|   const ScopedHandle& handle() const { return handle_; } | |
|   const std::string& filename() const { return filename_; } | |
| 
 | |
|  private: | |
|   const ScopedHandle handle_; | |
|   const std::string filename_; | |
| }; | |
| 
 | |
| class WindowsEnv : public Env { | |
|  public: | |
|   WindowsEnv(); | |
|   ~WindowsEnv() override { | |
|     static const char msg[] = | |
|         "WindowsEnv singleton destroyed. Unsupported behavior!\n"; | |
|     std::fwrite(msg, 1, sizeof(msg), stderr); | |
|     std::abort(); | |
|   } | |
| 
 | |
|   Status NewSequentialFile(const std::string& filename, | |
|                            SequentialFile** result) override { | |
|     *result = nullptr; | |
|     DWORD desired_access = GENERIC_READ; | |
|     DWORD share_mode = FILE_SHARE_READ; | |
|     ScopedHandle handle = ::CreateFileA( | |
|         filename.c_str(), desired_access, share_mode, | |
|         /*lpSecurityAttributes=*/nullptr, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, | |
|         /*hTemplateFile=*/nullptr); | |
|     if (!handle.is_valid()) { | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
| 
 | |
|     *result = new WindowsSequentialFile(filename, std::move(handle)); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status NewRandomAccessFile(const std::string& filename, | |
|                              RandomAccessFile** result) override { | |
|     *result = nullptr; | |
|     DWORD desired_access = GENERIC_READ; | |
|     DWORD share_mode = FILE_SHARE_READ; | |
|     ScopedHandle handle = | |
|         ::CreateFileA(filename.c_str(), desired_access, share_mode, | |
|                       /*lpSecurityAttributes=*/nullptr, OPEN_EXISTING, | |
|                       FILE_ATTRIBUTE_READONLY, | |
|                       /*hTemplateFile=*/nullptr); | |
|     if (!handle.is_valid()) { | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
|     if (!mmap_limiter_.Acquire()) { | |
|       *result = new WindowsRandomAccessFile(filename, std::move(handle)); | |
|       return Status::OK(); | |
|     } | |
| 
 | |
|     LARGE_INTEGER file_size; | |
|     Status status; | |
|     if (!::GetFileSizeEx(handle.get(), &file_size)) { | |
|       mmap_limiter_.Release(); | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
| 
 | |
|     ScopedHandle mapping = | |
|         ::CreateFileMappingA(handle.get(), | |
|                              /*security attributes=*/nullptr, PAGE_READONLY, | |
|                              /*dwMaximumSizeHigh=*/0, | |
|                              /*dwMaximumSizeLow=*/0, | |
|                              /*lpName=*/nullptr); | |
|     if (mapping.is_valid()) { | |
|       void* mmap_base = ::MapViewOfFile(mapping.get(), FILE_MAP_READ, | |
|                                         /*dwFileOffsetHigh=*/0, | |
|                                         /*dwFileOffsetLow=*/0, | |
|                                         /*dwNumberOfBytesToMap=*/0); | |
|       if (mmap_base) { | |
|         *result = new WindowsMmapReadableFile( | |
|             filename, reinterpret_cast<char*>(mmap_base), | |
|             static_cast<size_t>(file_size.QuadPart), &mmap_limiter_); | |
|         return Status::OK(); | |
|       } | |
|     } | |
|     mmap_limiter_.Release(); | |
|     return WindowsError(filename, ::GetLastError()); | |
|   } | |
| 
 | |
|   Status NewWritableFile(const std::string& filename, | |
|                          WritableFile** result) override { | |
|     DWORD desired_access = GENERIC_WRITE; | |
|     DWORD share_mode = 0;  // Exclusive access. | |
|     ScopedHandle handle = ::CreateFileA( | |
|         filename.c_str(), desired_access, share_mode, | |
|         /*lpSecurityAttributes=*/nullptr, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, | |
|         /*hTemplateFile=*/nullptr); | |
|     if (!handle.is_valid()) { | |
|       *result = nullptr; | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
| 
 | |
|     *result = new WindowsWritableFile(filename, std::move(handle)); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status NewAppendableFile(const std::string& filename, | |
|                            WritableFile** result) override { | |
|     DWORD desired_access = FILE_APPEND_DATA; | |
|     DWORD share_mode = 0;  // Exclusive access. | |
|     ScopedHandle handle = ::CreateFileA( | |
|         filename.c_str(), desired_access, share_mode, | |
|         /*lpSecurityAttributes=*/nullptr, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, | |
|         /*hTemplateFile=*/nullptr); | |
|     if (!handle.is_valid()) { | |
|       *result = nullptr; | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
| 
 | |
|     *result = new WindowsWritableFile(filename, std::move(handle)); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   bool FileExists(const std::string& filename) override { | |
|     return GetFileAttributesA(filename.c_str()) != INVALID_FILE_ATTRIBUTES; | |
|   } | |
| 
 | |
|   Status GetChildren(const std::string& directory_path, | |
|                      std::vector<std::string>* result) override { | |
|     const std::string find_pattern = directory_path + "\\*"; | |
|     WIN32_FIND_DATAA find_data; | |
|     HANDLE dir_handle = ::FindFirstFileA(find_pattern.c_str(), &find_data); | |
|     if (dir_handle == INVALID_HANDLE_VALUE) { | |
|       DWORD last_error = ::GetLastError(); | |
|       if (last_error == ERROR_FILE_NOT_FOUND) { | |
|         return Status::OK(); | |
|       } | |
|       return WindowsError(directory_path, last_error); | |
|     } | |
|     do { | |
|       char base_name[_MAX_FNAME]; | |
|       char ext[_MAX_EXT]; | |
| 
 | |
|       if (!_splitpath_s(find_data.cFileName, nullptr, 0, nullptr, 0, base_name, | |
|                         ARRAYSIZE(base_name), ext, ARRAYSIZE(ext))) { | |
|         result->emplace_back(std::string(base_name) + ext); | |
|       } | |
|     } while (::FindNextFileA(dir_handle, &find_data)); | |
|     DWORD last_error = ::GetLastError(); | |
|     ::FindClose(dir_handle); | |
|     if (last_error != ERROR_NO_MORE_FILES) { | |
|       return WindowsError(directory_path, last_error); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status RemoveFile(const std::string& filename) override { | |
|     if (!::DeleteFileA(filename.c_str())) { | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status CreateDir(const std::string& dirname) override { | |
|     if (!::CreateDirectoryA(dirname.c_str(), nullptr)) { | |
|       return WindowsError(dirname, ::GetLastError()); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status RemoveDir(const std::string& dirname) override { | |
|     if (!::RemoveDirectoryA(dirname.c_str())) { | |
|       return WindowsError(dirname, ::GetLastError()); | |
|     } | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status GetFileSize(const std::string& filename, uint64_t* size) override { | |
|     WIN32_FILE_ATTRIBUTE_DATA file_attributes; | |
|     if (!::GetFileAttributesExA(filename.c_str(), GetFileExInfoStandard, | |
|                                 &file_attributes)) { | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } | |
|     ULARGE_INTEGER file_size; | |
|     file_size.HighPart = file_attributes.nFileSizeHigh; | |
|     file_size.LowPart = file_attributes.nFileSizeLow; | |
|     *size = file_size.QuadPart; | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status RenameFile(const std::string& from, const std::string& to) override { | |
|     // Try a simple move first. It will only succeed when |to| doesn't already | |
|     // exist. | |
|     if (::MoveFileA(from.c_str(), to.c_str())) { | |
|       return Status::OK(); | |
|     } | |
|     DWORD move_error = ::GetLastError(); | |
| 
 | |
|     // Try the full-blown replace if the move fails, as ReplaceFile will only | |
|     // succeed when |to| does exist. When writing to a network share, we may not | |
|     // be able to change the ACLs. Ignore ACL errors then | |
|     // (REPLACEFILE_IGNORE_MERGE_ERRORS). | |
|     if (::ReplaceFileA(to.c_str(), from.c_str(), /*lpBackupFileName=*/nullptr, | |
|                        REPLACEFILE_IGNORE_MERGE_ERRORS, | |
|                        /*lpExclude=*/nullptr, /*lpReserved=*/nullptr)) { | |
|       return Status::OK(); | |
|     } | |
|     DWORD replace_error = ::GetLastError(); | |
|     // In the case of FILE_ERROR_NOT_FOUND from ReplaceFile, it is likely that | |
|     // |to| does not exist. In this case, the more relevant error comes from the | |
|     // call to MoveFile. | |
|     if (replace_error == ERROR_FILE_NOT_FOUND || | |
|         replace_error == ERROR_PATH_NOT_FOUND) { | |
|       return WindowsError(from, move_error); | |
|     } else { | |
|       return WindowsError(from, replace_error); | |
|     } | |
|   } | |
| 
 | |
|   Status LockFile(const std::string& filename, FileLock** lock) override { | |
|     *lock = nullptr; | |
|     Status result; | |
|     ScopedHandle handle = ::CreateFileA( | |
|         filename.c_str(), GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, | |
|         /*lpSecurityAttributes=*/nullptr, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, | |
|         nullptr); | |
|     if (!handle.is_valid()) { | |
|       result = WindowsError(filename, ::GetLastError()); | |
|     } else if (!LockOrUnlock(handle.get(), true)) { | |
|       result = WindowsError("lock " + filename, ::GetLastError()); | |
|     } else { | |
|       *lock = new WindowsFileLock(std::move(handle), filename); | |
|     } | |
|     return result; | |
|   } | |
| 
 | |
|   Status UnlockFile(FileLock* lock) override { | |
|     WindowsFileLock* windows_file_lock = | |
|         reinterpret_cast<WindowsFileLock*>(lock); | |
|     if (!LockOrUnlock(windows_file_lock->handle().get(), false)) { | |
|       return WindowsError("unlock " + windows_file_lock->filename(), | |
|                           ::GetLastError()); | |
|     } | |
|     delete windows_file_lock; | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   void Schedule(void (*background_work_function)(void* background_work_arg), | |
|                 void* background_work_arg) override; | |
| 
 | |
|   void StartThread(void (*thread_main)(void* thread_main_arg), | |
|                    void* thread_main_arg) override { | |
|     std::thread new_thread(thread_main, thread_main_arg); | |
|     new_thread.detach(); | |
|   } | |
| 
 | |
|   Status GetTestDirectory(std::string* result) override { | |
|     const char* env = getenv("TEST_TMPDIR"); | |
|     if (env && env[0] != '\0') { | |
|       *result = env; | |
|       return Status::OK(); | |
|     } | |
| 
 | |
|     char tmp_path[MAX_PATH]; | |
|     if (!GetTempPathA(ARRAYSIZE(tmp_path), tmp_path)) { | |
|       return WindowsError("GetTempPath", ::GetLastError()); | |
|     } | |
|     std::stringstream ss; | |
|     ss << tmp_path << "leveldbtest-" << std::this_thread::get_id(); | |
|     *result = ss.str(); | |
| 
 | |
|     // Directory may already exist | |
|     CreateDir(*result); | |
|     return Status::OK(); | |
|   } | |
| 
 | |
|   Status NewLogger(const std::string& filename, Logger** result) override { | |
|     std::FILE* fp = std::fopen(filename.c_str(), "wN"); | |
|     if (fp == nullptr) { | |
|       *result = nullptr; | |
|       return WindowsError(filename, ::GetLastError()); | |
|     } else { | |
|       *result = new WindowsLogger(fp); | |
|       return Status::OK(); | |
|     } | |
|   } | |
| 
 | |
|   uint64_t NowMicros() override { | |
|     // GetSystemTimeAsFileTime typically has a resolution of 10-20 msec. | |
|     // TODO(cmumford): Switch to GetSystemTimePreciseAsFileTime which is | |
|     // available in Windows 8 and later. | |
|     FILETIME ft; | |
|     ::GetSystemTimeAsFileTime(&ft); | |
|     // Each tick represents a 100-nanosecond intervals since January 1, 1601 | |
|     // (UTC). | |
|     uint64_t num_ticks = | |
|         (static_cast<uint64_t>(ft.dwHighDateTime) << 32) + ft.dwLowDateTime; | |
|     return num_ticks / 10; | |
|   } | |
| 
 | |
|   void SleepForMicroseconds(int micros) override { | |
|     std::this_thread::sleep_for(std::chrono::microseconds(micros)); | |
|   } | |
| 
 | |
|  private: | |
|   void BackgroundThreadMain(); | |
| 
 | |
|   static void BackgroundThreadEntryPoint(WindowsEnv* env) { | |
|     env->BackgroundThreadMain(); | |
|   } | |
| 
 | |
|   // Stores the work item data in a Schedule() call. | |
|   // | |
|   // Instances are constructed on the thread calling Schedule() and used on the | |
|   // background thread. | |
|   // | |
|   // This structure is thread-safe because it is immutable. | |
|   struct BackgroundWorkItem { | |
|     explicit BackgroundWorkItem(void (*function)(void* arg), void* arg) | |
|         : function(function), arg(arg) {} | |
| 
 | |
|     void (*const function)(void*); | |
|     void* const arg; | |
|   }; | |
| 
 | |
|   port::Mutex background_work_mutex_; | |
|   port::CondVar background_work_cv_ GUARDED_BY(background_work_mutex_); | |
|   bool started_background_thread_ GUARDED_BY(background_work_mutex_); | |
| 
 | |
|   std::queue<BackgroundWorkItem> background_work_queue_ | |
|       GUARDED_BY(background_work_mutex_); | |
| 
 | |
|   Limiter mmap_limiter_;  // Thread-safe. | |
| }; | |
| 
 | |
| // Return the maximum number of concurrent mmaps. | |
| int MaxMmaps() { return g_mmap_limit; } | |
| 
 | |
| WindowsEnv::WindowsEnv() | |
|     : background_work_cv_(&background_work_mutex_), | |
|       started_background_thread_(false), | |
|       mmap_limiter_(MaxMmaps()) {} | |
| 
 | |
| void WindowsEnv::Schedule( | |
|     void (*background_work_function)(void* background_work_arg), | |
|     void* background_work_arg) { | |
|   background_work_mutex_.Lock(); | |
| 
 | |
|   // Start the background thread, if we haven't done so already. | |
|   if (!started_background_thread_) { | |
|     started_background_thread_ = true; | |
|     std::thread background_thread(WindowsEnv::BackgroundThreadEntryPoint, this); | |
|     background_thread.detach(); | |
|   } | |
| 
 | |
|   // If the queue is empty, the background thread may be waiting for work. | |
|   if (background_work_queue_.empty()) { | |
|     background_work_cv_.Signal(); | |
|   } | |
| 
 | |
|   background_work_queue_.emplace(background_work_function, background_work_arg); | |
|   background_work_mutex_.Unlock(); | |
| } | |
| 
 | |
| void WindowsEnv::BackgroundThreadMain() { | |
|   while (true) { | |
|     background_work_mutex_.Lock(); | |
| 
 | |
|     // Wait until there is work to be done. | |
|     while (background_work_queue_.empty()) { | |
|       background_work_cv_.Wait(); | |
|     } | |
| 
 | |
|     assert(!background_work_queue_.empty()); | |
|     auto background_work_function = background_work_queue_.front().function; | |
|     void* background_work_arg = background_work_queue_.front().arg; | |
|     background_work_queue_.pop(); | |
| 
 | |
|     background_work_mutex_.Unlock(); | |
|     background_work_function(background_work_arg); | |
|   } | |
| } | |
| 
 | |
| // Wraps an Env instance whose destructor is never created. | |
| // | |
| // Intended usage: | |
| //   using PlatformSingletonEnv = SingletonEnv<PlatformEnv>; | |
| //   void ConfigurePosixEnv(int param) { | |
| //     PlatformSingletonEnv::AssertEnvNotInitialized(); | |
| //     // set global configuration flags. | |
| //   } | |
| //   Env* Env::Default() { | |
| //     static PlatformSingletonEnv default_env; | |
| //     return default_env.env(); | |
| //   } | |
| template <typename EnvType> | |
| class SingletonEnv { | |
|  public: | |
|   SingletonEnv() { | |
| #if !defined(NDEBUG) | |
|     env_initialized_.store(true, std::memory_order_relaxed); | |
| #endif  // !defined(NDEBUG) | |
|     static_assert(sizeof(env_storage_) >= sizeof(EnvType), | |
|                   "env_storage_ will not fit the Env"); | |
|     static_assert(alignof(decltype(env_storage_)) >= alignof(EnvType), | |
|                   "env_storage_ does not meet the Env's alignment needs"); | |
|     new (&env_storage_) EnvType(); | |
|   } | |
|   ~SingletonEnv() = default; | |
| 
 | |
|   SingletonEnv(const SingletonEnv&) = delete; | |
|   SingletonEnv& operator=(const SingletonEnv&) = delete; | |
| 
 | |
|   Env* env() { return reinterpret_cast<Env*>(&env_storage_); } | |
| 
 | |
|   static void AssertEnvNotInitialized() { | |
| #if !defined(NDEBUG) | |
|     assert(!env_initialized_.load(std::memory_order_relaxed)); | |
| #endif  // !defined(NDEBUG) | |
|   } | |
| 
 | |
|  private: | |
|   typename std::aligned_storage<sizeof(EnvType), alignof(EnvType)>::type | |
|       env_storage_; | |
| #if !defined(NDEBUG) | |
|   static std::atomic<bool> env_initialized_; | |
| #endif  // !defined(NDEBUG) | |
| }; | |
| 
 | |
| #if !defined(NDEBUG) | |
| template <typename EnvType> | |
| std::atomic<bool> SingletonEnv<EnvType>::env_initialized_; | |
| #endif  // !defined(NDEBUG) | |
|  | |
| using WindowsDefaultEnv = SingletonEnv<WindowsEnv>; | |
| 
 | |
| }  // namespace | |
|  | |
| void EnvWindowsTestHelper::SetReadOnlyMMapLimit(int limit) { | |
|   WindowsDefaultEnv::AssertEnvNotInitialized(); | |
|   g_mmap_limit = limit; | |
| } | |
| 
 | |
| Env* Env::Default() { | |
|   static WindowsDefaultEnv env_container; | |
|   return env_container.env(); | |
| } | |
| 
 | |
| }  // namespace leveldb
 |