|  | // Copyright 2015 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. | 
|  |  | 
|  | #include <stdio.h> | 
|  | #include <windows.h> | 
|  |  | 
|  | #include <string> | 
|  |  | 
|  | #include "src/main/cpp/util/errors.h" | 
|  | #include "src/main/cpp/util/file_platform.h" | 
|  | #include "src/main/cpp/util/logging.h" | 
|  | #include "src/main/cpp/util/strings.h" | 
|  | #include "third_party/ijar/mapped_file.h" | 
|  |  | 
|  | #define MAX_ERROR 2048 | 
|  |  | 
|  | namespace devtools_ijar { | 
|  |  | 
|  | using std::string; | 
|  | using std::wstring; | 
|  |  | 
|  | static char errmsg[MAX_ERROR] = ""; | 
|  |  | 
|  | struct MappedInputFileImpl { | 
|  | HANDLE file_; | 
|  | HANDLE mapping_; | 
|  |  | 
|  | MappedInputFileImpl(HANDLE file, HANDLE mapping) { | 
|  | file_ = file; | 
|  | mapping_ = mapping; | 
|  | } | 
|  | }; | 
|  |  | 
|  | MappedInputFile::MappedInputFile(const char* name) { | 
|  | impl_ = NULL; | 
|  | opened_ = false; | 
|  | errmsg_ = errmsg; | 
|  |  | 
|  | wstring wname; | 
|  | string error; | 
|  | if (!blaze_util::AsAbsoluteWindowsPath(name, &wname, &error)) { | 
|  | BAZEL_DIE(255) << "MappedInputFile(" << name | 
|  | << "): AsAbsoluteWindowsPath failed: " << error; | 
|  | } | 
|  | HANDLE file = CreateFileW(wname.c_str(), GENERIC_READ, FILE_SHARE_READ, NULL, | 
|  | OPEN_EXISTING, 0, NULL); | 
|  | if (file == INVALID_HANDLE_VALUE) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile(" << name << "): CreateFileW(" | 
|  | << blaze_util::WstringToString(wname) | 
|  | << ") failed: " << errormsg; | 
|  | } | 
|  |  | 
|  | LARGE_INTEGER size; | 
|  | if (!GetFileSizeEx(file, &size)) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile(" << name | 
|  | << "): GetFileSizeEx failed: " << errormsg; | 
|  | } | 
|  |  | 
|  | HANDLE mapping = CreateFileMapping(file, NULL, PAGE_READONLY, | 
|  | size.HighPart, size.LowPart, NULL); | 
|  | if (mapping == NULL || mapping == INVALID_HANDLE_VALUE) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile(" << name | 
|  | << "): CreateFileMapping failed: " << errormsg; | 
|  | } | 
|  |  | 
|  | void *view = MapViewOfFileEx(mapping, FILE_MAP_READ, 0, 0, 0, NULL); | 
|  | if (view == NULL) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile(" << name | 
|  | << "): MapViewOfFileEx failed: " << errormsg; | 
|  | } | 
|  |  | 
|  | impl_ = new MappedInputFileImpl(file, mapping); | 
|  | length_ = size.QuadPart; | 
|  | buffer_ = reinterpret_cast<u1*>(view); | 
|  | opened_ = true; | 
|  | } | 
|  |  | 
|  | MappedInputFile::~MappedInputFile() { | 
|  | delete impl_; | 
|  | } | 
|  |  | 
|  | void MappedInputFile::Discard(size_t bytes) { | 
|  | // This is not supported on Windows for now. I'm not sure if we can unmap | 
|  | // parts of an existing view and that this is necessary for Windows at all. | 
|  | // At any rate, this only matters for >2GB (or maybe >4GB?) input files. | 
|  | } | 
|  |  | 
|  | int MappedInputFile::Close() { | 
|  | if (!UnmapViewOfFile(buffer_)) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile::Close: UnmapViewOfFile failed: " | 
|  | << errormsg; | 
|  | } | 
|  |  | 
|  | if (!CloseHandle(impl_->mapping_)) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile::Close: CloseHandle for mapping failed: " | 
|  | << errormsg; | 
|  | } | 
|  |  | 
|  | if (!CloseHandle(impl_->file_)) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedInputFile::Close: CloseHandle for file failed: " | 
|  | << errormsg; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | struct MappedOutputFileImpl { | 
|  | HANDLE file_; | 
|  | HANDLE mapping_; | 
|  |  | 
|  | MappedOutputFileImpl(HANDLE file, HANDLE mapping) { | 
|  | file_ = file; | 
|  | mapping_ = mapping; | 
|  | } | 
|  | }; | 
|  |  | 
|  | MappedOutputFile::MappedOutputFile(const char* name, size_t estimated_size) { | 
|  | impl_ = NULL; | 
|  | opened_ = false; | 
|  | errmsg_ = errmsg; | 
|  |  | 
|  | wstring wname; | 
|  | string error; | 
|  | if (!blaze_util::AsAbsoluteWindowsPath(name, &wname, &error)) { | 
|  | BAZEL_DIE(255) << "MappedOutputFile(" << name | 
|  | << "): AsAbsoluteWindowsPath failed: " << error; | 
|  | } | 
|  | HANDLE file = CreateFileW(wname.c_str(), GENERIC_READ | GENERIC_WRITE, 0, | 
|  | NULL, CREATE_ALWAYS, 0, NULL); | 
|  | if (file == INVALID_HANDLE_VALUE) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedOutputFile(" << name << "): CreateFileW(" | 
|  | << blaze_util::WstringToString(wname) | 
|  | << ") failed: " << errormsg; | 
|  | } | 
|  |  | 
|  | HANDLE mapping = CreateFileMapping(file, NULL, PAGE_READWRITE, | 
|  | estimated_size >> 32, estimated_size & 0xffffffffUL, NULL); | 
|  | if (mapping == NULL || mapping == INVALID_HANDLE_VALUE) { | 
|  | BAZEL_DIE(255) << "MappedOutputFile(" << name | 
|  | << "): CreateFileMapping failed"; | 
|  | } | 
|  |  | 
|  | void *view = MapViewOfFileEx(mapping, FILE_MAP_ALL_ACCESS, 0, 0, 0, NULL); | 
|  | if (view == NULL) { | 
|  | string errormsg = blaze_util::GetLastErrorString(); | 
|  | BAZEL_DIE(255) << "MappedOutputFile(" << name | 
|  | << "): MapViewOfFileEx failed: " << errormsg; | 
|  | CloseHandle(mapping); | 
|  | CloseHandle(file); | 
|  | return; | 
|  | } | 
|  |  | 
|  | impl_ = new MappedOutputFileImpl(file, mapping); | 
|  | buffer_ = reinterpret_cast<u1*>(view); | 
|  | opened_ = true; | 
|  | } | 
|  |  | 
|  | MappedOutputFile::~MappedOutputFile() { | 
|  | delete impl_; | 
|  | } | 
|  |  | 
|  | int MappedOutputFile::Close(size_t size) { | 
|  | if (!UnmapViewOfFile(buffer_)) { | 
|  | BAZEL_DIE(255) << "MappedOutputFile::Close: UnmapViewOfFile failed: " | 
|  | << blaze_util::GetLastErrorString(); | 
|  | } | 
|  |  | 
|  | if (!CloseHandle(impl_->mapping_)) { | 
|  | BAZEL_DIE(255) | 
|  | << "MappedOutputFile::Close: CloseHandle for mapping failed: " | 
|  | << blaze_util::GetLastErrorString(); | 
|  | } | 
|  |  | 
|  | if (!SetFilePointer(impl_->file_, size, NULL, FILE_BEGIN)) { | 
|  | BAZEL_DIE(255) << "MappedOutputFile::Close: SetFilePointer failed: " | 
|  | << blaze_util::GetLastErrorString(); | 
|  | } | 
|  |  | 
|  | if (!SetEndOfFile(impl_->file_)) { | 
|  | BAZEL_DIE(255) << "MappedOutputFile::Close: SetEndOfFile failed: " | 
|  | << blaze_util::GetLastErrorString(); | 
|  | } | 
|  |  | 
|  | if (!CloseHandle(impl_->file_)) { | 
|  | BAZEL_DIE(255) << "MappedOutputFile::Close: CloseHandle for file failed: " | 
|  | << blaze_util::GetLastErrorString(); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | }  // namespace devtools_ijar |