| #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); |
| } |
| |
| |