hb-aat-layout-kerx-table.hh 9.4 KB
Newer Older
1 2
/*
 * Copyright © 2018  Ebrahim Byagowi
3
 * Copyright © 2018  Google, Inc.
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
 *
 *  This is part of HarfBuzz, a text shaping library.
 *
 * Permission is hereby granted, without written agreement and without
 * license or royalty fees, to use, copy, modify, and distribute this
 * software and its documentation for any purpose, provided that the
 * above copyright notice and the following two paragraphs appear in
 * all copies of this software.
 *
 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
 * DAMAGE.
 *
 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
 *
 * Google Author(s): Behdad Esfahbod
 */

#ifndef HB_AAT_LAYOUT_KERX_TABLE_HH
#define HB_AAT_LAYOUT_KERX_TABLE_HH

31
#include "hb-open-type-private.hh"
32 33
#include "hb-aat-layout-common-private.hh"

34
#define HB_AAT_TAG_kerx HB_TAG('k','e','r','x')
35 36


37
namespace AAT {
38

39
using namespace OT;
40 41


42
struct KerxFormat0Records
43 44 45 46 47 48 49 50 51 52 53 54
{
  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
    return_trace (c->check_struct (this));
  }

  protected:
  GlyphID	left;
  GlyphID	right;
  FWORD		value;
  public:
55
  DEFINE_SIZE_STATIC (6);
56 57 58 59
};

struct KerxSubTableFormat0
{
60 61 62 63 64 65 66 67 68
  // TODO(ebraminio) Enable when we got suitable BinSearchArrayOf
  // inline int get_kerning (hb_codepoint_t left, hb_codepoint_t right) const
  // {
  //   hb_glyph_pair_t pair = {left, right};
  //   int i = pairs.bsearch (pair);
  //   if (i == -1)
  //     return 0;
  //   return pairs[i].get_kerning ();
  // }
69 70 71 72

  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
73 74
    return_trace (c->check_struct (this) &&
      c->check_array (records, records[0].static_size, nPairs));
75 76 77
  }

  protected:
78 79 80 81 82 83 84 85 86
  // TODO(ebraminio): A custom version of "BinSearchArrayOf<KerxPair> pairs;" is
  // needed here to use HBUINT32 instead
  HBUINT32 nPairs;	/* The number of kerning pairs in this subtable */
  HBUINT32 searchRange; /* The largest power of two less than or equal to the value of nPairs,
                         * multiplied by the size in bytes of an entry in the subtable. */
  HBUINT32 entrySelector; /* This is calculated as log2 of the largest power of two less
                           * than or equal to the value of nPairs. */
  HBUINT32 rangeShift;	/* The value of nPairs minus the largest power of two less than or equal to nPairs. */
  KerxFormat0Records records[VAR]; /* VAR=nPairs */
87
  public:
88
  DEFINE_SIZE_ARRAY (16, records);
89 90 91 92 93 94
};

struct KerxSubTableFormat1
{
  inline bool sanitize (hb_sanitize_context_t *c) const
  {
95 96 97
    TRACE_SANITIZE (this);
    return_trace (c->check_struct (this) &&
      stateHeader.sanitize (c));
98 99 100
  }

  protected:
101 102
  StateTable<HBUINT16>		stateHeader;
  LOffsetTo<ArrayOf<HBUINT16> >	valueTable;
103
  public:
104
  DEFINE_SIZE_STATIC (20);
105 106
};

107
// TODO(ebraminio): Maybe this can be replaced with Lookup<HBUINT16>?
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
struct KerxClassTable
{
  inline unsigned int get_class (hb_codepoint_t g) const { return classes[g - firstGlyph]; }

  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
    return_trace (firstGlyph.sanitize (c) && classes.sanitize (c));
  }

  protected:
  HBUINT16		firstGlyph;	/* First glyph in class range. */
  ArrayOf<HBUINT16>	classes;	/* Glyph classes. */
  public:
  DEFINE_SIZE_ARRAY (4, classes);
};

