1 /****************************************************************************
2  *
3  * pfrgload.c
4  *
5  *   FreeType PFR glyph loader (body).
6  *
7  * Copyright 2002-2018 by
8  * David Turner, Robert Wilhelm, and Werner Lemberg.
9  *
10  * This file is part of the FreeType project, and may only be used,
11  * modified, and distributed under the terms of the FreeType project
12  * license, LICENSE.TXT.  By continuing to use, modify, or distribute
13  * this file you indicate that you have read the license and
14  * understand and accept it fully.
15  *
16  */
17 
18 
19 #include "pfrgload.h"
20 #include "pfrsbit.h"
21 #include "pfrload.h"            /* for macro definitions */
22 #include FT_INTERNAL_DEBUG_H
23 
24 #include "pfrerror.h"
25 
26 #undef  FT_COMPONENT
27 #define FT_COMPONENT  trace_pfr
28 
29 
30   /*************************************************************************/
31   /*************************************************************************/
32   /*****                                                               *****/
33   /*****                      PFR GLYPH BUILDER                        *****/
34   /*****                                                               *****/
35   /*************************************************************************/
36   /*************************************************************************/
37 
38 
39   FT_LOCAL_DEF( void )
pfr_glyph_init(PFR_Glyph glyph,FT_GlyphLoader loader)40   pfr_glyph_init( PFR_Glyph       glyph,
41                   FT_GlyphLoader  loader )
42   {
43     FT_ZERO( glyph );
44 
45     glyph->loader     = loader;
46     glyph->path_begun = 0;
47 
48     FT_GlyphLoader_Rewind( loader );
49   }
50 
51 
52   FT_LOCAL_DEF( void )
pfr_glyph_done(PFR_Glyph glyph)53   pfr_glyph_done( PFR_Glyph  glyph )
54   {
55     FT_Memory  memory = glyph->loader->memory;
56 
57 
58     FT_FREE( glyph->x_control );
59     glyph->y_control = NULL;
60 
61     glyph->max_xy_control = 0;
62 #if 0
63     glyph->num_x_control  = 0;
64     glyph->num_y_control  = 0;
65 #endif
66 
67     FT_FREE( glyph->subs );
68 
69     glyph->max_subs = 0;
70     glyph->num_subs = 0;
71 
72     glyph->loader     = NULL;
73     glyph->path_begun = 0;
74   }
75 
76 
77   /* close current contour, if any */
78   static void
pfr_glyph_close_contour(PFR_Glyph glyph)79   pfr_glyph_close_contour( PFR_Glyph  glyph )
80   {
81     FT_GlyphLoader  loader  = glyph->loader;
82     FT_Outline*     outline = &loader->current.outline;
83     FT_Int          last, first;
84 
85 
86     if ( !glyph->path_begun )
87       return;
88 
89     /* compute first and last point indices in current glyph outline */
90     last  = outline->n_points - 1;
91     first = 0;
92     if ( outline->n_contours > 0 )
93       first = outline->contours[outline->n_contours - 1];
94 
95     /* if the last point falls on the same location as the first one */
96     /* we need to delete it                                          */
97     if ( last > first )
98     {
99       FT_Vector*  p1 = outline->points + first;
100       FT_Vector*  p2 = outline->points + last;
101 
102 
103       if ( p1->x == p2->x && p1->y == p2->y )
104       {
105         outline->n_points--;
106         last--;
107       }
108     }
109 
110     /* don't add empty contours */
111     if ( last >= first )
112       outline->contours[outline->n_contours++] = (short)last;
113 
114     glyph->path_begun = 0;
115   }
116 
117 
118   /* reset glyph to start the loading of a new glyph */
119   static void
pfr_glyph_start(PFR_Glyph glyph)120   pfr_glyph_start( PFR_Glyph  glyph )
121   {
122     glyph->path_begun = 0;
123   }
124 
125 
126   static FT_Error
pfr_glyph_line_to(PFR_Glyph glyph,FT_Vector * to)127   pfr_glyph_line_to( PFR_Glyph   glyph,
128                      FT_Vector*  to )
129   {
130     FT_GlyphLoader  loader  = glyph->loader;
131     FT_Outline*     outline = &loader->current.outline;
132     FT_Error        error;
133 
134 
135     /* check that we have begun a new path */
136     if ( !glyph->path_begun )
137     {
138       error = FT_THROW( Invalid_Table );
139       FT_ERROR(( "pfr_glyph_line_to: invalid glyph data\n" ));
140       goto Exit;
141     }
142 
143     error = FT_GLYPHLOADER_CHECK_POINTS( loader, 1, 0 );
144     if ( !error )
145     {
146       FT_Int  n = outline->n_points;
147 
148 
149       outline->points[n] = *to;
150       outline->tags  [n] = FT_CURVE_TAG_ON;
151 
152       outline->n_points++;
153     }
154 
155   Exit:
156     return error;
157   }
158 
159 
160   static FT_Error
pfr_glyph_curve_to(PFR_Glyph glyph,FT_Vector * control1,FT_Vector * control2,FT_Vector * to)161   pfr_glyph_curve_to( PFR_Glyph   glyph,
162                       FT_Vector*  control1,
163                       FT_Vector*  control2,
164                       FT_Vector*  to )
165   {
166     FT_GlyphLoader  loader  = glyph->loader;
167     FT_Outline*     outline = &loader->current.outline;
168     FT_Error        error;
169 
170 
171     /* check that we have begun a new path */
172     if ( !glyph->path_begun )
173     {
174       error = FT_THROW( Invalid_Table );
175       FT_ERROR(( "pfr_glyph_line_to: invalid glyph data\n" ));
176       goto Exit;
177     }
178 
179     error = FT_GLYPHLOADER_CHECK_POINTS( loader, 3, 0 );
180     if ( !error )
181     {
182       FT_Vector*  vec = outline->points         + outline->n_points;
183       FT_Byte*    tag = (FT_Byte*)outline->tags + outline->n_points;
184 
185 
186       vec[0] = *control1;
187       vec[1] = *control2;
188       vec[2] = *to;
189       tag[0] = FT_CURVE_TAG_CUBIC;
190       tag[1] = FT_CURVE_TAG_CUBIC;
191       tag[2] = FT_CURVE_TAG_ON;
192 
193       outline->n_points = (FT_Short)( outline->n_points + 3 );
194     }
195 
196   Exit:
197     return error;
198   }
199 
200 
201   static FT_Error
pfr_glyph_move_to(PFR_Glyph glyph,FT_Vector * to)202   pfr_glyph_move_to( PFR_Glyph   glyph,
203                      FT_Vector*  to )
204   {
205     FT_GlyphLoader  loader  = glyph->loader;
206     FT_Error        error;
207 
208 
209     /* close current contour if any */
210     pfr_glyph_close_contour( glyph );
211 
212     /* indicate that a new contour has started */
213     glyph->path_begun = 1;
214 
215     /* check that there is space for a new contour and a new point */
216     error = FT_GLYPHLOADER_CHECK_POINTS( loader, 1, 1 );
217     if ( !error )
218     {
219       /* add new start point */
220       error = pfr_glyph_line_to( glyph, to );
221     }
222 
223     return error;
224   }
225 
226 
227   static void
pfr_glyph_end(PFR_Glyph glyph)228   pfr_glyph_end( PFR_Glyph  glyph )
229   {
230     /* close current contour if any */
231     pfr_glyph_close_contour( glyph );
232 
233     /* merge the current glyph into the stack */
234     FT_GlyphLoader_Add( glyph->loader );
235   }
236 
237 
238   /*************************************************************************/
239   /*************************************************************************/
240   /*****                                                               *****/
241   /*****                      PFR GLYPH LOADER                         *****/
242   /*****                                                               *****/
243   /*************************************************************************/
244   /*************************************************************************/
245 
246 
247   /* load a simple glyph */
248   static FT_Error
pfr_glyph_load_simple(PFR_Glyph glyph,FT_Byte * p,FT_Byte * limit)249   pfr_glyph_load_simple( PFR_Glyph  glyph,
250                          FT_Byte*   p,
251                          FT_Byte*   limit )
252   {
253     FT_Error   error  = FT_Err_Ok;
254     FT_Memory  memory = glyph->loader->memory;
255     FT_UInt    flags, x_count, y_count, i, count, mask;
256     FT_Int     x;
257 
258 
259     PFR_CHECK( 1 );
260     flags = PFR_NEXT_BYTE( p );
261 
262     /* test for composite glyphs */
263     if ( flags & PFR_GLYPH_IS_COMPOUND )
264       goto Failure;
265 
266     x_count = 0;
267     y_count = 0;
268 
269     if ( flags & PFR_GLYPH_1BYTE_XYCOUNT )
270     {
271       PFR_CHECK( 1 );
272       count   = PFR_NEXT_BYTE( p );
273       x_count = count & 15;
274       y_count = count >> 4;
275     }
276     else
277     {
278       if ( flags & PFR_GLYPH_XCOUNT )
279       {
280         PFR_CHECK( 1 );
281         x_count = PFR_NEXT_BYTE( p );
282       }
283 
284       if ( flags & PFR_GLYPH_YCOUNT )
285       {
286         PFR_CHECK( 1 );
287         y_count = PFR_NEXT_BYTE( p );
288       }
289     }
290 
291     count = x_count + y_count;
292 
293     /* re-allocate array when necessary */
294     if ( count > glyph->max_xy_control )
295     {
296       FT_UInt  new_max = FT_PAD_CEIL( count, 8 );
297 
298 
299       if ( FT_RENEW_ARRAY( glyph->x_control,
300                            glyph->max_xy_control,
301                            new_max ) )
302         goto Exit;
303 
304       glyph->max_xy_control = new_max;
305     }
306 
307     glyph->y_control = glyph->x_control + x_count;
308 
309     mask = 0;
310     x    = 0;
311 
312     for ( i = 0; i < count; i++ )
313     {
314       if ( ( i & 7 ) == 0 )
315       {
316         PFR_CHECK( 1 );
317         mask = PFR_NEXT_BYTE( p );
318       }
319 
320       if ( mask & 1 )
321       {
322         PFR_CHECK( 2 );
323         x = PFR_NEXT_SHORT( p );
324       }
325       else
326       {
327         PFR_CHECK( 1 );
328         x += PFR_NEXT_BYTE( p );
329       }
330 
331       glyph->x_control[i] = x;
332 
333       mask >>= 1;
334     }
335 
336     /* XXX: we ignore the secondary stroke and edge definitions */
337     /*      since we don't support native PFR hinting           */
338     /*                                                          */
339     if ( flags & PFR_GLYPH_SINGLE_EXTRA_ITEMS )
340     {
341       error = pfr_extra_items_skip( &p, limit );
342       if ( error )
343         goto Exit;
344     }
345 
346     pfr_glyph_start( glyph );
347 
348     /* now load a simple glyph */
349     {
350       FT_Vector   pos[4];
351       FT_Vector*  cur;
352 
353 
354       pos[0].x = pos[0].y = 0;
355       pos[3]   = pos[0];
356 
357       for (;;)
358       {
359         FT_UInt  format, format_low, args_format = 0, args_count, n;
360 
361 
362         /****************************************************************
363          * read instruction
364          */
365         PFR_CHECK( 1 );
366         format     = PFR_NEXT_BYTE( p );
367         format_low = format & 15;
368 
369         switch ( format >> 4 )
370         {
371         case 0:                                               /* end glyph */
372           FT_TRACE6(( "- end glyph" ));
373           args_count = 0;
374           break;
375 
376         case 1:                                  /* general line operation */
377           FT_TRACE6(( "- general line" ));
378           goto Line1;
379 
380         case 4:                                 /* move to inside contour  */
381           FT_TRACE6(( "- move to inside" ));
382           goto Line1;
383 
384         case 5:                                 /* move to outside contour */
385           FT_TRACE6(( "- move to outside" ));
386         Line1:
387           args_format = format_low;
388           args_count  = 1;
389           break;
390 
391         case 2:                                      /* horizontal line to */
392           FT_TRACE6(( "- horizontal line to cx.%d", format_low ));
393           if ( format_low >= x_count )
394             goto Failure;
395           pos[0].x   = glyph->x_control[format_low];
396           pos[0].y   = pos[3].y;
397           pos[3]     = pos[0];
398           args_count = 0;
399           break;
400 
401         case 3:                                        /* vertical line to */
402           FT_TRACE6(( "- vertical line to cy.%d", format_low ));
403           if ( format_low >= y_count )
404             goto Failure;
405           pos[0].x   = pos[3].x;
406           pos[0].y   = glyph->y_control[format_low];
407           pos[3]     = pos[0];
408           args_count = 0;
409           break;
410 
411         case 6:                            /* horizontal to vertical curve */
412           FT_TRACE6(( "- hv curve " ));
413           args_format = 0xB8E;
414           args_count  = 3;
415           break;
416 
417         case 7:                            /* vertical to horizontal curve */
418           FT_TRACE6(( "- vh curve" ));
419           args_format = 0xE2B;
420           args_count  = 3;
421           break;
422 
423         default:                                       /* general curve to */
424           FT_TRACE6(( "- general curve" ));
425           args_count  = 4;
426           args_format = format_low;
427         }
428 
429         /************************************************************
430          * now read arguments
431          */
432         cur = pos;
433         for ( n = 0; n < args_count; n++ )
434         {
435           FT_UInt  idx;
436           FT_Int   delta;
437 
438 
439           /* read the X argument */
440           switch ( args_format & 3 )
441           {
442           case 0:                           /* 8-bit index */
443             PFR_CHECK( 1 );
444             idx = PFR_NEXT_BYTE( p );
445             if ( idx >= x_count )
446               goto Failure;
447             cur->x = glyph->x_control[idx];
448             FT_TRACE7(( " cx#%d", idx ));
449             break;
450 
451           case 1:                           /* 16-bit absolute value */
452             PFR_CHECK( 2 );
453             cur->x = PFR_NEXT_SHORT( p );
454             FT_TRACE7(( " x.%d", cur->x ));
455             break;
456 
457           case 2:                           /* 8-bit delta */
458             PFR_CHECK( 1 );
459             delta  = PFR_NEXT_INT8( p );
460             cur->x = pos[3].x + delta;
461             FT_TRACE7(( " dx.%d", delta ));
462             break;
463 
464           default:
465             FT_TRACE7(( " |" ));
466             cur->x = pos[3].x;
467           }
468 
469           /* read the Y argument */
470           switch ( ( args_format >> 2 ) & 3 )
471           {
472           case 0:                           /* 8-bit index */
473             PFR_CHECK( 1 );
474             idx  = PFR_NEXT_BYTE( p );
475             if ( idx >= y_count )
476               goto Failure;
477             cur->y = glyph->y_control[idx];
478             FT_TRACE7(( " cy#%d", idx ));
479             break;
480 
481           case 1:                           /* 16-bit absolute value */
482             PFR_CHECK( 2 );
483             cur->y = PFR_NEXT_SHORT( p );
484             FT_TRACE7(( " y.%d", cur->y ));
485             break;
486 
487           case 2:                           /* 8-bit delta */
488             PFR_CHECK( 1 );
489             delta  = PFR_NEXT_INT8( p );
490             cur->y = pos[3].y + delta;
491             FT_TRACE7(( " dy.%d", delta ));
492             break;
493 
494           default:
495             FT_TRACE7(( " -" ));
496             cur->y = pos[3].y;
497           }
498 
499           /* read the additional format flag for the general curve */
500           if ( n == 0 && args_count == 4 )
501           {
502             PFR_CHECK( 1 );
503             args_format = PFR_NEXT_BYTE( p );
504             args_count--;
505           }
506           else
507             args_format >>= 4;
508 
509           /* save the previous point */
510           pos[3] = cur[0];
511           cur++;
512         }
513 
514         FT_TRACE7(( "\n" ));
515 
516         /************************************************************
517          * finally, execute instruction
518          */
519         switch ( format >> 4 )
520         {
521         case 0:                                       /* end glyph => EXIT */
522           pfr_glyph_end( glyph );
523           goto Exit;
524 
525         case 1:                                         /* line operations */
526         case 2:
527         case 3:
528           error = pfr_glyph_line_to( glyph, pos );
529           goto Test_Error;
530 
531         case 4:                                 /* move to inside contour  */
532         case 5:                                 /* move to outside contour */
533           error = pfr_glyph_move_to( glyph, pos );
534           goto Test_Error;
535 
536         default:                                       /* curve operations */
537           error = pfr_glyph_curve_to( glyph, pos, pos + 1, pos + 2 );
538 
539         Test_Error:  /* test error condition */
540           if ( error )
541             goto Exit;
542         }
543       } /* for (;;) */
544     }
545 
546   Exit:
547     return error;
548 
549   Failure:
550   Too_Short:
551     error = FT_THROW( Invalid_Table );
552     FT_ERROR(( "pfr_glyph_load_simple: invalid glyph data\n" ));
553     goto Exit;
554   }
555 
556 
557   /* load a composite/compound glyph */
558   static FT_Error
pfr_glyph_load_compound(PFR_Glyph glyph,FT_Byte * p,FT_Byte * limit)559   pfr_glyph_load_compound( PFR_Glyph  glyph,
560                            FT_Byte*   p,
561                            FT_Byte*   limit )
562   {
563     FT_Error        error  = FT_Err_Ok;
564     FT_GlyphLoader  loader = glyph->loader;
565     FT_Memory       memory = loader->memory;
566     PFR_SubGlyph    subglyph;
567     FT_UInt         flags, i, count, org_count;
568     FT_Int          x_pos, y_pos;
569 
570 
571     PFR_CHECK( 1 );
572     flags = PFR_NEXT_BYTE( p );
573 
574     /* test for composite glyphs */
575     if ( !( flags & PFR_GLYPH_IS_COMPOUND ) )
576       goto Failure;
577 
578     count = flags & 0x3F;
579 
580     /* ignore extra items when present */
581     /*                                 */
582     if ( flags & PFR_GLYPH_COMPOUND_EXTRA_ITEMS )
583     {
584       error = pfr_extra_items_skip( &p, limit );
585       if ( error )
586         goto Exit;
587     }
588 
589     /* we can't rely on the FT_GlyphLoader to load sub-glyphs, because   */
590     /* the PFR format is dumb, using direct file offsets to point to the */
591     /* sub-glyphs (instead of glyph indices).  Sigh.                     */
592     /*                                                                   */
593     /* For now, we load the list of sub-glyphs into a different array    */
594     /* but this will prevent us from using the auto-hinter at its best   */
595     /* quality.                                                          */
596     /*                                                                   */
597     org_count = glyph->num_subs;
598 
599     if ( org_count + count > glyph->max_subs )
600     {
601       FT_UInt  new_max = ( org_count + count + 3 ) & (FT_UInt)-4;
602 
603 
604       /* we arbitrarily limit the number of subglyphs */
605       /* to avoid endless recursion                   */
606       if ( new_max > 64 )
607       {
608         error = FT_THROW( Invalid_Table );
609         FT_ERROR(( "pfr_glyph_load_compound:"
610                    " too many compound glyphs components\n" ));
611         goto Exit;
612       }
613 
614       if ( FT_RENEW_ARRAY( glyph->subs, glyph->max_subs, new_max ) )
615         goto Exit;
616 
617       glyph->max_subs = new_max;
618     }
619 
620     subglyph = glyph->subs + org_count;
621 
622     for ( i = 0; i < count; i++, subglyph++ )
623     {
624       FT_UInt  format;
625 
626 
627       x_pos = 0;
628       y_pos = 0;
629 
630       PFR_CHECK( 1 );
631       format = PFR_NEXT_BYTE( p );
632 
633       /* read scale when available */
634       subglyph->x_scale = 0x10000L;
635       if ( format & PFR_SUBGLYPH_XSCALE )
636       {
637         PFR_CHECK( 2 );
638         subglyph->x_scale = PFR_NEXT_SHORT( p ) * 16;
639       }
640 
641       subglyph->y_scale = 0x10000L;
642       if ( format & PFR_SUBGLYPH_YSCALE )
643       {
644         PFR_CHECK( 2 );
645         subglyph->y_scale = PFR_NEXT_SHORT( p ) * 16;
646       }
647 
648       /* read offset */
649       switch ( format & 3 )
650       {
651       case 1:
652         PFR_CHECK( 2 );
653         x_pos = PFR_NEXT_SHORT( p );
654         break;
655 
656       case 2:
657         PFR_CHECK( 1 );
658         x_pos += PFR_NEXT_INT8( p );
659         break;
660 
661       default:
662         ;
663       }
664 
665       switch ( ( format >> 2 ) & 3 )
666       {
667       case 1:
668         PFR_CHECK( 2 );
669         y_pos = PFR_NEXT_SHORT( p );
670         break;
671 
672       case 2:
673         PFR_CHECK( 1 );
674         y_pos += PFR_NEXT_INT8( p );
675         break;
676 
677       default:
678         ;
679       }
680 
681       subglyph->x_delta = x_pos;
682       subglyph->y_delta = y_pos;
683 
684       /* read glyph position and size now */
685       if ( format & PFR_SUBGLYPH_2BYTE_SIZE )
686       {
687         PFR_CHECK( 2 );
688         subglyph->gps_size = PFR_NEXT_USHORT( p );
689       }
690       else
691       {
692         PFR_CHECK( 1 );
693         subglyph->gps_size = PFR_NEXT_BYTE( p );
694       }
695 
696       if ( format & PFR_SUBGLYPH_3BYTE_OFFSET )
697       {
698         PFR_CHECK( 3 );
699         subglyph->gps_offset = PFR_NEXT_ULONG( p );
700       }
701       else
702       {
703         PFR_CHECK( 2 );
704         subglyph->gps_offset = PFR_NEXT_USHORT( p );
705       }
706 
707       glyph->num_subs++;
708     }
709 
710   Exit:
711     return error;
712 
713   Failure:
714   Too_Short:
715     error = FT_THROW( Invalid_Table );
716     FT_ERROR(( "pfr_glyph_load_compound: invalid glyph data\n" ));
717     goto Exit;
718   }
719 
720 
721   static FT_Error
pfr_glyph_load_rec(PFR_Glyph glyph,FT_Stream stream,FT_ULong gps_offset,FT_ULong offset,FT_ULong size)722   pfr_glyph_load_rec( PFR_Glyph  glyph,
723                       FT_Stream  stream,
724                       FT_ULong   gps_offset,
725                       FT_ULong   offset,
726                       FT_ULong   size )
727   {
728     FT_Error  error;
729     FT_Byte*  p;
730     FT_Byte*  limit;
731 
732 
733     if ( FT_STREAM_SEEK( gps_offset + offset ) ||
734          FT_FRAME_ENTER( size )                )
735       goto Exit;
736 
737     p     = (FT_Byte*)stream->cursor;
738     limit = p + size;
739 
740     if ( size > 0 && *p & PFR_GLYPH_IS_COMPOUND )
741     {
742       FT_UInt         n, old_count, count;
743       FT_GlyphLoader  loader = glyph->loader;
744       FT_Outline*     base   = &loader->base.outline;
745 
746 
747       old_count = glyph->num_subs;
748 
749       /* this is a compound glyph - load it */
750       error = pfr_glyph_load_compound( glyph, p, limit );
751 
752       FT_FRAME_EXIT();
753 
754       if ( error )
755         goto Exit;
756 
757       count = glyph->num_subs - old_count;
758 
759       FT_TRACE4(( "compound glyph with %d element%s (offset %lu):\n",
760                   count,
761                   count == 1 ? "" : "s",
762                   offset ));
763 
764       /* now, load each individual glyph */
765       for ( n = 0; n < count; n++ )
766       {
767         FT_Int        i, old_points, num_points;
768         PFR_SubGlyph  subglyph;
769 
770 
771         FT_TRACE4(( "  subglyph %d:\n", n ));
772 
773         subglyph   = glyph->subs + old_count + n;
774         old_points = base->n_points;
775 
776         error = pfr_glyph_load_rec( glyph, stream, gps_offset,
777                                     subglyph->gps_offset,
778                                     subglyph->gps_size );
779         if ( error )
780           break;
781 
782         /* note that `glyph->subs' might have been re-allocated */
783         subglyph   = glyph->subs + old_count + n;
784         num_points = base->n_points - old_points;
785 
786         /* translate and eventually scale the new glyph points */
787         if ( subglyph->x_scale != 0x10000L || subglyph->y_scale != 0x10000L )
788         {
789           FT_Vector*  vec = base->points + old_points;
790 
791 
792           for ( i = 0; i < num_points; i++, vec++ )
793           {
794             vec->x = FT_MulFix( vec->x, subglyph->x_scale ) +
795                        subglyph->x_delta;
796             vec->y = FT_MulFix( vec->y, subglyph->y_scale ) +
797                        subglyph->y_delta;
798           }
799         }
800         else
801         {
802           FT_Vector*  vec = loader->base.outline.points + old_points;
803 
804 
805           for ( i = 0; i < num_points; i++, vec++ )
806           {
807             vec->x += subglyph->x_delta;
808             vec->y += subglyph->y_delta;
809           }
810         }
811 
812         /* proceed to next sub-glyph */
813       }
814 
815       FT_TRACE4(( "end compound glyph with %d element%s\n",
816                   count,
817                   count == 1 ? "" : "s" ));
818     }
819     else
820     {
821       FT_TRACE4(( "simple glyph (offset %lu)\n", offset ));
822 
823       /* load a simple glyph */
824       error = pfr_glyph_load_simple( glyph, p, limit );
825 
826       FT_FRAME_EXIT();
827     }
828 
829   Exit:
830     return error;
831   }
832 
833 
834   FT_LOCAL_DEF( FT_Error )
pfr_glyph_load(PFR_Glyph glyph,FT_Stream stream,FT_ULong gps_offset,FT_ULong offset,FT_ULong size)835   pfr_glyph_load( PFR_Glyph  glyph,
836                   FT_Stream  stream,
837                   FT_ULong   gps_offset,
838                   FT_ULong   offset,
839                   FT_ULong   size )
840   {
841     /* initialize glyph loader */
842     FT_GlyphLoader_Rewind( glyph->loader );
843 
844     glyph->num_subs = 0;
845 
846     /* load the glyph, recursively when needed */
847     return pfr_glyph_load_rec( glyph, stream, gps_offset, offset, size );
848   }
849 
850 
851 /* END */
852