// Copyright 2015 The Crashpad 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 "util/file/file_reader.h" #include "base/logging.h" #include "base/numerics/safe_conversions.h" #include "build/build_config.h" namespace crashpad { bool FileReaderInterface::ReadExactly(void* data, size_t size) { FileOperationResult expect = base::checked_cast(size); FileOperationResult rv = Read(data, size); if (rv < 0) { // Read() will have logged its own error. return false; } else if (rv != expect) { LOG(ERROR) << "ReadExactly(): expected " << expect << ", observed " << rv; return false; } return true; } WeakFileHandleFileReader::WeakFileHandleFileReader(FileHandle file_handle) : file_handle_(file_handle) { } WeakFileHandleFileReader::~WeakFileHandleFileReader() { } FileOperationResult WeakFileHandleFileReader::Read(void* data, size_t size) { DCHECK_NE(file_handle_, kInvalidFileHandle); // Don’t use LoggingReadFile(), which insists on a full read and only returns // a bool. This method permits short reads and returns the number of bytes // read. base::checked_cast(size); FileOperationResult rv = ReadFile(file_handle_, data, size); if (rv < 0) { PLOG(ERROR) << "read"; return -1; } return rv; } FileOffset WeakFileHandleFileReader::Seek(FileOffset offset, int whence) { DCHECK_NE(file_handle_, kInvalidFileHandle); return LoggingSeekFile(file_handle_, offset, whence); } FileReader::FileReader() : file_(), weak_file_handle_file_reader_(kInvalidFileHandle) { } FileReader::~FileReader() { } bool FileReader::Open(const base::FilePath& path) { CHECK(!file_.is_valid()); file_.reset(LoggingOpenFileForRead(path)); if (!file_.is_valid()) { return false; } weak_file_handle_file_reader_.set_file_handle(file_.get()); return true; } void FileReader::Close() { CHECK(file_.is_valid()); weak_file_handle_file_reader_.set_file_handle(kInvalidFileHandle); file_.reset(); } FileOperationResult FileReader::Read(void* data, size_t size) { DCHECK(file_.is_valid()); return weak_file_handle_file_reader_.Read(data, size); } FileOffset FileReader::Seek(FileOffset offset, int whence) { DCHECK(file_.is_valid()); return weak_file_handle_file_reader_.Seek(offset, whence); } WeakStdioFileReader::WeakStdioFileReader(FILE* file) : file_(file) { } WeakStdioFileReader::~WeakStdioFileReader() { } FileOperationResult WeakStdioFileReader::Read(void* data, size_t size) { DCHECK(file_); size_t rv = fread(data, 1, size, file_); if (rv != size && ferror(file_)) { STDIO_PLOG(ERROR) << "fread"; return -1; } if (rv > size) { LOG(ERROR) << "fread: expected " << size << ", observed " << rv; return -1; } return rv; } FileOffset WeakStdioFileReader::Seek(FileOffset offset, int whence) { DCHECK(file_); if (fseeko(file_, offset, whence) == -1) { STDIO_PLOG(ERROR) << "fseeko"; return -1; } FileOffset new_offset = ftello(file_); if (new_offset == -1) { STDIO_PLOG(ERROR) << "ftello"; return -1; } return new_offset; } } // namespace crashpad