implemented get_glyph_extents with CFF1
Implemented path operators in a new struct PathProc hooked up to CSOpSet Added current point to CSInterpEnv
This commit is contained in:
parent
f2c4720ae6
commit
c0c85b852d
|
@ -237,6 +237,35 @@ struct Number
|
||||||
else
|
else
|
||||||
return (float)u.int_val;
|
return (float)u.int_val;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline int ceil (void) const
|
||||||
|
{
|
||||||
|
switch (format)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case NumInt:
|
||||||
|
return u.int_val;
|
||||||
|
case NumFixed:
|
||||||
|
return (u.fixed_val + 0xFFFF) >> 16;
|
||||||
|
case NumReal:
|
||||||
|
return (int)ceilf (u.real_val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int floor (void) const
|
||||||
|
{
|
||||||
|
switch (format)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case NumInt:
|
||||||
|
return u.int_val;
|
||||||
|
case NumFixed:
|
||||||
|
return u.fixed_val >> 16;
|
||||||
|
case NumReal:
|
||||||
|
return (int)floorf (u.real_val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inline bool in_int_range (void) const
|
inline bool in_int_range (void) const
|
||||||
{
|
{
|
||||||
if (is_int ())
|
if (is_int ())
|
||||||
|
@ -247,6 +276,44 @@ struct Number
|
||||||
return ((float)(int16_t)to_int () == u.real_val);
|
return ((float)(int16_t)to_int () == u.real_val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline bool operator > (const Number &n) const
|
||||||
|
{
|
||||||
|
switch (format)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case NumInt: return u.int_val > n.to_int ();
|
||||||
|
case NumFixed: return u.fixed_val > n.to_fixed ();
|
||||||
|
case NumReal: return u.real_val > n.to_real ();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
inline bool operator < (const Number &n) const
|
||||||
|
{ return n > *this; }
|
||||||
|
|
||||||
|
inline bool operator >= (const Number &n) const
|
||||||
|
{ return ! (*this < n); }
|
||||||
|
|
||||||
|
inline bool operator <= (const Number &n) const
|
||||||
|
{ return ! (*this > n); }
|
||||||
|
|
||||||
|
inline const Number &operator += (const Number &n)
|
||||||
|
{
|
||||||
|
switch (format)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
case NumInt:
|
||||||
|
u.int_val += n.to_int ();
|
||||||
|
break;
|
||||||
|
case NumFixed:
|
||||||
|
u.fixed_val += n.to_fixed ();
|
||||||
|
break;
|
||||||
|
case NumReal:
|
||||||
|
u.real_val += n.to_real ();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
enum NumFormat {
|
enum NumFormat {
|
||||||
NumInt,
|
NumInt,
|
||||||
|
@ -390,6 +457,12 @@ struct Stack
|
||||||
elements[i].fini ();
|
elements[i].fini ();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline const ELEM& operator [] (unsigned int i) const
|
||||||
|
{ return elements[i]; }
|
||||||
|
|
||||||
|
inline ELEM& operator [] (unsigned int i)
|
||||||
|
{ return elements[i]; }
|
||||||
|
|
||||||
inline void push (const ELEM &v)
|
inline void push (const ELEM &v)
|
||||||
{
|
{
|
||||||
if (likely (count < elements.len))
|
if (likely (count < elements.len))
|
||||||
|
@ -412,6 +485,12 @@ struct Stack
|
||||||
return Null(ELEM);
|
return Null(ELEM);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline void pop (unsigned int n)
|
||||||
|
{
|
||||||
|
if (likely (count >= n))
|
||||||
|
count -= n;
|
||||||
|
}
|
||||||
|
|
||||||
inline const ELEM& peek (void)
|
inline const ELEM& peek (void)
|
||||||
{
|
{
|
||||||
if (likely (count > 0))
|
if (likely (count > 0))
|
||||||
|
@ -436,6 +515,7 @@ struct Stack
|
||||||
|
|
||||||
static const unsigned int kSizeLimit = LIMIT;
|
static const unsigned int kSizeLimit = LIMIT;
|
||||||
|
|
||||||
|
protected:
|
||||||
unsigned int count;
|
unsigned int count;
|
||||||
hb_vector_t<ELEM, kSizeLimit> elements;
|
hb_vector_t<ELEM, kSizeLimit> elements;
|
||||||
};
|
};
|
||||||
|
@ -597,13 +677,13 @@ struct InterpEnv
|
||||||
|
|
||||||
inline void pop_n_args (unsigned int n)
|
inline void pop_n_args (unsigned int n)
|
||||||
{
|
{
|
||||||
assert (n <= argStack.count);
|
assert (n <= argStack.get_count ());
|
||||||
argStack.count -= n;
|
argStack.pop (n);
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void clear_args (void)
|
inline void clear_args (void)
|
||||||
{
|
{
|
||||||
pop_n_args (argStack.count);
|
pop_n_args (argStack.get_count ());
|
||||||
}
|
}
|
||||||
|
|
||||||
SubByteStr substr;
|
SubByteStr substr;
|
||||||
|
|
|
@ -57,6 +57,22 @@ struct BiasedSubrs
|
||||||
unsigned int bias;
|
unsigned int bias;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct Point
|
||||||
|
{
|
||||||
|
inline void init (void)
|
||||||
|
{
|
||||||
|
x.init ();
|
||||||
|
y.init ();
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void move_x (const Number &dx) { x += dx; }
|
||||||
|
inline void move_y (const Number &dy) { y += dy; }
|
||||||
|
inline void move (const Number &dx, const Number &dy) { move_x (dx); move_y (dy); }
|
||||||
|
|
||||||
|
Number x;
|
||||||
|
Number y;
|
||||||
|
};
|
||||||
|
|
||||||
template <typename ARG, typename SUBRS>
|
template <typename ARG, typename SUBRS>
|
||||||
struct CSInterpEnv : InterpEnv<ARG>
|
struct CSInterpEnv : InterpEnv<ARG>
|
||||||
{
|
{
|
||||||
|
@ -68,6 +84,7 @@ struct CSInterpEnv : InterpEnv<ARG>
|
||||||
seen_hintmask = false;
|
seen_hintmask = false;
|
||||||
hstem_count = 0;
|
hstem_count = 0;
|
||||||
vstem_count = 0;
|
vstem_count = 0;
|
||||||
|
pt.init ();
|
||||||
callStack.init ();
|
callStack.init ();
|
||||||
globalSubrs.init (globalSubrs_);
|
globalSubrs.init (globalSubrs_);
|
||||||
localSubrs.init (localSubrs_);
|
localSubrs.init (localSubrs_);
|
||||||
|
@ -129,6 +146,30 @@ struct CSInterpEnv : InterpEnv<ARG>
|
||||||
inline void set_endchar (bool endchar_flag_) { endchar_flag = endchar_flag_; }
|
inline void set_endchar (bool endchar_flag_) { endchar_flag = endchar_flag_; }
|
||||||
inline bool is_endchar (void) const { return endchar_flag; }
|
inline bool is_endchar (void) const { return endchar_flag; }
|
||||||
|
|
||||||
|
inline const Number &get_x (void) const { return pt.x; }
|
||||||
|
inline const Number &get_y (void) const { return pt.y; }
|
||||||
|
inline const Point &get_pt (void) const { return pt; }
|
||||||
|
|
||||||
|
inline void moveto (const Point &pt_ ) { pt = pt_; }
|
||||||
|
|
||||||
|
inline unsigned int move_x_with_arg (unsigned int i)
|
||||||
|
{
|
||||||
|
pt.move_x (SUPER::argStack[i]);
|
||||||
|
return i + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline unsigned int move_y_with_arg (unsigned int i)
|
||||||
|
{
|
||||||
|
pt.move_y (SUPER::argStack[i]);
|
||||||
|
return i + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline unsigned int move_xy_with_arg (unsigned int i)
|
||||||
|
{
|
||||||
|
pt.move (SUPER::argStack[i], SUPER::argStack[i+1]);
|
||||||
|
return i + 2;
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
bool endchar_flag;
|
bool endchar_flag;
|
||||||
bool seen_moveto;
|
bool seen_moveto;
|
||||||
|
@ -142,10 +183,33 @@ struct CSInterpEnv : InterpEnv<ARG>
|
||||||
BiasedSubrs<SUBRS> localSubrs;
|
BiasedSubrs<SUBRS> localSubrs;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
Point pt;
|
||||||
|
|
||||||
typedef InterpEnv<ARG> SUPER;
|
typedef InterpEnv<ARG> SUPER;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename ARG, typename OPSET, typename ENV, typename PARAM>
|
template <typename ENV, typename PARAM>
|
||||||
|
struct PathProcsNull
|
||||||
|
{
|
||||||
|
static inline void rmoveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void hmoveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void vmoveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void rlineto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void hlineto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void vlineto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void rrcurveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void rcurveline (ENV &env, PARAM& param) {}
|
||||||
|
static inline void rlinecurve (ENV &env, PARAM& param) {}
|
||||||
|
static inline void vvcurveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void hhcurveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void vhcurveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void hvcurveto (ENV &env, PARAM& param) {}
|
||||||
|
static inline void moveto (ENV &env, PARAM& param, const Point &pt) {}
|
||||||
|
static inline void line (ENV &env, PARAM& param, const Point &pt1) {}
|
||||||
|
static inline void curve (ENV &env, PARAM& param, const Point &pt1, const Point &pt2, const Point &pt3) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename ARG, typename OPSET, typename ENV, typename PARAM, typename PATH=PathProcsNull<ENV, PARAM> >
|
||||||
struct CSOpSet : OpSet<ARG>
|
struct CSOpSet : OpSet<ARG>
|
||||||
{
|
{
|
||||||
static inline bool process_op (OpCode op, ENV &env, PARAM& param)
|
static inline bool process_op (OpCode op, ENV &env, PARAM& param)
|
||||||
|
@ -180,23 +244,57 @@ struct CSOpSet : OpSet<ARG>
|
||||||
case OpCode_cntrmask:
|
case OpCode_cntrmask:
|
||||||
OPSET::process_hintmask (op, env, param);
|
OPSET::process_hintmask (op, env, param);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case OpCode_vmoveto:
|
|
||||||
case OpCode_rlineto:
|
|
||||||
case OpCode_hlineto:
|
|
||||||
case OpCode_vlineto:
|
|
||||||
case OpCode_rmoveto:
|
case OpCode_rmoveto:
|
||||||
|
PATH::rmoveto (env, param);
|
||||||
|
process_post_move (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_hmoveto:
|
case OpCode_hmoveto:
|
||||||
OPSET::process_moveto (op, env, param);
|
PATH::hmoveto (env, param);
|
||||||
|
process_post_move (op, env, param);
|
||||||
|
break;
|
||||||
|
case OpCode_vmoveto:
|
||||||
|
PATH::vmoveto (env, param);
|
||||||
|
process_post_move (op, env, param);
|
||||||
|
break;
|
||||||
|
case OpCode_rlineto:
|
||||||
|
PATH::rlineto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
|
case OpCode_hlineto:
|
||||||
|
PATH::hlineto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
|
case OpCode_vlineto:
|
||||||
|
PATH::vlineto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
break;
|
break;
|
||||||
case OpCode_rrcurveto:
|
case OpCode_rrcurveto:
|
||||||
|
PATH::rrcurveto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_rcurveline:
|
case OpCode_rcurveline:
|
||||||
|
PATH::rcurveline (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_rlinecurve:
|
case OpCode_rlinecurve:
|
||||||
|
PATH::rlinecurve (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_vvcurveto:
|
case OpCode_vvcurveto:
|
||||||
|
PATH::vvcurveto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_hhcurveto:
|
case OpCode_hhcurveto:
|
||||||
|
PATH::hhcurveto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_vhcurveto:
|
case OpCode_vhcurveto:
|
||||||
|
PATH::vhcurveto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
|
break;
|
||||||
case OpCode_hvcurveto:
|
case OpCode_hvcurveto:
|
||||||
OPSET::process_path (op, env, param);
|
PATH::hvcurveto (env, param);
|
||||||
|
process_post_path (op, env, param);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case OpCode_hflex:
|
case OpCode_hflex:
|
||||||
|
@ -214,13 +312,13 @@ struct CSOpSet : OpSet<ARG>
|
||||||
|
|
||||||
static inline void process_hstem (OpCode op, ENV &env, PARAM& param)
|
static inline void process_hstem (OpCode op, ENV &env, PARAM& param)
|
||||||
{
|
{
|
||||||
env.hstem_count += env.argStack.count / 2;
|
env.hstem_count += env.argStack.get_count () / 2;
|
||||||
OPSET::flush_args_and_op (op, env, param);
|
OPSET::flush_args_and_op (op, env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void process_vstem (OpCode op, ENV &env, PARAM& param)
|
static inline void process_vstem (OpCode op, ENV &env, PARAM& param)
|
||||||
{
|
{
|
||||||
env.vstem_count += env.argStack.count / 2;
|
env.vstem_count += env.argStack.get_count () / 2;
|
||||||
OPSET::flush_args_and_op (op, env, param);
|
OPSET::flush_args_and_op (op, env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -239,7 +337,7 @@ struct CSOpSet : OpSet<ARG>
|
||||||
OPSET::flush_args_and_op (op, env, param);
|
OPSET::flush_args_and_op (op, env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void process_moveto (OpCode op, ENV &env, PARAM& param)
|
static inline void process_post_move (OpCode op, ENV &env, PARAM& param)
|
||||||
{
|
{
|
||||||
if (!env.seen_moveto)
|
if (!env.seen_moveto)
|
||||||
{
|
{
|
||||||
|
@ -249,14 +347,14 @@ struct CSOpSet : OpSet<ARG>
|
||||||
OPSET::flush_args_and_op (op, env, param);
|
OPSET::flush_args_and_op (op, env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void process_path (OpCode op, ENV &env, PARAM& param)
|
static inline void process_post_path (OpCode op, ENV &env, PARAM& param)
|
||||||
{
|
{
|
||||||
OPSET::flush_args_and_op (op, env, param);
|
OPSET::flush_args_and_op (op, env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void flush_args_and_op (OpCode op, ENV &env, PARAM& param)
|
static inline void flush_args_and_op (OpCode op, ENV &env, PARAM& param)
|
||||||
{
|
{
|
||||||
OPSET::flush_n_args_and_op (op, env.argStack.count, env, param);
|
OPSET::flush_n_args_and_op (op, env.argStack.get_count (), env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void flush_n_args_and_op (OpCode op, unsigned int n, ENV &env, PARAM& param)
|
static inline void flush_n_args_and_op (OpCode op, unsigned int n, ENV &env, PARAM& param)
|
||||||
|
@ -267,7 +365,7 @@ struct CSOpSet : OpSet<ARG>
|
||||||
|
|
||||||
static inline void flush_args (ENV &env, PARAM& param)
|
static inline void flush_args (ENV &env, PARAM& param)
|
||||||
{
|
{
|
||||||
OPSET::flush_n_args (env.argStack.count, env, param);
|
OPSET::flush_n_args (env.argStack.get_count (), env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void flush_n_args (unsigned int n, ENV &env, PARAM& param)
|
static inline void flush_n_args (unsigned int n, ENV &env, PARAM& param)
|
||||||
|
@ -297,9 +395,312 @@ struct CSOpSet : OpSet<ARG>
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
typedef OpSet<ARG> SUPER;
|
typedef OpSet<ARG> SUPER;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template <typename PATH, typename ENV, typename PARAM>
|
||||||
|
struct PathProcs
|
||||||
|
{
|
||||||
|
static inline void rmoveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
const Number &dy = env.argStack.pop ();
|
||||||
|
const Number &dx = env.argStack.pop ();
|
||||||
|
pt1.move (dx, dy);
|
||||||
|
env.moveto (pt1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void hmoveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move_x (env.argStack.pop ());
|
||||||
|
env.moveto (pt1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void vmoveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move_y (env.argStack.pop ());
|
||||||
|
env.moveto (pt1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void rlineto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
for (unsigned int i = 0; i + 2 <= env.argStack.get_count (); i += 2)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move (env.argStack[i], env.argStack[i+1]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void hlineto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1;
|
||||||
|
unsigned int i = 0;
|
||||||
|
for (; i + 2 <= env.argStack.get_count (); i += 2)
|
||||||
|
{
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_x (env.argStack[i]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
pt1.move_y (env.argStack[i+1]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
if (i < env.argStack.get_count ())
|
||||||
|
{
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_x (env.argStack[i]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void vlineto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1;
|
||||||
|
unsigned int i = 0;
|
||||||
|
for (; i + 2 <= env.argStack.get_count (); i += 2)
|
||||||
|
{
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_y (env.argStack[i]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
pt1.move_x (env.argStack[i+1]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
if (i < env.argStack.get_count ())
|
||||||
|
{
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_y (env.argStack[i]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void rrcurveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
for (unsigned int i = 0; i + 6 <= env.argStack.get_count (); i += 6)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move (env.argStack[i], env.argStack[i+1]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+2], env.argStack[i+3]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move (env.argStack[i+4], env.argStack[i+5]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void rcurveline (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
unsigned int i = 0;
|
||||||
|
for (; i + 6 <= env.argStack.get_count (); i += 6)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move (env.argStack[i], env.argStack[i+1]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+2], env.argStack[i+3]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move (env.argStack[i+4], env.argStack[i+5]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
for (; i + 2 <= env.argStack.get_count (); i += 2)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move (env.argStack[i], env.argStack[i+1]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void rlinecurve (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
unsigned int i = 0;
|
||||||
|
unsigned int line_limit = (env.argStack.get_count () % 6);
|
||||||
|
for (; i + 2 <= line_limit; i += 2)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move (env.argStack[i], env.argStack[i+1]);
|
||||||
|
PATH::line (env, param, pt1);
|
||||||
|
}
|
||||||
|
for (; i + 6 <= env.argStack.get_count (); i += 6)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move (env.argStack[i], env.argStack[i+1]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+2], env.argStack[i+3]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move (env.argStack[i+4], env.argStack[i+5]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void vvcurveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
unsigned int i = 0;
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
if ((env.argStack.get_count () & 1) != 0)
|
||||||
|
pt1.move_x (env.argStack[i++]);
|
||||||
|
for (; i + 4 <= env.argStack.get_count (); i += 4)
|
||||||
|
{
|
||||||
|
pt1.move_y (env.argStack[i]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move_y (env.argStack[i+3]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void hhcurveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
unsigned int i = 0;
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
if ((env.argStack.get_count () & 1) != 0)
|
||||||
|
pt1.move_y (env.argStack[i++]);
|
||||||
|
for (; i + 4 <= env.argStack.get_count (); i += 4)
|
||||||
|
{
|
||||||
|
pt1.move_x (env.argStack[i]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move_x (env.argStack[i+3]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void vhcurveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1, pt2, pt3;
|
||||||
|
unsigned int i = 0;
|
||||||
|
if ((env.argStack.get_count () % 8) >= 4)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move_y (env.argStack[i]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move_x (env.argStack[i+3]);
|
||||||
|
i += 4;
|
||||||
|
|
||||||
|
for (; i + 8 <= env.argStack.get_count (); i += 8)
|
||||||
|
{
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_x (env.argStack[i]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_y (env.argStack[i+3]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
|
||||||
|
pt1 = pt3;
|
||||||
|
pt1.move_y (env.argStack[i+4]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+5], env.argStack[i+6]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_x (env.argStack[i+7]);
|
||||||
|
}
|
||||||
|
if (i < env.argStack.get_count ())
|
||||||
|
pt3.move_y (env.argStack[i]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (; i + 8 <= env.argStack.get_count (); i += 8)
|
||||||
|
{
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_y (env.argStack[i]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_x (env.argStack[i+3]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
|
||||||
|
pt1 = pt3;
|
||||||
|
pt1.move_x (env.argStack[i+4]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+5], env.argStack[i+6]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_y (env.argStack[i+7]);
|
||||||
|
if ((env.argStack.get_count () - i < 16) && ((env.argStack.get_count () & 1) != 0))
|
||||||
|
pt3.move_x (env.argStack[i+8]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void hvcurveto (ENV &env, PARAM& param)
|
||||||
|
{
|
||||||
|
Point pt1, pt2, pt3;
|
||||||
|
unsigned int i = 0;
|
||||||
|
if ((env.argStack.get_count () % 8) >= 4)
|
||||||
|
{
|
||||||
|
Point pt1 = env.get_pt ();
|
||||||
|
pt1.move_x (env.argStack[i]);
|
||||||
|
Point pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
Point pt3 = pt2;
|
||||||
|
pt3.move_y (env.argStack[i+3]);
|
||||||
|
i += 4;
|
||||||
|
|
||||||
|
for (; i + 8 <= env.argStack.get_count (); i += 8)
|
||||||
|
{
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_y (env.argStack[i]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_x (env.argStack[i+3]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
|
||||||
|
pt1 = pt3;
|
||||||
|
pt1.move_x (env.argStack[i+4]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+5], env.argStack[i+6]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_y (env.argStack[i+7]);
|
||||||
|
}
|
||||||
|
if (i < env.argStack.get_count ())
|
||||||
|
pt3.move_x (env.argStack[i]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (; i + 8 <= env.argStack.get_count (); i += 8)
|
||||||
|
{
|
||||||
|
pt1 = env.get_pt ();
|
||||||
|
pt1.move_x (env.argStack[i]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+1], env.argStack[i+2]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_y (env.argStack[i+3]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
|
||||||
|
pt1 = pt3;
|
||||||
|
pt1.move_y (env.argStack[i+4]);
|
||||||
|
pt2 = pt1;
|
||||||
|
pt2.move (env.argStack[i+5], env.argStack[i+6]);
|
||||||
|
pt3 = pt2;
|
||||||
|
pt3.move_x (env.argStack[i+7]);
|
||||||
|
if ((env.argStack.get_count () - i < 16) && ((env.argStack.get_count () & 1) != 0))
|
||||||
|
pt3.move_y (env.argStack[i+8]);
|
||||||
|
PATH::curve (env, param, pt1, pt2, pt3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* default actions to be overridden */
|
||||||
|
static inline void moveto (ENV &env, PARAM& param, const Point &pt)
|
||||||
|
{ env.moveto (pt); }
|
||||||
|
|
||||||
|
static inline void line (ENV &env, PARAM& param, const Point &pt1)
|
||||||
|
{ PATH::moveto (env, param, pt1); }
|
||||||
|
|
||||||
|
static inline void curve (ENV &env, PARAM& param, const Point &pt1, const Point &pt2, const Point &pt3)
|
||||||
|
{ PATH::moveto (env, param, pt3); }
|
||||||
|
};
|
||||||
|
|
||||||
template <typename ENV, typename OPSET, typename PARAM>
|
template <typename ENV, typename OPSET, typename PARAM>
|
||||||
struct CSInterpreter : Interpreter<ENV>
|
struct CSInterpreter : Interpreter<ENV>
|
||||||
{
|
{
|
||||||
|
|
|
@ -52,9 +52,9 @@ struct CFF1CSInterpEnv : CSInterpEnv<Number, CFF1Subrs>
|
||||||
{
|
{
|
||||||
if (!processed_width)
|
if (!processed_width)
|
||||||
{
|
{
|
||||||
if ((SUPER::argStack.count & 1) != 0)
|
if ((SUPER::argStack.get_count () & 1) != 0)
|
||||||
{
|
{
|
||||||
width = SUPER::argStack.elements[0];
|
width = SUPER::argStack[0];
|
||||||
has_width = true;
|
has_width = true;
|
||||||
}
|
}
|
||||||
processed_width = true;
|
processed_width = true;
|
||||||
|
@ -72,8 +72,8 @@ struct CFF1CSInterpEnv : CSInterpEnv<Number, CFF1Subrs>
|
||||||
typedef CSInterpEnv<Number, CFF1Subrs> SUPER;
|
typedef CSInterpEnv<Number, CFF1Subrs> SUPER;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename OPSET, typename PARAM>
|
template <typename OPSET, typename PARAM, typename PATH=PathProcsNull<CFF1CSInterpEnv, PARAM> >
|
||||||
struct CFF1CSOpSet : CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM>
|
struct CFF1CSOpSet : CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM, PATH>
|
||||||
{
|
{
|
||||||
static inline bool process_op (OpCode op, CFF1CSInterpEnv &env, PARAM& param)
|
static inline bool process_op (OpCode op, CFF1CSInterpEnv &env, PARAM& param)
|
||||||
{
|
{
|
||||||
|
@ -167,8 +167,8 @@ struct CFF1CSOpSet : CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM>
|
||||||
if (unlikely (!env.argStack.check_pop_num (n1))) return false;
|
if (unlikely (!env.argStack.check_pop_num (n1))) return false;
|
||||||
int i = n1.to_int ();
|
int i = n1.to_int ();
|
||||||
if (i < 0) i = 0;
|
if (i < 0) i = 0;
|
||||||
if (unlikely (i >= env.argStack.count || !env.argStack.check_overflow (1))) return false;
|
if (unlikely (i >= env.argStack.get_count () || !env.argStack.check_overflow (1))) return false;
|
||||||
env.argStack.push (env.argStack.elements[env.argStack.count - i - 1]);
|
env.argStack.push (env.argStack[env.argStack.get_count () - i - 1]);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case OpCode_roll:
|
case OpCode_roll:
|
||||||
|
@ -176,13 +176,13 @@ struct CFF1CSOpSet : CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM>
|
||||||
if (unlikely (!env.argStack.check_pop_num2 (n1, n2))) return false;
|
if (unlikely (!env.argStack.check_pop_num2 (n1, n2))) return false;
|
||||||
int n = n1.to_int ();
|
int n = n1.to_int ();
|
||||||
int j = n2.to_int ();
|
int j = n2.to_int ();
|
||||||
if (unlikely (n < 0 || n > env.argStack.count)) return false;
|
if (unlikely (n < 0 || n > env.argStack.get_count ())) return false;
|
||||||
if (likely (n > 0))
|
if (likely (n > 0))
|
||||||
{
|
{
|
||||||
if (j < 0)
|
if (j < 0)
|
||||||
j = n - (-j % n);
|
j = n - (-j % n);
|
||||||
j %= n;
|
j %= n;
|
||||||
unsigned int top = env.argStack.count - 1;
|
unsigned int top = env.argStack.get_count () - 1;
|
||||||
unsigned int bot = top - n + 1;
|
unsigned int bot = top - n + 1;
|
||||||
env.argStack.reverse_range (top - j + 1, top);
|
env.argStack.reverse_range (top - j + 1, top);
|
||||||
env.argStack.reverse_range (bot, top - j);
|
env.argStack.reverse_range (bot, top - j);
|
||||||
|
@ -205,7 +205,7 @@ struct CFF1CSOpSet : CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM>
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
typedef CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM> SUPER;
|
typedef CSOpSet<Number, OPSET, CFF1CSInterpEnv, PARAM, PATH> SUPER;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename OPSET, typename PARAM>
|
template <typename OPSET, typename PARAM>
|
||||||
|
|
|
@ -120,8 +120,8 @@ struct CFF2CSInterpEnv : CSInterpEnv<BlendArg, CFF2Subrs>
|
||||||
typedef CSInterpEnv<BlendArg, CFF2Subrs> SUPER;
|
typedef CSInterpEnv<BlendArg, CFF2Subrs> SUPER;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename OPSET, typename PARAM>
|
template <typename OPSET, typename PARAM, typename PATH=PathProcsNull<CFF2CSInterpEnv, PARAM> >
|
||||||
struct CFF2CSOpSet : CSOpSet<BlendArg, OPSET, CFF2CSInterpEnv, PARAM>
|
struct CFF2CSOpSet : CSOpSet<BlendArg, OPSET, CFF2CSInterpEnv, PARAM, PATH>
|
||||||
{
|
{
|
||||||
static inline bool process_op (OpCode op, CFF2CSInterpEnv &env, PARAM& param)
|
static inline bool process_op (OpCode op, CFF2CSInterpEnv &env, PARAM& param)
|
||||||
{
|
{
|
||||||
|
@ -158,10 +158,10 @@ struct CFF2CSOpSet : CSOpSet<BlendArg, OPSET, CFF2CSInterpEnv, PARAM>
|
||||||
/* copy the blend values into blend array of the default values */
|
/* copy the blend values into blend array of the default values */
|
||||||
unsigned int start = env.argStack.get_count () - ((k+1) * n);
|
unsigned int start = env.argStack.get_count () - ((k+1) * n);
|
||||||
for (unsigned int i = 0; i < n; i++)
|
for (unsigned int i = 0; i < n; i++)
|
||||||
env.argStack.elements[start + i].set_blends (n, i, k, &env.argStack.elements[start + n + (i * k)]);
|
env.argStack[start + i].set_blends (n, i, k, &env.argStack[start + n + (i * k)]);
|
||||||
|
|
||||||
/* pop off blend values leaving default values now adorned with blend values */
|
/* pop off blend values leaving default values now adorned with blend values */
|
||||||
env.argStack.count -= k * n;
|
env.argStack.pop (k * n);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -172,7 +172,7 @@ struct CFF2CSOpSet : CSOpSet<BlendArg, OPSET, CFF2CSInterpEnv, PARAM>
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
typedef CSOpSet<BlendArg, OPSET, CFF2CSInterpEnv, PARAM> SUPER;
|
typedef CSOpSet<BlendArg, OPSET, CFF2CSInterpEnv, PARAM, PATH> SUPER;
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename OPSET, typename PARAM>
|
template <typename OPSET, typename PARAM>
|
||||||
|
|
|
@ -36,7 +36,7 @@
|
||||||
|
|
||||||
/* Global nul-content Null pool. Enlarge as necessary. */
|
/* Global nul-content Null pool. Enlarge as necessary. */
|
||||||
|
|
||||||
#define HB_NULL_POOL_SIZE 800
|
#define HB_NULL_POOL_SIZE 9880
|
||||||
|
|
||||||
extern HB_INTERNAL
|
extern HB_INTERNAL
|
||||||
hb_vector_size_impl_t const _hb_NullPool[(HB_NULL_POOL_SIZE + sizeof (hb_vector_size_impl_t) - 1) / sizeof (hb_vector_size_impl_t)];
|
hb_vector_size_impl_t const _hb_NullPool[(HB_NULL_POOL_SIZE + sizeof (hb_vector_size_impl_t) - 1) / sizeof (hb_vector_size_impl_t)];
|
||||||
|
|
|
@ -25,6 +25,9 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "hb-ot-cff1-table.hh"
|
#include "hb-ot-cff1-table.hh"
|
||||||
|
#include "hb-cff1-interp-cs.hh"
|
||||||
|
|
||||||
|
using namespace CFF;
|
||||||
|
|
||||||
/* SID to code */
|
/* SID to code */
|
||||||
static const uint8_t standard_encoding [] =
|
static const uint8_t standard_encoding [] =
|
||||||
|
@ -128,3 +131,82 @@ hb_codepoint_t OT::cff1::lookup_expert_subset_charset (hb_codepoint_t glyph)
|
||||||
else
|
else
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct ExtentsParam
|
||||||
|
{
|
||||||
|
inline void init (void)
|
||||||
|
{
|
||||||
|
min_x.set_int (INT32_MAX);
|
||||||
|
min_y.set_int (INT32_MAX);
|
||||||
|
max_x.set_int (INT32_MIN);
|
||||||
|
max_y.set_int (INT32_MIN);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void update_bounds (const Point &pt)
|
||||||
|
{
|
||||||
|
if (pt.x < min_x) min_x = pt.x;
|
||||||
|
if (pt.x > max_x) max_x = pt.x;
|
||||||
|
if (pt.y < min_y) min_y = pt.y;
|
||||||
|
if (pt.y > max_y) max_y = pt.y;
|
||||||
|
}
|
||||||
|
|
||||||
|
Number min_x;
|
||||||
|
Number min_y;
|
||||||
|
Number max_x;
|
||||||
|
Number max_y;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct CFF1PathProcs_Extents : PathProcs<CFF1PathProcs_Extents, CFF1CSInterpEnv, ExtentsParam>
|
||||||
|
{
|
||||||
|
static inline void line (CFF1CSInterpEnv &env, ExtentsParam& param, const Point &pt1)
|
||||||
|
{
|
||||||
|
param.update_bounds (env.get_pt ());
|
||||||
|
env.moveto (pt1);
|
||||||
|
param.update_bounds (env.get_pt ());
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void curve (CFF1CSInterpEnv &env, ExtentsParam& param, const Point &pt1, const Point &pt2, const Point &pt3)
|
||||||
|
{
|
||||||
|
param.update_bounds (env.get_pt ());
|
||||||
|
env.moveto (pt3);
|
||||||
|
param.update_bounds (env.get_pt ());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct CFF1CSOpSet_Extents : CFF1CSOpSet<CFF1CSOpSet_Extents, ExtentsParam, CFF1PathProcs_Extents> {};
|
||||||
|
|
||||||
|
bool OT::cff1::accelerator_t::get_extents (hb_codepoint_t glyph, hb_glyph_extents_t *extents) const
|
||||||
|
{
|
||||||
|
if (unlikely (!is_valid () || (glyph >= num_glyphs))) return false;
|
||||||
|
|
||||||
|
unsigned int fd = fdSelect->get_fd (glyph);
|
||||||
|
CFF1CSInterpreter<CFF1CSOpSet_Extents, ExtentsParam> interp;
|
||||||
|
const ByteStr str = (*charStrings)[glyph];
|
||||||
|
interp.env.init (str, *this, fd);
|
||||||
|
ExtentsParam param;
|
||||||
|
param.init ();
|
||||||
|
if (unlikely (!interp.interpret (param))) return false;
|
||||||
|
|
||||||
|
if (param.min_x >= param.max_x)
|
||||||
|
{
|
||||||
|
extents->width = 0;
|
||||||
|
extents->x_bearing = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
extents->x_bearing = (int32_t)param.min_x.floor ();
|
||||||
|
extents->width = (int32_t)param.max_x.ceil () - extents->x_bearing;
|
||||||
|
}
|
||||||
|
if (param.min_y >= param.max_y)
|
||||||
|
{
|
||||||
|
extents->height = 0;
|
||||||
|
extents->y_bearing = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
extents->y_bearing = (int32_t)param.max_y.ceil ();
|
||||||
|
extents->height = (int32_t)param.min_y.floor () - extents->y_bearing;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
|
@ -1007,11 +1007,6 @@ struct cff1
|
||||||
if (unlikely (!top_interp.interpret (topDict))) { fini (); return; }
|
if (unlikely (!top_interp.interpret (topDict))) { fini (); return; }
|
||||||
}
|
}
|
||||||
|
|
||||||
encoding = &Null(Encoding);
|
|
||||||
charset = &StructAtOffsetOrNull<Charset> (cff, topDict.CharsetOffset);
|
|
||||||
if (unlikely (is_CID () && (charset == &Null(Charset))))
|
|
||||||
{ fini (); return; }
|
|
||||||
|
|
||||||
fdCount = 1;
|
fdCount = 1;
|
||||||
if (is_CID ())
|
if (is_CID ())
|
||||||
{
|
{
|
||||||
|
@ -1027,12 +1022,6 @@ struct cff1
|
||||||
{
|
{
|
||||||
fdArray = &Null(CFF1FDArray);
|
fdArray = &Null(CFF1FDArray);
|
||||||
fdSelect = &Null(CFF1FDSelect);
|
fdSelect = &Null(CFF1FDSelect);
|
||||||
if (!is_predef_encoding ())
|
|
||||||
{
|
|
||||||
encoding = &StructAtOffsetOrNull<Encoding> (cff, topDict.EncodingOffset);
|
|
||||||
if ((encoding == &Null (Encoding)) || !encoding->sanitize (&sc))
|
|
||||||
{ fini (); return; }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
stringIndex = &StructAtOffset<CFF1StringIndex> (topDictIndex, topDictIndex->get_size ());
|
stringIndex = &StructAtOffset<CFF1StringIndex> (topDictIndex, topDictIndex->get_size ());
|
||||||
|
@ -1112,6 +1101,56 @@ struct cff1
|
||||||
inline bool is_valid (void) const { return blob != nullptr; }
|
inline bool is_valid (void) const { return blob != nullptr; }
|
||||||
inline bool is_CID (void) const { return topDict.is_CID (); }
|
inline bool is_CID (void) const { return topDict.is_CID (); }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
hb_blob_t *blob;
|
||||||
|
hb_sanitize_context_t sc;
|
||||||
|
|
||||||
|
public:
|
||||||
|
const CFF1NameIndex *nameIndex;
|
||||||
|
const CFF1TopDictIndex *topDictIndex;
|
||||||
|
const CFF1StringIndex *stringIndex;
|
||||||
|
const CFF1Subrs *globalSubrs;
|
||||||
|
const CFF1CharStrings *charStrings;
|
||||||
|
const CFF1FDArray *fdArray;
|
||||||
|
const CFF1FDSelect *fdSelect;
|
||||||
|
unsigned int fdCount;
|
||||||
|
|
||||||
|
CFF1TopDictValues topDict;
|
||||||
|
hb_vector_t<CFF1FontDictValues> fontDicts;
|
||||||
|
hb_vector_t<PRIVDICTVAL> privateDicts;
|
||||||
|
|
||||||
|
unsigned int num_glyphs;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct accelerator_t : accelerator_templ_t<CFF1PrivateDictOpSet, CFF1PrivateDictValues>
|
||||||
|
{
|
||||||
|
bool get_extents (hb_codepoint_t glyph, hb_glyph_extents_t *extents) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct accelerator_subset_t : accelerator_templ_t<CFF1PrivateDictOpSet_Subset, CFF1PrivateDictValues_Subset>
|
||||||
|
{
|
||||||
|
inline void init (hb_face_t *face)
|
||||||
|
{
|
||||||
|
SUPER::init (face);
|
||||||
|
if (blob == nullptr) return;
|
||||||
|
|
||||||
|
const OT::cff1 *cff = this->blob->template as<OT::cff1> ();
|
||||||
|
encoding = &Null(Encoding);
|
||||||
|
charset = &StructAtOffsetOrNull<Charset> (cff, topDict.CharsetOffset);
|
||||||
|
if (is_CID ())
|
||||||
|
{
|
||||||
|
if (unlikely (charset == &Null(Charset))) { fini (); return; }
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!is_predef_encoding ())
|
||||||
|
{
|
||||||
|
encoding = &StructAtOffsetOrNull<Encoding> (cff, topDict.EncodingOffset);
|
||||||
|
if (unlikely ((encoding == &Null (Encoding)) || !encoding->sanitize (&sc))) { fini (); return; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
inline bool is_predef_encoding (void) const { return topDict.EncodingOffset <= ExpertEncoding; }
|
inline bool is_predef_encoding (void) const { return topDict.EncodingOffset <= ExpertEncoding; }
|
||||||
inline bool is_predef_charset (void) const { return topDict.CharsetOffset <= ExpertSubsetCharset; }
|
inline bool is_predef_charset (void) const { return topDict.CharsetOffset <= ExpertSubsetCharset; }
|
||||||
|
|
||||||
|
@ -1164,42 +1203,13 @@ struct cff1
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inline bool get_extents (hb_codepoint_t glyph,
|
|
||||||
hb_glyph_extents_t *extents) const
|
|
||||||
{
|
|
||||||
// XXX: TODO
|
|
||||||
if (glyph >= num_glyphs)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
hb_blob_t *blob;
|
|
||||||
hb_sanitize_context_t sc;
|
|
||||||
|
|
||||||
public:
|
|
||||||
const CFF1NameIndex *nameIndex;
|
|
||||||
const CFF1TopDictIndex *topDictIndex;
|
|
||||||
const CFF1StringIndex *stringIndex;
|
|
||||||
const Encoding *encoding;
|
const Encoding *encoding;
|
||||||
const Charset *charset;
|
const Charset *charset;
|
||||||
const CFF1Subrs *globalSubrs;
|
|
||||||
const CFF1CharStrings *charStrings;
|
|
||||||
const CFF1FDArray *fdArray;
|
|
||||||
const CFF1FDSelect *fdSelect;
|
|
||||||
unsigned int fdCount;
|
|
||||||
|
|
||||||
CFF1TopDictValues topDict;
|
private:
|
||||||
hb_vector_t<CFF1FontDictValues> fontDicts;
|
typedef accelerator_templ_t<CFF1PrivateDictOpSet_Subset, CFF1PrivateDictValues_Subset> SUPER;
|
||||||
hb_vector_t<PRIVDICTVAL> privateDicts;
|
|
||||||
|
|
||||||
unsigned int num_glyphs;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef accelerator_templ_t<CFF1PrivateDictOpSet, CFF1PrivateDictValues> accelerator_t;
|
|
||||||
typedef accelerator_templ_t<CFF1PrivateDictOpSet_Subset, CFF1PrivateDictValues_Subset> accelerator_subset_t;
|
|
||||||
|
|
||||||
inline bool subset (hb_subset_plan_t *plan) const
|
inline bool subset (hb_subset_plan_t *plan) const
|
||||||
{
|
{
|
||||||
hb_blob_t *cff_prime = nullptr;
|
hb_blob_t *cff_prime = nullptr;
|
||||||
|
@ -1233,6 +1243,7 @@ struct cff1
|
||||||
DEFINE_SIZE_STATIC (4);
|
DEFINE_SIZE_STATIC (4);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct cff1_accelerator_t : cff1::accelerator_t {};
|
||||||
} /* namespace OT */
|
} /* namespace OT */
|
||||||
|
|
||||||
#endif /* HB_OT_CFF1_TABLE_HH */
|
#endif /* HB_OT_CFF1_TABLE_HH */
|
||||||
|
|
|
@ -28,6 +28,7 @@
|
||||||
|
|
||||||
#include "hb-ot-cmap-table.hh"
|
#include "hb-ot-cmap-table.hh"
|
||||||
#include "hb-ot-glyf-table.hh"
|
#include "hb-ot-glyf-table.hh"
|
||||||
|
#include "hb-ot-cff1-table.hh"
|
||||||
#include "hb-ot-hmtx-table.hh"
|
#include "hb-ot-hmtx-table.hh"
|
||||||
#include "hb-ot-kern-table.hh"
|
#include "hb-ot-kern-table.hh"
|
||||||
#include "hb-ot-post-table.hh"
|
#include "hb-ot-post-table.hh"
|
||||||
|
|
|
@ -68,6 +68,7 @@
|
||||||
HB_OT_LAYOUT_ACCELERATOR(OT, post) \
|
HB_OT_LAYOUT_ACCELERATOR(OT, post) \
|
||||||
HB_OT_LAYOUT_ACCELERATOR(OT, kern) \
|
HB_OT_LAYOUT_ACCELERATOR(OT, kern) \
|
||||||
HB_OT_LAYOUT_ACCELERATOR(OT, glyf) \
|
HB_OT_LAYOUT_ACCELERATOR(OT, glyf) \
|
||||||
|
HB_OT_LAYOUT_ACCELERATOR(OT, cff1) \
|
||||||
HB_OT_LAYOUT_ACCELERATOR(OT, CBDT) \
|
HB_OT_LAYOUT_ACCELERATOR(OT, CBDT) \
|
||||||
/* */
|
/* */
|
||||||
|
|
||||||
|
|
|
@ -145,6 +145,8 @@ hb_ot_get_glyph_extents (hb_font_t *font,
|
||||||
{
|
{
|
||||||
const hb_ot_face_data_t *ot_face = (const hb_ot_face_data_t *) font_data;
|
const hb_ot_face_data_t *ot_face = (const hb_ot_face_data_t *) font_data;
|
||||||
bool ret = ot_face->glyf->get_extents (glyph, extents);
|
bool ret = ot_face->glyf->get_extents (glyph, extents);
|
||||||
|
if (!ret)
|
||||||
|
ret = ot_face->cff1->get_extents (glyph, extents);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
ret = ot_face->CBDT->get_extents (glyph, extents);
|
ret = ot_face->CBDT->get_extents (glyph, extents);
|
||||||
// TODO Hook up side-bearings variations.
|
// TODO Hook up side-bearings variations.
|
||||||
|
|
|
@ -299,8 +299,8 @@ struct CFF1CSOpSet_Flatten : CFF1CSOpSet<CFF1CSOpSet_Flatten, FlattenParam>
|
||||||
|
|
||||||
static inline void flush_n_args (unsigned int n, CFF1CSInterpEnv &env, FlattenParam& param)
|
static inline void flush_n_args (unsigned int n, CFF1CSInterpEnv &env, FlattenParam& param)
|
||||||
{
|
{
|
||||||
for (unsigned int i = env.argStack.count - n; i < env.argStack.count; i++)
|
for (unsigned int i = env.argStack.get_count () - n; i < env.argStack.get_count (); i++)
|
||||||
param.flatStr.encode_num (env.argStack.elements[i]);
|
param.flatStr.encode_num (env.argStack[i]);
|
||||||
SUPER::flush_n_args (n, env, param);
|
SUPER::flush_n_args (n, env, param);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -111,9 +111,9 @@ struct CFF2CSOpSet_Flatten : CFF2CSOpSet<CFF2CSOpSet_Flatten, FlattenParam>
|
||||||
|
|
||||||
static inline void flush_n_args (unsigned int n, CFF2CSInterpEnv &env, FlattenParam& param)
|
static inline void flush_n_args (unsigned int n, CFF2CSInterpEnv &env, FlattenParam& param)
|
||||||
{
|
{
|
||||||
for (unsigned int i = env.argStack.count - n; i < env.argStack.count;)
|
for (unsigned int i = env.argStack.get_count () - n; i < env.argStack.get_count ();)
|
||||||
{
|
{
|
||||||
const BlendArg &arg = env.argStack.elements[i];
|
const BlendArg &arg = env.argStack[i];
|
||||||
if (arg.blended ())
|
if (arg.blended ())
|
||||||
{
|
{
|
||||||
assert ((arg.numValues > 0) && (n >= arg.numValues));
|
assert ((arg.numValues > 0) && (n >= arg.numValues));
|
||||||
|
@ -134,7 +134,7 @@ struct CFF2CSOpSet_Flatten : CFF2CSOpSet<CFF2CSOpSet_Flatten, FlattenParam>
|
||||||
/* flatten the default values */
|
/* flatten the default values */
|
||||||
for (unsigned int j = 0; j < arg.numValues; j++)
|
for (unsigned int j = 0; j < arg.numValues; j++)
|
||||||
{
|
{
|
||||||
const BlendArg &arg1 = env.argStack.elements[i + j];
|
const BlendArg &arg1 = env.argStack[i + j];
|
||||||
assert (arg1.blended () && (arg.numValues == arg1.numValues) && (arg1.valueIndex == j) &&
|
assert (arg1.blended () && (arg.numValues == arg1.numValues) && (arg1.valueIndex == j) &&
|
||||||
(arg1.deltas.len == env.get_region_count ()));
|
(arg1.deltas.len == env.get_region_count ()));
|
||||||
param.flatStr.encode_num (arg1);
|
param.flatStr.encode_num (arg1);
|
||||||
|
@ -142,7 +142,7 @@ struct CFF2CSOpSet_Flatten : CFF2CSOpSet<CFF2CSOpSet_Flatten, FlattenParam>
|
||||||
/* flatten deltas for each value */
|
/* flatten deltas for each value */
|
||||||
for (unsigned int j = 0; j < arg.numValues; j++)
|
for (unsigned int j = 0; j < arg.numValues; j++)
|
||||||
{
|
{
|
||||||
const BlendArg &arg1 = env.argStack.elements[i + j];
|
const BlendArg &arg1 = env.argStack[i + j];
|
||||||
for (unsigned int k = 0; k < arg1.deltas.len; k++)
|
for (unsigned int k = 0; k < arg1.deltas.len; k++)
|
||||||
param.flatStr.encode_num (arg1.deltas[k]);
|
param.flatStr.encode_num (arg1.deltas[k]);
|
||||||
}
|
}
|
||||||
|
@ -165,7 +165,7 @@ struct CFF2CSOpSet_Flatten : CFF2CSOpSet<CFF2CSOpSet_Flatten, FlattenParam>
|
||||||
|
|
||||||
private:
|
private:
|
||||||
typedef CFF2CSOpSet<CFF2CSOpSet_Flatten, FlattenParam> SUPER;
|
typedef CFF2CSOpSet<CFF2CSOpSet_Flatten, FlattenParam> SUPER;
|
||||||
typedef CSOpSet<BlendArg, CFF2CSOpSet_Flatten, CFF2CSInterpEnv, FlattenParam> CSOPSET;
|
typedef CSOpSet<BlendArg, CFF2CSOpSet_Flatten, CFF2CSOpSet_Flatten, CFF2CSInterpEnv, FlattenParam> CSOPSET;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct cff2_subset_plan {
|
struct cff2_subset_plan {
|
||||||
|
|
Loading…
Reference in New Issue