struct KerxSubTableFormat2
{
  inline int get_kerning (hb_codepoint_t left, hb_codepoint_t right, const char *end) const
  {
    unsigned int l = (this+leftClassTable).get_class (left);
    unsigned int r = (this+leftClassTable).get_class (left);
    unsigned int offset = l * rowWidth + r * sizeof (FWORD);
    const FWORD *arr = &(this+array);
    if (unlikely ((const void *) arr < (const void *) this || (const void *) arr >= (const void *) end))
      return 0;
    const FWORD *v = &StructAtOffset<FWORD> (arr, offset);
    if (unlikely ((const void *) v < (const void *) arr || (const void *) (v + 1) > (const void *) end))
      return 0;
    return *v;
  }

  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
144 145
    return_trace (c->check_struct (this) &&
      rowWidth.sanitize (c) &&
146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162
		  leftClassTable.sanitize (c, this) &&
		  rightClassTable.sanitize (c, this) &&
		  array.sanitize (c, this));
  }

  protected:
  HBUINT32	rowWidth;	/* The width, in bytes, of a row in the table. */
  LOffsetTo<KerxClassTable>
		leftClassTable;	/* Offset from beginning of this subtable to
				 * left-hand class table. */
  LOffsetTo<KerxClassTable>
		rightClassTable;/* Offset from beginning of this subtable to
				 * right-hand class table. */
  LOffsetTo<FWORD>
		array;		/* Offset from beginning of this subtable to
				 * the start of the kerning array. */
  public:
163
  DEFINE_SIZE_STATIC (16);
164 165 166 167 168 169 170
};

struct KerxSubTableFormat4
{
  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
171 172
    return_trace (c->check_struct (this) &&
      rowWidth.sanitize (c) &&
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
		  leftClassTable.sanitize (c, this) &&
		  rightClassTable.sanitize (c, this) &&
		  array.sanitize (c, this));
  }

  protected:
  HBUINT32	rowWidth;	/* The width, in bytes, of a row in the table. */
  LOffsetTo<KerxClassTable>
		leftClassTable;	/* Offset from beginning of this subtable to
				 * left-hand class table. */
  LOffsetTo<KerxClassTable>
		rightClassTable;/* Offset from beginning of this subtable to
				 * right-hand class table. */
  LOffsetTo<FWORD>
		array;		/* Offset from beginning of this subtable to
				 * the start of the kerning array. */
  public:
190
  DEFINE_SIZE_STATIC (16);
191 192 193 194 195 196
};

struct KerxSubTableFormat6
{
  inline bool sanitize (hb_sanitize_context_t *c) const
  {
197
    TRACE_SANITIZE (this);
198 199 200 201 202
    return_trace (c->check_struct (this) &&
      rowIndexTable.sanitize (c, this) &&
      columnIndexTable.sanitize (c, this) &&
      kerningArray.sanitize (c, this) &&
      kerningVector.sanitize (c, this));
203 204 205
  }

  protected:
206 207 208
  HBUINT32	flags;
  HBUINT16	rowCount;
  HBUINT16	columnCount;
209 210 211 212
  LOffsetTo<Lookup<HBUINT16> >	rowIndexTable;
  LOffsetTo<Lookup<HBUINT16> >	columnIndexTable;
  LOffsetTo<Lookup<HBUINT16> >	kerningArray;
  LOffsetTo<Lookup<HBUINT16> >	kerningVector;
213
  public:
214
  DEFINE_SIZE_STATIC (24);
215 216
};

217
enum coverage_flags_t
218
{
219 220 221 222 223 224 225 226 227
  COVERAGE_VERTICAL_FLAG	= 0x80u,
  COVERAGE_CROSSSTREAM_FLAG	= 0x40u,
  COVERAGE_VARIATION_FLAG	= 0x20u,
  COVERAGE_PROCESS_DIRECTION	= 0x10u,
};

