| #include "SkFlattenable.h" |
| #include "SkTypeface.h" |
| |
| void SkFlattenable::flatten(SkFlattenableWriteBuffer&) |
| { |
| /* we don't write anything at the moment, but this allows our subclasses |
| to not know that, since we want them to always call INHERITED::flatten() |
| in their code. |
| */ |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkFlattenableReadBuffer::SkFlattenableReadBuffer() { |
| fRCArray = NULL; |
| fRCCount = 0; |
| |
| fTFArray = NULL; |
| fTFCount = 0; |
| |
| fFactoryArray = NULL; |
| fFactoryCount = 0; |
| } |
| |
| SkFlattenableReadBuffer::SkFlattenableReadBuffer(const void* data) : |
| INHERITED(data, 1024 * 1024) { |
| fRCArray = NULL; |
| fRCCount = 0; |
| |
| fTFArray = NULL; |
| fTFCount = 0; |
| |
| fFactoryArray = NULL; |
| fFactoryCount = 0; |
| } |
| |
| SkFlattenableReadBuffer::SkFlattenableReadBuffer(const void* data, size_t size) |
| : INHERITED(data, size) { |
| fRCArray = NULL; |
| fRCCount = 0; |
| |
| fTFArray = NULL; |
| fTFCount = 0; |
| |
| fFactoryArray = NULL; |
| fFactoryCount = 0; |
| } |
| |
| SkTypeface* SkFlattenableReadBuffer::readTypeface() { |
| uint32_t index = this->readU32(); |
| if (0 == index || index > (unsigned)fTFCount) { |
| if (index) { |
| SkDebugf("====== typeface index %d\n", index); |
| } |
| return NULL; |
| } else { |
| SkASSERT(fTFArray); |
| return fTFArray[index - 1]; |
| } |
| } |
| |
| SkRefCnt* SkFlattenableReadBuffer::readRefCnt() { |
| uint32_t index = this->readU32(); |
| if (0 == index || index > (unsigned)fRCCount) { |
| return NULL; |
| } else { |
| SkASSERT(fRCArray); |
| return fRCArray[index - 1]; |
| } |
| } |
| |
| SkFlattenable* SkFlattenableReadBuffer::readFlattenable() { |
| SkFlattenable::Factory factory = NULL; |
| |
| if (fFactoryCount > 0) { |
| uint32_t index = this->readU32(); |
| if (index > 0) { |
| index -= 1; |
| SkASSERT(index < (unsigned)fFactoryCount); |
| factory = fFactoryArray[index]; |
| // if we recorded an index, but failed to get a factory, we need |
| // to skip the flattened data in the buffer |
| if (NULL == factory) { |
| uint32_t size = this->readU32(); |
| this->skip(size); |
| // fall through and return NULL for the object |
| } |
| } |
| } else { |
| factory = (SkFlattenable::Factory)readFunctionPtr(); |
| } |
| |
| SkFlattenable* obj = NULL; |
| if (factory) { |
| uint32_t sizeRecorded = this->readU32(); |
| uint32_t offset = this->offset(); |
| obj = (*factory)(*this); |
| // check that we read the amount we expected |
| uint32_t sizeRead = this->offset() - offset; |
| if (sizeRecorded != sizeRead) { |
| // we could try to fix up the offset... |
| sk_throw(); |
| } |
| } |
| return obj; |
| } |
| |
| void* SkFlattenableReadBuffer::readFunctionPtr() { |
| void* proc; |
| this->read(&proc, sizeof(proc)); |
| return proc; |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkFlattenableWriteBuffer::SkFlattenableWriteBuffer(size_t minSize) : |
| INHERITED(minSize) { |
| fFlags = (Flags)0; |
| fRCRecorder = NULL; |
| fTFRecorder = NULL; |
| fFactoryRecorder = NULL; |
| } |
| |
| SkFlattenableWriteBuffer::~SkFlattenableWriteBuffer() { |
| fRCRecorder->safeUnref(); |
| fTFRecorder->safeUnref(); |
| fFactoryRecorder->safeUnref(); |
| } |
| |
| SkRefCntRecorder* SkFlattenableWriteBuffer::setRefCntRecorder( |
| SkRefCntRecorder* rec) { |
| SkRefCnt_SafeAssign(fRCRecorder, rec); |
| return rec; |
| } |
| |
| SkRefCntRecorder* SkFlattenableWriteBuffer::setTypefaceRecorder( |
| SkRefCntRecorder* rec) { |
| SkRefCnt_SafeAssign(fTFRecorder, rec); |
| return rec; |
| } |
| |
| SkFactoryRecorder* SkFlattenableWriteBuffer::setFactoryRecorder( |
| SkFactoryRecorder* rec) { |
| SkRefCnt_SafeAssign(fFactoryRecorder, rec); |
| return rec; |
| } |
| |
| void SkFlattenableWriteBuffer::writeTypeface(SkTypeface* obj) { |
| if (NULL == obj || NULL == fTFRecorder) { |
| this->write32(0); |
| } else { |
| this->write32(fTFRecorder->record(obj)); |
| } |
| } |
| |
| void SkFlattenableWriteBuffer::writeRefCnt(SkRefCnt* obj) { |
| if (NULL == obj || NULL == fRCRecorder) { |
| this->write32(0); |
| } else { |
| this->write32(fRCRecorder->record(obj)); |
| } |
| } |
| |
| void SkFlattenableWriteBuffer::writeFlattenable(SkFlattenable* flattenable) { |
| SkFlattenable::Factory factory = NULL; |
| if (flattenable) { |
| factory = flattenable->getFactory(); |
| } |
| |
| if (fFactoryRecorder) { |
| this->write32(fFactoryRecorder->record(factory)); |
| } else { |
| this->writeFunctionPtr((void*)factory); |
| } |
| |
| if (factory) { |
| // make room for the size of the flatttened object |
| (void)this->reserve(sizeof(uint32_t)); |
| // record the current size, so we can subtract after the object writes. |
| uint32_t offset = this->size(); |
| // now flatten the object |
| flattenable->flatten(*this); |
| uint32_t objSize = this->size() - offset; |
| // record the obj's size |
| *this->peek32(offset - sizeof(uint32_t)) = objSize; |
| } |
| } |
| |
| void SkFlattenableWriteBuffer::writeFunctionPtr(void* proc) { |
| *(void**)this->reserve(sizeof(void*)) = proc; |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkRefCntRecorder::~SkRefCntRecorder() { |
| // call this now, while our decPtr() is sill in scope |
| this->reset(); |
| } |
| |
| void SkRefCntRecorder::incPtr(void* ptr) { |
| ((SkRefCnt*)ptr)->ref(); |
| } |
| |
| void SkRefCntRecorder::decPtr(void* ptr) { |
| ((SkRefCnt*)ptr)->unref(); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| /////////////////////////////////////////////////////////////////////////////// |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| #define MAX_PAIR_COUNT 64 |
| |
| struct Pair { |
| const char* fName; |
| SkFlattenable::Factory fFactory; |
| }; |
| |
| static int gCount; |
| static Pair gPairs[MAX_PAIR_COUNT]; |
| |
| void SkFlattenable::Register(const char name[], Factory factory) { |
| SkASSERT(name); |
| SkASSERT(factory); |
| |
| static bool gOnce; |
| if (!gOnce) { |
| gCount = 0; |
| gOnce = true; |
| } |
| |
| SkASSERT(gCount < MAX_PAIR_COUNT); |
| |
| gPairs[gCount].fName = name; |
| gPairs[gCount].fFactory = factory; |
| gCount += 1; |
| } |
| |
| SkFlattenable::Factory SkFlattenable::NameToFactory(const char name[]) { |
| const Pair* pairs = gPairs; |
| for (int i = gCount - 1; i >= 0; --i) { |
| if (strcmp(pairs[i].fName, name) == 0) { |
| return pairs[i].fFactory; |
| } |
| } |
| return NULL; |
| } |
| |
| const char* SkFlattenable::FactoryToName(Factory fact) { |
| const Pair* pairs = gPairs; |
| for (int i = gCount - 1; i >= 0; --i) { |
| if (pairs[i].fFactory == fact) { |
| return pairs[i].fName; |
| } |
| } |
| return NULL; |
| } |
| |
| bool SkFlattenable::toDumpString(SkString* str) const { |
| return false; |
| } |
| |