| // Copyright 2018 The Bazel Authors. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| #ifndef BAZEL_SRC_TOOLS_SINGLEJAR_MAPPED_FILE_WINDOWS_H_ |
| #define BAZEL_SRC_TOOLS_SINGLEJAR_MAPPED_FILE_WINDOWS_H_ 1 |
| |
| #if !defined(_WIN64) |
| #error This code is for 64 bit Windows. |
| #endif |
| |
| #include "src/main/cpp/util/path_platform.h" |
| #include "src/tools/singlejar/diag.h" |
| |
| #ifndef WIN32_LEAN_AND_MEAN |
| #define WIN32_LEAN_AND_MEAN |
| #endif |
| |
| #include <windows.h> |
| |
| #include <string> |
| |
| MappedFile::MappedFile() |
| : mapped_start_(nullptr), |
| mapped_end_(nullptr), |
| hFile_(INVALID_HANDLE_VALUE), |
| hMapFile_(INVALID_HANDLE_VALUE) {} |
| |
| bool MappedFile::is_open() const { return hFile_ != INVALID_HANDLE_VALUE; } |
| |
| bool MappedFile::Open(const std::string& path) { |
| if (is_open()) { |
| diag_errx(1, "%s:%d: This instance is already open", __FILE__, __LINE__); |
| } |
| |
| std::wstring wpath; |
| std::string error; |
| if (!blaze_util::AsAbsoluteWindowsPath(path, &wpath, &error)) { |
| diag_warn("%s:%d: AsAbsoluteWindowsPath failed: %s", __FILE__, __LINE__, |
| error.c_str()); |
| return false; |
| } |
| |
| hFile_ = CreateFileW(wpath.c_str(), GENERIC_READ, FILE_SHARE_READ, NULL, |
| OPEN_EXISTING, 0, NULL); |
| if (hFile_ == INVALID_HANDLE_VALUE) { |
| diag_warn("%s:%d: CreateFileW failed for %S", __FILE__, __LINE__, |
| wpath.c_str()); |
| return false; |
| } |
| |
| LARGE_INTEGER temp; |
| ::GetFileSizeEx(hFile_, &temp); |
| size_t fileSize = temp.QuadPart; |
| |
| if (fileSize == 0) { |
| // Handle empty files specially, because CreateFileMapping cannot map them: |
| // |
| // From |
| // https://docs.microsoft.com/en-us/windows/desktop/api/winbase/nf-winbase-createfilemappinga |
| // |
| // An attempt to map a file with a length of 0 (zero) fails with an error |
| // code of ERROR_FILE_INVALID. Applications should test for files with a |
| // length of 0 (zero) and reject those files. |
| mapped_start_ = nullptr; |
| mapped_end_ = nullptr; |
| hMapFile_ = INVALID_HANDLE_VALUE; |
| return true; |
| } |
| |
| hMapFile_ = ::CreateFileMapping( |
| hFile_, |
| nullptr, // default security |
| PAGE_READONLY, // read-only permission |
| static_cast<DWORD>(fileSize >> 32), // size of mapping object, high |
| static_cast<DWORD>(fileSize), // size of mapping object, low |
| nullptr); // name of mapping object |
| |
| if (hMapFile_ == nullptr) { |
| diag_warn("%s:%d: CreateFileMapping for %s failed", __FILE__, __LINE__, |
| path.c_str()); |
| ::CloseHandle(hFile_); |
| hFile_ = INVALID_HANDLE_VALUE; |
| return false; |
| } |
| |
| mapped_start_ = static_cast<unsigned char*>( |
| MapViewOfFile(hMapFile_, |
| FILE_MAP_READ | FILE_MAP_COPY, // PROT_READ | MAP_PRIVATE |
| 0, // file offset, high |
| 0, // file offset, low |
| fileSize)); // file size |
| if (mapped_start_ == nullptr) { |
| diag_warn("%s:%d: MapViewOfFile for %s failed", __FILE__, __LINE__, |
| path.c_str()); |
| ::CloseHandle(hMapFile_); |
| ::CloseHandle(hFile_); |
| hFile_ = INVALID_HANDLE_VALUE; |
| hMapFile_ = INVALID_HANDLE_VALUE; |
| return false; |
| } |
| |
| mapped_end_ = mapped_start_ + fileSize; |
| return true; |
| } |
| |
| void MappedFile::Close() { |
| if (is_open()) { |
| if (mapped_start_) { |
| ::UnmapViewOfFile(mapped_start_); |
| } |
| if (hMapFile_ != INVALID_HANDLE_VALUE) { |
| ::CloseHandle(hMapFile_); |
| hMapFile_ = INVALID_HANDLE_VALUE; |
| } |
| ::CloseHandle(hFile_); |
| hFile_ = INVALID_HANDLE_VALUE; |
| mapped_start_ = mapped_end_ = nullptr; |
| } |
| } |
| |
| #endif // BAZEL_SRC_TOOLS_SINGLEJAR_MAPPED_FILE_WINDOWS_H_ |