struct KerxTable
{
  inline bool apply (hb_aat_apply_context_t *c, const AAT::ankr *ankr) const
228
  {
229 230 231
    TRACE_APPLY (this);
    /* TODO */
    return_trace (false);
232 233
  }

234 235 236
  inline unsigned int get_size (void) const { return length; }

  inline bool sanitize (hb_sanitize_context_t *c) const
237 238
  {
    TRACE_SANITIZE (this);
239 240 241
    if (!c->check_struct (this))
      return_trace (false);

242
    switch (format) {
243 244 245 246 247 248
    case 0: return u.format0.sanitize (c);
    case 1: return u.format1.sanitize (c);
    case 2: return u.format2.sanitize (c);
    case 4: return u.format4.sanitize (c);
    case 6: return u.format6.sanitize (c);
    default:return_trace (false);
249 250 251
    }
  }

252 253 254 255 256 257
protected:
  HBUINT32	length;
  HBUINT8	coverage;
  HBUINT16	unused;
  HBUINT8	format;
  HBUINT32	tupleIndex;
258 259
  union {
  KerxSubTableFormat0	format0;
260
  KerxSubTableFormat1	format1;
261 262 263 264
  KerxSubTableFormat2	format2;
  KerxSubTableFormat4	format4;
  KerxSubTableFormat6	format6;
  } u;
265 266
public:
  DEFINE_SIZE_MIN (12);
267 268
};

269 270 271 272 273 274 275
struct SubtableGlyphCoverageArray
{
  inline bool sanitize (hb_sanitize_context_t *c) const
  {
    TRACE_SANITIZE (this);
    return_trace (c->check_struct (this));
  }
276

277 278 279 280 281 282 283
  protected:
  HBUINT32	length;
  HBUINT32	coverage;
  HBUINT32	tupleCount;
  public:
  DEFINE_SIZE_STATIC (12);
};
284 285 286

struct kerx
{
287
  static const hb_tag_t tableTag = HB_AAT_TAG_kerx;
288

E
Ebrahim Byagowi 已提交
289
  inline bool apply (hb_aat_apply_context_t *c, const AAT::ankr *ankr) const
290 291
  {
    TRACE_APPLY (this);
292 293
    const KerxTable &table = StructAfter<KerxTable> (*this);
    return_trace (table.apply (c, ankr));
294 295
  }

296
  inline bool sanitize (hb_sanitize_context_t *c) const
297
  {
298 299 300
    TRACE_SANITIZE (this);
    if (!(c->check_struct (this)))
     return_trace (false);
301

E
Ebrahim Byagowi 已提交
302
    /* TODO: Something like `morx`s ChainSubtable should be done here instead */
303 304 305
    const KerxTable *table = &StructAfter<KerxTable> (*this);
    if (!(table->sanitize (c)))
      return_trace (false);
306

307
    for (unsigned int i = 0; i < nTables - 1; ++i)
308
    {
309 310 311
      table = &StructAfter<KerxTable> (*table);
      if (!(table->sanitize (c)))
        return_trace (false);
312 313
    }

314 315 316
    // If version is less than 3, we are done here; otherwise better to check footer also
    if (version < 3)
      return_trace (true);
317

318 319 320 321
    // TODO: Investigate why this just work on some fonts no matter of version
    // const SubtableGlyphCoverageArray &footer =
    //   StructAfter<SubtableGlyphCoverageArray> (*table);
    // return_trace (footer.sanitize (c));
322 323 324 325 326 327 328

    return_trace (true);
  }

  protected:
  HBUINT16		version;
  HBUINT16		padding;
329 330 331
  HBUINT32		nTables;
/*KerxTable tables[VAR];*/
/*SubtableGlyphCoverageArray coverage_array;*/
332
  public:
333
  DEFINE_SIZE_STATIC (8);
334 335 336 337 338 339
};

} /* namespace AAT */


#endif /* HB_AAT_LAYOUT_KERX_TABLE_HH */