/*
* 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