/* * Copyright 2012 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef SkOTTable_glyf_DEFINED #define SkOTTable_glyf_DEFINED #include "SkEndian.h" #include "SkOTTableTypes.h" #include "SkOTTable_head.h" #include "SkOTTable_loca.h" #pragma pack(push, 1) struct SkOTTableGlyphData; extern uint8_t const * const SK_OT_GlyphData_NoOutline; struct SkOTTableGlyph { static const SK_OT_CHAR TAG0 = 'g'; static const SK_OT_CHAR TAG1 = 'l'; static const SK_OT_CHAR TAG2 = 'y'; static const SK_OT_CHAR TAG3 = 'f'; static const SK_OT_ULONG TAG = SkOTTableTAG<SkOTTableGlyph>::value; class Iterator { public: Iterator(const SkOTTableGlyph& glyf, const SkOTTableIndexToLocation& loca, SkOTTableHead::IndexToLocFormat locaFormat) : fGlyf(glyf) , fLocaFormat(SkOTTableHead::IndexToLocFormat::ShortOffsets == locaFormat.value ? 0 : 1) , fCurrentGlyphOffset(0) { fLocaPtr.shortOffset = reinterpret_cast<const SK_OT_USHORT*>(&loca); } void advance(uint16_t num) { fLocaPtr.shortOffset += num << fLocaFormat; fCurrentGlyphOffset = fLocaFormat ? SkEndian_SwapBE32(*fLocaPtr.longOffset) : uint32_t(SkEndian_SwapBE16(*fLocaPtr.shortOffset) << 1); } const SkOTTableGlyphData* next() { uint32_t previousGlyphOffset = fCurrentGlyphOffset; advance(1); if (previousGlyphOffset == fCurrentGlyphOffset) { return reinterpret_cast<const SkOTTableGlyphData*>(&SK_OT_GlyphData_NoOutline); } else { return reinterpret_cast<const SkOTTableGlyphData*>( reinterpret_cast<const SK_OT_BYTE*>(&fGlyf) + previousGlyphOffset ); } } private: const SkOTTableGlyph& fGlyf; uint16_t fLocaFormat; //0 or 1 uint32_t fCurrentGlyphOffset; union LocaPtr { const SK_OT_USHORT* shortOffset; const SK_OT_ULONG* longOffset; } fLocaPtr; }; }; struct SkOTTableGlyphData { SK_OT_SHORT numberOfContours; //== -1 Composite, > 0 Simple SK_OT_FWORD xMin; SK_OT_FWORD yMin; SK_OT_FWORD xMax; SK_OT_FWORD yMax; struct Simple { SK_OT_USHORT endPtsOfContours[1/*numberOfContours*/]; struct Instructions { SK_OT_USHORT length; SK_OT_BYTE data[1/*length*/]; }; union Flags { struct Field { SK_OT_BYTE_BITFIELD( OnCurve, xShortVector, yShortVector, Repeat, xIsSame_xShortVectorPositive, yIsSame_yShortVectorPositive, Reserved6, Reserved7) } field; struct Raw { static const SK_OT_USHORT OnCurveMask = SkTEndian_SwapBE16(1 << 0); static const SK_OT_USHORT xShortVectorMask = SkTEndian_SwapBE16(1 << 1); static const SK_OT_USHORT yShortVectorMask = SkTEndian_SwapBE16(1 << 2); static const SK_OT_USHORT RepeatMask = SkTEndian_SwapBE16(1 << 3); static const SK_OT_USHORT xIsSame_xShortVectorPositiveMask = SkTEndian_SwapBE16(1 << 4); static const SK_OT_USHORT yIsSame_yShortVectorPositiveMask = SkTEndian_SwapBE16(1 << 5); SK_OT_BYTE value; } raw; }; //xCoordinates //yCoordinates }; struct Composite { struct Component { union Flags { struct Field { //8-15 SK_OT_BYTE_BITFIELD( WE_HAVE_INSTRUCTIONS, USE_MY_METRICS, OVERLAP_COMPOUND, SCALED_COMPONENT_OFFSET, UNSCALED_COMPONENT_OFFSET, Reserved13, Reserved14, Reserved15) //0-7 SK_OT_BYTE_BITFIELD( ARG_1_AND_2_ARE_WORDS, ARGS_ARE_XY_VALUES, ROUND_XY_TO_GRID, WE_HAVE_A_SCALE, RESERVED, MORE_COMPONENTS, WE_HAVE_AN_X_AND_Y_SCALE, WE_HAVE_A_TWO_BY_TWO) } field; struct Raw { static const SK_OT_USHORT ARG_1_AND_2_ARE_WORDS_Mask = SkTEndian_SwapBE16(1 << 0); static const SK_OT_USHORT ARGS_ARE_XY_VALUES_Mask = SkTEndian_SwapBE16(1 << 1); static const SK_OT_USHORT ROUND_XY_TO_GRID_Mask = SkTEndian_SwapBE16(1 << 2); static const SK_OT_USHORT WE_HAVE_A_SCALE_Mask = SkTEndian_SwapBE16(1 << 3); static const SK_OT_USHORT RESERVED_Mask = SkTEndian_SwapBE16(1 << 4); static const SK_OT_USHORT MORE_COMPONENTS_Mask = SkTEndian_SwapBE16(1 << 5); static const SK_OT_USHORT WE_HAVE_AN_X_AND_Y_SCALE_Mask = SkTEndian_SwapBE16(1 << 6); static const SK_OT_USHORT WE_HAVE_A_TWO_BY_TWO_Mask = SkTEndian_SwapBE16(1 << 7); static const SK_OT_USHORT WE_HAVE_INSTRUCTIONS_Mask = SkTEndian_SwapBE16(1 << 8); static const SK_OT_USHORT USE_MY_METRICS_Mask = SkTEndian_SwapBE16(1 << 9); static const SK_OT_USHORT OVERLAP_COMPOUND_Mask = SkTEndian_SwapBE16(1 << 10); static const SK_OT_USHORT SCALED_COMPONENT_OFFSET_Mask = SkTEndian_SwapBE16(1 << 11); static const SK_OT_USHORT UNSCALED_COMPONENT_OFFSET_mask = SkTEndian_SwapBE16(1 << 12); //Reserved //Reserved //Reserved SK_OT_USHORT value; } raw; } flags; SK_OT_USHORT glyphIndex; union Transform { union Matrix { /** !WE_HAVE_A_SCALE & !WE_HAVE_AN_X_AND_Y_SCALE & !WE_HAVE_A_TWO_BY_TWO */ struct None { } none; /** WE_HAVE_A_SCALE */ struct Scale { SK_OT_F2DOT14 a_d; } scale; /** WE_HAVE_AN_X_AND_Y_SCALE */ struct ScaleXY { SK_OT_F2DOT14 a; SK_OT_F2DOT14 d; } scaleXY; /** WE_HAVE_A_TWO_BY_TWO */ struct TwoByTwo { SK_OT_F2DOT14 a; SK_OT_F2DOT14 b; SK_OT_F2DOT14 c; SK_OT_F2DOT14 d; } twoByTwo; }; /** ARG_1_AND_2_ARE_WORDS & ARGS_ARE_XY_VALUES */ struct WordValue { SK_OT_FWORD e; SK_OT_FWORD f; SkOTTableGlyphData::Composite::Component::Transform::Matrix matrix; } wordValue; /** !ARG_1_AND_2_ARE_WORDS & ARGS_ARE_XY_VALUES */ struct ByteValue { SK_OT_CHAR e; SK_OT_CHAR f; SkOTTableGlyphData::Composite::Component::Transform::Matrix matrix; } byteValue; /** ARG_1_AND_2_ARE_WORDS & !ARGS_ARE_XY_VALUES */ struct WordIndex { SK_OT_USHORT compoundPointIndex; SK_OT_USHORT componentPointIndex; SkOTTableGlyphData::Composite::Component::Transform::Matrix matrix; } wordIndex; /** !ARG_1_AND_2_ARE_WORDS & !ARGS_ARE_XY_VALUES */ struct ByteIndex { SK_OT_BYTE compoundPointIndex; SK_OT_BYTE componentPointIndex; SkOTTableGlyphData::Composite::Component::Transform::Matrix matrix; } byteIndex; } transform; } component;//[] last element does not set MORE_COMPONENTS /** Comes after the last Component if the last component has WE_HAVE_INSTR. */ struct Instructions { SK_OT_USHORT length; SK_OT_BYTE data[1/*length*/]; }; }; }; #pragma pack(pop) #endif