| /* |
| * 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/BigBufferStream.h" |
| |
| #include <algorithm> |
| |
| namespace android { |
| |
| // |
| // BigBufferInputStream |
| // |
| |
| bool BigBufferInputStream::Next(const void** data, size_t* size) { |
| if (iter_ == buffer_->end()) { |
| return false; |
| } |
| |
| if (offset_ == iter_->size) { |
| ++iter_; |
| if (iter_ == buffer_->end()) { |
| return false; |
| } |
| offset_ = 0; |
| } |
| |
| *data = iter_->buffer.get() + offset_; |
| *size = iter_->size - offset_; |
| bytes_read_ += iter_->size - offset_; |
| offset_ = iter_->size; |
| return true; |
| } |
| |
| void BigBufferInputStream::BackUp(size_t count) { |
| if (count > offset_) { |
| bytes_read_ -= offset_; |
| offset_ = 0; |
| } else { |
| offset_ -= count; |
| bytes_read_ -= count; |
| } |
| } |
| |
| bool BigBufferInputStream::CanRewind() const { |
| return true; |
| } |
| |
| bool BigBufferInputStream::Rewind() { |
| iter_ = buffer_->begin(); |
| offset_ = 0; |
| bytes_read_ = 0; |
| return true; |
| } |
| |
| size_t BigBufferInputStream::ByteCount() const { |
| return bytes_read_; |
| } |
| |
| bool BigBufferInputStream::HadError() const { |
| return false; |
| } |
| |
| size_t BigBufferInputStream::TotalSize() const { |
| return buffer_->size(); |
| } |
| |
| bool BigBufferInputStream::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 > buffer_->size()) { |
| return false; |
| } |
| auto p = reinterpret_cast<uint8_t*>(data); |
| for (auto iter = buffer_->begin(); iter != buffer_->end() && byte_count > 0; ++iter) { |
| if (offset < iter->size) { |
| size_t to_read = std::min(byte_count, (size_t)(iter->size - offset)); |
| memcpy(p, iter->buffer.get() + offset, to_read); |
| byte_count -= to_read; |
| p += to_read; |
| offset = 0; |
| } else { |
| offset -= iter->size; |
| } |
| } |
| return byte_count == 0; |
| } |
| |
| // |
| // BigBufferOutputStream |
| // |
| |
| bool BigBufferOutputStream::Next(void** data, size_t* size) { |
| *data = buffer_->NextBlock(size); |
| return true; |
| } |
| |
| void BigBufferOutputStream::BackUp(size_t count) { |
| buffer_->BackUp(count); |
| } |
| |
| size_t BigBufferOutputStream::ByteCount() const { |
| return buffer_->size(); |
| } |
| |
| bool BigBufferOutputStream::HadError() const { |
| return false; |
| } |
| |
| } // namespace android |