1 /*
2  * Copyright 2006-2012 The Android Open Source Project
3  * Copyright 2012 Mozilla Foundation
4  *
5  * Use of this source code is governed by a BSD-style license that can be
6  * found in the LICENSE file.
7  */
8 
9 #include "SkBitmap.h"
10 #include "SkCanvas.h"
11 #include "SkColor.h"
12 #include "SkColorPriv.h"
13 #include "SkFDot6.h"
14 #include "SkFontHost_FreeType_common.h"
15 #include "SkPath.h"
16 
17 #include <ft2build.h>
18 #include FT_FREETYPE_H
19 #include FT_BITMAP_H
20 #include FT_IMAGE_H
21 #include FT_OUTLINE_H
22 // In the past, FT_GlyphSlot_Own_Bitmap was defined in this header file.
23 #include FT_SYNTHESIS_H
24 
25 // FT_LOAD_COLOR and the corresponding FT_Pixel_Mode::FT_PIXEL_MODE_BGRA
26 // were introduced in FreeType 2.5.0.
27 // The following may be removed once FreeType 2.5.0 is required to build.
28 #ifndef FT_LOAD_COLOR
29 #    define FT_LOAD_COLOR ( 1L << 20 )
30 #    define FT_PIXEL_MODE_BGRA 7
31 #endif
32 
33 //#define SK_SHOW_TEXT_BLIT_COVERAGE
34 
compute_pixel_mode(SkMask::Format format)35 static FT_Pixel_Mode compute_pixel_mode(SkMask::Format format) {
36     switch (format) {
37         case SkMask::kBW_Format:
38             return FT_PIXEL_MODE_MONO;
39         case SkMask::kA8_Format:
40         default:
41             return FT_PIXEL_MODE_GRAY;
42     }
43 }
44 
45 ///////////////////////////////////////////////////////////////////////////////
46 
packTriple(U8CPU r,U8CPU g,U8CPU b)47 static uint16_t packTriple(U8CPU r, U8CPU g, U8CPU b) {
48 #ifdef SK_SHOW_TEXT_BLIT_COVERAGE
49     r = SkTMax(r, (U8CPU)0x40);
50     g = SkTMax(g, (U8CPU)0x40);
51     b = SkTMax(b, (U8CPU)0x40);
52 #endif
53     return SkPack888ToRGB16(r, g, b);
54 }
55 
grayToRGB16(U8CPU gray)56 static uint16_t grayToRGB16(U8CPU gray) {
57 #ifdef SK_SHOW_TEXT_BLIT_COVERAGE
58     gray = SkTMax(gray, (U8CPU)0x40);
59 #endif
60     return SkPack888ToRGB16(gray, gray, gray);
61 }
62 
bittst(const uint8_t data[],int bitOffset)63 static int bittst(const uint8_t data[], int bitOffset) {
64     SkASSERT(bitOffset >= 0);
65     int lowBit = data[bitOffset >> 3] >> (~bitOffset & 7);
66     return lowBit & 1;
67 }
68 
69 /**
70  *  Copies a FT_Bitmap into an SkMask with the same dimensions.
71  *
72  *  FT_PIXEL_MODE_MONO
73  *  FT_PIXEL_MODE_GRAY
74  *  FT_PIXEL_MODE_LCD
75  *  FT_PIXEL_MODE_LCD_V
76  */
77 template<bool APPLY_PREBLEND>
copyFT2LCD16(const FT_Bitmap & bitmap,const SkMask & mask,int lcdIsBGR,const uint8_t * tableR,const uint8_t * tableG,const uint8_t * tableB)78 static void copyFT2LCD16(const FT_Bitmap& bitmap, const SkMask& mask, int lcdIsBGR,
79                          const uint8_t* tableR, const uint8_t* tableG, const uint8_t* tableB)
80 {
81     SkASSERT(SkMask::kLCD16_Format == mask.fFormat);
82     if (FT_PIXEL_MODE_LCD != bitmap.pixel_mode) {
83         SkASSERT(mask.fBounds.width() == static_cast<int>(bitmap.width));
84     }
85     if (FT_PIXEL_MODE_LCD_V != bitmap.pixel_mode) {
86         SkASSERT(mask.fBounds.height() == static_cast<int>(bitmap.rows));
87     }
88 
89     const uint8_t* src = bitmap.buffer;
90     uint16_t* dst = reinterpret_cast<uint16_t*>(mask.fImage);
91     const size_t dstRB = mask.fRowBytes;
92 
93     const int width = mask.fBounds.width();
94     const int height = mask.fBounds.height();
95 
96     switch (bitmap.pixel_mode) {
97         case FT_PIXEL_MODE_MONO:
98             for (int y = height; y --> 0;) {
99                 for (int x = 0; x < width; ++x) {
100                     dst[x] = -bittst(src, x);
101                 }
102                 dst = (uint16_t*)((char*)dst + dstRB);
103                 src += bitmap.pitch;
104             }
105             break;
106         case FT_PIXEL_MODE_GRAY:
107             for (int y = height; y --> 0;) {
108                 for (int x = 0; x < width; ++x) {
109                     dst[x] = grayToRGB16(src[x]);
110                 }
111                 dst = (uint16_t*)((char*)dst + dstRB);
112                 src += bitmap.pitch;
113             }
114             break;
115         case FT_PIXEL_MODE_LCD:
116             SkASSERT(3 * mask.fBounds.width() == static_cast<int>(bitmap.width));
117             for (int y = height; y --> 0;) {
118                 const uint8_t* triple = src;
119                 if (lcdIsBGR) {
120                     for (int x = 0; x < width; x++) {
121                         dst[x] = packTriple(sk_apply_lut_if<APPLY_PREBLEND>(triple[2], tableR),
122                                             sk_apply_lut_if<APPLY_PREBLEND>(triple[1], tableG),
123                                             sk_apply_lut_if<APPLY_PREBLEND>(triple[0], tableB));
124                         triple += 3;
125                     }
126                 } else {
127                     for (int x = 0; x < width; x++) {
128                         dst[x] = packTriple(sk_apply_lut_if<APPLY_PREBLEND>(triple[0], tableR),
129                                             sk_apply_lut_if<APPLY_PREBLEND>(triple[1], tableG),
130                                             sk_apply_lut_if<APPLY_PREBLEND>(triple[2], tableB));
131                         triple += 3;
132                     }
133                 }
134                 src += bitmap.pitch;
135                 dst = (uint16_t*)((char*)dst + dstRB);
136             }
137             break;
138         case FT_PIXEL_MODE_LCD_V:
139             SkASSERT(3 * mask.fBounds.height() == static_cast<int>(bitmap.rows));
140             for (int y = height; y --> 0;) {
141                 const uint8_t* srcR = src;
142                 const uint8_t* srcG = srcR + bitmap.pitch;
143                 const uint8_t* srcB = srcG + bitmap.pitch;
144                 if (lcdIsBGR) {
145                     SkTSwap(srcR, srcB);
146                 }
147                 for (int x = 0; x < width; x++) {
148                     dst[x] = packTriple(sk_apply_lut_if<APPLY_PREBLEND>(*srcR++, tableR),
149                                         sk_apply_lut_if<APPLY_PREBLEND>(*srcG++, tableG),
150                                         sk_apply_lut_if<APPLY_PREBLEND>(*srcB++, tableB));
151                 }
152                 src += 3 * bitmap.pitch;
153                 dst = (uint16_t*)((char*)dst + dstRB);
154             }
155             break;
156         default:
157             SkDEBUGF(("FT_Pixel_Mode %d", bitmap.pixel_mode));
158             SkDEBUGFAIL("unsupported FT_Pixel_Mode for LCD16");
159             break;
160     }
161 }
162 
163 /**
164  *  Copies a FT_Bitmap into an SkMask with the same dimensions.
165  *
166  *  Yes, No, Never Requested, Never Produced
167  *
168  *                        kBW kA8 k3D kARGB32 kLCD16
169  *  FT_PIXEL_MODE_MONO     Y   Y  NR     N       Y
170  *  FT_PIXEL_MODE_GRAY     N   Y  NR     N       Y
171  *  FT_PIXEL_MODE_GRAY2   NP  NP  NR    NP      NP
172  *  FT_PIXEL_MODE_GRAY4   NP  NP  NR    NP      NP
173  *  FT_PIXEL_MODE_LCD     NP  NP  NR    NP      NP
174  *  FT_PIXEL_MODE_LCD_V   NP  NP  NR    NP      NP
175  *  FT_PIXEL_MODE_BGRA     N   N  NR     Y       N
176  *
177  *  TODO: All of these N need to be Y or otherwise ruled out.
178  */
copyFTBitmap(const FT_Bitmap & srcFTBitmap,SkMask & dstMask)179 static void copyFTBitmap(const FT_Bitmap& srcFTBitmap, SkMask& dstMask) {
180     SkASSERT(dstMask.fBounds.width() == static_cast<int>(srcFTBitmap.width));
181     SkASSERT(dstMask.fBounds.height() == static_cast<int>(srcFTBitmap.rows));
182 
183     const uint8_t* src = reinterpret_cast<const uint8_t*>(srcFTBitmap.buffer);
184     const FT_Pixel_Mode srcFormat = static_cast<FT_Pixel_Mode>(srcFTBitmap.pixel_mode);
185     // FT_Bitmap::pitch is an int and allowed to be negative.
186     const int srcPitch = srcFTBitmap.pitch;
187     const size_t srcRowBytes = SkTAbs(srcPitch);
188 
189     uint8_t* dst = dstMask.fImage;
190     const SkMask::Format dstFormat = static_cast<SkMask::Format>(dstMask.fFormat);
191     const size_t dstRowBytes = dstMask.fRowBytes;
192 
193     const size_t width = srcFTBitmap.width;
194     const size_t height = srcFTBitmap.rows;
195 
196     if (SkMask::kLCD16_Format == dstFormat) {
197         copyFT2LCD16<false>(srcFTBitmap, dstMask, false, NULL, NULL, NULL);
198         return;
199     }
200 
201     if ((FT_PIXEL_MODE_MONO == srcFormat && SkMask::kBW_Format == dstFormat) ||
202         (FT_PIXEL_MODE_GRAY == srcFormat && SkMask::kA8_Format == dstFormat))
203     {
204         size_t commonRowBytes = SkTMin(srcRowBytes, dstRowBytes);
205         for (size_t y = height; y --> 0;) {
206             memcpy(dst, src, commonRowBytes);
207             src += srcPitch;
208             dst += dstRowBytes;
209         }
210     } else if (FT_PIXEL_MODE_MONO == srcFormat && SkMask::kA8_Format == dstFormat) {
211         for (size_t y = height; y --> 0;) {
212             uint8_t byte = 0;
213             int bits = 0;
214             const uint8_t* src_row = src;
215             uint8_t* dst_row = dst;
216             for (size_t x = width; x --> 0;) {
217                 if (0 == bits) {
218                     byte = *src_row++;
219                     bits = 8;
220                 }
221                 *dst_row++ = byte & 0x80 ? 0xff : 0x00;
222                 bits--;
223                 byte <<= 1;
224             }
225             src += srcPitch;
226             dst += dstRowBytes;
227         }
228     } else if (FT_PIXEL_MODE_BGRA == srcFormat && SkMask::kARGB32_Format == dstFormat) {
229         // FT_PIXEL_MODE_BGRA is pre-multiplied.
230         for (size_t y = height; y --> 0;) {
231             const uint8_t* src_row = src;
232             SkPMColor* dst_row = reinterpret_cast<SkPMColor*>(dst);
233             for (size_t x = 0; x < width; ++x) {
234                 uint8_t b = *src_row++;
235                 uint8_t g = *src_row++;
236                 uint8_t r = *src_row++;
237                 uint8_t a = *src_row++;
238                 *dst_row++ = SkPackARGB32(a, r, g, b);
239 #ifdef SK_SHOW_TEXT_BLIT_COVERAGE
240                 *(dst_row-1) = SkFourByteInterp256(*(dst_row-1), SK_ColorWHITE, 0x40);
241 #endif
242             }
243             src += srcPitch;
244             dst += dstRowBytes;
245         }
246     } else {
247         SkDEBUGF(("FT_Pixel_Mode %d, SkMask::Format %d\n", srcFormat, dstFormat));
248         SkDEBUGFAIL("unsupported combination of FT_Pixel_Mode and SkMask::Format");
249     }
250 }
251 
convert_8_to_1(unsigned byte)252 static inline int convert_8_to_1(unsigned byte) {
253     SkASSERT(byte <= 0xFF);
254     // Arbitrary decision that making the cutoff at 1/4 instead of 1/2 in general looks better.
255     return (byte >> 6) != 0;
256 }
257 
pack_8_to_1(const uint8_t alpha[8])258 static uint8_t pack_8_to_1(const uint8_t alpha[8]) {
259     unsigned bits = 0;
260     for (int i = 0; i < 8; ++i) {
261         bits <<= 1;
262         bits |= convert_8_to_1(alpha[i]);
263     }
264     return SkToU8(bits);
265 }
266 
packA8ToA1(const SkMask & mask,const uint8_t * src,size_t srcRB)267 static void packA8ToA1(const SkMask& mask, const uint8_t* src, size_t srcRB) {
268     const int height = mask.fBounds.height();
269     const int width = mask.fBounds.width();
270     const int octs = width >> 3;
271     const int leftOverBits = width & 7;
272 
273     uint8_t* dst = mask.fImage;
274     const int dstPad = mask.fRowBytes - SkAlign8(width)/8;
275     SkASSERT(dstPad >= 0);
276 
277     const int srcPad = srcRB - width;
278     SkASSERT(srcPad >= 0);
279 
280     for (int y = 0; y < height; ++y) {
281         for (int i = 0; i < octs; ++i) {
282             *dst++ = pack_8_to_1(src);
283             src += 8;
284         }
285         if (leftOverBits > 0) {
286             unsigned bits = 0;
287             int shift = 7;
288             for (int i = 0; i < leftOverBits; ++i, --shift) {
289                 bits |= convert_8_to_1(*src++) << shift;
290             }
291             *dst++ = bits;
292         }
293         src += srcPad;
294         dst += dstPad;
295     }
296 }
297 
SkMaskFormat_for_SkColorType(SkColorType colorType)298 inline SkMask::Format SkMaskFormat_for_SkColorType(SkColorType colorType) {
299     switch (colorType) {
300         case kAlpha_8_SkColorType:
301             return SkMask::kA8_Format;
302         case kN32_SkColorType:
303             return SkMask::kARGB32_Format;
304         default:
305             SkDEBUGFAIL("unsupported SkBitmap::Config");
306             return SkMask::kA8_Format;
307     }
308 }
309 
SkColorType_for_FTPixelMode(FT_Pixel_Mode pixel_mode)310 inline SkColorType SkColorType_for_FTPixelMode(FT_Pixel_Mode pixel_mode) {
311     switch (pixel_mode) {
312         case FT_PIXEL_MODE_MONO:
313         case FT_PIXEL_MODE_GRAY:
314             return kAlpha_8_SkColorType;
315         case FT_PIXEL_MODE_BGRA:
316             return kN32_SkColorType;
317         default:
318             SkDEBUGFAIL("unsupported FT_PIXEL_MODE");
319             return kAlpha_8_SkColorType;
320     }
321 }
322 
SkColorType_for_SkMaskFormat(SkMask::Format format)323 inline SkColorType SkColorType_for_SkMaskFormat(SkMask::Format format) {
324     switch (format) {
325         case SkMask::kBW_Format:
326         case SkMask::kA8_Format:
327         case SkMask::kLCD16_Format:
328             return kAlpha_8_SkColorType;
329         case SkMask::kARGB32_Format:
330             return kN32_SkColorType;
331         default:
332             SkDEBUGFAIL("unsupported destination SkBitmap::Config");
333             return kAlpha_8_SkColorType;
334     }
335 }
336 
generateGlyphImage(FT_Face face,const SkGlyph & glyph)337 void SkScalerContext_FreeType_Base::generateGlyphImage(FT_Face face, const SkGlyph& glyph) {
338     const bool doBGR = SkToBool(fRec.fFlags & SkScalerContext::kLCD_BGROrder_Flag);
339     const bool doVert = SkToBool(fRec.fFlags & SkScalerContext::kLCD_Vertical_Flag);
340 
341     switch ( face->glyph->format ) {
342         case FT_GLYPH_FORMAT_OUTLINE: {
343             FT_Outline* outline = &face->glyph->outline;
344             FT_BBox     bbox;
345             FT_Bitmap   target;
346 
347             int dx = 0, dy = 0;
348             if (fRec.fFlags & SkScalerContext::kSubpixelPositioning_Flag) {
349                 dx = SkFixedToFDot6(glyph.getSubXFixed());
350                 dy = SkFixedToFDot6(glyph.getSubYFixed());
351                 // negate dy since freetype-y-goes-up and skia-y-goes-down
352                 dy = -dy;
353             }
354             FT_Outline_Get_CBox(outline, &bbox);
355             /*
356                 what we really want to do for subpixel is
357                     offset(dx, dy)
358                     compute_bounds
359                     offset(bbox & !63)
360                 but that is two calls to offset, so we do the following, which
361                 achieves the same thing with only one offset call.
362             */
363             FT_Outline_Translate(outline, dx - ((bbox.xMin + dx) & ~63),
364                                           dy - ((bbox.yMin + dy) & ~63));
365 
366             if (SkMask::kLCD16_Format == glyph.fMaskFormat) {
367                 FT_Render_Glyph(face->glyph, doVert ? FT_RENDER_MODE_LCD_V : FT_RENDER_MODE_LCD);
368                 SkMask mask;
369                 glyph.toMask(&mask);
370                 if (fPreBlend.isApplicable()) {
371                     copyFT2LCD16<true>(face->glyph->bitmap, mask, doBGR,
372                                        fPreBlend.fR, fPreBlend.fG, fPreBlend.fB);
373                 } else {
374                     copyFT2LCD16<false>(face->glyph->bitmap, mask, doBGR,
375                                         fPreBlend.fR, fPreBlend.fG, fPreBlend.fB);
376                 }
377             } else {
378                 target.width = glyph.fWidth;
379                 target.rows = glyph.fHeight;
380                 target.pitch = glyph.rowBytes();
381                 target.buffer = reinterpret_cast<uint8_t*>(glyph.fImage);
382                 target.pixel_mode = compute_pixel_mode( (SkMask::Format)fRec.fMaskFormat);
383                 target.num_grays = 256;
384 
385                 memset(glyph.fImage, 0, glyph.rowBytes() * glyph.fHeight);
386                 FT_Outline_Get_Bitmap(face->glyph->library, outline, &target);
387             }
388         } break;
389 
390         case FT_GLYPH_FORMAT_BITMAP: {
391             FT_Pixel_Mode pixel_mode = static_cast<FT_Pixel_Mode>(face->glyph->bitmap.pixel_mode);
392             SkMask::Format maskFormat = static_cast<SkMask::Format>(glyph.fMaskFormat);
393 
394             // Assume that the other formats do not exist.
395             SkASSERT(FT_PIXEL_MODE_MONO == pixel_mode ||
396                      FT_PIXEL_MODE_GRAY == pixel_mode ||
397                      FT_PIXEL_MODE_BGRA == pixel_mode);
398 
399             // These are the only formats this ScalerContext should request.
400             SkASSERT(SkMask::kBW_Format == maskFormat ||
401                      SkMask::kA8_Format == maskFormat ||
402                      SkMask::kARGB32_Format == maskFormat ||
403                      SkMask::kLCD16_Format == maskFormat);
404 
405             if (fRec.fFlags & SkScalerContext::kEmbolden_Flag &&
406                 !(face->style_flags & FT_STYLE_FLAG_BOLD))
407             {
408                 FT_GlyphSlot_Own_Bitmap(face->glyph);
409                 FT_Bitmap_Embolden(face->glyph->library, &face->glyph->bitmap,
410                                    kBitmapEmboldenStrength, 0);
411             }
412 
413             // If no scaling needed, directly copy glyph bitmap.
414             if (glyph.fWidth == face->glyph->bitmap.width &&
415                 glyph.fHeight == face->glyph->bitmap.rows &&
416                 glyph.fTop == -face->glyph->bitmap_top &&
417                 glyph.fLeft == face->glyph->bitmap_left)
418             {
419                 SkMask dstMask;
420                 glyph.toMask(&dstMask);
421                 copyFTBitmap(face->glyph->bitmap, dstMask);
422                 break;
423             }
424 
425             // Otherwise, scale the bitmap.
426 
427             // Copy the FT_Bitmap into an SkBitmap (either A8 or ARGB)
428             SkBitmap unscaledBitmap;
429             unscaledBitmap.allocPixels(SkImageInfo::Make(face->glyph->bitmap.width,
430                                                          face->glyph->bitmap.rows,
431                                                          SkColorType_for_FTPixelMode(pixel_mode),
432                                                          kPremul_SkAlphaType));
433 
434             SkMask unscaledBitmapAlias;
435             unscaledBitmapAlias.fImage = reinterpret_cast<uint8_t*>(unscaledBitmap.getPixels());
436             unscaledBitmapAlias.fBounds.set(0, 0, unscaledBitmap.width(), unscaledBitmap.height());
437             unscaledBitmapAlias.fRowBytes = unscaledBitmap.rowBytes();
438             unscaledBitmapAlias.fFormat = SkMaskFormat_for_SkColorType(unscaledBitmap.colorType());
439             copyFTBitmap(face->glyph->bitmap, unscaledBitmapAlias);
440 
441             // Wrap the glyph's mask in a bitmap, unless the glyph's mask is BW or LCD.
442             // BW requires an A8 target for resizing, which can then be down sampled.
443             // LCD should use a 4x A8 target, which will then be down sampled.
444             // For simplicity, LCD uses A8 and is replicated.
445             int bitmapRowBytes = 0;
446             if (SkMask::kBW_Format != maskFormat && SkMask::kLCD16_Format != maskFormat) {
447                 bitmapRowBytes = glyph.rowBytes();
448             }
449             SkBitmap dstBitmap;
450             dstBitmap.setInfo(SkImageInfo::Make(glyph.fWidth, glyph.fHeight,
451                                                 SkColorType_for_SkMaskFormat(maskFormat),
452                                                 kPremul_SkAlphaType),
453                               bitmapRowBytes);
454             if (SkMask::kBW_Format == maskFormat || SkMask::kLCD16_Format == maskFormat) {
455                 dstBitmap.allocPixels();
456             } else {
457                 dstBitmap.setPixels(glyph.fImage);
458             }
459 
460             // Scale unscaledBitmap into dstBitmap.
461             SkCanvas canvas(dstBitmap);
462             canvas.clear(SK_ColorTRANSPARENT);
463             canvas.scale(SkIntToScalar(glyph.fWidth) / SkIntToScalar(face->glyph->bitmap.width),
464                          SkIntToScalar(glyph.fHeight) / SkIntToScalar(face->glyph->bitmap.rows));
465             SkPaint paint;
466             paint.setFilterQuality(kMedium_SkFilterQuality);
467             canvas.drawBitmap(unscaledBitmap, 0, 0, &paint);
468 
469             // If the destination is BW or LCD, convert from A8.
470             if (SkMask::kBW_Format == maskFormat) {
471                 // Copy the A8 dstBitmap into the A1 glyph.fImage.
472                 SkMask dstMask;
473                 glyph.toMask(&dstMask);
474                 packA8ToA1(dstMask, dstBitmap.getAddr8(0, 0), dstBitmap.rowBytes());
475             } else if (SkMask::kLCD16_Format == maskFormat) {
476                 // Copy the A8 dstBitmap into the LCD16 glyph.fImage.
477                 uint8_t* src = dstBitmap.getAddr8(0, 0);
478                 uint16_t* dst = reinterpret_cast<uint16_t*>(glyph.fImage);
479                 for (int y = dstBitmap.height(); y --> 0;) {
480                     for (int x = 0; x < dstBitmap.width(); ++x) {
481                         dst[x] = grayToRGB16(src[x]);
482                     }
483                     dst = (uint16_t*)((char*)dst + glyph.rowBytes());
484                     src += dstBitmap.rowBytes();
485                 }
486             }
487 
488         } break;
489 
490         default:
491             SkDEBUGFAIL("unknown glyph format");
492             memset(glyph.fImage, 0, glyph.rowBytes() * glyph.fHeight);
493             return;
494     }
495 
496 // We used to always do this pre-USE_COLOR_LUMINANCE, but with colorlum,
497 // it is optional
498 #if defined(SK_GAMMA_APPLY_TO_A8)
499     if (SkMask::kA8_Format == glyph.fMaskFormat && fPreBlend.isApplicable()) {
500         uint8_t* SK_RESTRICT dst = (uint8_t*)glyph.fImage;
501         unsigned rowBytes = glyph.rowBytes();
502 
503         for (int y = glyph.fHeight - 1; y >= 0; --y) {
504             for (int x = glyph.fWidth - 1; x >= 0; --x) {
505                 dst[x] = fPreBlend.fG[dst[x]];
506             }
507             dst += rowBytes;
508         }
509     }
510 #endif
511 }
512 
513 ///////////////////////////////////////////////////////////////////////////////
514 
move_proc(const FT_Vector * pt,void * ctx)515 static int move_proc(const FT_Vector* pt, void* ctx) {
516     SkPath* path = (SkPath*)ctx;
517     path->close();  // to close the previous contour (if any)
518     path->moveTo(SkFDot6ToScalar(pt->x), -SkFDot6ToScalar(pt->y));
519     return 0;
520 }
521 
line_proc(const FT_Vector * pt,void * ctx)522 static int line_proc(const FT_Vector* pt, void* ctx) {
523     SkPath* path = (SkPath*)ctx;
524     path->lineTo(SkFDot6ToScalar(pt->x), -SkFDot6ToScalar(pt->y));
525     return 0;
526 }
527 
quad_proc(const FT_Vector * pt0,const FT_Vector * pt1,void * ctx)528 static int quad_proc(const FT_Vector* pt0, const FT_Vector* pt1,
529                      void* ctx) {
530     SkPath* path = (SkPath*)ctx;
531     path->quadTo(SkFDot6ToScalar(pt0->x), -SkFDot6ToScalar(pt0->y),
532                  SkFDot6ToScalar(pt1->x), -SkFDot6ToScalar(pt1->y));
533     return 0;
534 }
535 
cubic_proc(const FT_Vector * pt0,const FT_Vector * pt1,const FT_Vector * pt2,void * ctx)536 static int cubic_proc(const FT_Vector* pt0, const FT_Vector* pt1,
537                       const FT_Vector* pt2, void* ctx) {
538     SkPath* path = (SkPath*)ctx;
539     path->cubicTo(SkFDot6ToScalar(pt0->x), -SkFDot6ToScalar(pt0->y),
540                   SkFDot6ToScalar(pt1->x), -SkFDot6ToScalar(pt1->y),
541                   SkFDot6ToScalar(pt2->x), -SkFDot6ToScalar(pt2->y));
542     return 0;
543 }
544 
generateGlyphPath(FT_Face face,SkPath * path)545 void SkScalerContext_FreeType_Base::generateGlyphPath(FT_Face face,
546                                                       SkPath* path)
547 {
548     FT_Outline_Funcs    funcs;
549 
550     funcs.move_to   = move_proc;
551     funcs.line_to   = line_proc;
552     funcs.conic_to  = quad_proc;
553     funcs.cubic_to  = cubic_proc;
554     funcs.shift     = 0;
555     funcs.delta     = 0;
556 
557     FT_Error err = FT_Outline_Decompose(&face->glyph->outline, &funcs, path);
558 
559     if (err != 0) {
560         path->reset();
561         return;
562     }
563 
564     path->close();
565 }
566