/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkFlattenable.h" #include "SkPtrRecorder.h" #include "SkReadBuffer.h" #include <algorithm> SkNamedFactorySet::SkNamedFactorySet() : fNextAddedFactory(0) {} uint32_t SkNamedFactorySet::find(SkFlattenable::Factory factory) { uint32_t index = fFactorySet.find(factory); if (index > 0) { return index; } const char* name = SkFlattenable::FactoryToName(factory); if (nullptr == name) { return 0; } *fNames.append() = name; return fFactorySet.add(factory); } const char* SkNamedFactorySet::getNextAddedFactoryName() { if (fNextAddedFactory < fNames.count()) { return fNames[fNextAddedFactory++]; } return nullptr; } /////////////////////////////////////////////////////////////////////////////// SkRefCntSet::~SkRefCntSet() { // call this now, while our decPtr() is sill in scope this->reset(); } void SkRefCntSet::incPtr(void* ptr) { ((SkRefCnt*)ptr)->ref(); } void SkRefCntSet::decPtr(void* ptr) { ((SkRefCnt*)ptr)->unref(); } /////////////////////////////////////////////////////////////////////////////// namespace { struct Entry { const char* fName; SkFlattenable::Factory fFactory; SkFlattenable::Type fType; }; struct EntryComparator { bool operator()(const Entry& a, const Entry& b) const { return strcmp(a.fName, b.fName) < 0; } bool operator()(const Entry& a, const char* b) const { return strcmp(a.fName, b) < 0; } bool operator()(const char* a, const Entry& b) const { return strcmp(a, b.fName) < 0; } }; int gCount = 0; Entry gEntries[128]; } // namespace void SkFlattenable::Finalize() { std::sort(gEntries, gEntries + gCount, EntryComparator()); } void SkFlattenable::Register(const char name[], Factory factory, SkFlattenable::Type type) { SkASSERT(name); SkASSERT(factory); SkASSERT(gCount < (int)SK_ARRAY_COUNT(gEntries)); gEntries[gCount].fName = name; gEntries[gCount].fFactory = factory; gEntries[gCount].fType = type; gCount += 1; } #ifdef SK_DEBUG static void report_no_entries(const char* functionName) { if (!gCount) { SkDebugf("%s has no registered name/factory/type entries." " Call SkFlattenable::InitializeFlattenablesIfNeeded() before using gEntries", functionName); } } #endif SkFlattenable::Factory SkFlattenable::NameToFactory(const char name[]) { InitializeFlattenablesIfNeeded(); SkASSERT(std::is_sorted(gEntries, gEntries + gCount, EntryComparator())); #ifdef SK_DEBUG report_no_entries(__FUNCTION__); #endif auto pair = std::equal_range(gEntries, gEntries + gCount, name, EntryComparator()); if (pair.first == pair.second) return nullptr; return pair.first->fFactory; } bool SkFlattenable::NameToType(const char name[], SkFlattenable::Type* type) { SkASSERT(type); InitializeFlattenablesIfNeeded(); SkASSERT(std::is_sorted(gEntries, gEntries + gCount, EntryComparator())); #ifdef SK_DEBUG report_no_entries(__FUNCTION__); #endif auto pair = std::equal_range(gEntries, gEntries + gCount, name, EntryComparator()); if (pair.first == pair.second) return false; *type = pair.first->fType; return true; } const char* SkFlattenable::FactoryToName(Factory fact) { InitializeFlattenablesIfNeeded(); #ifdef SK_DEBUG report_no_entries(__FUNCTION__); #endif const Entry* entries = gEntries; for (int i = gCount - 1; i >= 0; --i) { if (entries[i].fFactory == fact) { return entries[i].fName; } } return nullptr; } /////////////////////////////////////////////////////////////////////////////////////////////////// sk_sp<SkData> SkFlattenable::serialize(const SkSerialProcs* procs) const { SkBinaryWriteBuffer writer; if (procs) { writer.setSerialProcs(*procs); } writer.writeFlattenable(this); size_t size = writer.bytesWritten(); auto data = SkData::MakeUninitialized(size); writer.writeToMemory(data->writable_data()); return data; } sk_sp<SkFlattenable> SkFlattenable::Deserialize(SkFlattenable::Type type, const void* data, size_t size, const SkDeserialProcs* procs) { SkReadBuffer buffer(data, size); if (procs) { buffer.setDeserialProcs(*procs); } return sk_sp<SkFlattenable>(buffer.readFlattenable(type)); }