|  | // 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_ |