Clean up buffering and serializatoin.
BUG=skia:7515 Change-Id: Id580e8f464c6263fa32a5d6347f505b00bba29d7 Reviewed-on: https://skia-review.googlesource.com/118460 Reviewed-by: Herb Derby <herb@google.com> Commit-Queue: Herb Derby <herb@google.com>
This commit is contained in:
parent
62473ad906
commit
46117c83d2
@ -15,8 +15,6 @@
|
|||||||
#include "SkFindAndPlaceGlyph.h"
|
#include "SkFindAndPlaceGlyph.h"
|
||||||
#include "SkTypeface_remote.h"
|
#include "SkTypeface_remote.h"
|
||||||
|
|
||||||
static const size_t kPageSize = 4096;
|
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
class ArraySlice final : public std::tuple<const T*, size_t> {
|
class ArraySlice final : public std::tuple<const T*, size_t> {
|
||||||
public:
|
public:
|
||||||
@ -45,21 +43,43 @@ private:
|
|||||||
size_t fSize;
|
size_t fSize;
|
||||||
};
|
};
|
||||||
|
|
||||||
// -- SkRemoteStrikeTransport ----------------------------------------------------------------------------------
|
// -- SkRemoteStrikeTransport ----------------------------------------------------------------------
|
||||||
|
|
||||||
|
static SkRemoteStrikeTransport::IOResult write_data(
|
||||||
|
size_t size, const uint8_t* data, SkRemoteStrikeTransport* t) {
|
||||||
|
|
||||||
|
if (t->write(&size, sizeof(size)) == SkRemoteStrikeTransport::kFail) {
|
||||||
|
return SkRemoteStrikeTransport::kFail;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (t->write(data, size) == SkRemoteStrikeTransport::kFail) {
|
||||||
|
return SkRemoteStrikeTransport::kFail;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SkRemoteStrikeTransport::kSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
|
static SkRemoteStrikeTransport::IOResult read_data(
|
||||||
|
size_t size, uint8_t* data, SkRemoteStrikeTransport* t) {
|
||||||
|
|
||||||
|
size_t totalRead = 0;
|
||||||
|
while (totalRead < size) {
|
||||||
|
size_t sizeRead;
|
||||||
|
SkRemoteStrikeTransport::IOResult result;
|
||||||
|
std::tie(sizeRead, result) = t->read(&data[totalRead], size - totalRead);
|
||||||
|
if (result == SkRemoteStrikeTransport::kFail || sizeRead == 0) {
|
||||||
|
return SkRemoteStrikeTransport::kFail;
|
||||||
|
}
|
||||||
|
totalRead += sizeRead;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SkRemoteStrikeTransport::kSuccess;
|
||||||
|
}
|
||||||
|
|
||||||
SkRemoteStrikeTransport::IOResult SkRemoteStrikeTransport::writeSkData(const SkData& data) {
|
SkRemoteStrikeTransport::IOResult SkRemoteStrikeTransport::writeSkData(const SkData& data) {
|
||||||
size_t size = data.size();
|
return write_data(data.size(), (uint8_t*)data.data(), this);
|
||||||
|
|
||||||
if (this->write(&size, sizeof(size)) == kFail) {
|
|
||||||
return kFail;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this->write(data.data(), size) == kFail) {
|
|
||||||
return kFail;
|
|
||||||
}
|
|
||||||
|
|
||||||
return kSuccess;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sk_sp<SkData> SkRemoteStrikeTransport::readSkData() {
|
sk_sp<SkData> SkRemoteStrikeTransport::readSkData() {
|
||||||
size_t size;
|
size_t size;
|
||||||
if(std::get<1>(this->read(&size, sizeof(size))) == kFail) {
|
if(std::get<1>(this->read(&size, sizeof(size))) == kFail) {
|
||||||
@ -67,164 +87,112 @@ sk_sp<SkData> SkRemoteStrikeTransport::readSkData() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
auto data = std::unique_ptr<uint8_t[]>{new uint8_t[size]};
|
auto data = std::unique_ptr<uint8_t[]>{new uint8_t[size]};
|
||||||
size_t totalRead = 0;
|
if (read_data(size, data.get(), this) == kFail) {
|
||||||
while (totalRead < size) {
|
return nullptr;
|
||||||
size_t sizeRead;
|
|
||||||
IOResult result;
|
|
||||||
std::tie(sizeRead, result) = this->read(&data[totalRead], size - totalRead);
|
|
||||||
if (result == kFail || sizeRead == 0) {
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
totalRead += sizeRead;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return SkData::MakeWithCopy(data.get(), size);
|
return SkData::MakeWithCopy(data.get(), size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SkRemoteStrikeTransport::IOResult SkRemoteStrikeTransport::writeVector(
|
||||||
|
const std::vector<uint8_t>& vector) {
|
||||||
|
return write_data(vector.size(), vector.data(), this);
|
||||||
|
}
|
||||||
|
|
||||||
|
SkRemoteStrikeTransport::IOResult SkRemoteStrikeTransport::readVector(
|
||||||
|
std::vector<uint8_t>* vector) {
|
||||||
|
size_t vectorSize = 0;
|
||||||
|
size_t readSize = 0;
|
||||||
|
SkRemoteStrikeTransport::IOResult result;
|
||||||
|
std::tie(readSize, result) = this->read(&vectorSize, sizeof(vectorSize));
|
||||||
|
if(result == kFail || readSize == 0) {
|
||||||
|
return kFail;
|
||||||
|
}
|
||||||
|
|
||||||
|
vector->resize(vectorSize);
|
||||||
|
|
||||||
|
return read_data(vectorSize, vector->data(), this);
|
||||||
|
}
|
||||||
|
|
||||||
// -- Serializer ----------------------------------------------------------------------------------
|
// -- Serializer ----------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
static size_t pad(size_t size, size_t alignment) {
|
||||||
|
return (size + (alignment - 1)) & ~(alignment - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// N.B. pointers are only valid until the next call.
|
||||||
class Serializer {
|
class Serializer {
|
||||||
public:
|
public:
|
||||||
void startWrite() {
|
Serializer(std::vector<uint8_t>* buffer) : fBuffer{buffer} { }
|
||||||
fCursor = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void startWrite(const T& data) {
|
T* push_back(const T& data) {
|
||||||
this->startWrite();
|
auto result = allocate(sizeof(T), alignof(T));
|
||||||
this->write<T>(data);
|
return new (result) T(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T, typename... Args>
|
template <typename T, typename... Args>
|
||||||
T* startEmplace(Args&&... args) {
|
T* emplace_back(Args&& ... args) {
|
||||||
this->startWrite();
|
auto result = allocate(sizeof(T), alignof(T));
|
||||||
return this->emplace<T>(std::forward<Args>(args)...);
|
return new (result) T{std::forward<Args>(args)...};
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T, typename... Args>
|
|
||||||
T* emplace(Args&&... args) {
|
|
||||||
T* result = new (&fBuffer[fCursor]) T{std::forward<Args>(args)...};
|
|
||||||
fCursor += sizeof(T);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
void write(const T& data) {
|
|
||||||
// TODO: guard against bad T.
|
|
||||||
memcpy(&fBuffer[fCursor], &data, sizeof(data));
|
|
||||||
fCursor += sizeof(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
T* allocate() {
|
|
||||||
// TODO: guard against bad T.
|
|
||||||
T* result = (T*)&fBuffer[fCursor];
|
|
||||||
fCursor += sizeof(T);
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void writeDescriptor(const SkDescriptor& desc) {
|
void writeDescriptor(const SkDescriptor& desc) {
|
||||||
memcpy(&fBuffer[fCursor], &desc, desc.getLength());
|
auto result = allocate(desc.getLength(), alignof(SkDescriptor));
|
||||||
fCursor += desc.getLength();
|
memcpy(result, &desc, desc.getLength());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T* allocateArray(int count) {
|
T* allocateArray(int count) {
|
||||||
T* result = (T*)&fBuffer[fCursor];
|
auto result = allocate(sizeof(T) * count, alignof(T));
|
||||||
fCursor += count * sizeof(T);
|
return new (result) T[count];
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
SkRemoteStrikeTransport::IOResult endWrite(SkRemoteStrikeTransport* transport) {
|
|
||||||
return transport->write(fBuffer.get(), fCursor);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static constexpr size_t kBufferSize = kPageSize * 2000;
|
void* allocate(size_t size, size_t alignment) {
|
||||||
std::unique_ptr<uint8_t[]> fBuffer{new uint8_t[kBufferSize]};
|
size_t aligned = pad(fBuffer->size(), alignment);
|
||||||
size_t fCursor{0};
|
fBuffer->resize(aligned + size);
|
||||||
//size_t fEnd{0};
|
return &(*fBuffer)[aligned];
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<uint8_t>* fBuffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
// -- Deserializer -------------------------------------------------------------------------------
|
// -- Deserializer -------------------------------------------------------------------------------
|
||||||
|
|
||||||
class Deserializer {
|
class Deserializer {
|
||||||
public:
|
public:
|
||||||
void startRead(SkRemoteStrikeTransport* transport) {
|
Deserializer(const std::vector<uint8_t>& buffer) : fBuffer{buffer} { }
|
||||||
fCursor = 0;
|
|
||||||
fEnd = 0;
|
|
||||||
fTransport = transport;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
T* startRead(SkRemoteStrikeTransport* transport) {
|
|
||||||
this->startRead(transport);
|
|
||||||
return this->read<T>();
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T* read() {
|
T* read() {
|
||||||
T* result = (T*)this->ensureAtLeast(sizeof(T));
|
size_t padded = pad(fCursor, alignof(T));
|
||||||
fCursor += sizeof(T);
|
fCursor = padded + sizeof(T);
|
||||||
return result;
|
return (T*)&fBuffer[padded];
|
||||||
}
|
}
|
||||||
|
|
||||||
SkDescriptor* readDescriptor() {
|
SkDescriptor* readDescriptor() {
|
||||||
SkDescriptor* result = (SkDescriptor*)this->ensureAtLeast(sizeof(SkDescriptor));
|
size_t padded = pad(fCursor, alignof(SkDescriptor));
|
||||||
size_t size = result->getLength();
|
SkDescriptor* result = (SkDescriptor*)&fBuffer[padded];
|
||||||
this->ensureAtLeast(size);
|
fCursor = padded + result->getLength();
|
||||||
fCursor += size;
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
ArraySlice<T> readArray(int count) {
|
ArraySlice<T> readArray(int count) {
|
||||||
|
size_t padded = pad(fCursor, alignof(T));
|
||||||
size_t size = count * sizeof(T);
|
size_t size = count * sizeof(T);
|
||||||
const T* base = (const T*)this->ensureAtLeast(size);
|
const T* base = (const T*)&fBuffer[padded];
|
||||||
ArraySlice<T> result = ArraySlice<T>{base, (uint32_t)count};
|
ArraySlice<T> result = ArraySlice<T>{base, (uint32_t)count};
|
||||||
fCursor += size;
|
fCursor = padded + size;
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t endRead() {
|
|
||||||
fTransport = nullptr;
|
|
||||||
return size();
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t size() {return fCursor;}
|
size_t size() {return fCursor;}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void* ensureAtLeast(size_t size) {
|
const std::vector<uint8_t>& fBuffer;
|
||||||
if (size > fEnd - fCursor) {
|
size_t fCursor{0};
|
||||||
if (readAtLeast(size) == SkRemoteStrikeTransport::kFail) {
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return &fBuffer[fCursor];
|
|
||||||
}
|
|
||||||
|
|
||||||
SkRemoteStrikeTransport::IOResult readAtLeast(size_t size) {
|
|
||||||
size_t readSoFar = 0;
|
|
||||||
size_t bufferLeft = kBufferSize - fCursor;
|
|
||||||
size_t needed = size - (fEnd - fCursor);
|
|
||||||
while (readSoFar < needed) {
|
|
||||||
SkRemoteStrikeTransport::IOResult result;
|
|
||||||
size_t readSize;
|
|
||||||
std::tie(readSize, result) =
|
|
||||||
fTransport->read(&fBuffer[fEnd+readSoFar], bufferLeft - readSoFar);
|
|
||||||
if (result == SkRemoteStrikeTransport::kFail || readSize == 0) {return SkRemoteStrikeTransport::kFail;}
|
|
||||||
readSoFar += readSize;
|
|
||||||
}
|
|
||||||
fEnd += readSoFar;
|
|
||||||
return SkRemoteStrikeTransport::kSuccess;
|
|
||||||
}
|
|
||||||
|
|
||||||
SkRemoteStrikeTransport* fTransport;
|
|
||||||
|
|
||||||
static constexpr size_t kBufferSize = kPageSize * 2000;
|
|
||||||
std::unique_ptr<uint8_t[]> fBuffer{new uint8_t[kBufferSize]};
|
|
||||||
size_t fCursor{0};
|
|
||||||
size_t fEnd{0};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// -- TrackLayerDevice -----------------------------------------------------------------------------
|
// -- TrackLayerDevice -----------------------------------------------------------------------------
|
||||||
@ -259,7 +227,8 @@ SkStrikeCacheDifferenceSpec::findStrikeDifferences(const SkDescriptor& desc,
|
|||||||
auto newDescPtr = newDesc.get();
|
auto newDescPtr = newDesc.get();
|
||||||
StrikeDifferences strikeDiffs{typefaceID, std::move(newDesc)};
|
StrikeDifferences strikeDiffs{typefaceID, std::move(newDesc)};
|
||||||
|
|
||||||
mapIter = fDescriptorToDifferencesMap.emplace_hint(mapIter, newDescPtr, std::move(strikeDiffs));
|
mapIter = fDescriptorToDifferencesMap.emplace_hint(
|
||||||
|
mapIter, newDescPtr, std::move(strikeDiffs));
|
||||||
}
|
}
|
||||||
|
|
||||||
return mapIter->second;
|
return mapIter->second;
|
||||||
@ -462,22 +431,12 @@ enum class OpCode : int32_t {
|
|||||||
kPrepopulateCache = 3,
|
kPrepopulateCache = 3,
|
||||||
};
|
};
|
||||||
|
|
||||||
class Op {
|
struct StrikeDiffHeader {
|
||||||
public:
|
StrikeDiffHeader() {}
|
||||||
Op(OpCode opCode, SkFontID typefaceId, const SkScalerContextRec& rec)
|
StrikeDiffHeader(int strikeCount_) : strikeCount{strikeCount_} {}
|
||||||
: opCode{opCode}
|
int strikeCount;
|
||||||
, typefaceId{typefaceId}
|
|
||||||
, descriptor{rec} { }
|
|
||||||
const OpCode opCode;
|
|
||||||
const SkFontID typefaceId;
|
|
||||||
const SkScalerContextRecDescriptor descriptor;
|
|
||||||
SkPackedGlyphID glyphID;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Header {
|
|
||||||
Header(int strikeCount_) : strikeCount{strikeCount_} {}
|
|
||||||
const int strikeCount;
|
|
||||||
};
|
|
||||||
struct StrikeSpec {
|
struct StrikeSpec {
|
||||||
StrikeSpec(SkFontID typefaceID_, uint32_t descLength_, int glyphCount_)
|
StrikeSpec(SkFontID typefaceID_, uint32_t descLength_, int glyphCount_)
|
||||||
: typefaceID{typefaceID_}
|
: typefaceID{typefaceID_}
|
||||||
@ -491,54 +450,85 @@ struct StrikeSpec {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct WireTypeface {
|
struct WireTypeface {
|
||||||
// std::thread::id thread_id; // TODO:need to figure a good solution
|
|
||||||
SkFontID typefaceID;
|
SkFontID typefaceID;
|
||||||
int glyphCount;
|
int glyphCount;
|
||||||
SkFontStyle style;
|
SkFontStyle style;
|
||||||
bool isFixed;
|
bool isFixed;
|
||||||
};
|
};
|
||||||
|
|
||||||
static void write_strikes_spec(const SkStrikeCacheDifferenceSpec &spec,
|
class Op {
|
||||||
Serializer* serializer,
|
public:
|
||||||
SkRemoteStrikeTransport* transport) {
|
Op(OpCode opCode, SkFontID typefaceId, const SkScalerContextRec& rec)
|
||||||
serializer->startEmplace<Op>(OpCode::kPrepopulateCache, SkFontID{0}, SkScalerContextRec{});
|
: opCode{opCode}
|
||||||
|
, typefaceId{typefaceId}
|
||||||
|
, descriptor{rec} { }
|
||||||
|
const OpCode opCode;
|
||||||
|
const SkFontID typefaceId;
|
||||||
|
const SkScalerContextRecDescriptor descriptor;
|
||||||
|
union {
|
||||||
|
// kGlyphPath and kGlyphMetricsAndImage
|
||||||
|
SkPackedGlyphID glyphID;
|
||||||
|
// kPrepopulateCache
|
||||||
|
StrikeDiffHeader strikeSpecHeader;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
serializer->emplace<Header>(spec.strikeCount());
|
class Result {
|
||||||
|
union {
|
||||||
|
SkPaint::FontMetrics fontMetrics;
|
||||||
|
SkGlyph glyph;
|
||||||
|
StrikeDiffHeader strikeDiffHeader;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
size_t SkStrikeCacheDifferenceSpec::sizeBytes() const {
|
||||||
|
size_t sum = sizeof(Op) + sizeof(StrikeDiffHeader);
|
||||||
|
for (auto& pair : fDescriptorToDifferencesMap) {
|
||||||
|
const auto& strike = pair.second;
|
||||||
|
sum += sizeof(StrikeSpec)
|
||||||
|
+ strike.fDesc->getLength()
|
||||||
|
+ strike.fGlyphIDs->count() * sizeof(SkPackedGlyphID);
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void write_strikes_spec(const SkStrikeCacheDifferenceSpec &spec,
|
||||||
|
Serializer* serializer) {
|
||||||
|
serializer->emplace_back<Op>(OpCode::kPrepopulateCache, SkFontID{0}, SkScalerContextRec{});
|
||||||
|
|
||||||
|
serializer->emplace_back<StrikeDiffHeader>(spec.strikeCount());
|
||||||
|
|
||||||
auto perStrike = [serializer](SkFontID typefaceID, const SkDescriptor& desc, int glyphCount) {
|
auto perStrike = [serializer](SkFontID typefaceID, const SkDescriptor& desc, int glyphCount) {
|
||||||
serializer->emplace<StrikeSpec>(typefaceID, desc.getLength(), glyphCount);
|
serializer->emplace_back<StrikeSpec>(typefaceID, desc.getLength(), glyphCount);
|
||||||
serializer->writeDescriptor(desc);
|
serializer->writeDescriptor(desc);
|
||||||
};
|
};
|
||||||
|
|
||||||
auto perGlyph = [serializer](SkPackedGlyphID glyphID) {
|
auto perGlyph = [serializer](SkPackedGlyphID glyphID) {
|
||||||
serializer->write<SkPackedGlyphID>(glyphID);
|
serializer->push_back<SkPackedGlyphID>(glyphID);
|
||||||
};
|
};
|
||||||
|
|
||||||
spec.iterateDifferences(perStrike, perGlyph);
|
spec.iterateDifferences(perStrike, perGlyph);
|
||||||
|
|
||||||
serializer->endWrite(transport);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void read_strikes_spec_write_strikes_data(
|
static void read_strikes_spec_write_strikes_data(
|
||||||
Deserializer* deserializer, Serializer* serializer, SkRemoteStrikeTransport* transport,
|
Deserializer* deserializer, Serializer* serializer, SkStrikeServer* server)
|
||||||
SkStrikeServer* rc)
|
|
||||||
{
|
{
|
||||||
// Don't start because the op started this deserialization.
|
// Don't start because the op started this deserialization.
|
||||||
auto header = deserializer->read<Header>();
|
auto header = deserializer->read<StrikeDiffHeader>();
|
||||||
serializer->startWrite<Header>(*header);
|
serializer->push_back<StrikeDiffHeader>(*header);
|
||||||
for (int i = 0; i < header->strikeCount; i++) {
|
for (int i = 0; i < header->strikeCount; i++) {
|
||||||
auto spec = deserializer->read<StrikeSpec>();
|
auto spec = deserializer->read<StrikeSpec>();
|
||||||
auto desc = deserializer->readDescriptor();
|
auto desc = deserializer->readDescriptor();
|
||||||
serializer->write<StrikeSpec>(*spec);
|
serializer->push_back<StrikeSpec>(*spec);
|
||||||
serializer->writeDescriptor(*desc);
|
serializer->writeDescriptor(*desc);
|
||||||
SkScalerContextRecDescriptor recDesc{*desc};
|
SkScalerContextRecDescriptor recDesc{*desc};
|
||||||
auto scaler = rc->generateScalerContext(recDesc, spec->typefaceID);
|
auto scaler = server->generateScalerContext(recDesc, spec->typefaceID);
|
||||||
SkPaint::FontMetrics fontMetrics;
|
SkPaint::FontMetrics fontMetrics;
|
||||||
scaler->getFontMetrics(&fontMetrics);
|
scaler->getFontMetrics(&fontMetrics);
|
||||||
serializer->write<SkPaint::FontMetrics>(fontMetrics);
|
serializer->push_back<SkPaint::FontMetrics>(fontMetrics);
|
||||||
auto glyphIDs = deserializer->readArray<SkPackedGlyphID>(spec->glyphCount);
|
auto glyphIDs = deserializer->readArray<SkPackedGlyphID>(spec->glyphCount);
|
||||||
for (auto glyphID : glyphIDs) {
|
for (auto glyphID : glyphIDs) {
|
||||||
auto glyph = serializer->allocate<SkGlyph>();
|
auto glyph = serializer->emplace_back<SkGlyph>();
|
||||||
glyph->initWithGlyphID(glyphID);
|
glyph->initWithGlyphID(glyphID);
|
||||||
scaler->getMetrics(glyph);
|
scaler->getMetrics(glyph);
|
||||||
auto imageSize = glyph->computeImageSize();
|
auto imageSize = glyph->computeImageSize();
|
||||||
@ -546,20 +536,18 @@ static void read_strikes_spec_write_strikes_data(
|
|||||||
glyph->fImage = nullptr;
|
glyph->fImage = nullptr;
|
||||||
|
|
||||||
if (imageSize > 0) {
|
if (imageSize > 0) {
|
||||||
glyph->fImage = serializer->allocateArray<uint8_t>(imageSize);
|
// Since the allocateArray can move glyph, make one that stays in one place.
|
||||||
scaler->getImage(*glyph);
|
SkGlyph stationaryGlyph = *glyph;
|
||||||
|
stationaryGlyph.fImage = serializer->allocateArray<uint8_t>(imageSize);
|
||||||
|
scaler->getImage(stationaryGlyph);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
deserializer->endRead();
|
|
||||||
serializer->endWrite(transport);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void update_caches_from_strikes_data(SkStrikeClient *client,
|
static void update_caches_from_strikes_data(SkStrikeClient *client,
|
||||||
Deserializer *deserializer,
|
Deserializer *deserializer) {
|
||||||
SkRemoteStrikeTransport *transport) {
|
auto header = deserializer->read<StrikeDiffHeader>();
|
||||||
deserializer->startRead(transport);
|
|
||||||
auto header = deserializer->read<Header>();
|
|
||||||
for (int i = 0; i < header->strikeCount; i++) {
|
for (int i = 0; i < header->strikeCount; i++) {
|
||||||
auto spec = deserializer->read<StrikeSpec>();
|
auto spec = deserializer->read<StrikeSpec>();
|
||||||
auto desc = deserializer->readDescriptor();
|
auto desc = deserializer->readDescriptor();
|
||||||
@ -586,7 +574,6 @@ static void update_caches_from_strikes_data(SkStrikeClient *client,
|
|||||||
memcpy(allocatedGlyph->fImage, image.data(), image.size());
|
memcpy(allocatedGlyph->fImage, image.data(), image.size());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
deserializer->endRead();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// -- SkStrikeServer -------------------------------------------------------------------------------
|
// -- SkStrikeServer -------------------------------------------------------------------------------
|
||||||
@ -599,22 +586,30 @@ SkStrikeServer::~SkStrikeServer() {
|
|||||||
|
|
||||||
int SkStrikeServer::serve() {
|
int SkStrikeServer::serve() {
|
||||||
|
|
||||||
auto serializer = skstd::make_unique<Serializer>();
|
std::vector<uint8_t> inBuffer;
|
||||||
auto deserializer = skstd::make_unique<Deserializer>();;
|
std::vector<uint8_t> outBuffer;
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
Op* op = deserializer->startRead<Op>(fTransport);
|
inBuffer.clear();
|
||||||
if (op == nullptr) { break; }
|
auto result = fTransport->readVector(&inBuffer);
|
||||||
|
if (result == SkRemoteStrikeTransport::kFail) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
Deserializer deserializer{inBuffer};
|
||||||
|
Op* op = deserializer.read<Op>();
|
||||||
|
|
||||||
fOpCount += 1;
|
fOpCount += 1;
|
||||||
|
|
||||||
|
outBuffer.clear();
|
||||||
|
Serializer serializer{&outBuffer};
|
||||||
|
|
||||||
switch (op->opCode) {
|
switch (op->opCode) {
|
||||||
case OpCode::kFontMetrics : {
|
case OpCode::kFontMetrics : {
|
||||||
auto sc = this->generateScalerContext(op->descriptor, op->typefaceId);
|
auto sc = this->generateScalerContext(op->descriptor, op->typefaceId);
|
||||||
SkPaint::FontMetrics metrics;
|
SkPaint::FontMetrics metrics;
|
||||||
sc->getFontMetrics(&metrics);
|
sc->getFontMetrics(&metrics);
|
||||||
serializer->startWrite<SkPaint::FontMetrics>(metrics);
|
serializer.push_back<SkPaint::FontMetrics>(metrics);
|
||||||
serializer->endWrite(fTransport);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::kGlyphPath : {
|
case OpCode::kGlyphPath : {
|
||||||
@ -623,41 +618,41 @@ int SkStrikeServer::serve() {
|
|||||||
SkPath path;
|
SkPath path;
|
||||||
sc->getPath(op->glyphID, &path);
|
sc->getPath(op->glyphID, &path);
|
||||||
size_t pathSize = path.writeToMemory(nullptr);
|
size_t pathSize = path.writeToMemory(nullptr);
|
||||||
serializer->startWrite<size_t>(pathSize);
|
serializer.push_back<size_t>(pathSize);
|
||||||
auto pathData = serializer->allocateArray<uint8_t>(pathSize);
|
auto pathData = serializer.allocateArray<uint8_t>(pathSize);
|
||||||
path.writeToMemory(pathData);
|
path.writeToMemory(pathData);
|
||||||
serializer->endWrite(fTransport);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::kGlyphMetricsAndImage : {
|
case OpCode::kGlyphMetricsAndImage : {
|
||||||
auto sc = this->generateScalerContext(op->descriptor, op->typefaceId);
|
auto scaler = this->generateScalerContext(op->descriptor, op->typefaceId);
|
||||||
|
|
||||||
serializer->startWrite();
|
auto glyph = serializer.emplace_back<SkGlyph>();
|
||||||
auto glyph = serializer->allocate<SkGlyph>();
|
|
||||||
// TODO: check for buffer overflow.
|
// TODO: check for buffer overflow.
|
||||||
glyph->initWithGlyphID(op->glyphID);
|
glyph->initWithGlyphID(op->glyphID);
|
||||||
sc->getMetrics(glyph);
|
scaler->getMetrics(glyph);
|
||||||
auto imageSize = glyph->computeImageSize();
|
auto imageSize = glyph->computeImageSize();
|
||||||
glyph->fPathData = nullptr;
|
glyph->fPathData = nullptr;
|
||||||
glyph->fImage = nullptr;
|
glyph->fImage = nullptr;
|
||||||
|
|
||||||
if (imageSize > 0) {
|
if (imageSize > 0) {
|
||||||
glyph->fImage = serializer->allocateArray<uint8_t>(imageSize);
|
// Since the allocateArray can move glyph, make one that stays in one place.
|
||||||
sc->getImage(*glyph);
|
SkGlyph stationaryGlyph = *glyph;
|
||||||
|
stationaryGlyph.fImage = serializer.allocateArray<uint8_t>(imageSize);
|
||||||
|
scaler->getImage(stationaryGlyph);
|
||||||
}
|
}
|
||||||
|
|
||||||
serializer->endWrite(fTransport);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::kPrepopulateCache : {
|
case OpCode::kPrepopulateCache : {
|
||||||
read_strikes_spec_write_strikes_data(
|
read_strikes_spec_write_strikes_data(
|
||||||
deserializer.get(), serializer.get(), fTransport, this);
|
&deserializer, &serializer, this);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
SK_ABORT("Bad op");
|
SK_ABORT("Bad op");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fTransport->writeVector(outBuffer);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -709,13 +704,8 @@ sk_sp<SkData> SkStrikeServer::encodeTypeface(SkTypeface* tf) {
|
|||||||
|
|
||||||
// -- SkStrikeClient -------------------------------------------------------------------------------
|
// -- SkStrikeClient -------------------------------------------------------------------------------
|
||||||
|
|
||||||
static Op* start_op_write(
|
SkStrikeClient::SkStrikeClient(SkRemoteStrikeTransport* transport)
|
||||||
OpCode opCode, const SkTypefaceProxy& tf, const SkScalerContextRec& rec, Serializer* serializer)
|
: fTransport{transport} { }
|
||||||
{
|
|
||||||
return serializer->startEmplace<Op>(opCode, tf.remoteTypefaceID(), rec);
|
|
||||||
}
|
|
||||||
|
|
||||||
SkStrikeClient::SkStrikeClient(SkRemoteStrikeTransport* transport) : fTransport{transport} { }
|
|
||||||
|
|
||||||
void SkStrikeClient::generateFontMetrics(
|
void SkStrikeClient::generateFontMetrics(
|
||||||
const SkTypefaceProxy& typefaceProxy,
|
const SkTypefaceProxy& typefaceProxy,
|
||||||
@ -723,18 +713,18 @@ void SkStrikeClient::generateFontMetrics(
|
|||||||
SkPaint::FontMetrics* metrics) {
|
SkPaint::FontMetrics* metrics) {
|
||||||
// Send generateFontMetrics
|
// Send generateFontMetrics
|
||||||
{
|
{
|
||||||
Serializer serializer;
|
fBuffer.clear();
|
||||||
serializer.startEmplace<Op>(OpCode::kFontMetrics, typefaceProxy.remoteTypefaceID(), rec);
|
Serializer serializer{&fBuffer};
|
||||||
start_op_write(OpCode::kFontMetrics, typefaceProxy, rec, &serializer);
|
serializer.emplace_back<Op>(OpCode::kFontMetrics, typefaceProxy.remoteTypefaceID(), rec);
|
||||||
serializer.endWrite(fTransport);
|
fTransport->writeVector(fBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Receive generateFontMetrics
|
// Receive generateFontMetrics
|
||||||
{
|
{
|
||||||
Deserializer deserializer;
|
fBuffer.clear();
|
||||||
deserializer.startRead(fTransport);
|
fTransport->readVector(&fBuffer);
|
||||||
|
Deserializer deserializer(fBuffer);
|
||||||
*metrics = *deserializer.read<SkPaint::FontMetrics>();
|
*metrics = *deserializer.read<SkPaint::FontMetrics>();
|
||||||
deserializer.endRead();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -743,30 +733,30 @@ void SkStrikeClient::generateMetricsAndImage(
|
|||||||
const SkScalerContextRec& rec,
|
const SkScalerContextRec& rec,
|
||||||
SkArenaAlloc* alloc,
|
SkArenaAlloc* alloc,
|
||||||
SkGlyph* glyph) {
|
SkGlyph* glyph) {
|
||||||
SkScalerContextRecDescriptor rd{rec};
|
|
||||||
|
|
||||||
{
|
{
|
||||||
Serializer serializer;
|
fBuffer.clear();
|
||||||
Op *op = serializer.startEmplace<Op>(
|
Serializer serializer(&fBuffer);
|
||||||
|
Op *op = serializer.emplace_back<Op>(
|
||||||
OpCode::kGlyphMetricsAndImage, typefaceProxy.remoteTypefaceID(), rec);
|
OpCode::kGlyphMetricsAndImage, typefaceProxy.remoteTypefaceID(), rec);
|
||||||
op->glyphID = glyph->getPackedID();
|
op->glyphID = glyph->getPackedID();
|
||||||
serializer.endWrite(fTransport);
|
fTransport->writeVector(fBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Receive generateMetricsAndImage
|
// Receive generateMetricsAndImage
|
||||||
{
|
{
|
||||||
Deserializer deserializer;
|
fBuffer.clear();
|
||||||
*glyph = *deserializer.startRead<SkGlyph>(fTransport);
|
fTransport->readVector(&fBuffer);
|
||||||
|
Deserializer deserializer(fBuffer);
|
||||||
|
*glyph = *deserializer.read<SkGlyph>();
|
||||||
auto imageSize = glyph->computeImageSize();
|
auto imageSize = glyph->computeImageSize();
|
||||||
glyph->fPathData = nullptr;
|
glyph->fPathData = nullptr;
|
||||||
glyph->fImage = nullptr;
|
glyph->fImage = nullptr;
|
||||||
if (imageSize > 0) {
|
if (imageSize > 0) {
|
||||||
auto image = deserializer.readArray<uint8_t>(imageSize);
|
auto image = deserializer.readArray<uint8_t>(imageSize);
|
||||||
SkASSERT(imageSize == image.size());
|
SkASSERT(imageSize == image.size());
|
||||||
glyph->allocImage(alloc);
|
glyph->allocImage(alloc);
|
||||||
memcpy(glyph->fImage, image.data(), imageSize);
|
memcpy(glyph->fImage, image.data(), imageSize);
|
||||||
}
|
}
|
||||||
deserializer.endRead();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -776,30 +766,37 @@ void SkStrikeClient::generatePath(
|
|||||||
SkGlyphID glyphID,
|
SkGlyphID glyphID,
|
||||||
SkPath* path) {
|
SkPath* path) {
|
||||||
{
|
{
|
||||||
Serializer serializer;
|
fBuffer.clear();
|
||||||
Op *op = serializer.startEmplace<Op>(
|
Serializer serializer{&fBuffer};
|
||||||
|
Op *op = serializer.emplace_back<Op>(
|
||||||
OpCode::kGlyphPath, typefaceProxy.remoteTypefaceID(), rec);
|
OpCode::kGlyphPath, typefaceProxy.remoteTypefaceID(), rec);
|
||||||
op->glyphID = glyphID;
|
op->glyphID = glyphID;
|
||||||
serializer.endWrite(fTransport);
|
fTransport->writeVector(fBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
Deserializer deserializer;
|
fBuffer.clear();
|
||||||
size_t pathSize = *deserializer.startRead<size_t>(fTransport);
|
fTransport->readVector(&fBuffer);
|
||||||
|
Deserializer deserializer{fBuffer};
|
||||||
|
size_t pathSize = *deserializer.read<size_t>();
|
||||||
auto rawPath = deserializer.readArray<uint8_t>(pathSize);
|
auto rawPath = deserializer.readArray<uint8_t>(pathSize);
|
||||||
path->readFromMemory(rawPath.data(), rawPath.size());
|
path->readFromMemory(rawPath.data(), rawPath.size());
|
||||||
deserializer.endRead();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void SkStrikeClient::primeStrikeCache(const SkStrikeCacheDifferenceSpec& strikeDifferences) {
|
void SkStrikeClient::primeStrikeCache(const SkStrikeCacheDifferenceSpec& strikeDifferences) {
|
||||||
{
|
{
|
||||||
auto serializer = skstd::make_unique<Serializer>();
|
fBuffer.clear();
|
||||||
write_strikes_spec(strikeDifferences, serializer.get(), fTransport);
|
fBuffer.reserve(strikeDifferences.sizeBytes());
|
||||||
|
Serializer serializer{&fBuffer};
|
||||||
|
write_strikes_spec(strikeDifferences, &serializer);
|
||||||
|
fTransport->writeVector(fBuffer);
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
auto deserializer = skstd::make_unique<Deserializer>();;
|
fBuffer.clear();
|
||||||
update_caches_from_strikes_data(this, deserializer.get(), fTransport);
|
fTransport->readVector(&fBuffer);
|
||||||
|
Deserializer deserializer{fBuffer};
|
||||||
|
update_caches_from_strikes_data(this, &deserializer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,6 +36,8 @@ public:
|
|||||||
virtual std::tuple<size_t, IOResult> read(void*, size_t) = 0;
|
virtual std::tuple<size_t, IOResult> read(void*, size_t) = 0;
|
||||||
IOResult writeSkData(const SkData&);
|
IOResult writeSkData(const SkData&);
|
||||||
sk_sp<SkData> readSkData();
|
sk_sp<SkData> readSkData();
|
||||||
|
IOResult writeVector(const std::vector<uint8_t>&);
|
||||||
|
IOResult readVector(std::vector<uint8_t>*);
|
||||||
};
|
};
|
||||||
|
|
||||||
class SkScalerContextRecDescriptor {
|
class SkScalerContextRecDescriptor {
|
||||||
@ -98,6 +100,7 @@ class SkStrikeCacheDifferenceSpec {
|
|||||||
public:
|
public:
|
||||||
StrikeDifferences& findStrikeDifferences(const SkDescriptor& desc, SkFontID typefaceID);
|
StrikeDifferences& findStrikeDifferences(const SkDescriptor& desc, SkFontID typefaceID);
|
||||||
int strikeCount() const { return fDescriptorToDifferencesMap.size(); }
|
int strikeCount() const { return fDescriptorToDifferencesMap.size(); }
|
||||||
|
size_t sizeBytes() const;
|
||||||
template <typename PerStrike, typename PerGlyph>
|
template <typename PerStrike, typename PerGlyph>
|
||||||
void iterateDifferences(PerStrike perStrike, PerGlyph perGlyph) const;
|
void iterateDifferences(PerStrike perStrike, PerGlyph perGlyph) const;
|
||||||
|
|
||||||
@ -211,6 +214,8 @@ private:
|
|||||||
SkTHashMap<SkFontID, sk_sp<SkTypefaceProxy>> fMapIdToTypeface;
|
SkTHashMap<SkFontID, sk_sp<SkTypefaceProxy>> fMapIdToTypeface;
|
||||||
|
|
||||||
SkRemoteStrikeTransport* const fTransport;
|
SkRemoteStrikeTransport* const fTransport;
|
||||||
|
|
||||||
|
std::vector<uint8_t> fBuffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // SkRemoteGlyphCache_DEFINED
|
#endif // SkRemoteGlyphCache_DEFINED
|
||||||
|
@ -90,7 +90,6 @@ static void build_prime_cache_spec(const SkIRect &bounds,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void final_draw(std::string outFilename,
|
static void final_draw(std::string outFilename,
|
||||||
SkRemoteStrikeTransport* transport,
|
|
||||||
SkDeserialProcs* procs,
|
SkDeserialProcs* procs,
|
||||||
SkData* picData,
|
SkData* picData,
|
||||||
SkStrikeClient* client) {
|
SkStrikeClient* client) {
|
||||||
@ -146,7 +145,7 @@ static void gpu(int readFd, int writeFd) {
|
|||||||
SkDeserialProcs procs;
|
SkDeserialProcs procs;
|
||||||
client.prepareDeserializeProcs(&procs);
|
client.prepareDeserializeProcs(&procs);
|
||||||
|
|
||||||
final_draw("test.png", &rwTransport, &procs, picData.get(), &client);
|
final_draw("test.png", &procs, picData.get(), &client);
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("GPU is exiting\n");
|
printf("GPU is exiting\n");
|
||||||
@ -177,7 +176,7 @@ static int renderer(
|
|||||||
return server.serve();
|
return server.serve();
|
||||||
} else {
|
} else {
|
||||||
stream = skpData;
|
stream = skpData;
|
||||||
final_draw("test-correct.png", &rwTransport, nullptr, stream.get(), nullptr);
|
final_draw("test-correct.png", nullptr, stream.get(), nullptr);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user