comparison mupdf-source/thirdparty/harfbuzz/src/hb-cff-interp-cs-common.hh @ 2:b50eed0cc0ef upstream

ADD: MuPDF v1.26.7: the MuPDF source as downloaded by a default build of PyMuPDF 1.26.4. The directory name has changed: no version number in the expanded directory now.
author Franz Glasner <fzglas.hg@dom66.de>
date Mon, 15 Sep 2025 11:43:07 +0200
parents
children
comparison
equal deleted inserted replaced
1:1d09e1dec1d9 2:b50eed0cc0ef
1 /*
2 * Copyright © 2018 Adobe Inc.
3 *
4 * This is part of HarfBuzz, a text shaping library.
5 *
6 * Permission is hereby granted, without written agreement and without
7 * license or royalty fees, to use, copy, modify, and distribute this
8 * software and its documentation for any purpose, provided that the
9 * above copyright notice and the following two paragraphs appear in
10 * all copies of this software.
11 *
12 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
13 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
14 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
15 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
16 * DAMAGE.
17 *
18 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
19 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
20 * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS
21 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
22 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
23 *
24 * Adobe Author(s): Michiharu Ariza
25 */
26 #ifndef HB_CFF_INTERP_CS_COMMON_HH
27 #define HB_CFF_INTERP_CS_COMMON_HH
28
29 #include "hb.hh"
30 #include "hb-cff-interp-common.hh"
31
32 namespace CFF {
33
34 using namespace OT;
35
36 enum cs_type_t {
37 CSType_CharString,
38 CSType_GlobalSubr,
39 CSType_LocalSubr
40 };
41
42 struct call_context_t
43 {
44 void init (const byte_str_ref_t substr_=byte_str_ref_t (), cs_type_t type_=CSType_CharString, unsigned int subr_num_=0)
45 {
46 str_ref = substr_;
47 type = type_;
48 subr_num = subr_num_;
49 }
50
51 void fini () {}
52
53 byte_str_ref_t str_ref;
54 cs_type_t type;
55 unsigned int subr_num;
56 };
57
58 /* call stack */
59 const unsigned int kMaxCallLimit = 10;
60 const unsigned int kMaxOps = 10000;
61 struct call_stack_t : cff_stack_t<call_context_t, kMaxCallLimit> {};
62
63 template <typename SUBRS>
64 struct biased_subrs_t
65 {
66 void init (const SUBRS *subrs_)
67 {
68 subrs = subrs_;
69 unsigned int nSubrs = get_count ();
70 if (nSubrs < 1240)
71 bias = 107;
72 else if (nSubrs < 33900)
73 bias = 1131;
74 else
75 bias = 32768;
76 }
77
78 void fini () {}
79
80 unsigned int get_count () const { return subrs ? subrs->count : 0; }
81 unsigned int get_bias () const { return bias; }
82
83 hb_ubytes_t operator [] (unsigned int index) const
84 {
85 if (unlikely (!subrs || index >= subrs->count))
86 return hb_ubytes_t ();
87 else
88 return (*subrs)[index];
89 }
90
91 protected:
92 unsigned int bias;
93 const SUBRS *subrs;
94 };
95
96 struct point_t
97 {
98 void set_int (int _x, int _y)
99 {
100 x.set_int (_x);
101 y.set_int (_y);
102 }
103
104 void move_x (const number_t &dx) { x += dx; }
105 void move_y (const number_t &dy) { y += dy; }
106 void move (const number_t &dx, const number_t &dy) { move_x (dx); move_y (dy); }
107 void move (const point_t &d) { move_x (d.x); move_y (d.y); }
108
109 number_t x;
110 number_t y;
111 };
112
113 template <typename ARG, typename SUBRS>
114 struct cs_interp_env_t : interp_env_t<ARG>
115 {
116 cs_interp_env_t (const hb_ubytes_t &str, const SUBRS *globalSubrs_, const SUBRS *localSubrs_) :
117 interp_env_t<ARG> (str)
118 {
119 context.init (str, CSType_CharString);
120 seen_moveto = true;
121 seen_hintmask = false;
122 hstem_count = 0;
123 vstem_count = 0;
124 hintmask_size = 0;
125 pt.set_int (0, 0);
126 globalSubrs.init (globalSubrs_);
127 localSubrs.init (localSubrs_);
128 }
129 ~cs_interp_env_t ()
130 {
131 globalSubrs.fini ();
132 localSubrs.fini ();
133 }
134
135 bool in_error () const
136 {
137 return callStack.in_error () || SUPER::in_error ();
138 }
139
140 bool pop_subr_num (const biased_subrs_t<SUBRS>& biasedSubrs, unsigned int &subr_num)
141 {
142 subr_num = 0;
143 int n = SUPER::argStack.pop_int ();
144 n += biasedSubrs.get_bias ();
145 if (unlikely ((n < 0) || ((unsigned int)n >= biasedSubrs.get_count ())))
146 return false;
147
148 subr_num = (unsigned int)n;
149 return true;
150 }
151
152 void call_subr (const biased_subrs_t<SUBRS>& biasedSubrs, cs_type_t type)
153 {
154 unsigned int subr_num = 0;
155
156 if (unlikely (!pop_subr_num (biasedSubrs, subr_num)
157 || callStack.get_count () >= kMaxCallLimit))
158 {
159 SUPER::set_error ();
160 return;
161 }
162 context.str_ref = SUPER::str_ref;
163 callStack.push (context);
164
165 context.init ( biasedSubrs[subr_num], type, subr_num);
166 SUPER::str_ref = context.str_ref;
167 }
168
169 void return_from_subr ()
170 {
171 if (unlikely (SUPER::str_ref.in_error ()))
172 SUPER::set_error ();
173 context = callStack.pop ();
174 SUPER::str_ref = context.str_ref;
175 }
176
177 void determine_hintmask_size ()
178 {
179 if (!seen_hintmask)
180 {
181 vstem_count += SUPER::argStack.get_count() / 2;
182 hintmask_size = (hstem_count + vstem_count + 7) >> 3;
183 seen_hintmask = true;
184 }
185 }
186
187 void set_endchar (bool endchar_flag_) { endchar_flag = endchar_flag_; }
188 bool is_endchar () const { return endchar_flag; }
189
190 const number_t &get_x () const { return pt.x; }
191 const number_t &get_y () const { return pt.y; }
192 const point_t &get_pt () const { return pt; }
193
194 void moveto (const point_t &pt_ ) { pt = pt_; }
195
196 public:
197 call_context_t context;
198 bool endchar_flag;
199 bool seen_moveto;
200 bool seen_hintmask;
201
202 unsigned int hstem_count;
203 unsigned int vstem_count;
204 unsigned int hintmask_size;
205 call_stack_t callStack;
206 biased_subrs_t<SUBRS> globalSubrs;
207 biased_subrs_t<SUBRS> localSubrs;
208
209 private:
210 point_t pt;
211
212 typedef interp_env_t<ARG> SUPER;
213 };
214
215 template <typename ENV, typename PARAM>
216 struct path_procs_null_t
217 {
218 static void rmoveto (ENV &env, PARAM& param) {}
219 static void hmoveto (ENV &env, PARAM& param) {}
220 static void vmoveto (ENV &env, PARAM& param) {}
221 static void rlineto (ENV &env, PARAM& param) {}
222 static void hlineto (ENV &env, PARAM& param) {}
223 static void vlineto (ENV &env, PARAM& param) {}
224 static void rrcurveto (ENV &env, PARAM& param) {}
225 static void rcurveline (ENV &env, PARAM& param) {}
226 static void rlinecurve (ENV &env, PARAM& param) {}
227 static void vvcurveto (ENV &env, PARAM& param) {}
228 static void hhcurveto (ENV &env, PARAM& param) {}
229 static void vhcurveto (ENV &env, PARAM& param) {}
230 static void hvcurveto (ENV &env, PARAM& param) {}
231 static void moveto (ENV &env, PARAM& param, const point_t &pt) {}
232 static void line (ENV &env, PARAM& param, const point_t &pt1) {}
233 static void curve (ENV &env, PARAM& param, const point_t &pt1, const point_t &pt2, const point_t &pt3) {}
234 static void hflex (ENV &env, PARAM& param) {}
235 static void flex (ENV &env, PARAM& param) {}
236 static void hflex1 (ENV &env, PARAM& param) {}
237 static void flex1 (ENV &env, PARAM& param) {}
238 };
239
240 template <typename ARG, typename OPSET, typename ENV, typename PARAM, typename PATH=path_procs_null_t<ENV, PARAM>>
241 struct cs_opset_t : opset_t<ARG>
242 {
243 static void process_op (op_code_t op, ENV &env, PARAM& param)
244 {
245 switch (op) {
246
247 case OpCode_return:
248 env.return_from_subr ();
249 break;
250 case OpCode_endchar:
251 OPSET::check_width (op, env, param);
252 env.set_endchar (true);
253 OPSET::flush_args_and_op (op, env, param);
254 break;
255
256 case OpCode_fixedcs:
257 env.argStack.push_fixed_from_substr (env.str_ref);
258 break;
259
260 case OpCode_callsubr:
261 env.call_subr (env.localSubrs, CSType_LocalSubr);
262 break;
263
264 case OpCode_callgsubr:
265 env.call_subr (env.globalSubrs, CSType_GlobalSubr);
266 break;
267
268 case OpCode_hstem:
269 case OpCode_hstemhm:
270 OPSET::check_width (op, env, param);
271 OPSET::process_hstem (op, env, param);
272 break;
273 case OpCode_vstem:
274 case OpCode_vstemhm:
275 OPSET::check_width (op, env, param);
276 OPSET::process_vstem (op, env, param);
277 break;
278 case OpCode_hintmask:
279 case OpCode_cntrmask:
280 OPSET::check_width (op, env, param);
281 OPSET::process_hintmask (op, env, param);
282 break;
283 case OpCode_rmoveto:
284 OPSET::check_width (op, env, param);
285 PATH::rmoveto (env, param);
286 OPSET::process_post_move (op, env, param);
287 break;
288 case OpCode_hmoveto:
289 OPSET::check_width (op, env, param);
290 PATH::hmoveto (env, param);
291 OPSET::process_post_move (op, env, param);
292 break;
293 case OpCode_vmoveto:
294 OPSET::check_width (op, env, param);
295 PATH::vmoveto (env, param);
296 OPSET::process_post_move (op, env, param);
297 break;
298 case OpCode_rlineto:
299 PATH::rlineto (env, param);
300 process_post_path (op, env, param);
301 break;
302 case OpCode_hlineto:
303 PATH::hlineto (env, param);
304 process_post_path (op, env, param);
305 break;
306 case OpCode_vlineto:
307 PATH::vlineto (env, param);
308 process_post_path (op, env, param);
309 break;
310 case OpCode_rrcurveto:
311 PATH::rrcurveto (env, param);
312 process_post_path (op, env, param);
313 break;
314 case OpCode_rcurveline:
315 PATH::rcurveline (env, param);
316 process_post_path (op, env, param);
317 break;
318 case OpCode_rlinecurve:
319 PATH::rlinecurve (env, param);
320 process_post_path (op, env, param);
321 break;
322 case OpCode_vvcurveto:
323 PATH::vvcurveto (env, param);
324 process_post_path (op, env, param);
325 break;
326 case OpCode_hhcurveto:
327 PATH::hhcurveto (env, param);
328 process_post_path (op, env, param);
329 break;
330 case OpCode_vhcurveto:
331 PATH::vhcurveto (env, param);
332 process_post_path (op, env, param);
333 break;
334 case OpCode_hvcurveto:
335 PATH::hvcurveto (env, param);
336 process_post_path (op, env, param);
337 break;
338
339 case OpCode_hflex:
340 PATH::hflex (env, param);
341 OPSET::process_post_flex (op, env, param);
342 break;
343
344 case OpCode_flex:
345 PATH::flex (env, param);
346 OPSET::process_post_flex (op, env, param);
347 break;
348
349 case OpCode_hflex1:
350 PATH::hflex1 (env, param);
351 OPSET::process_post_flex (op, env, param);
352 break;
353
354 case OpCode_flex1:
355 PATH::flex1 (env, param);
356 OPSET::process_post_flex (op, env, param);
357 break;
358
359 default:
360 SUPER::process_op (op, env);
361 break;
362 }
363 }
364
365 static void process_hstem (op_code_t op, ENV &env, PARAM& param)
366 {
367 env.hstem_count += env.argStack.get_count () / 2;
368 OPSET::flush_args_and_op (op, env, param);
369 }
370
371 static void process_vstem (op_code_t op, ENV &env, PARAM& param)
372 {
373 env.vstem_count += env.argStack.get_count () / 2;
374 OPSET::flush_args_and_op (op, env, param);
375 }
376
377 static void process_hintmask (op_code_t op, ENV &env, PARAM& param)
378 {
379 env.determine_hintmask_size ();
380 if (likely (env.str_ref.avail (env.hintmask_size)))
381 {
382 OPSET::flush_hintmask (op, env, param);
383 env.str_ref.inc (env.hintmask_size);
384 }
385 }
386
387 static void process_post_flex (op_code_t op, ENV &env, PARAM& param)
388 {
389 OPSET::flush_args_and_op (op, env, param);
390 }
391
392 static void check_width (op_code_t op, ENV &env, PARAM& param)
393 {}
394
395 static void process_post_move (op_code_t op, ENV &env, PARAM& param)
396 {
397 if (!env.seen_moveto)
398 {
399 env.determine_hintmask_size ();
400 env.seen_moveto = true;
401 }
402 OPSET::flush_args_and_op (op, env, param);
403 }
404
405 static void process_post_path (op_code_t op, ENV &env, PARAM& param)
406 {
407 OPSET::flush_args_and_op (op, env, param);
408 }
409
410 static void flush_args_and_op (op_code_t op, ENV &env, PARAM& param)
411 {
412 OPSET::flush_args (env, param);
413 OPSET::flush_op (op, env, param);
414 }
415
416 static void flush_args (ENV &env, PARAM& param)
417 {
418 env.pop_n_args (env.argStack.get_count ());
419 }
420
421 static void flush_op (op_code_t op, ENV &env, PARAM& param)
422 {
423 }
424
425 static void flush_hintmask (op_code_t op, ENV &env, PARAM& param)
426 {
427 OPSET::flush_args_and_op (op, env, param);
428 }
429
430 static bool is_number_op (op_code_t op)
431 {
432 switch (op)
433 {
434 case OpCode_shortint:
435 case OpCode_fixedcs:
436 case OpCode_TwoBytePosInt0: case OpCode_TwoBytePosInt1:
437 case OpCode_TwoBytePosInt2: case OpCode_TwoBytePosInt3:
438 case OpCode_TwoByteNegInt0: case OpCode_TwoByteNegInt1:
439 case OpCode_TwoByteNegInt2: case OpCode_TwoByteNegInt3:
440 return true;
441
442 default:
443 /* 1-byte integer */
444 return (OpCode_OneByteIntFirst <= op) && (op <= OpCode_OneByteIntLast);
445 }
446 }
447
448 protected:
449 typedef opset_t<ARG> SUPER;
450 };
451
452 template <typename PATH, typename ENV, typename PARAM>
453 struct path_procs_t
454 {
455 static void rmoveto (ENV &env, PARAM& param)
456 {
457 point_t pt1 = env.get_pt ();
458 const number_t &dy = env.pop_arg ();
459 const number_t &dx = env.pop_arg ();
460 pt1.move (dx, dy);
461 PATH::moveto (env, param, pt1);
462 }
463
464 static void hmoveto (ENV &env, PARAM& param)
465 {
466 point_t pt1 = env.get_pt ();
467 pt1.move_x (env.pop_arg ());
468 PATH::moveto (env, param, pt1);
469 }
470
471 static void vmoveto (ENV &env, PARAM& param)
472 {
473 point_t pt1 = env.get_pt ();
474 pt1.move_y (env.pop_arg ());
475 PATH::moveto (env, param, pt1);
476 }
477
478 static void rlineto (ENV &env, PARAM& param)
479 {
480 for (unsigned int i = 0; i + 2 <= env.argStack.get_count (); i += 2)
481 {
482 point_t pt1 = env.get_pt ();
483 pt1.move (env.eval_arg (i), env.eval_arg (i+1));
484 PATH::line (env, param, pt1);
485 }
486 }
487
488 static void hlineto (ENV &env, PARAM& param)
489 {
490 point_t pt1;
491 unsigned int i = 0;
492 for (; i + 2 <= env.argStack.get_count (); i += 2)
493 {
494 pt1 = env.get_pt ();
495 pt1.move_x (env.eval_arg (i));
496 PATH::line (env, param, pt1);
497 pt1.move_y (env.eval_arg (i+1));
498 PATH::line (env, param, pt1);
499 }
500 if (i < env.argStack.get_count ())
501 {
502 pt1 = env.get_pt ();
503 pt1.move_x (env.eval_arg (i));
504 PATH::line (env, param, pt1);
505 }
506 }
507
508 static void vlineto (ENV &env, PARAM& param)
509 {
510 point_t pt1;
511 unsigned int i = 0;
512 for (; i + 2 <= env.argStack.get_count (); i += 2)
513 {
514 pt1 = env.get_pt ();
515 pt1.move_y (env.eval_arg (i));
516 PATH::line (env, param, pt1);
517 pt1.move_x (env.eval_arg (i+1));
518 PATH::line (env, param, pt1);
519 }
520 if (i < env.argStack.get_count ())
521 {
522 pt1 = env.get_pt ();
523 pt1.move_y (env.eval_arg (i));
524 PATH::line (env, param, pt1);
525 }
526 }
527
528 static void rrcurveto (ENV &env, PARAM& param)
529 {
530 for (unsigned int i = 0; i + 6 <= env.argStack.get_count (); i += 6)
531 {
532 point_t pt1 = env.get_pt ();
533 pt1.move (env.eval_arg (i), env.eval_arg (i+1));
534 point_t pt2 = pt1;
535 pt2.move (env.eval_arg (i+2), env.eval_arg (i+3));
536 point_t pt3 = pt2;
537 pt3.move (env.eval_arg (i+4), env.eval_arg (i+5));
538 PATH::curve (env, param, pt1, pt2, pt3);
539 }
540 }
541
542 static void rcurveline (ENV &env, PARAM& param)
543 {
544 unsigned int arg_count = env.argStack.get_count ();
545 if (unlikely (arg_count < 8))
546 return;
547
548 unsigned int i = 0;
549 unsigned int curve_limit = arg_count - 2;
550 for (; i + 6 <= curve_limit; i += 6)
551 {
552 point_t pt1 = env.get_pt ();
553 pt1.move (env.eval_arg (i), env.eval_arg (i+1));
554 point_t pt2 = pt1;
555 pt2.move (env.eval_arg (i+2), env.eval_arg (i+3));
556 point_t pt3 = pt2;
557 pt3.move (env.eval_arg (i+4), env.eval_arg (i+5));
558 PATH::curve (env, param, pt1, pt2, pt3);
559 }
560
561 point_t pt1 = env.get_pt ();
562 pt1.move (env.eval_arg (i), env.eval_arg (i+1));
563 PATH::line (env, param, pt1);
564 }
565
566 static void rlinecurve (ENV &env, PARAM& param)
567 {
568 unsigned int arg_count = env.argStack.get_count ();
569 if (unlikely (arg_count < 8))
570 return;
571
572 unsigned int i = 0;
573 unsigned int line_limit = arg_count - 6;
574 for (; i + 2 <= line_limit; i += 2)
575 {
576 point_t pt1 = env.get_pt ();
577 pt1.move (env.eval_arg (i), env.eval_arg (i+1));
578 PATH::line (env, param, pt1);
579 }
580
581 point_t pt1 = env.get_pt ();
582 pt1.move (env.eval_arg (i), env.eval_arg (i+1));
583 point_t pt2 = pt1;
584 pt2.move (env.eval_arg (i+2), env.eval_arg (i+3));
585 point_t pt3 = pt2;
586 pt3.move (env.eval_arg (i+4), env.eval_arg (i+5));
587 PATH::curve (env, param, pt1, pt2, pt3);
588 }
589
590 static void vvcurveto (ENV &env, PARAM& param)
591 {
592 unsigned int i = 0;
593 point_t pt1 = env.get_pt ();
594 if ((env.argStack.get_count () & 1) != 0)
595 pt1.move_x (env.eval_arg (i++));
596 for (; i + 4 <= env.argStack.get_count (); i += 4)
597 {
598 pt1.move_y (env.eval_arg (i));
599 point_t pt2 = pt1;
600 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
601 point_t pt3 = pt2;
602 pt3.move_y (env.eval_arg (i+3));
603 PATH::curve (env, param, pt1, pt2, pt3);
604 pt1 = env.get_pt ();
605 }
606 }
607
608 static void hhcurveto (ENV &env, PARAM& param)
609 {
610 unsigned int i = 0;
611 point_t pt1 = env.get_pt ();
612 if ((env.argStack.get_count () & 1) != 0)
613 pt1.move_y (env.eval_arg (i++));
614 for (; i + 4 <= env.argStack.get_count (); i += 4)
615 {
616 pt1.move_x (env.eval_arg (i));
617 point_t pt2 = pt1;
618 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
619 point_t pt3 = pt2;
620 pt3.move_x (env.eval_arg (i+3));
621 PATH::curve (env, param, pt1, pt2, pt3);
622 pt1 = env.get_pt ();
623 }
624 }
625
626 static void vhcurveto (ENV &env, PARAM& param)
627 {
628 point_t pt1, pt2, pt3;
629 unsigned int i = 0;
630 if ((env.argStack.get_count () % 8) >= 4)
631 {
632 point_t pt1 = env.get_pt ();
633 pt1.move_y (env.eval_arg (i));
634 point_t pt2 = pt1;
635 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
636 point_t pt3 = pt2;
637 pt3.move_x (env.eval_arg (i+3));
638 i += 4;
639
640 for (; i + 8 <= env.argStack.get_count (); i += 8)
641 {
642 PATH::curve (env, param, pt1, pt2, pt3);
643 pt1 = env.get_pt ();
644 pt1.move_x (env.eval_arg (i));
645 pt2 = pt1;
646 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
647 pt3 = pt2;
648 pt3.move_y (env.eval_arg (i+3));
649 PATH::curve (env, param, pt1, pt2, pt3);
650
651 pt1 = pt3;
652 pt1.move_y (env.eval_arg (i+4));
653 pt2 = pt1;
654 pt2.move (env.eval_arg (i+5), env.eval_arg (i+6));
655 pt3 = pt2;
656 pt3.move_x (env.eval_arg (i+7));
657 }
658 if (i < env.argStack.get_count ())
659 pt3.move_y (env.eval_arg (i));
660 PATH::curve (env, param, pt1, pt2, pt3);
661 }
662 else
663 {
664 for (; i + 8 <= env.argStack.get_count (); i += 8)
665 {
666 pt1 = env.get_pt ();
667 pt1.move_y (env.eval_arg (i));
668 pt2 = pt1;
669 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
670 pt3 = pt2;
671 pt3.move_x (env.eval_arg (i+3));
672 PATH::curve (env, param, pt1, pt2, pt3);
673
674 pt1 = pt3;
675 pt1.move_x (env.eval_arg (i+4));
676 pt2 = pt1;
677 pt2.move (env.eval_arg (i+5), env.eval_arg (i+6));
678 pt3 = pt2;
679 pt3.move_y (env.eval_arg (i+7));
680 if ((env.argStack.get_count () - i < 16) && ((env.argStack.get_count () & 1) != 0))
681 pt3.move_x (env.eval_arg (i+8));
682 PATH::curve (env, param, pt1, pt2, pt3);
683 }
684 }
685 }
686
687 static void hvcurveto (ENV &env, PARAM& param)
688 {
689 point_t pt1, pt2, pt3;
690 unsigned int i = 0;
691 if ((env.argStack.get_count () % 8) >= 4)
692 {
693 point_t pt1 = env.get_pt ();
694 pt1.move_x (env.eval_arg (i));
695 point_t pt2 = pt1;
696 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
697 point_t pt3 = pt2;
698 pt3.move_y (env.eval_arg (i+3));
699 i += 4;
700
701 for (; i + 8 <= env.argStack.get_count (); i += 8)
702 {
703 PATH::curve (env, param, pt1, pt2, pt3);
704 pt1 = env.get_pt ();
705 pt1.move_y (env.eval_arg (i));
706 pt2 = pt1;
707 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
708 pt3 = pt2;
709 pt3.move_x (env.eval_arg (i+3));
710 PATH::curve (env, param, pt1, pt2, pt3);
711
712 pt1 = pt3;
713 pt1.move_x (env.eval_arg (i+4));
714 pt2 = pt1;
715 pt2.move (env.eval_arg (i+5), env.eval_arg (i+6));
716 pt3 = pt2;
717 pt3.move_y (env.eval_arg (i+7));
718 }
719 if (i < env.argStack.get_count ())
720 pt3.move_x (env.eval_arg (i));
721 PATH::curve (env, param, pt1, pt2, pt3);
722 }
723 else
724 {
725 for (; i + 8 <= env.argStack.get_count (); i += 8)
726 {
727 pt1 = env.get_pt ();
728 pt1.move_x (env.eval_arg (i));
729 pt2 = pt1;
730 pt2.move (env.eval_arg (i+1), env.eval_arg (i+2));
731 pt3 = pt2;
732 pt3.move_y (env.eval_arg (i+3));
733 PATH::curve (env, param, pt1, pt2, pt3);
734
735 pt1 = pt3;
736 pt1.move_y (env.eval_arg (i+4));
737 pt2 = pt1;
738 pt2.move (env.eval_arg (i+5), env.eval_arg (i+6));
739 pt3 = pt2;
740 pt3.move_x (env.eval_arg (i+7));
741 if ((env.argStack.get_count () - i < 16) && ((env.argStack.get_count () & 1) != 0))
742 pt3.move_y (env.eval_arg (i+8));
743 PATH::curve (env, param, pt1, pt2, pt3);
744 }
745 }
746 }
747
748 /* default actions to be overridden */
749 static void moveto (ENV &env, PARAM& param, const point_t &pt)
750 { env.moveto (pt); }
751
752 static void line (ENV &env, PARAM& param, const point_t &pt1)
753 { PATH::moveto (env, param, pt1); }
754
755 static void curve (ENV &env, PARAM& param, const point_t &pt1, const point_t &pt2, const point_t &pt3)
756 { PATH::moveto (env, param, pt3); }
757
758 static void hflex (ENV &env, PARAM& param)
759 {
760 if (likely (env.argStack.get_count () == 7))
761 {
762 point_t pt1 = env.get_pt ();
763 pt1.move_x (env.eval_arg (0));
764 point_t pt2 = pt1;
765 pt2.move (env.eval_arg (1), env.eval_arg (2));
766 point_t pt3 = pt2;
767 pt3.move_x (env.eval_arg (3));
768 point_t pt4 = pt3;
769 pt4.move_x (env.eval_arg (4));
770 point_t pt5 = pt4;
771 pt5.move_x (env.eval_arg (5));
772 pt5.y = pt1.y;
773 point_t pt6 = pt5;
774 pt6.move_x (env.eval_arg (6));
775
776 curve2 (env, param, pt1, pt2, pt3, pt4, pt5, pt6);
777 }
778 else
779 env.set_error ();
780 }
781
782 static void flex (ENV &env, PARAM& param)
783 {
784 if (likely (env.argStack.get_count () == 13))
785 {
786 point_t pt1 = env.get_pt ();
787 pt1.move (env.eval_arg (0), env.eval_arg (1));
788 point_t pt2 = pt1;
789 pt2.move (env.eval_arg (2), env.eval_arg (3));
790 point_t pt3 = pt2;
791 pt3.move (env.eval_arg (4), env.eval_arg (5));
792 point_t pt4 = pt3;
793 pt4.move (env.eval_arg (6), env.eval_arg (7));
794 point_t pt5 = pt4;
795 pt5.move (env.eval_arg (8), env.eval_arg (9));
796 point_t pt6 = pt5;
797 pt6.move (env.eval_arg (10), env.eval_arg (11));
798
799 curve2 (env, param, pt1, pt2, pt3, pt4, pt5, pt6);
800 }
801 else
802 env.set_error ();
803 }
804
805 static void hflex1 (ENV &env, PARAM& param)
806 {
807 if (likely (env.argStack.get_count () == 9))
808 {
809 point_t pt1 = env.get_pt ();
810 pt1.move (env.eval_arg (0), env.eval_arg (1));
811 point_t pt2 = pt1;
812 pt2.move (env.eval_arg (2), env.eval_arg (3));
813 point_t pt3 = pt2;
814 pt3.move_x (env.eval_arg (4));
815 point_t pt4 = pt3;
816 pt4.move_x (env.eval_arg (5));
817 point_t pt5 = pt4;
818 pt5.move (env.eval_arg (6), env.eval_arg (7));
819 point_t pt6 = pt5;
820 pt6.move_x (env.eval_arg (8));
821 pt6.y = env.get_pt ().y;
822
823 curve2 (env, param, pt1, pt2, pt3, pt4, pt5, pt6);
824 }
825 else
826 env.set_error ();
827 }
828
829 static void flex1 (ENV &env, PARAM& param)
830 {
831 if (likely (env.argStack.get_count () == 11))
832 {
833 point_t d;
834 for (unsigned int i = 0; i < 10; i += 2)
835 d.move (env.eval_arg (i), env.eval_arg (i+1));
836
837 point_t pt1 = env.get_pt ();
838 pt1.move (env.eval_arg (0), env.eval_arg (1));
839 point_t pt2 = pt1;
840 pt2.move (env.eval_arg (2), env.eval_arg (3));
841 point_t pt3 = pt2;
842 pt3.move (env.eval_arg (4), env.eval_arg (5));
843 point_t pt4 = pt3;
844 pt4.move (env.eval_arg (6), env.eval_arg (7));
845 point_t pt5 = pt4;
846 pt5.move (env.eval_arg (8), env.eval_arg (9));
847 point_t pt6 = pt5;
848
849 if (fabs (d.x.to_real ()) > fabs (d.y.to_real ()))
850 {
851 pt6.move_x (env.eval_arg (10));
852 pt6.y = env.get_pt ().y;
853 }
854 else
855 {
856 pt6.x = env.get_pt ().x;
857 pt6.move_y (env.eval_arg (10));
858 }
859
860 curve2 (env, param, pt1, pt2, pt3, pt4, pt5, pt6);
861 }
862 else
863 env.set_error ();
864 }
865
866 protected:
867 static void curve2 (ENV &env, PARAM& param,
868 const point_t &pt1, const point_t &pt2, const point_t &pt3,
869 const point_t &pt4, const point_t &pt5, const point_t &pt6)
870 {
871 PATH::curve (env, param, pt1, pt2, pt3);
872 PATH::curve (env, param, pt4, pt5, pt6);
873 }
874 };
875
876 template <typename ENV, typename OPSET, typename PARAM>
877 struct cs_interpreter_t : interpreter_t<ENV>
878 {
879 cs_interpreter_t (ENV& env_) : interpreter_t<ENV> (env_) {}
880
881 bool interpret (PARAM& param)
882 {
883 SUPER::env.set_endchar (false);
884
885 unsigned max_ops = kMaxOps;
886 for (;;) {
887 if (unlikely (!--max_ops))
888 {
889 SUPER::env.set_error ();
890 break;
891 }
892 OPSET::process_op (SUPER::env.fetch_op (), SUPER::env, param);
893 if (unlikely (SUPER::env.in_error ()))
894 return false;
895 if (SUPER::env.is_endchar ())
896 break;
897 }
898
899 return true;
900 }
901
902 private:
903 typedef interpreter_t<ENV> SUPER;
904 };
905
906 } /* namespace CFF */
907
908 #endif /* HB_CFF_INTERP_CS_COMMON_HH */