#include "SkWriter32.h"
struct SkWriter32::Block {
Block* fNext;
size_t fSize;
size_t fAllocated;
size_t available() const { return fSize - fAllocated; }
char* base() { return (char*)(this + 1); }
const char* base() const { return (const char*)(this + 1); }
uint32_t* alloc(size_t size) {
SkASSERT(SkAlign4(size) == size);
SkASSERT(this->available() >= size);
void* ptr = this->base() + fAllocated;
fAllocated += size;
SkASSERT(fAllocated <= fSize);
return (uint32_t*)ptr;
}
uint32_t* peek32(size_t offset) {
SkASSERT(offset <= fAllocated + 4);
void* ptr = this->base() + offset;
return (uint32_t*)ptr;
}
static Block* Create(size_t size) {
SkASSERT(SkAlign4(size) == size);
Block* block = (Block*)sk_malloc_throw(sizeof(Block) + size);
block->fNext = NULL;
block->fSize = size;
block->fAllocated = 0;
return block;
}
};
///////////////////////////////////////////////////////////////////////////////
SkWriter32::~SkWriter32() {
this->reset();
}
void SkWriter32::reset() {
Block* block = fHead;
while (block) {
Block* next = block->fNext;
sk_free(block);
block = next;
}
fSize = 0;
fHead = fTail = NULL;
fSingleBlock = NULL;
}
void SkWriter32::reset(void* block, size_t size) {
this->reset();
SkASSERT(0 == ((fSingleBlock - (char*)0) & 3)); // need 4-byte alignment
fSingleBlock = (char*)block;
fSingleBlockSize = (size & ~3);
}
uint32_t* SkWriter32::reserve(size_t size) {
SkASSERT(SkAlign4(size) == size);
if (fSingleBlock) {
uint32_t* ptr = (uint32_t*)(fSingleBlock + fSize);
fSize += size;
SkASSERT(fSize <= fSingleBlockSize);
return ptr;
}
Block* block = fTail;
if (NULL == block) {
SkASSERT(NULL == fHead);
fHead = fTail = block = Block::Create(SkMax32(size, fMinSize));
} else if (block->available() < size) {
fTail = Block::Create(SkMax32(size, fMinSize));
block->fNext = fTail;
block = fTail;
}
fSize += size;
return block->alloc(size);
}
uint32_t* SkWriter32::peek32(size_t offset) {
SkASSERT(SkAlign4(offset) == offset);
SkASSERT(offset <= fSize);
if (fSingleBlock) {
return (uint32_t*)(fSingleBlock + offset);
}
Block* block = fHead;
SkASSERT(NULL != block);
while (offset >= block->fAllocated) {
offset -= block->fAllocated;
block = block->fNext;
SkASSERT(NULL != block);
}
return block->peek32(offset);
}
void SkWriter32::flatten(void* dst) const {
if (fSingleBlock) {
memcpy(dst, fSingleBlock, fSize);
return;
}
const Block* block = fHead;
SkDEBUGCODE(size_t total = 0;)
while (block) {
size_t allocated = block->fAllocated;
memcpy(dst, block->base(), allocated);
dst = (char*)dst + allocated;
block = block->fNext;
SkDEBUGCODE(total += allocated;)
SkASSERT(total <= fSize);
}
SkASSERT(total == fSize);
}
void SkWriter32::writePad(const void* src, size_t size) {
size_t alignedSize = SkAlign4(size);
char* dst = (char*)this->reserve(alignedSize);
memcpy(dst, src, size);
dst += size;
int n = alignedSize - size;
while (--n >= 0) {
*dst++ = 0;
}
}
#include "SkStream.h"
size_t SkWriter32::readFromStream(SkStream* stream, size_t length) {
if (fSingleBlock) {
SkASSERT(fSingleBlockSize >= fSize);
size_t remaining = fSingleBlockSize - fSize;
if (length > remaining) {
length = remaining;
}
stream->read(fSingleBlock + fSize, length);
fSize += length;
return length;
}
char scratch[1024];
const size_t MAX = sizeof(scratch);
size_t remaining = length;
while (remaining != 0) {
size_t n = remaining;
if (n > MAX) {
n = MAX;
}
size_t bytes = stream->read(scratch, n);
this->writePad(scratch, bytes);
remaining -= bytes;
if (bytes != n) {
break;
}
}
return length - remaining;
}
bool SkWriter32::writeToStream(SkWStream* stream) {
if (fSingleBlock) {
return stream->write(fSingleBlock, fSize);
}
const Block* block = fHead;
while (block) {
if (!stream->write(block->base(), block->fAllocated)) {
return false;
}
block = block->fNext;
}
return true;
}
///////////////////////////////////////////////////////////////////////////////
#include "SkReader32.h"
const char* SkReader32::readString(size_t* outLen) {
// we need to read at least 1-4 bytes
SkASSERT(this->isAvailable(4));
const uint8_t* base = (const uint8_t*)this->peek();
const uint8_t* ptr = base;
size_t len = *ptr++;
if (0xFF == len) {
len = (ptr[0] << 8) | ptr[1];
ptr += 2;
SkASSERT(len < 0xFFFF);
}
// skip what we've read, and 0..3 pad bytes
// add 1 for the terminating 0 that writeString() included
size_t alignedSize = SkAlign4(len + (ptr - base) + 1);
this->skip(alignedSize);
if (outLen) {
*outLen = len;
}
return (const char*)ptr;
}
void SkWriter32::writeString(const char str[], size_t len) {
if ((long)len < 0) {
SkASSERT(str);
len = strlen(str);
}
size_t lenBytes = 1;
if (len >= 0xFF) {
lenBytes = 3;
SkASSERT(len < 0xFFFF);
}
// add 1 since we also write a terminating 0
size_t alignedLen = SkAlign4(lenBytes + len + 1);
uint8_t* ptr = (uint8_t*)this->reserve(alignedLen);
if (1 == lenBytes) {
*ptr++ = SkToU8(len);
} else {
*ptr++ = 0xFF;
*ptr++ = SkToU8(len >> 8);
*ptr++ = len & 0xFF;
}
memcpy(ptr, str, len);
ptr[len] = 0;
// we may have left 0,1,2,3 bytes uninitialized, since we reserved align4
// number of bytes. That's ok, since the reader will know to skip those
}
size_t SkWriter32::WriteStringSize(const char* str, size_t len) {
if ((long)len < 0) {
SkASSERT(str);
len = strlen(str);
}
size_t lenBytes = 1;
if (len >= 0xFF) {
lenBytes = 3;
SkASSERT(len < 0xFFFF);
}
// add 1 since we also write a terminating 0
return SkAlign4(lenBytes + len + 1);
}