/* * 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. */ #define LOG_TAG "libprotoutil" #include <stdlib.h> #include <android/util/EncodedBuffer.h> #include <android/util/protobuf.h> #include <cutils/log.h> namespace android { namespace util { const size_t BUFFER_SIZE = 8 * 1024; // 8 KB EncodedBuffer::Pointer::Pointer() : Pointer(BUFFER_SIZE) { } EncodedBuffer::Pointer::Pointer(size_t chunkSize) :mIndex(0), mOffset(0) { mChunkSize = chunkSize == 0 ? BUFFER_SIZE : chunkSize; } size_t EncodedBuffer::Pointer::pos() const { return mIndex * mChunkSize + mOffset; } size_t EncodedBuffer::Pointer::index() const { return mIndex; } size_t EncodedBuffer::Pointer::offset() const { return mOffset; } EncodedBuffer::Pointer* EncodedBuffer::Pointer::move(size_t amt) { size_t newOffset = mOffset + amt; mIndex += newOffset / mChunkSize; mOffset = newOffset % mChunkSize; return this; } EncodedBuffer::Pointer* EncodedBuffer::Pointer::rewind() { mIndex = 0; mOffset = 0; return this; } EncodedBuffer::Pointer EncodedBuffer::Pointer::copy() const { Pointer p = Pointer(mChunkSize); p.mIndex = mIndex; p.mOffset = mOffset; return p; } // =========================================================== EncodedBuffer::EncodedBuffer() : EncodedBuffer(0) { } EncodedBuffer::EncodedBuffer(size_t chunkSize) :mBuffers() { mChunkSize = chunkSize == 0 ? BUFFER_SIZE : chunkSize; mWp = Pointer(mChunkSize); mEp = Pointer(mChunkSize); } EncodedBuffer::~EncodedBuffer() { for (size_t i=0; i<mBuffers.size(); i++) { uint8_t* buf = mBuffers[i]; free(buf); } } inline uint8_t* EncodedBuffer::at(const Pointer& p) const { return mBuffers[p.index()] + p.offset(); } void EncodedBuffer::clear() { mWp.rewind(); mEp.rewind(); } /******************************** Write APIs ************************************************/ size_t EncodedBuffer::size() const { return mWp.pos(); } EncodedBuffer::Pointer* EncodedBuffer::wp() { return &mWp; } uint8_t* EncodedBuffer::writeBuffer() { // This prevents write pointer move too fast than allocating the buffer. if (mWp.index() > mBuffers.size()) return NULL; uint8_t* buf = NULL; if (mWp.index() == mBuffers.size()) { buf = (uint8_t*)malloc(mChunkSize); if (buf == NULL) return NULL; // This indicates NO_MEMORY mBuffers.push_back(buf); } return at(mWp); } size_t EncodedBuffer::currentToWrite() { return mChunkSize - mWp.offset(); } void EncodedBuffer::writeRawByte(uint8_t val) { *writeBuffer() = val; mWp.move(); } size_t EncodedBuffer::writeRawVarint64(uint64_t val) { size_t size = 0; while (true) { size++; if ((val & ~0x7F) == 0) { writeRawByte((uint8_t) val); return size; } else { writeRawByte((uint8_t)((val & 0x7F) | 0x80)); val >>= 7; } } } size_t EncodedBuffer::writeRawVarint32(uint32_t val) { uint64_t v =(uint64_t)val; return writeRawVarint64(v); } void EncodedBuffer::writeRawFixed32(uint32_t val) { writeRawByte((uint8_t) val); writeRawByte((uint8_t) (val>>8)); writeRawByte((uint8_t) (val>>16)); writeRawByte((uint8_t) (val>>24)); } void EncodedBuffer::writeRawFixed64(uint64_t val) { writeRawByte((uint8_t) val); writeRawByte((uint8_t) (val>>8)); writeRawByte((uint8_t) (val>>16)); writeRawByte((uint8_t) (val>>24)); writeRawByte((uint8_t) (val>>32)); writeRawByte((uint8_t) (val>>40)); writeRawByte((uint8_t) (val>>48)); writeRawByte((uint8_t) (val>>56)); } size_t EncodedBuffer::writeHeader(uint32_t fieldId, uint8_t wireType) { return writeRawVarint32((fieldId << FIELD_ID_SHIFT) | wireType); } /******************************** Edit APIs ************************************************/ EncodedBuffer::Pointer* EncodedBuffer::ep() { return &mEp; } uint8_t EncodedBuffer::readRawByte() { uint8_t val = *at(mEp); mEp.move(); return val; } uint64_t EncodedBuffer::readRawVarint() { uint64_t val = 0, shift = 0; size_t start = mEp.pos(); while (true) { uint8_t byte = readRawByte(); val |= (UINT64_C(0x7F) & byte) << shift; if ((byte & 0x80) == 0) break; shift += 7; } return val; } uint32_t EncodedBuffer::readRawFixed32() { uint32_t val = 0; for (auto i=0; i<32; i+=8) { val += (uint32_t)readRawByte() << i; } return val; } uint64_t EncodedBuffer::readRawFixed64() { uint64_t val = 0; for (auto i=0; i<64; i+=8) { val += (uint64_t)readRawByte() << i; } return val; } void EncodedBuffer::editRawFixed32(size_t pos, uint32_t val) { size_t oldPos = mEp.pos(); mEp.rewind()->move(pos); for (auto i=0; i<32; i+=8) { *at(mEp) = (uint8_t) (val >> i); mEp.move(); } mEp.rewind()->move(oldPos); } void EncodedBuffer::copy(size_t srcPos, size_t size) { if (size == 0) return; Pointer cp(mChunkSize); cp.move(srcPos); while (cp.pos() < srcPos + size) { writeRawByte(*at(cp)); cp.move(); } } /********************************* Read APIs ************************************************/ EncodedBuffer::iterator EncodedBuffer::begin() const { return EncodedBuffer::iterator(*this); } EncodedBuffer::iterator::iterator(const EncodedBuffer& buffer) :mData(buffer), mRp(buffer.mChunkSize) { } size_t EncodedBuffer::iterator::size() const { return mData.size(); } size_t EncodedBuffer::iterator::bytesRead() const { return mRp.pos(); } EncodedBuffer::Pointer* EncodedBuffer::iterator::rp() { return &mRp; } uint8_t const* EncodedBuffer::iterator::readBuffer() { return hasNext() ? const_cast<uint8_t const*>(mData.at(mRp)) : NULL; } size_t EncodedBuffer::iterator::currentToRead() { return (mData.mWp.index() > mRp.index()) ? mData.mChunkSize - mRp.offset() : mData.mWp.offset() - mRp.offset(); } bool EncodedBuffer::iterator::hasNext() { return mRp.pos() < mData.mWp.pos(); } uint8_t EncodedBuffer::iterator::next() { uint8_t res = *(mData.at(mRp)); mRp.move(); return res; } uint64_t EncodedBuffer::iterator::readRawVarint() { uint64_t val = 0, shift = 0; while (true) { uint8_t byte = next(); val |= (INT64_C(0x7F) & byte) << shift; if ((byte & 0x80) == 0) break; shift += 7; } return val; } } // util } // android