/* * 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 "io/StringStream.h" using ::android::StringPiece; namespace aapt { namespace io { StringInputStream::StringInputStream(StringPiece str) : str_(str), offset_(0u) { } bool StringInputStream::Next(const void** data, size_t* size) { if (offset_ == str_.size()) { return false; } *data = str_.data() + offset_; *size = str_.size() - offset_; offset_ = str_.size(); return true; } void StringInputStream::BackUp(size_t count) { if (count > offset_) { offset_ = 0u; } else { offset_ -= count; } } size_t StringInputStream::ByteCount() const { return offset_; } size_t StringInputStream::TotalSize() const { return str_.size(); } bool StringInputStream::ReadFullyAtOffset(void* data, size_t byte_count, off64_t offset) { if (byte_count == 0) { return true; } if (offset < 0) { return false; } if (offset > std::numeric_limits::max() - byte_count) { return false; } if (offset + byte_count > str_.size()) { return false; } memcpy(data, str_.data() + offset, byte_count); return true; } StringOutputStream::StringOutputStream(std::string* str, size_t buffer_capacity) : str_(str), buffer_capacity_(buffer_capacity), buffer_offset_(0u), buffer_(new char[buffer_capacity]) { } StringOutputStream::~StringOutputStream() { Flush(); } bool StringOutputStream::Next(void** data, size_t* size) { if (buffer_offset_ == buffer_capacity_) { FlushImpl(); } *data = buffer_.get() + buffer_offset_; *size = buffer_capacity_ - buffer_offset_; buffer_offset_ = buffer_capacity_; return true; } void StringOutputStream::BackUp(size_t count) { if (count > buffer_offset_) { buffer_offset_ = 0u; } else { buffer_offset_ -= count; } } size_t StringOutputStream::ByteCount() const { return str_->size() + buffer_offset_; } void StringOutputStream::Flush() { if (buffer_offset_ != 0u) { FlushImpl(); } } void StringOutputStream::FlushImpl() { str_->append(buffer_.get(), buffer_offset_); buffer_offset_ = 0u; } } // namespace io } // namespace aapt