harfbuzz/src/hb-ot-layout-gpos-table.hh

1553 lines
48 KiB
C++
Raw Normal View History

2009-05-18 23:09:33 +02:00
/*
2011-04-21 23:14:28 +02:00
* Copyright © 2007,2008,2009,2010 Red Hat, Inc.
2012-04-24 04:26:13 +02:00
* Copyright © 2010,2012 Google, Inc.
2009-05-18 23:09:33 +02:00
*
2010-04-22 06:11:43 +02:00
* This is part of HarfBuzz, a text shaping library.
2009-05-18 23:09:33 +02:00
*
* 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.
*
* Red Hat Author(s): Behdad Esfahbod
* Google Author(s): Behdad Esfahbod
2009-05-18 23:09:33 +02:00
*/
#ifndef HB_OT_LAYOUT_GPOS_TABLE_HH
#define HB_OT_LAYOUT_GPOS_TABLE_HH
2009-05-18 23:09:33 +02:00
#include "hb-ot-layout-gsubgpos-private.hh"
2009-05-18 23:09:33 +02:00
2010-07-23 21:11:18 +02:00
namespace OT {
2010-05-05 07:13:09 +02:00
2011-07-28 22:48:43 +02:00
/* buffer **position** var allocations */
2010-10-28 05:09:10 +02:00
#define attach_lookback() var.u16[0] /* number of glyphs to go back to attach this glyph to its base */
#define cursive_chain() var.i16[1] /* character to which this connects, may be positive or negative */
2009-05-18 23:09:33 +02:00
/* Shared Tables: ValueRecord, Anchor Table, and MarkArray */
2010-05-06 19:06:15 +02:00
typedef USHORT Value;
typedef Value ValueRecord[VAR];
2009-05-18 23:09:33 +02:00
2009-05-20 05:42:30 +02:00
struct ValueFormat : USHORT
{
enum Flags {
xPlacement = 0x0001, /* Includes horizontal adjustment for placement */
yPlacement = 0x0002, /* Includes vertical adjustment for placement */
xAdvance = 0x0004, /* Includes horizontal adjustment for advance */
yAdvance = 0x0008, /* Includes vertical adjustment for advance */
xPlaDevice = 0x0010, /* Includes horizontal Device table for placement */
yPlaDevice = 0x0020, /* Includes vertical Device table for placement */
xAdvDevice = 0x0040, /* Includes horizontal Device table for advance */
yAdvDevice = 0x0080, /* Includes vertical Device table for advance */
ignored = 0x0F00, /* Was used in TrueType Open for MM fonts */
reserved = 0xF000, /* For future use */
devices = 0x00F0 /* Mask for having any Device table */
2010-04-21 08:02:57 +02:00
};
2010-04-21 08:02:57 +02:00
/* All fields are options. Only those available advance the value pointer. */
#if 0
SHORT xPlacement; /* Horizontal adjustment for
* placement--in design units */
SHORT yPlacement; /* Vertical adjustment for
* placement--in design units */
SHORT xAdvance; /* Horizontal adjustment for
* advance--in design units (only used
* for horizontal writing) */
SHORT yAdvance; /* Vertical adjustment for advance--in
* design units (only used for vertical
* writing) */
Offset xPlaDevice; /* Offset to Device table for
* horizontal placement--measured from
* beginning of PosTable (may be NULL) */
Offset yPlaDevice; /* Offset to Device table for vertical
* placement--measured from beginning
* of PosTable (may be NULL) */
Offset xAdvDevice; /* Offset to Device table for
* horizontal advance--measured from
* beginning of PosTable (may be NULL) */
Offset yAdvDevice; /* Offset to Device table for vertical
* advance--measured from beginning of
* PosTable (may be NULL) */
#endif
2010-10-02 00:58:50 +02:00
inline unsigned int get_len (void) const
2010-04-21 08:02:57 +02:00
{ return _hb_popcount32 ((unsigned int) *this); }
2010-10-02 00:58:50 +02:00
inline unsigned int get_size (void) const
{ return get_len () * Value::static_size; }
2010-04-21 08:02:57 +02:00
void apply_value (hb_font_t *font,
hb_direction_t direction,
const void *base,
const Value *values,
hb_glyph_position_t &glyph_pos) const
2010-04-21 08:02:57 +02:00
{
unsigned int x_ppem, y_ppem;
unsigned int format = *this;
hb_bool_t horizontal = HB_DIRECTION_IS_HORIZONTAL (direction);
2010-04-21 08:02:57 +02:00
if (!format) return;
2011-05-11 06:04:15 +02:00
if (format & xPlacement) glyph_pos.x_offset += font->em_scale_x (get_short (values++));
if (format & yPlacement) glyph_pos.y_offset += font->em_scale_y (get_short (values++));
if (format & xAdvance) {
if (likely (horizontal)) glyph_pos.x_advance += font->em_scale_x (get_short (values++)); else values++;
}
/* y_advance values grow downward but font-space grows upward, hence negation */
if (format & yAdvance) {
if (unlikely (!horizontal)) glyph_pos.y_advance -= font->em_scale_y (get_short (values++)); else values++;
}
2010-04-29 20:31:56 +02:00
if (!has_device ()) return;
x_ppem = font->x_ppem;
y_ppem = font->y_ppem;
2010-04-29 20:31:56 +02:00
if (!x_ppem && !y_ppem) return;
/* pixel -> fractional pixel */
if (format & xPlaDevice) {
if (x_ppem) glyph_pos.x_offset += (base + get_device (values++)).get_x_delta (font); else values++;
}
if (format & yPlaDevice) {
if (y_ppem) glyph_pos.y_offset += (base + get_device (values++)).get_y_delta (font); else values++;
}
if (format & xAdvDevice) {
if (horizontal && x_ppem) glyph_pos.x_advance += (base + get_device (values++)).get_x_delta (font); else values++;
}
if (format & yAdvDevice) {
/* y_advance values grow downward but font-space grows upward, hence negation */
if (!horizontal && y_ppem) glyph_pos.y_advance -= (base + get_device (values++)).get_y_delta (font); else values++;
}
}
2010-04-21 08:02:57 +02:00
private:
2010-05-13 20:18:49 +02:00
inline bool sanitize_value_devices (hb_sanitize_context_t *c, void *base, Value *values) {
2010-04-21 08:02:57 +02:00
unsigned int format = *this;
if (format & xPlacement) values++;
if (format & yPlacement) values++;
if (format & xAdvance) values++;
if (format & yAdvance) values++;
2010-05-13 20:18:49 +02:00
if ((format & xPlaDevice) && !get_device (values++).sanitize (c, base)) return false;
if ((format & yPlaDevice) && !get_device (values++).sanitize (c, base)) return false;
if ((format & xAdvDevice) && !get_device (values++).sanitize (c, base)) return false;
if ((format & yAdvDevice) && !get_device (values++).sanitize (c, base)) return false;
2010-04-21 08:02:57 +02:00
return true;
}
2010-05-06 19:06:15 +02:00
static inline OffsetTo<Device>& get_device (Value* value)
{ return *CastP<OffsetTo<Device> > (value); }
static inline const OffsetTo<Device>& get_device (const Value* value)
{ return *CastP<OffsetTo<Device> > (value); }
static inline const SHORT& get_short (const Value* value)
{ return *CastP<SHORT> (value); }
2010-04-21 08:02:57 +02:00
public:
2010-10-02 00:58:50 +02:00
inline bool has_device (void) const {
2010-04-21 08:02:57 +02:00
unsigned int format = *this;
return (format & devices) != 0;
}
2010-05-13 20:18:49 +02:00
inline bool sanitize_value (hb_sanitize_context_t *c, void *base, Value *values) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_range (values, get_size ()) && (!has_device () || sanitize_value_devices (c, base, values)));
2010-04-21 08:02:57 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize_values (hb_sanitize_context_t *c, void *base, Value *values, unsigned int count) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
2010-04-21 08:02:57 +02:00
unsigned int len = get_len ();
if (!c->check_array (values, get_size (), count)) return TRACE_RETURN (false);
2010-04-21 08:02:57 +02:00
if (!has_device ()) return TRACE_RETURN (true);
2010-04-21 08:02:57 +02:00
for (unsigned int i = 0; i < count; i++) {
2010-05-13 20:18:49 +02:00
if (!sanitize_value_devices (c, base, values))
return TRACE_RETURN (false);
2010-04-21 08:02:57 +02:00
values += len;
}
return TRACE_RETURN (true);
2010-04-21 08:02:57 +02:00
}
/* Just sanitize referenced Device tables. Doesn't check the values themselves. */
2010-05-13 20:18:49 +02:00
inline bool sanitize_values_stride_unsafe (hb_sanitize_context_t *c, void *base, Value *values, unsigned int count, unsigned int stride) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
2010-04-21 08:02:57 +02:00
if (!has_device ()) return TRACE_RETURN (true);
2010-04-21 08:02:57 +02:00
for (unsigned int i = 0; i < count; i++) {
2010-05-13 20:18:49 +02:00
if (!sanitize_value_devices (c, base, values))
return TRACE_RETURN (false);
2010-04-21 08:02:57 +02:00
values += stride;
}
return TRACE_RETURN (true);
2010-04-21 08:02:57 +02:00
}
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct AnchorFormat1
{
inline void get_anchor (hb_font_t *font, hb_codepoint_t glyph_id HB_UNUSED,
2009-05-20 04:16:04 +02:00
hb_position_t *x, hb_position_t *y) const
{
2011-05-11 06:04:15 +02:00
*x = font->em_scale_x (xCoordinate);
*y = font->em_scale_y (yCoordinate);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
SHORT xCoordinate; /* Horizontal value--in design units */
SHORT yCoordinate; /* Vertical value--in design units */
public:
DEFINE_SIZE_STATIC (6);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct AnchorFormat2
{
inline void get_anchor (hb_font_t *font, hb_codepoint_t glyph_id,
2009-05-20 04:16:04 +02:00
hb_position_t *x, hb_position_t *y) const
{
unsigned int x_ppem = font->x_ppem;
unsigned int y_ppem = font->y_ppem;
2010-04-29 09:59:06 +02:00
hb_position_t cx, cy;
2010-11-03 20:40:07 +01:00
hb_bool_t ret = false;
2010-04-29 09:59:06 +02:00
if (x_ppem || y_ppem)
2012-08-02 01:03:46 +02:00
ret = font->get_glyph_contour_point_for_origin (glyph_id, anchorPoint, HB_DIRECTION_LTR, &cx, &cy);
2011-05-11 06:04:15 +02:00
*x = x_ppem && ret ? cx : font->em_scale_x (xCoordinate);
*y = y_ppem && ret ? cy : font->em_scale_y (yCoordinate);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 2 */
SHORT xCoordinate; /* Horizontal value--in design units */
SHORT yCoordinate; /* Vertical value--in design units */
USHORT anchorPoint; /* Index to glyph contour point */
public:
DEFINE_SIZE_STATIC (8);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct AnchorFormat3
{
inline void get_anchor (hb_font_t *font, hb_codepoint_t glyph_id HB_UNUSED,
2009-05-20 04:16:04 +02:00
hb_position_t *x, hb_position_t *y) const
{
2011-05-11 06:04:15 +02:00
*x = font->em_scale_x (xCoordinate);
*y = font->em_scale_y (yCoordinate);
if (font->x_ppem)
*x += (this+xDeviceTable).get_x_delta (font);
if (font->y_ppem)
*y += (this+yDeviceTable).get_x_delta (font);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && xDeviceTable.sanitize (c, this) && yDeviceTable.sanitize (c, this));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 3 */
SHORT xCoordinate; /* Horizontal value--in design units */
SHORT yCoordinate; /* Vertical value--in design units */
OffsetTo<Device>
xDeviceTable; /* Offset to Device table for X
* coordinate-- from beginning of
* Anchor table (may be NULL) */
OffsetTo<Device>
yDeviceTable; /* Offset to Device table for Y
* coordinate-- from beginning of
* Anchor table (may be NULL) */
public:
DEFINE_SIZE_STATIC (10);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct Anchor
{
inline void get_anchor (hb_font_t *font, hb_codepoint_t glyph_id,
2009-05-20 04:16:04 +02:00
hb_position_t *x, hb_position_t *y) const
{
2009-05-18 23:09:33 +02:00
*x = *y = 0;
switch (u.format) {
case 1: u.format1.get_anchor (font, glyph_id, x, y); return;
case 2: u.format2.get_anchor (font, glyph_id, x, y); return;
case 3: u.format3.get_anchor (font, glyph_id, x, y); return;
default: return;
2009-05-18 23:09:33 +02:00
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
case 2: return TRACE_RETURN (u.format2.sanitize (c));
case 3: return TRACE_RETURN (u.format3.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
AnchorFormat1 format1;
AnchorFormat2 format2;
AnchorFormat3 format3;
2009-05-18 23:09:33 +02:00
} u;
2010-05-11 00:08:46 +02:00
public:
2010-05-11 00:47:48 +02:00
DEFINE_SIZE_UNION (2, format);
2009-05-18 23:09:33 +02:00
};
2009-08-15 00:14:03 +02:00
struct AnchorMatrix
{
inline const Anchor& get_anchor (unsigned int row, unsigned int col, unsigned int cols) const {
if (unlikely (row >= rows || col >= cols)) return Null(Anchor);
2009-08-15 00:14:03 +02:00
return this+matrix[row * cols + col];
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c, unsigned int cols) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!c->check_struct (this)) return TRACE_RETURN (false);
if (unlikely (rows > 0 && cols >= ((unsigned int) -1) / rows)) return TRACE_RETURN (false);
2009-08-15 00:14:03 +02:00
unsigned int count = rows * cols;
if (!c->check_array (matrix, matrix[0].static_size, count)) return TRACE_RETURN (false);
2009-08-15 00:14:03 +02:00
for (unsigned int i = 0; i < count; i++)
if (!matrix[i].sanitize (c, this)) return TRACE_RETURN (false);
return TRACE_RETURN (true);
2009-08-15 00:14:03 +02:00
}
USHORT rows; /* Number of rows */
protected:
2009-08-15 00:14:03 +02:00
OffsetTo<Anchor>
matrix[VAR]; /* Matrix of offsets to Anchor tables--
2009-08-15 00:14:03 +02:00
* from beginning of AnchorMatrix table */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_ARRAY (2, matrix);
2009-08-15 00:14:03 +02:00
};
2009-05-20 05:42:30 +02:00
struct MarkRecord
{
2009-05-21 10:58:24 +02:00
friend struct MarkArray;
2009-05-18 23:09:33 +02:00
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c, void *base) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && markAnchor.sanitize (c, base));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT klass; /* Class defined for this mark */
OffsetTo<Anchor>
markAnchor; /* Offset to Anchor table--from
* beginning of MarkArray table */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_STATIC (4);
2009-05-18 23:09:33 +02:00
};
2010-05-10 23:28:16 +02:00
struct MarkArray : ArrayOf<MarkRecord> /* Array of MarkRecords--in Coverage order */
2009-05-20 05:42:30 +02:00
{
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c,
unsigned int mark_index, unsigned int glyph_index,
const AnchorMatrix &anchors, unsigned int class_count,
unsigned int glyph_pos) const
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
2010-05-10 23:28:16 +02:00
const MarkRecord &record = ArrayOf<MarkRecord>::operator[](mark_index);
unsigned int mark_class = record.klass;
const Anchor& mark_anchor = this + record.markAnchor;
const Anchor& glyph_anchor = anchors.get_anchor (glyph_index, mark_class, class_count);
hb_position_t mark_x, mark_y, base_x, base_y;
mark_anchor.get_anchor (c->font, c->buffer->cur().codepoint, &mark_x, &mark_y);
glyph_anchor.get_anchor (c->font, c->buffer->info[glyph_pos].codepoint, &base_x, &base_y);
hb_glyph_position_t &o = c->buffer->cur_pos();
2010-10-28 05:09:10 +02:00
o.x_offset = base_x - mark_x;
o.y_offset = base_y - mark_y;
2011-07-22 17:28:07 +02:00
o.attach_lookback() = c->buffer->idx - glyph_pos;
2011-07-22 17:28:07 +02:00
c->buffer->idx++;
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (true);
}
2009-05-18 23:09:33 +02:00
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (ArrayOf<MarkRecord>::sanitize (c, this));
2009-08-04 19:30:49 +02:00
}
2009-05-18 23:09:33 +02:00
};
/* Lookups */
2009-05-20 05:42:30 +02:00
struct SinglePosFormat1
{
inline const Coverage &get_coverage (void) const
{
return this+coverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 04:16:04 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
valueFormat.apply_value (c->font, c->direction, this,
values, c->buffer->cur_pos());
2011-07-22 17:28:07 +02:00
c->buffer->idx++;
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (true);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && coverage.sanitize (c, this) && valueFormat.sanitize_value (c, this, values));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
OffsetTo<Coverage>
coverage; /* Offset to Coverage table--from
* beginning of subtable */
ValueFormat valueFormat; /* Defines the types of data in the
2009-05-18 23:09:33 +02:00
* ValueRecord */
ValueRecord values; /* Defines positioning
* value(s)--applied to all glyphs in
* the Coverage table */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_ARRAY (6, values);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct SinglePosFormat2
{
inline const Coverage &get_coverage (void) const
{
return this+coverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 04:16:04 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
2012-05-11 02:33:11 +02:00
if (likely (index >= valueCount)) return TRACE_RETURN (false);
valueFormat.apply_value (c->font, c->direction, this,
&values[index * valueFormat.get_len ()],
c->buffer->cur_pos());
2011-07-22 17:28:07 +02:00
c->buffer->idx++;
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (true);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && coverage.sanitize (c, this) && valueFormat.sanitize_values (c, this, values, valueCount));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 2 */
OffsetTo<Coverage>
coverage; /* Offset to Coverage table--from
* beginning of subtable */
ValueFormat valueFormat; /* Defines the types of data in the
2009-05-18 23:09:33 +02:00
* ValueRecord */
USHORT valueCount; /* Number of ValueRecords */
ValueRecord values; /* Array of ValueRecords--positioning
* values applied to glyphs */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_ARRAY (8, values);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct SinglePos
{
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (u.format) {
2012-11-23 23:23:41 +01:00
case 1: return TRACE_RETURN (c->process (u.format1));
case 2: return TRACE_RETURN (c->process (u.format2));
default:return TRACE_RETURN (c->default_return_value ());
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
case 2: return TRACE_RETURN (u.format2.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
SinglePosFormat1 format1;
SinglePosFormat2 format2;
2009-05-18 23:09:33 +02:00
} u;
};
2009-05-20 05:42:30 +02:00
struct PairValueRecord
{
2010-05-11 06:23:50 +02:00
friend struct PairSet;
2009-05-18 23:09:33 +02:00
protected:
2009-05-18 23:09:33 +02:00
GlyphID secondGlyph; /* GlyphID of second glyph in the
* pair--first glyph is listed in the
* Coverage table */
ValueRecord values; /* Positioning data for the first glyph
* followed by for second glyph */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_ARRAY (2, values);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct PairSet
{
2009-05-20 04:16:04 +02:00
friend struct PairPosFormat1;
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c,
2010-05-11 06:23:50 +02:00
const ValueFormat *valueFormats,
unsigned int pos) const
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
2010-05-11 06:23:50 +02:00
unsigned int len1 = valueFormats[0].get_len ();
unsigned int len2 = valueFormats[1].get_len ();
unsigned int record_size = USHORT::static_size * (1 + len1 + len2);
unsigned int count = len;
const PairValueRecord *record = CastP<PairValueRecord> (array);
for (unsigned int i = 0; i < count; i++)
{
if (c->buffer->info[pos].codepoint == record->secondGlyph)
2010-05-11 06:23:50 +02:00
{
valueFormats[0].apply_value (c->font, c->direction, this,
&record->values[0], c->buffer->cur_pos());
valueFormats[1].apply_value (c->font, c->direction, this,
&record->values[len1], c->buffer->pos[pos]);
2010-05-11 06:23:50 +02:00
if (len2)
pos++;
2011-07-22 17:28:07 +02:00
c->buffer->idx = pos;
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (true);
2010-05-11 06:23:50 +02:00
}
record = &StructAtOffset<PairValueRecord> (record, record_size);
}
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (false);
2010-05-11 06:23:50 +02:00
}
struct sanitize_closure_t {
void *base;
ValueFormat *valueFormats;
unsigned int len1; /* valueFormats[0].get_len() */
unsigned int stride; /* 1 + len1 + len2 */
};
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c, const sanitize_closure_t *closure) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
2010-05-13 20:18:49 +02:00
if (!(c->check_struct (this)
&& c->check_array (array, USHORT::static_size * closure->stride, len))) return TRACE_RETURN (false);
2010-05-11 06:23:50 +02:00
unsigned int count = len;
PairValueRecord *record = CastP<PairValueRecord> (array);
return TRACE_RETURN (closure->valueFormats[0].sanitize_values_stride_unsafe (c, closure->base, &record->values[0], count, closure->stride)
&& closure->valueFormats[1].sanitize_values_stride_unsafe (c, closure->base, &record->values[closure->len1], count, closure->stride));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-20 04:16:04 +02:00
USHORT len; /* Number of PairValueRecords */
USHORT array[VAR]; /* Array of PairValueRecords--ordered
2009-05-20 04:16:04 +02:00
* by GlyphID of the second glyph */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_ARRAY (2, array);
2009-05-20 04:16:04 +02:00
};
2009-05-18 23:09:33 +02:00
2009-05-20 05:42:30 +02:00
struct PairPosFormat1
{
inline const Coverage &get_coverage (void) const
{
return this+coverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 04:16:04 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
2012-01-17 04:05:08 +01:00
hb_apply_context_t::mark_skipping_forward_iterator_t skippy_iter (c, c->buffer->idx, 1);
2012-05-11 02:33:11 +02:00
if (skippy_iter.has_no_chance ()) return TRACE_RETURN (false);
2009-05-20 04:16:04 +02:00
unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
2009-05-20 04:16:04 +02:00
2012-05-11 02:33:11 +02:00
if (!skippy_iter.next ()) return TRACE_RETURN (false);
2009-05-20 04:16:04 +02:00
2012-05-11 02:33:11 +02:00
return TRACE_RETURN ((this+pairSet[index]).apply (c, &valueFormat1, skippy_iter.idx));
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
unsigned int len1 = valueFormat1.get_len ();
unsigned int len2 = valueFormat2.get_len ();
2010-05-11 06:23:50 +02:00
PairSet::sanitize_closure_t closure = {
this,
&valueFormat1,
len1,
1 + len1 + len2
};
return TRACE_RETURN (c->check_struct (this) && coverage.sanitize (c, this) && pairSet.sanitize (c, this, &closure));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
OffsetTo<Coverage>
coverage; /* Offset to Coverage table--from
* beginning of subtable */
ValueFormat valueFormat1; /* Defines the types of data in
2009-05-18 23:09:33 +02:00
* ValueRecord1--for the first glyph
* in the pair--may be zero (0) */
ValueFormat valueFormat2; /* Defines the types of data in
2009-05-18 23:09:33 +02:00
* ValueRecord2--for the second glyph
* in the pair--may be zero (0) */
OffsetArrayOf<PairSet>
pairSet; /* Array of PairSet tables
* ordered by Coverage Index */
public:
DEFINE_SIZE_ARRAY (10, pairSet);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct PairPosFormat2
{
inline const Coverage &get_coverage (void) const
{
return this+coverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 04:30:09 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
2012-01-17 04:05:08 +01:00
hb_apply_context_t::mark_skipping_forward_iterator_t skippy_iter (c, c->buffer->idx, 1);
2012-05-11 02:33:11 +02:00
if (skippy_iter.has_no_chance ()) return TRACE_RETURN (false);
2009-05-20 04:30:09 +02:00
unsigned int index = (this+coverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (index == NOT_COVERED)) return TRACE_RETURN (false);
2009-05-20 04:30:09 +02:00
2012-05-11 02:33:11 +02:00
if (!skippy_iter.next ()) return TRACE_RETURN (false);
2009-05-20 04:30:09 +02:00
unsigned int len1 = valueFormat1.get_len ();
unsigned int len2 = valueFormat2.get_len ();
unsigned int record_len = len1 + len2;
unsigned int klass1 = (this+classDef1) (c->buffer->cur().codepoint);
2012-01-17 04:05:08 +01:00
unsigned int klass2 = (this+classDef2) (c->buffer->info[skippy_iter.idx].codepoint);
2012-05-11 02:33:11 +02:00
if (unlikely (klass1 >= class1Count || klass2 >= class2Count)) return TRACE_RETURN (false);
2009-05-20 04:30:09 +02:00
const Value *v = &values[record_len * (klass1 * class2Count + klass2)];
valueFormat1.apply_value (c->font, c->direction, this,
v, c->buffer->cur_pos());
valueFormat2.apply_value (c->font, c->direction, this,
2012-01-17 04:05:08 +01:00
v + len1, c->buffer->pos[skippy_iter.idx]);
2009-05-20 04:30:09 +02:00
2012-01-17 04:05:08 +01:00
c->buffer->idx = skippy_iter.idx;
2009-05-20 04:30:09 +02:00
if (len2)
2012-01-17 04:05:08 +01:00
c->buffer->idx++;
2009-05-20 04:30:09 +02:00
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (true);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
2010-05-13 20:18:49 +02:00
if (!(c->check_struct (this)
&& coverage.sanitize (c, this)
&& classDef1.sanitize (c, this)
&& classDef2.sanitize (c, this))) return TRACE_RETURN (false);
unsigned int len1 = valueFormat1.get_len ();
unsigned int len2 = valueFormat2.get_len ();
unsigned int stride = len1 + len2;
unsigned int record_size = valueFormat1.get_size () + valueFormat2.get_size ();
unsigned int count = (unsigned int) class1Count * (unsigned int) class2Count;
return TRACE_RETURN (c->check_array (values, record_size, count) &&
valueFormat1.sanitize_values_stride_unsafe (c, this, &values[0], count, stride) &&
valueFormat2.sanitize_values_stride_unsafe (c, this, &values[len1], count, stride));
2009-08-04 19:30:49 +02:00
}
2009-05-20 04:30:09 +02:00
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 2 */
OffsetTo<Coverage>
coverage; /* Offset to Coverage table--from
* beginning of subtable */
ValueFormat valueFormat1; /* ValueRecord definition--for the
2009-05-18 23:09:33 +02:00
* first glyph of the pair--may be zero
* (0) */
ValueFormat valueFormat2; /* ValueRecord definition--for the
2009-05-18 23:09:33 +02:00
* second glyph of the pair--may be
* zero (0) */
OffsetTo<ClassDef>
classDef1; /* Offset to ClassDef table--from
* beginning of PairPos subtable--for
* the first glyph of the pair */
OffsetTo<ClassDef>
classDef2; /* Offset to ClassDef table--from
* beginning of PairPos subtable--for
* the second glyph of the pair */
USHORT class1Count; /* Number of classes in ClassDef1
* table--includes Class0 */
USHORT class2Count; /* Number of classes in ClassDef2
* table--includes Class0 */
ValueRecord values; /* Matrix of value pairs:
* class1-major, class2-minor,
* Each entry has value1 and value2 */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_ARRAY (16, values);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct PairPos
{
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (u.format) {
2012-11-23 23:23:41 +01:00
case 1: return TRACE_RETURN (c->process (u.format1));
case 2: return TRACE_RETURN (c->process (u.format2));
default:return TRACE_RETURN (c->default_return_value ());
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
case 2: return TRACE_RETURN (u.format2.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
PairPosFormat1 format1;
PairPosFormat2 format2;
2009-05-18 23:09:33 +02:00
} u;
};
2009-05-20 05:42:30 +02:00
struct EntryExitRecord
{
2010-05-10 22:38:32 +02:00
friend struct CursivePosFormat1;
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c, void *base) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (entryAnchor.sanitize (c, base) && exitAnchor.sanitize (c, base));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
OffsetTo<Anchor>
entryAnchor; /* Offset to EntryAnchor table--from
* beginning of CursivePos
* subtable--may be NULL */
OffsetTo<Anchor>
exitAnchor; /* Offset to ExitAnchor table--from
* beginning of CursivePos
* subtable--may be NULL */
2010-05-10 22:38:32 +02:00
public:
DEFINE_SIZE_STATIC (4);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct CursivePosFormat1
{
inline const Coverage &get_coverage (void) const
{
return this+coverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 05:25:41 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
2009-05-20 10:16:35 +02:00
/* We don't handle mark glyphs here. */
if (c->property & HB_OT_LAYOUT_GLYPH_PROPS_MARK) return TRACE_RETURN (false);
2009-05-20 10:16:35 +02:00
2012-01-17 04:05:08 +01:00
hb_apply_context_t::mark_skipping_forward_iterator_t skippy_iter (c, c->buffer->idx, 1);
2012-05-11 02:33:11 +02:00
if (skippy_iter.has_no_chance ()) return TRACE_RETURN (false);
2009-05-20 10:16:35 +02:00
const EntryExitRecord &this_record = entryExitRecord[(this+coverage) (c->buffer->cur().codepoint)];
2012-05-11 02:33:11 +02:00
if (!this_record.exitAnchor) return TRACE_RETURN (false);
2012-05-11 02:33:11 +02:00
if (!skippy_iter.next ()) return TRACE_RETURN (false);
2012-01-17 04:05:08 +01:00
const EntryExitRecord &next_record = entryExitRecord[(this+coverage) (c->buffer->info[skippy_iter.idx].codepoint)];
2012-05-11 02:33:11 +02:00
if (!next_record.entryAnchor) return TRACE_RETURN (false);
2009-05-20 05:25:41 +02:00
2011-07-22 17:28:07 +02:00
unsigned int i = c->buffer->idx;
2012-01-17 04:05:08 +01:00
unsigned int j = skippy_iter.idx;
2009-05-20 05:25:41 +02:00
hb_position_t entry_x, entry_y, exit_x, exit_y;
(this+this_record.exitAnchor).get_anchor (c->font, c->buffer->info[i].codepoint, &exit_x, &exit_y);
(this+next_record.entryAnchor).get_anchor (c->font, c->buffer->info[j].codepoint, &entry_x, &entry_y);
2009-05-27 06:17:37 +02:00
hb_glyph_position_t *pos = c->buffer->pos;
hb_position_t d;
/* Main-direction adjustment */
switch (c->direction) {
case HB_DIRECTION_LTR:
pos[i].x_advance = exit_x + pos[i].x_offset;
d = entry_x + pos[j].x_offset;
pos[j].x_advance -= d;
pos[j].x_offset -= d;
break;
case HB_DIRECTION_RTL:
d = exit_x + pos[i].x_offset;
pos[i].x_advance -= d;
pos[i].x_offset -= d;
pos[j].x_advance = entry_x + pos[j].x_offset;
break;
case HB_DIRECTION_TTB:
pos[i].y_advance = exit_y + pos[i].y_offset;
d = entry_y + pos[j].y_offset;
pos[j].y_advance -= d;
pos[j].y_offset -= d;
break;
case HB_DIRECTION_BTT:
d = exit_y + pos[i].y_offset;
pos[i].y_advance -= d;
pos[i].y_offset -= d;
pos[j].y_advance = entry_y;
break;
case HB_DIRECTION_INVALID:
default:
break;
2009-05-20 05:25:41 +02:00
}
/* Cross-direction adjustment */
if (c->lookup_props & LookupFlag::RightToLeft) {
pos[i].cursive_chain() = j - i;
if (likely (HB_DIRECTION_IS_HORIZONTAL (c->direction)))
pos[i].y_offset = entry_y - exit_y;
else
pos[i].x_offset = entry_x - exit_x;
} else {
pos[j].cursive_chain() = i - j;
if (likely (HB_DIRECTION_IS_HORIZONTAL (c->direction)))
pos[j].y_offset = exit_y - entry_y;
else
pos[j].x_offset = exit_x - entry_x;
2009-05-20 05:25:41 +02:00
}
2011-07-22 17:28:07 +02:00
c->buffer->idx = j;
2012-05-11 02:33:11 +02:00
return TRACE_RETURN (true);
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (coverage.sanitize (c, this) && entryExitRecord.sanitize (c, this));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
OffsetTo<Coverage>
coverage; /* Offset to Coverage table--from
* beginning of subtable */
ArrayOf<EntryExitRecord>
entryExitRecord; /* Array of EntryExit records--in
* Coverage Index order */
public:
DEFINE_SIZE_ARRAY (6, entryExitRecord);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct CursivePos
{
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (u.format) {
2012-11-23 23:23:41 +01:00
case 1: return TRACE_RETURN (c->process (u.format1));
default:return TRACE_RETURN (c->default_return_value ());
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
CursivePosFormat1 format1;
2009-05-18 23:09:33 +02:00
} u;
};
2009-08-15 00:14:03 +02:00
typedef AnchorMatrix BaseArray; /* base-major--
* in order of BaseCoverage Index--,
* mark-minor--
2009-05-21 10:47:05 +02:00
* ordered by class--zero-based. */
2009-05-18 23:09:33 +02:00
2009-05-20 05:42:30 +02:00
struct MarkBasePosFormat1
{
inline const Coverage &get_coverage (void) const
{
return this+markCoverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 05:42:30 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
unsigned int mark_index = (this+markCoverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (mark_index == NOT_COVERED)) return TRACE_RETURN (false);
2009-05-21 12:32:01 +02:00
/* now we search backwards for a non-mark glyph */
unsigned int property;
2012-01-17 04:05:08 +01:00
hb_apply_context_t::mark_skipping_backward_iterator_t skippy_iter (c, c->buffer->idx, 1);
do {
if (!skippy_iter.prev (&property, LookupFlag::IgnoreMarks)) return TRACE_RETURN (false);
/* We only want to attach to the first of a MultipleSubst sequence. Reject others. */
if (0 == get_lig_comp (c->buffer->info[skippy_iter.idx])) break;
skippy_iter.reject ();
} while (1);
/* The following assertion is too strong, so we've disabled it. */
if (!(property & HB_OT_LAYOUT_GLYPH_PROPS_BASE_GLYPH)) {/*return TRACE_RETURN (false);*/}
2009-05-21 12:32:01 +02:00
2012-01-17 04:05:08 +01:00
unsigned int base_index = (this+baseCoverage) (c->buffer->info[skippy_iter.idx].codepoint);
2012-05-11 02:33:11 +02:00
if (base_index == NOT_COVERED) return TRACE_RETURN (false);
2009-05-21 12:32:01 +02:00
2012-05-11 02:33:11 +02:00
return TRACE_RETURN ((this+markArray).apply (c, mark_index, base_index, this+baseArray, classCount, skippy_iter.idx));
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && markCoverage.sanitize (c, this) && baseCoverage.sanitize (c, this) &&
markArray.sanitize (c, this) && baseArray.sanitize (c, this, (unsigned int) classCount));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
2009-05-21 10:47:05 +02:00
OffsetTo<Coverage>
markCoverage; /* Offset to MarkCoverage table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkBasePos subtable */
2009-05-21 10:47:05 +02:00
OffsetTo<Coverage>
baseCoverage; /* Offset to BaseCoverage table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkBasePos subtable */
USHORT classCount; /* Number of classes defined for marks */
2009-05-21 10:47:05 +02:00
OffsetTo<MarkArray>
markArray; /* Offset to MarkArray table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkBasePos subtable */
2009-05-21 10:47:05 +02:00
OffsetTo<BaseArray>
baseArray; /* Offset to BaseArray table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkBasePos subtable */
public:
DEFINE_SIZE_STATIC (12);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct MarkBasePos
{
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (u.format) {
2012-11-23 23:23:41 +01:00
case 1: return TRACE_RETURN (c->process (u.format1));
default:return TRACE_RETURN (c->default_return_value ());
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
MarkBasePosFormat1 format1;
2009-05-18 23:09:33 +02:00
} u;
};
2009-08-15 00:14:03 +02:00
typedef AnchorMatrix LigatureAttach; /* component-major--
* in order of writing direction--,
* mark-minor--
* ordered by class--zero-based. */
2009-05-18 23:09:33 +02:00
2009-08-15 00:32:56 +02:00
typedef OffsetListOf<LigatureAttach> LigatureArray;
/* Array of LigatureAttach
2009-05-18 23:09:33 +02:00
* tables ordered by
* LigatureCoverage Index */
2009-05-20 05:42:30 +02:00
struct MarkLigPosFormat1
{
inline const Coverage &get_coverage (void) const
{
return this+markCoverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 05:42:30 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
unsigned int mark_index = (this+markCoverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (mark_index == NOT_COVERED)) return TRACE_RETURN (false);
/* now we search backwards for a non-mark glyph */
unsigned int property;
2012-01-17 04:05:08 +01:00
hb_apply_context_t::mark_skipping_backward_iterator_t skippy_iter (c, c->buffer->idx, 1);
2012-05-11 02:33:11 +02:00
if (!skippy_iter.prev (&property, LookupFlag::IgnoreMarks)) return TRACE_RETURN (false);
/* The following assertion is too strong, so we've disabled it. */
if (!(property & HB_OT_LAYOUT_GLYPH_PROPS_LIGATURE)) {/*return TRACE_RETURN (false);*/}
2012-01-17 04:05:08 +01:00
unsigned int j = skippy_iter.idx;
unsigned int lig_index = (this+ligatureCoverage) (c->buffer->info[j].codepoint);
2012-05-11 02:33:11 +02:00
if (lig_index == NOT_COVERED) return TRACE_RETURN (false);
const LigatureArray& lig_array = this+ligatureArray;
2009-08-15 00:32:56 +02:00
const LigatureAttach& lig_attach = lig_array[lig_index];
2009-08-15 00:14:03 +02:00
/* Find component to attach to */
2009-08-15 01:37:18 +02:00
unsigned int comp_count = lig_attach.rows;
2012-05-11 02:33:11 +02:00
if (unlikely (!comp_count)) return TRACE_RETURN (false);
/* We must now check whether the ligature ID of the current mark glyph
* is identical to the ligature ID of the found ligature. If yes, we
* can directly use the component index. If not, we attach the mark
* glyph to the last component of the ligature. */
2012-07-30 06:55:15 +02:00
unsigned int comp_index;
unsigned int lig_id = get_lig_id (c->buffer->info[j]);
unsigned int mark_id = get_lig_id (c->buffer->cur());
unsigned int mark_comp = get_lig_comp (c->buffer->cur());
if (lig_id && lig_id == mark_id && mark_comp > 0)
comp_index = MIN (comp_count, get_lig_comp (c->buffer->cur())) - 1;
else
2009-08-15 01:37:18 +02:00
comp_index = comp_count - 1;
2012-05-11 02:33:11 +02:00
return TRACE_RETURN ((this+markArray).apply (c, mark_index, comp_index, lig_attach, classCount, j));
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && markCoverage.sanitize (c, this) && ligatureCoverage.sanitize (c, this) &&
markArray.sanitize (c, this) && ligatureArray.sanitize (c, this, (unsigned int) classCount));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
OffsetTo<Coverage>
markCoverage; /* Offset to Mark Coverage table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkLigPos subtable */
OffsetTo<Coverage>
ligatureCoverage; /* Offset to Ligature Coverage
2009-05-18 23:09:33 +02:00
* table--from beginning of MarkLigPos
* subtable */
USHORT classCount; /* Number of defined mark classes */
OffsetTo<MarkArray>
markArray; /* Offset to MarkArray table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkLigPos subtable */
OffsetTo<LigatureArray>
ligatureArray; /* Offset to LigatureArray table--from
2009-05-18 23:09:33 +02:00
* beginning of MarkLigPos subtable */
public:
DEFINE_SIZE_STATIC (12);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct MarkLigPos
{
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (u.format) {
2012-11-23 23:23:41 +01:00
case 1: return TRACE_RETURN (c->process (u.format1));
default:return TRACE_RETURN (c->default_return_value ());
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
MarkLigPosFormat1 format1;
2009-05-18 23:09:33 +02:00
} u;
};
2009-08-15 00:14:03 +02:00
typedef AnchorMatrix Mark2Array; /* mark2-major--
* in order of Mark2Coverage Index--,
* mark1-minor--
* ordered by class--zero-based. */
2009-05-18 23:09:33 +02:00
2009-05-20 05:42:30 +02:00
struct MarkMarkPosFormat1
{
inline const Coverage &get_coverage (void) const
{
return this+mark1Coverage;
}
2010-05-13 20:18:49 +02:00
inline bool apply (hb_apply_context_t *c) const
2009-05-20 05:42:30 +02:00
{
2012-11-23 21:32:14 +01:00
TRACE_APPLY (this);
unsigned int mark1_index = (this+mark1Coverage) (c->buffer->cur().codepoint);
2012-05-11 02:33:11 +02:00
if (likely (mark1_index == NOT_COVERED)) return TRACE_RETURN (false);
/* now we search backwards for a suitable mark glyph until a non-mark glyph */
unsigned int property;
2012-01-17 04:05:08 +01:00
hb_apply_context_t::mark_skipping_backward_iterator_t skippy_iter (c, c->buffer->idx, 1);
2012-05-11 02:33:11 +02:00
if (!skippy_iter.prev (&property)) return TRACE_RETURN (false);
if (!(property & HB_OT_LAYOUT_GLYPH_PROPS_MARK)) return TRACE_RETURN (false);
2009-05-26 23:58:37 +02:00
2012-01-17 04:05:08 +01:00
unsigned int j = skippy_iter.idx;
unsigned int id1 = get_lig_id (c->buffer->cur());
unsigned int id2 = get_lig_id (c->buffer->info[j]);
unsigned int comp1 = get_lig_comp (c->buffer->cur());
unsigned int comp2 = get_lig_comp (c->buffer->info[j]);
if (likely (id1 == id2)) {
if (id1 == 0) /* Marks belonging to the same base. */
goto good;
else if (comp1 == comp2) /* Marks belonging to the same ligature component. */
goto good;
} else {
/* If ligature ids don't match, it may be the case that one of the marks
* itself is a ligature. In which case match. */
if ((id1 > 0 && !comp1) || (id2 > 0 && !comp2))
goto good;
}
/* Didn't match. */
return TRACE_RETURN (false);
good:
unsigned int mark2_index = (this+mark2Coverage) (c->buffer->info[j].codepoint);
2012-05-11 02:33:11 +02:00
if (mark2_index == NOT_COVERED) return TRACE_RETURN (false);
2012-05-11 02:33:11 +02:00
return TRACE_RETURN ((this+mark1Array).apply (c, mark1_index, mark2_index, this+mark2Array, classCount, j));
2009-05-18 23:09:33 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
return TRACE_RETURN (c->check_struct (this) && mark1Coverage.sanitize (c, this) &&
mark2Coverage.sanitize (c, this) && mark1Array.sanitize (c, this)
&& mark2Array.sanitize (c, this, (unsigned int) classCount));
2009-08-04 19:30:49 +02:00
}
protected:
2009-05-18 23:09:33 +02:00
USHORT format; /* Format identifier--format = 1 */
OffsetTo<Coverage>
mark1Coverage; /* Offset to Combining Mark1 Coverage
2009-05-18 23:09:33 +02:00
* table--from beginning of MarkMarkPos
* subtable */
OffsetTo<Coverage>
mark2Coverage; /* Offset to Combining Mark2 Coverage
2009-05-18 23:09:33 +02:00
* table--from beginning of MarkMarkPos
* subtable */
USHORT classCount; /* Number of defined mark classes */
OffsetTo<MarkArray>
mark1Array; /* Offset to Mark1Array table--from
* beginning of MarkMarkPos subtable */
OffsetTo<Mark2Array>
mark2Array; /* Offset to Mark2Array table--from
* beginning of MarkMarkPos subtable */
public:
DEFINE_SIZE_STATIC (12);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct MarkMarkPos
{
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (u.format) {
2012-11-23 23:23:41 +01:00
case 1: return TRACE_RETURN (c->process (u.format1));
default:return TRACE_RETURN (c->default_return_value ());
}
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.format.sanitize (c)) return TRACE_RETURN (false);
2009-08-04 19:30:49 +02:00
switch (u.format) {
case 1: return TRACE_RETURN (u.format1.sanitize (c));
default:return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
USHORT format; /* Format identifier */
2010-05-11 01:45:41 +02:00
MarkMarkPosFormat1 format1;
2009-05-18 23:09:33 +02:00
} u;
};
2012-11-23 22:51:43 +01:00
struct ContextPos : Context {};
2009-05-18 23:09:33 +02:00
2012-11-23 22:51:43 +01:00
struct ChainContextPos : ChainContext {};
2009-05-18 23:09:33 +02:00
2012-11-23 22:57:36 +01:00
struct ExtensionPos : Extension<ExtensionPos>
2009-05-20 05:42:30 +02:00
{
2012-11-23 23:04:55 +01:00
typedef struct PosLookupSubTable LookupSubTable;
2009-05-18 23:09:33 +02:00
};
2009-05-18 23:09:33 +02:00
/*
* PosLookup
*/
2009-05-20 05:42:30 +02:00
struct PosLookupSubTable
{
2009-05-18 23:09:33 +02:00
friend struct PosLookup;
enum Type {
Single = 1,
Pair = 2,
Cursive = 3,
MarkBase = 4,
MarkLig = 5,
MarkMark = 6,
Context = 7,
ChainContext = 8,
2009-08-18 22:41:59 +02:00
Extension = 9
};
template <typename context_t>
inline typename context_t::return_t process (context_t *c, unsigned int lookup_type) const
2012-06-09 09:02:36 +02:00
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
switch (lookup_type) {
2012-11-23 23:23:41 +01:00
case Single: return TRACE_RETURN (u.single.process (c));
case Pair: return TRACE_RETURN (u.pair.process (c));
case Cursive: return TRACE_RETURN (u.cursive.process (c));
case MarkBase: return TRACE_RETURN (u.markBase.process (c));
case MarkLig: return TRACE_RETURN (u.markLig.process (c));
case MarkMark: return TRACE_RETURN (u.markMark.process (c));
case Context: return TRACE_RETURN (u.context.process (c));
case ChainContext: return TRACE_RETURN (u.chainContext.process (c));
case Extension: return TRACE_RETURN (u.extension.process (c));
default: return TRACE_RETURN (c->default_return_value ());
}
2012-06-09 09:02:36 +02:00
}
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c, unsigned int lookup_type) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (!u.header.sub_format.sanitize (c))
2012-06-09 09:02:36 +02:00
return TRACE_RETURN (false);
switch (lookup_type) {
case Single: return TRACE_RETURN (u.single.sanitize (c));
case Pair: return TRACE_RETURN (u.pair.sanitize (c));
case Cursive: return TRACE_RETURN (u.cursive.sanitize (c));
case MarkBase: return TRACE_RETURN (u.markBase.sanitize (c));
case MarkLig: return TRACE_RETURN (u.markLig.sanitize (c));
case MarkMark: return TRACE_RETURN (u.markMark.sanitize (c));
case Context: return TRACE_RETURN (u.context.sanitize (c));
case ChainContext: return TRACE_RETURN (u.chainContext.sanitize (c));
case Extension: return TRACE_RETURN (u.extension.sanitize (c));
default: return TRACE_RETURN (true);
2009-08-04 19:30:49 +02:00
}
}
protected:
2009-05-18 23:09:33 +02:00
union {
struct {
USHORT sub_format;
} header;
2010-05-11 01:45:41 +02:00
SinglePos single;
PairPos pair;
CursivePos cursive;
MarkBasePos markBase;
MarkLigPos markLig;
MarkMarkPos markMark;
ContextPos context;
2010-05-11 01:45:41 +02:00
ChainContextPos chainContext;
ExtensionPos extension;
2009-05-18 23:09:33 +02:00
} u;
2010-05-11 00:08:46 +02:00
public:
DEFINE_SIZE_UNION (2, header.sub_format);
2009-05-18 23:09:33 +02:00
};
2009-05-20 05:42:30 +02:00
struct PosLookup : Lookup
{
inline const PosLookupSubTable& get_subtable (unsigned int i) const
{ return this+CastR<OffsetArrayOf<PosLookupSubTable> > (subTable)[i]; }
2009-05-18 23:09:33 +02:00
template <typename context_t>
inline typename context_t::return_t process (context_t *c) const
{
2012-11-23 23:23:41 +01:00
TRACE_PROCESS (this);
unsigned int lookup_type = get_type ();
unsigned int count = get_subtable_count ();
for (unsigned int i = 0; i < count; i++) {
typename context_t::return_t r = get_subtable (i).process (c, lookup_type);
if (c->stop_sublookup_iteration (r))
2012-11-23 23:23:41 +01:00
return TRACE_RETURN (r);
}
2012-11-23 23:23:41 +01:00
return TRACE_RETURN (c->default_return_value ());
}
template <typename context_t>
static inline typename context_t::return_t process_recurse_func (context_t *c, unsigned int lookup_index);
2012-08-02 03:18:54 +02:00
template <typename set_t>
inline void add_coverage (set_t *glyphs) const
{
hb_get_coverage_context_t c;
2012-08-02 03:18:54 +02:00
const Coverage *last = NULL;
unsigned int count = get_subtable_count ();
for (unsigned int i = 0; i < count; i++) {
const Coverage *coverage = &get_subtable (i).process (&c, get_type ());
if (coverage != last) {
coverage->add_coverage (glyphs);
last = coverage;
2012-08-02 03:18:54 +02:00
}
}
}
inline bool apply_once (hb_apply_context_t *c) const
2009-05-20 05:42:30 +02:00
{
2012-11-24 00:13:48 +01:00
TRACE_APPLY (this);
2012-07-31 01:47:53 +02:00
if (!c->check_glyph_property (&c->buffer->cur(), c->lookup_props, &c->property))
2012-11-24 00:13:48 +01:00
return TRACE_RETURN (false);
return TRACE_RETURN (process (c));
2009-05-18 23:09:33 +02:00
}
static bool apply_recurse_func (hb_apply_context_t *c, unsigned int lookup_index);
2012-09-04 21:15:19 +02:00
inline bool apply_string (hb_apply_context_t *c, const hb_set_digest_t *digest) const
2009-05-20 05:42:30 +02:00
{
2009-05-18 23:09:33 +02:00
bool ret = false;
2012-09-05 04:42:17 +02:00
if (unlikely (!c->buffer->len || !c->lookup_mask))
2009-05-18 23:09:33 +02:00
return false;
c->set_recurse_func (apply_recurse_func);
2012-04-23 19:17:09 +02:00
c->set_lookup (*this);
2012-04-23 19:17:09 +02:00
c->buffer->idx = 0;
while (c->buffer->idx < c->buffer->len)
{
if ((c->buffer->cur().mask & c->lookup_mask) &&
2012-09-04 21:15:19 +02:00
digest->may_have (c->buffer->cur().codepoint) &&
apply_once (c))
ret = true;
else
c->buffer->idx++;
}
2009-05-18 23:09:33 +02:00
return ret;
}
2009-08-04 19:30:49 +02:00
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (unlikely (!Lookup::sanitize (c))) return TRACE_RETURN (false);
OffsetArrayOf<PosLookupSubTable> &list = CastR<OffsetArrayOf<PosLookupSubTable> > (subTable);
return TRACE_RETURN (list.sanitize (c, this, get_type ()));
2009-08-04 19:30:49 +02:00
}
2009-05-18 23:09:33 +02:00
};
2009-08-04 19:30:49 +02:00
typedef OffsetListOf<PosLookup> PosLookupList;
2009-05-18 23:09:33 +02:00
/*
2011-08-17 14:43:45 +02:00
* GPOS -- The Glyph Positioning Table
2009-05-18 23:09:33 +02:00
*/
2009-05-20 05:42:30 +02:00
struct GPOS : GSUBGPOS
{
2009-08-05 02:27:05 +02:00
static const hb_tag_t Tag = HB_OT_TAG_GPOS;
2009-05-18 23:09:33 +02:00
2009-05-20 05:42:30 +02:00
inline const PosLookup& get_lookup (unsigned int i) const
{ return CastR<PosLookup> (GSUBGPOS::get_lookup (i)); }
2009-05-18 23:09:33 +02:00
2012-08-02 03:18:54 +02:00
template <typename set_t>
inline void add_coverage (set_t *glyphs, unsigned int lookup_index) const
{ get_lookup (lookup_index).add_coverage (glyphs); }
static inline void position_start (hb_font_t *font, hb_buffer_t *buffer);
static inline void position_finish (hb_font_t *font, hb_buffer_t *buffer, hb_bool_t zero_width_attahced_marks);
2010-10-28 04:48:31 +02:00
2010-05-13 20:18:49 +02:00
inline bool sanitize (hb_sanitize_context_t *c) {
2012-11-23 21:32:14 +01:00
TRACE_SANITIZE (this);
if (unlikely (!GSUBGPOS::sanitize (c))) return TRACE_RETURN (false);
OffsetTo<PosLookupList> &list = CastR<OffsetTo<PosLookupList> > (lookupList);
return TRACE_RETURN (list.sanitize (c, this));
2009-08-04 19:30:49 +02:00
}
public:
DEFINE_SIZE_STATIC (10);
2009-05-18 23:09:33 +02:00
};
static void
fix_cursive_minor_offset (hb_glyph_position_t *pos, unsigned int i, hb_direction_t direction)
{
2012-08-23 15:33:30 +02:00
unsigned int j = pos[i].cursive_chain();
if (likely (!j))
return;
2012-08-23 15:33:30 +02:00
j += i;
2012-08-23 15:33:30 +02:00
pos[i].cursive_chain() = 0;
2012-08-23 15:33:30 +02:00
fix_cursive_minor_offset (pos, j, direction);
2012-08-23 15:33:30 +02:00
if (HB_DIRECTION_IS_HORIZONTAL (direction))
pos[i].y_offset += pos[j].y_offset;
else
pos[i].x_offset += pos[j].x_offset;
}
static void
fix_mark_attachment (hb_glyph_position_t *pos, unsigned int i, hb_direction_t direction, hb_bool_t zero_width_attached_marks)
{
if (likely (!(pos[i].attach_lookback())))
return;
unsigned int j = i - pos[i].attach_lookback();
if (zero_width_attached_marks) {
pos[i].x_advance = 0;
pos[i].y_advance = 0;
}
pos[i].x_offset += pos[j].x_offset;
pos[i].y_offset += pos[j].y_offset;
if (HB_DIRECTION_IS_FORWARD (direction))
for (unsigned int k = j; k < i; k++) {
pos[i].x_offset -= pos[k].x_advance;
pos[i].y_offset -= pos[k].y_advance;
}
else
for (unsigned int k = j + 1; k < i + 1; k++) {
pos[i].x_offset += pos[k].x_advance;
pos[i].y_offset += pos[k].y_advance;
}
}
void
GPOS::position_start (hb_font_t *font HB_UNUSED, hb_buffer_t *buffer)
{
buffer->clear_positions ();
2011-07-28 22:48:43 +02:00
unsigned int count = buffer->len;
for (unsigned int i = 0; i < count; i++)
buffer->pos[i].attach_lookback() = buffer->pos[i].cursive_chain() = 0;
}
2010-10-28 04:48:31 +02:00
void
GPOS::position_finish (hb_font_t *font HB_UNUSED, hb_buffer_t *buffer, hb_bool_t zero_width_attached_marks)
2010-10-28 04:48:31 +02:00
{
unsigned int len;
hb_glyph_position_t *pos = hb_buffer_get_glyph_positions (buffer, &len);
2010-10-28 04:48:31 +02:00
hb_direction_t direction = buffer->props.direction;
/* Handle cursive connections */
for (unsigned int i = 0; i < len; i++)
fix_cursive_minor_offset (pos, i, direction);
2010-10-28 04:48:31 +02:00
/* Handle attachments */
for (unsigned int i = 0; i < len; i++)
fix_mark_attachment (pos, i, direction, zero_width_attached_marks);
2011-07-28 22:48:43 +02:00
HB_BUFFER_DEALLOCATE_VAR (buffer, syllable);
HB_BUFFER_DEALLOCATE_VAR (buffer, lig_props);
2012-07-31 01:54:50 +02:00
HB_BUFFER_DEALLOCATE_VAR (buffer, glyph_props);
2010-10-28 04:48:31 +02:00
}
2009-05-18 23:09:33 +02:00
/* Out-of-class implementation for methods recursing */
template <typename context_t>
inline typename context_t::return_t PosLookup::process_recurse_func (context_t *c, unsigned int lookup_index)
{
const GPOS &gpos = *(hb_ot_layout_from_face (c->face)->gpos);
const PosLookup &l = gpos.get_lookup (lookup_index);
return l.process (c);
}
inline bool PosLookup::apply_recurse_func (hb_apply_context_t *c, unsigned int lookup_index)
2009-05-20 05:42:30 +02:00
{
const GPOS &gpos = *(hb_ot_layout_from_face (c->face)->gpos);
2009-05-18 23:09:33 +02:00
const PosLookup &l = gpos.get_lookup (lookup_index);
c->set_lookup (l);
return l.apply_once (c);
2009-05-18 23:09:33 +02:00
}
2010-10-28 05:09:10 +02:00
#undef attach_lookback
#undef cursive_chain
2012-11-17 03:49:54 +01:00
} /* namespace OT */
2010-07-23 21:11:18 +02:00
#endif /* HB_OT_LAYOUT_GPOS_TABLE_HH */