/* * Copyright (C) 2017 The Android Open Source Project * * 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 "androidfw/FileStream.h" #include // for errno #include // for O_RDONLY #include // for read #include "android-base/errors.h" #include "android-base/file.h" // for O_BINARY #include "android-base/logging.h" #include "android-base/macros.h" #include "android-base/utf8.h" #if defined(_WIN32) // This is only needed for O_CLOEXEC. #include #define O_CLOEXEC O_NOINHERIT #endif using ::android::base::SystemErrorCodeToString; using ::android::base::unique_fd; namespace android { FileInputStream::FileInputStream(const std::string& path, size_t buffer_capacity) : should_close_(true), buffer_capacity_(buffer_capacity) { int mode = O_RDONLY | O_CLOEXEC | O_BINARY; fd_ = TEMP_FAILURE_RETRY(::android::base::utf8::open(path.c_str(), mode)); if (fd_ == -1) { error_ = SystemErrorCodeToString(errno); } else { buffer_.reset(new uint8_t[buffer_capacity_]); } } FileInputStream::FileInputStream(int fd, size_t buffer_capacity) : fd_(fd), should_close_(true), buffer_capacity_(buffer_capacity) { if (fd_ < 0) { error_ = "Bad File Descriptor"; } else { buffer_.reset(new uint8_t[buffer_capacity_]); } } FileInputStream::FileInputStream(android::base::borrowed_fd fd, size_t buffer_capacity) : fd_(fd.get()), should_close_(false), buffer_capacity_(buffer_capacity) { if (fd_ < 0) { error_ = "Bad File Descriptor"; } else { buffer_.reset(new uint8_t[buffer_capacity_]); } } bool FileInputStream::Next(const void** data, size_t* size) { if (HadError()) { return false; } // Deal with any remaining bytes after BackUp was called. if (buffer_offset_ != buffer_size_) { *data = buffer_.get() + buffer_offset_; *size = buffer_size_ - buffer_offset_; total_byte_count_ += buffer_size_ - buffer_offset_; buffer_offset_ = buffer_size_; return true; } ssize_t n = TEMP_FAILURE_RETRY(read(fd_, buffer_.get(), buffer_capacity_)); if (n < 0) { error_ = SystemErrorCodeToString(errno); if (fd_ != -1) { if (should_close_) { close(fd_); } fd_ = -1; } buffer_.reset(); return false; } buffer_size_ = static_cast(n); buffer_offset_ = buffer_size_; total_byte_count_ += buffer_size_; *data = buffer_.get(); *size = buffer_size_; return buffer_size_ != 0u; } void FileInputStream::BackUp(size_t count) { if (count > buffer_offset_) { count = buffer_offset_; } buffer_offset_ -= count; total_byte_count_ -= count; } size_t FileInputStream::ByteCount() const { return total_byte_count_; } bool FileInputStream::HadError() const { return fd_ == -1; } std::string FileInputStream::GetError() const { return error_; } bool FileInputStream::ReadFullyAtOffset(void* data, size_t byte_count, off64_t offset) { return base::ReadFullyAtOffset(fd_, data, byte_count, offset); } FileOutputStream::FileOutputStream(const std::string& path, size_t buffer_capacity) : buffer_capacity_(buffer_capacity) { int mode = O_WRONLY | O_CREAT | O_TRUNC | O_CLOEXEC | O_BINARY; owned_fd_.reset(TEMP_FAILURE_RETRY(::android::base::utf8::open(path.c_str(), mode, 0666))); fd_ = owned_fd_.get(); if (fd_ < 0) { error_ = SystemErrorCodeToString(errno); } else { buffer_.reset(new uint8_t[buffer_capacity_]); } } FileOutputStream::FileOutputStream(unique_fd fd, size_t buffer_capacity) : FileOutputStream(fd.get(), buffer_capacity) { owned_fd_ = std::move(fd); } FileOutputStream::FileOutputStream(int fd, size_t buffer_capacity) : fd_(fd), buffer_capacity_(buffer_capacity) { if (fd_ < 0) { error_ = "Bad File Descriptor"; } else { buffer_.reset(new uint8_t[buffer_capacity_]); } } FileOutputStream::~FileOutputStream() { // Flush the buffer. Flush(); } bool FileOutputStream::Next(void** data, size_t* size) { if (HadError()) { return false; } if (buffer_offset_ == buffer_capacity_) { if (!FlushImpl()) { return false; } } const size_t buffer_size = buffer_capacity_ - buffer_offset_; *data = buffer_.get() + buffer_offset_; *size = buffer_size; total_byte_count_ += buffer_size; buffer_offset_ = buffer_capacity_; return true; } void FileOutputStream::BackUp(size_t count) { if (count > buffer_offset_) { count = buffer_offset_; } buffer_offset_ -= count; total_byte_count_ -= count; } size_t FileOutputStream::ByteCount() const { return total_byte_count_; } bool FileOutputStream::Flush() { if (!HadError()) { return FlushImpl(); } return false; } bool FileOutputStream::FlushImpl() { ssize_t n = TEMP_FAILURE_RETRY(write(fd_, buffer_.get(), buffer_offset_)); if (n < 0) { error_ = SystemErrorCodeToString(errno); owned_fd_.reset(); fd_ = -1; buffer_.reset(); return false; } buffer_offset_ = 0u; return true; } bool FileOutputStream::HadError() const { return fd_ == -1; } std::string FileOutputStream::GetError() const { return error_; } } // namespace android