blob: bb3911b201759d96f08ead2be14ebea06c87580a [file] [log] [blame]
/*
* 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<off64_t>::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