2018-08-15 21:00:19 +02:00
|
|
|
/*
|
|
|
|
* Copyright © 2018 Adobe Systems Incorporated.
|
|
|
|
*
|
|
|
|
* This is part of HarfBuzz, a text shaping library.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, without written agreement and without
|
|
|
|
* license or royalty fees, to use, copy, modify, and distribute this
|
|
|
|
* software and its documentation for any purpose, provided that the
|
|
|
|
* above copyright notice and the following two paragraphs appear in
|
|
|
|
* all copies of this software.
|
|
|
|
*
|
|
|
|
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
|
|
|
|
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
|
|
|
|
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
|
|
|
|
* DAMAGE.
|
|
|
|
*
|
|
|
|
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
|
|
|
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
|
|
|
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
|
|
|
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
|
|
|
|
*
|
|
|
|
* Adobe Author(s): Michiharu Ariza
|
|
|
|
*/
|
2018-08-15 22:04:43 +02:00
|
|
|
#ifndef HB_CFF_INTERP_CS_COMMON_PRIVATE_HH
|
|
|
|
#define HB_CFF_INTERP_CS_COMMON_PRIVATE_HH
|
2018-08-15 21:00:19 +02:00
|
|
|
|
2018-08-29 22:26:17 +02:00
|
|
|
#include "hb.hh"
|
|
|
|
#include "hb-cff-interp-common.hh"
|
2018-08-15 21:00:19 +02:00
|
|
|
|
|
|
|
namespace CFF {
|
|
|
|
|
|
|
|
using namespace OT;
|
|
|
|
|
|
|
|
/* call stack */
|
|
|
|
struct CallStack : Stack<SubByteStr, 10> {};
|
|
|
|
|
|
|
|
template <typename SUBRS>
|
|
|
|
struct BiasedSubrs
|
|
|
|
{
|
|
|
|
inline void init (const SUBRS &subrs_)
|
|
|
|
{
|
|
|
|
subrs = &subrs_;
|
|
|
|
unsigned int nSubrs = subrs_.count;
|
|
|
|
if (nSubrs < 1240)
|
|
|
|
bias = 107;
|
|
|
|
else if (nSubrs < 33900)
|
|
|
|
bias = 1131;
|
|
|
|
else
|
|
|
|
bias = 32768;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void fini (void) {}
|
|
|
|
|
|
|
|
const SUBRS *subrs;
|
|
|
|
unsigned int bias;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename SUBRS>
|
|
|
|
struct CSInterpEnv : InterpEnv
|
|
|
|
{
|
|
|
|
inline void init (const ByteStr &str, const SUBRS &globalSubrs_, const SUBRS &localSubrs_)
|
|
|
|
{
|
|
|
|
InterpEnv::init (str);
|
|
|
|
|
2018-08-17 22:13:18 +02:00
|
|
|
stack_cleared = false;
|
|
|
|
seen_moveto = true;
|
|
|
|
seen_hintmask = false;
|
|
|
|
hstem_count = 0;
|
|
|
|
vstem_count = 0;
|
2018-08-15 21:00:19 +02:00
|
|
|
callStack.init ();
|
|
|
|
globalSubrs.init (globalSubrs_);
|
|
|
|
localSubrs.init (localSubrs_);
|
|
|
|
}
|
|
|
|
inline void fini (void)
|
|
|
|
{
|
|
|
|
InterpEnv::fini ();
|
|
|
|
|
|
|
|
callStack.fini ();
|
|
|
|
globalSubrs.fini ();
|
|
|
|
localSubrs.fini ();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool popSubrNum (const BiasedSubrs<SUBRS>& biasedSubrs, unsigned int &subr_num)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
if (unlikely ((!callStack.check_overflow (1) ||
|
|
|
|
!argStack.check_pop_int (n))))
|
|
|
|
return false;
|
|
|
|
n += biasedSubrs.bias;
|
|
|
|
if (unlikely ((n < 0) || (n >= biasedSubrs.subrs->count)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
subr_num = (unsigned int)n;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool callSubr (const BiasedSubrs<SUBRS>& biasedSubrs)
|
|
|
|
{
|
|
|
|
unsigned int subr_num;
|
|
|
|
|
|
|
|
if (unlikely (!popSubrNum (biasedSubrs, subr_num)))
|
|
|
|
return false;
|
|
|
|
callStack.push (substr);
|
|
|
|
substr = (*biasedSubrs.subrs)[subr_num];
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool returnFromSubr (void)
|
|
|
|
{
|
2018-08-17 22:13:18 +02:00
|
|
|
if (unlikely (!callStack.check_underflow ()))
|
2018-08-15 21:00:19 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
substr = callStack.pop ();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-17 22:13:18 +02:00
|
|
|
inline void determine_hintmask_size (void)
|
|
|
|
{
|
|
|
|
if (!seen_hintmask)
|
|
|
|
{
|
|
|
|
vstem_count += argStack.size / 2;
|
|
|
|
hintmask_size = (hstem_count + vstem_count + 7) >> 3;
|
|
|
|
seen_hintmask = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void clear_stack (void)
|
|
|
|
{
|
|
|
|
stack_cleared = true;
|
|
|
|
argStack.clear ();
|
|
|
|
}
|
|
|
|
|
2018-08-15 21:00:19 +02:00
|
|
|
inline void set_endchar (bool endchar_flag_) { endchar_flag = endchar_flag_; }
|
|
|
|
inline bool is_endchar (void) const { return endchar_flag; }
|
2018-08-17 22:13:18 +02:00
|
|
|
inline bool is_stack_cleared (void) const { return stack_cleared; }
|
2018-08-15 21:00:19 +02:00
|
|
|
|
2018-08-29 21:14:30 +02:00
|
|
|
public:
|
2018-08-17 22:13:18 +02:00
|
|
|
bool endchar_flag;
|
|
|
|
bool stack_cleared;
|
|
|
|
bool seen_moveto;
|
|
|
|
bool seen_hintmask;
|
2018-08-15 21:00:19 +02:00
|
|
|
|
2018-08-17 22:13:18 +02:00
|
|
|
unsigned int hstem_count;
|
|
|
|
unsigned int vstem_count;
|
|
|
|
unsigned int hintmask_size;
|
2018-08-15 21:00:19 +02:00
|
|
|
CallStack callStack;
|
|
|
|
BiasedSubrs<SUBRS> globalSubrs;
|
|
|
|
BiasedSubrs<SUBRS> localSubrs;
|
|
|
|
};
|
|
|
|
|
2018-08-29 21:14:30 +02:00
|
|
|
template <typename OPSET, typename ENV, typename PARAM>
|
2018-08-15 21:00:19 +02:00
|
|
|
struct CSOpSet : OpSet
|
|
|
|
{
|
2018-08-29 21:14:30 +02:00
|
|
|
static inline bool process_op (OpCode op, ENV &env, PARAM& param)
|
|
|
|
{
|
2018-08-15 21:00:19 +02:00
|
|
|
switch (op) {
|
|
|
|
|
2018-08-17 22:13:18 +02:00
|
|
|
case OpCode_return:
|
|
|
|
return env.returnFromSubr ();
|
|
|
|
case OpCode_endchar:
|
|
|
|
env.set_endchar (true);
|
|
|
|
return true;
|
|
|
|
|
2018-08-22 20:36:39 +02:00
|
|
|
case OpCode_fixedcs:
|
|
|
|
return env.argStack.push_fixed_from_substr (env.substr);
|
2018-08-15 21:00:19 +02:00
|
|
|
|
|
|
|
case OpCode_callsubr:
|
|
|
|
return env.callSubr (env.localSubrs);
|
|
|
|
|
|
|
|
case OpCode_callgsubr:
|
|
|
|
return env.callSubr (env.globalSubrs);
|
|
|
|
|
2018-08-17 22:13:18 +02:00
|
|
|
case OpCode_hstem:
|
|
|
|
case OpCode_hstemhm:
|
2018-08-29 21:14:30 +02:00
|
|
|
OPSET::process_hstem (env, param);
|
2018-08-17 22:13:18 +02:00
|
|
|
break;
|
|
|
|
case OpCode_vstem:
|
|
|
|
case OpCode_vstemhm:
|
2018-08-29 21:14:30 +02:00
|
|
|
OPSET::process_vstem (env, param);
|
2018-08-17 22:13:18 +02:00
|
|
|
break;
|
|
|
|
case OpCode_hintmask:
|
|
|
|
case OpCode_cntrmask:
|
|
|
|
env.determine_hintmask_size ();
|
2018-08-29 21:14:30 +02:00
|
|
|
OPSET::flush_stack (env, param);
|
2018-08-17 22:13:18 +02:00
|
|
|
if (unlikely (!env.substr.avail (env.hintmask_size)))
|
|
|
|
return false;
|
|
|
|
env.substr.inc (env.hintmask_size);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OpCode_vmoveto:
|
|
|
|
case OpCode_rlineto:
|
|
|
|
case OpCode_hlineto:
|
|
|
|
case OpCode_vlineto:
|
|
|
|
case OpCode_rmoveto:
|
|
|
|
case OpCode_hmoveto:
|
2018-08-29 21:14:30 +02:00
|
|
|
OPSET::process_moveto (env, param);
|
2018-08-17 22:13:18 +02:00
|
|
|
break;
|
|
|
|
case OpCode_rrcurveto:
|
|
|
|
case OpCode_rcurveline:
|
|
|
|
case OpCode_rlinecurve:
|
|
|
|
case OpCode_vvcurveto:
|
|
|
|
case OpCode_hhcurveto:
|
|
|
|
case OpCode_vhcurveto:
|
|
|
|
case OpCode_hvcurveto:
|
|
|
|
case OpCode_hflex:
|
|
|
|
case OpCode_flex:
|
|
|
|
case OpCode_hflex1:
|
|
|
|
case OpCode_flex1:
|
2018-08-29 21:14:30 +02:00
|
|
|
OPSET::flush_stack (env, param);
|
2018-08-17 22:13:18 +02:00
|
|
|
break;
|
|
|
|
|
2018-08-15 21:00:19 +02:00
|
|
|
default:
|
|
|
|
return OpSet::process_op (op, env);
|
|
|
|
}
|
2018-08-17 22:13:18 +02:00
|
|
|
return true;
|
2018-08-15 21:00:19 +02:00
|
|
|
}
|
2018-08-29 21:14:30 +02:00
|
|
|
|
|
|
|
static inline void process_hstem (ENV &env, PARAM& param)
|
|
|
|
{
|
|
|
|
env.hstem_count += env.argStack.size / 2;
|
|
|
|
OPSET::flush_stack (env, param);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void process_vstem (ENV &env, PARAM& param)
|
|
|
|
{
|
|
|
|
env.vstem_count += env.argStack.size / 2;
|
|
|
|
OPSET::flush_stack (env, param);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void process_moveto (ENV &env, PARAM& param)
|
|
|
|
{
|
|
|
|
if (!env.seen_moveto)
|
|
|
|
{
|
|
|
|
env.determine_hintmask_size ();
|
|
|
|
env.seen_moveto = true;
|
|
|
|
}
|
|
|
|
OPSET::flush_stack (env, param);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void flush_stack (ENV &env, PARAM& param)
|
|
|
|
{
|
|
|
|
env.clear_stack ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* numeric / logical / arithmetic operators */
|
|
|
|
static inline bool is_arg_op (OpCode op)
|
|
|
|
{
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case OpCode_shortint:
|
|
|
|
case OpCode_TwoBytePosInt0: case OpCode_TwoBytePosInt1:
|
|
|
|
case OpCode_TwoBytePosInt2: case OpCode_TwoBytePosInt3:
|
|
|
|
case OpCode_TwoByteNegInt0: case OpCode_TwoByteNegInt1:
|
|
|
|
case OpCode_TwoByteNegInt2: case OpCode_TwoByteNegInt3:
|
|
|
|
case OpCode_fixedcs:
|
|
|
|
case OpCode_and:
|
|
|
|
case OpCode_or:
|
|
|
|
case OpCode_not:
|
|
|
|
case OpCode_abs:
|
|
|
|
case OpCode_add:
|
|
|
|
case OpCode_sub:
|
|
|
|
case OpCode_div:
|
|
|
|
case OpCode_neg:
|
|
|
|
case OpCode_eq:
|
|
|
|
case OpCode_drop:
|
|
|
|
case OpCode_put:
|
|
|
|
case OpCode_get:
|
|
|
|
case OpCode_ifelse:
|
|
|
|
case OpCode_random:
|
|
|
|
case OpCode_mul:
|
|
|
|
case OpCode_sqrt:
|
|
|
|
case OpCode_dup:
|
|
|
|
case OpCode_exch:
|
|
|
|
case OpCode_index:
|
|
|
|
case OpCode_roll:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return (OpCode_OneByteIntFirst <= op) && (op <= OpCode_OneByteIntLast);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool is_subr_op (OpCode op)
|
|
|
|
{
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case OpCode_callsubr:
|
|
|
|
case OpCode_callgsubr:
|
|
|
|
case OpCode_return:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2018-08-15 21:00:19 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
template <typename ENV, typename OPSET, typename PARAM>
|
|
|
|
struct CSInterpreter : Interpreter<ENV>
|
|
|
|
{
|
|
|
|
inline bool interpret (PARAM& param)
|
|
|
|
{
|
|
|
|
param.init ();
|
|
|
|
Interpreter<ENV> &super = *this;
|
|
|
|
super.env.set_endchar (false);
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
OpCode op;
|
2018-08-17 22:13:18 +02:00
|
|
|
if (unlikely (!super.env.fetch_op (op) ||
|
2018-08-15 21:00:19 +02:00
|
|
|
!OPSET::process_op (op, super.env, param)))
|
|
|
|
return false;
|
|
|
|
if (super.env.is_endchar ())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} /* namespace CFF */
|
|
|
|
|
2018-08-15 22:04:43 +02:00
|
|
|
#endif /* HB_CFF_INTERP_CS_COMMON_PRIVATE_HH */
|