| /* |
| * Copyright 2018 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkGlyph.h" |
| |
| #include "SkArenaAlloc.h" |
| #include "SkMakeUnique.h" |
| #include "SkScalerContext.h" |
| |
| void SkGlyph::toMask(SkMask* mask) const { |
| SkASSERT(mask); |
| |
| mask->fImage = (uint8_t*)fImage; |
| mask->fBounds.set(fLeft, fTop, fLeft + fWidth, fTop + fHeight); |
| mask->fRowBytes = this->rowBytes(); |
| mask->fFormat = static_cast<SkMask::Format>(fMaskFormat); |
| } |
| |
| void SkGlyph::zeroMetrics() { |
| fAdvanceX = 0; |
| fAdvanceY = 0; |
| fWidth = 0; |
| fHeight = 0; |
| fTop = 0; |
| fLeft = 0; |
| } |
| |
| static size_t bits_to_bytes(size_t bits) { |
| return (bits + 7) >> 3; |
| } |
| |
| static size_t format_alignment(SkMask::Format format) { |
| switch (format) { |
| case SkMask::kBW_Format: |
| case SkMask::kA8_Format: |
| case SkMask::k3D_Format: |
| case SkMask::kSDF_Format: |
| return alignof(uint8_t); |
| case SkMask::kARGB32_Format: |
| return alignof(uint32_t); |
| case SkMask::kLCD16_Format: |
| return alignof(uint16_t); |
| default: |
| SK_ABORT("Unknown mask format."); |
| break; |
| } |
| return 0; |
| } |
| |
| static size_t format_rowbytes(int width, SkMask::Format format) { |
| return format == SkMask::kBW_Format ? bits_to_bytes(width) |
| : width * format_alignment(format); |
| } |
| |
| size_t SkGlyph::formatAlignment() const { |
| auto format = static_cast<SkMask::Format>(fMaskFormat); |
| return format_alignment(format); |
| } |
| |
| size_t SkGlyph::allocImage(SkArenaAlloc* alloc) { |
| auto size = this->computeImageSize(); |
| auto format = static_cast<SkMask::Format>(fMaskFormat); |
| fImage = alloc->makeBytesAlignedTo(size, format_alignment(format)); |
| |
| return size; |
| } |
| |
| size_t SkGlyph::rowBytes() const { |
| return format_rowbytes(fWidth, (SkMask::Format)fMaskFormat); |
| } |
| |
| size_t SkGlyph::rowBytesUsingFormat(SkMask::Format format) const { |
| return format_rowbytes(fWidth, format); |
| } |
| |
| size_t SkGlyph::computeImageSize() const { |
| size_t size = this->rowBytes() * fHeight; |
| |
| if (fMaskFormat == SkMask::k3D_Format) { |
| size *= 3; |
| } |
| |
| return size; |
| } |
| |
| size_t SkGlyph::copyImageData(const SkGlyph& from, SkArenaAlloc* alloc) { |
| fMaskFormat = from.fMaskFormat; |
| fWidth = from.fWidth; |
| fHeight = from.fHeight; |
| fLeft = from.fLeft; |
| fTop = from.fTop; |
| fForceBW = from.fForceBW; |
| |
| if (from.fImage != nullptr) { |
| auto imageSize = this->allocImage(alloc); |
| SkASSERT(imageSize == from.computeImageSize()); |
| |
| memcpy(fImage, from.fImage, imageSize); |
| return imageSize; |
| } |
| |
| return 0u; |
| } |
| |
| SkPath* SkGlyph::addPath(SkScalerContext* scalerContext, SkArenaAlloc* alloc) { |
| if (!this->isEmpty()) { |
| if (fPathData == nullptr) { |
| fPathData = alloc->make<SkGlyph::PathData>(); |
| if (scalerContext->getPath(this->getPackedID(), &fPathData->fPath)) { |
| fPathData->fPath.updateBoundsCache(); |
| fPathData->fPath.getGenerationID(); |
| fPathData->fHasPath = true; |
| } |
| } |
| } |
| return this->path(); |
| } |
| |