• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /***************************************************************************/
2 /*                                                                         */
3 /*  ftraster.c                                                             */
4 /*                                                                         */
5 /*    The FreeType glyph rasterizer (body).                                */
6 /*                                                                         */
7 /*  Copyright 1996-2015 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   /*                                                                       */
20   /* This file can be compiled without the rest of the FreeType engine, by */
21   /* defining the _STANDALONE_ macro when compiling it.  You also need to  */
22   /* put the files `ftimage.h' and `ftmisc.h' into the $(incdir)           */
23   /* directory.  Typically, you should do something like                   */
24   /*                                                                       */
25   /* - copy `src/raster/ftraster.c' (this file) to your current directory  */
26   /*                                                                       */
27   /* - copy `include/ftimage.h' and `src/raster/ftmisc.h' to your current  */
28   /*   directory                                                           */
29   /*                                                                       */
30   /* - compile `ftraster' with the _STANDALONE_ macro defined, as in       */
31   /*                                                                       */
32   /*     cc -c -D_STANDALONE_ ftraster.c                                   */
33   /*                                                                       */
34   /* The renderer can be initialized with a call to                        */
35   /* `ft_standard_raster.raster_new'; a bitmap can be generated            */
36   /* with a call to `ft_standard_raster.raster_render'.                    */
37   /*                                                                       */
38   /* See the comments and documentation in the file `ftimage.h' for more   */
39   /* details on how the raster works.                                      */
40   /*                                                                       */
41   /*************************************************************************/
42 
43 
44   /*************************************************************************/
45   /*                                                                       */
46   /* This is a rewrite of the FreeType 1.x scan-line converter             */
47   /*                                                                       */
48   /*************************************************************************/
49 
50 #ifdef _STANDALONE_
51 
52   /* The size in bytes of the render pool used by the scan-line converter  */
53   /* to do all of its work.                                                */
54 #define FT_RENDER_POOL_SIZE  16384L
55 
56 #define FT_CONFIG_STANDARD_LIBRARY_H  <stdlib.h>
57 
58 #include <string.h>           /* for memset */
59 
60 #include "ftmisc.h"
61 #include "ftimage.h"
62 
63 #else /* !_STANDALONE_ */
64 
65 #include <ft2build.h>
66 #include "ftraster.h"
67 #include FT_INTERNAL_CALC_H   /* for FT_MulDiv and FT_MulDiv_No_Round */
68 
69 #include "rastpic.h"
70 
71 #endif /* !_STANDALONE_ */
72 
73 
74   /*************************************************************************/
75   /*                                                                       */
76   /* A simple technical note on how the raster works                       */
77   /* -----------------------------------------------                       */
78   /*                                                                       */
79   /*   Converting an outline into a bitmap is achieved in several steps:   */
80   /*                                                                       */
81   /*   1 - Decomposing the outline into successive `profiles'.  Each       */
82   /*       profile is simply an array of scanline intersections on a given */
83   /*       dimension.  A profile's main attributes are                     */
84   /*                                                                       */
85   /*       o its scanline position boundaries, i.e. `Ymin' and `Ymax'      */
86   /*                                                                       */
87   /*       o an array of intersection coordinates for each scanline        */
88   /*         between `Ymin' and `Ymax'                                     */
89   /*                                                                       */
90   /*       o a direction, indicating whether it was built going `up' or    */
91   /*         `down', as this is very important for filling rules           */
92   /*                                                                       */
93   /*       o its drop-out mode                                             */
94   /*                                                                       */
95   /*   2 - Sweeping the target map's scanlines in order to compute segment */
96   /*       `spans' which are then filled.  Additionally, this pass         */
97   /*       performs drop-out control.                                      */
98   /*                                                                       */
99   /*   The outline data is parsed during step 1 only.  The profiles are    */
100   /*   built from the bottom of the render pool, used as a stack.  The     */
101   /*   following graphics shows the profile list under construction:       */
102   /*                                                                       */
103   /*     __________________________________________________________ _ _    */
104   /*    |         |                 |         |                 |          */
105   /*    | profile | coordinates for | profile | coordinates for |-->       */
106   /*    |    1    |  profile 1      |    2    |  profile 2      |-->       */
107   /*    |_________|_________________|_________|_________________|__ _ _    */
108   /*                                                                       */
109   /*    ^                                                       ^          */
110   /*    |                                                       |          */
111   /* start of render pool                                      top         */
112   /*                                                                       */
113   /*   The top of the profile stack is kept in the `top' variable.         */
114   /*                                                                       */
115   /*   As you can see, a profile record is pushed on top of the render     */
116   /*   pool, which is then followed by its coordinates/intersections.  If  */
117   /*   a change of direction is detected in the outline, a new profile is  */
118   /*   generated until the end of the outline.                             */
119   /*                                                                       */
120   /*   Note that when all profiles have been generated, the function       */
121   /*   Finalize_Profile_Table() is used to record, for each profile, its   */
122   /*   bottom-most scanline as well as the scanline above its upmost       */
123   /*   boundary.  These positions are called `y-turns' because they (sort  */
124   /*   of) correspond to local extrema.  They are stored in a sorted list  */
125   /*   built from the top of the render pool as a downwards stack:         */
126   /*                                                                       */
127   /*      _ _ _______________________________________                      */
128   /*                            |                    |                     */
129   /*                         <--| sorted list of     |                     */
130   /*                         <--|  extrema scanlines |                     */
131   /*      _ _ __________________|____________________|                     */
132   /*                                                                       */
133   /*                            ^                    ^                     */
134   /*                            |                    |                     */
135   /*                         maxBuff           sizeBuff = end of pool      */
136   /*                                                                       */
137   /*   This list is later used during the sweep phase in order to          */
138   /*   optimize performance (see technical note on the sweep below).       */
139   /*                                                                       */
140   /*   Of course, the raster detects whether the two stacks collide and    */
141   /*   handles the situation properly.                                     */
142   /*                                                                       */
143   /*************************************************************************/
144 
145 
146   /*************************************************************************/
147   /*************************************************************************/
148   /**                                                                     **/
149   /**  CONFIGURATION MACROS                                               **/
150   /**                                                                     **/
151   /*************************************************************************/
152   /*************************************************************************/
153 
154   /* define DEBUG_RASTER if you want to compile a debugging version */
155 /* #define DEBUG_RASTER */
156 
157 
158   /*************************************************************************/
159   /*************************************************************************/
160   /**                                                                     **/
161   /**  OTHER MACROS (do not change)                                       **/
162   /**                                                                     **/
163   /*************************************************************************/
164   /*************************************************************************/
165 
166   /*************************************************************************/
167   /*                                                                       */
168   /* The macro FT_COMPONENT is used in trace mode.  It is an implicit      */
169   /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log  */
170   /* messages during execution.                                            */
171   /*                                                                       */
172 #undef  FT_COMPONENT
173 #define FT_COMPONENT  trace_raster
174 
175 
176 #ifdef _STANDALONE_
177 
178   /* Auxiliary macros for token concatenation. */
179 #define FT_ERR_XCAT( x, y )  x ## y
180 #define FT_ERR_CAT( x, y )   FT_ERR_XCAT( x, y )
181 
182 #define FT_MAX( a, b )  ( (a) > (b) ? (a) : (b) )
183 
184   /* This macro is used to indicate that a function parameter is unused. */
185   /* Its purpose is simply to reduce compiler warnings.  Note also that  */
186   /* simply defining it as `(void)x' doesn't avoid warnings with certain */
187   /* ANSI compilers (e.g. LCC).                                          */
188 #define FT_UNUSED( x )  (x) = (x)
189 
190   /* Disable the tracing mechanism for simplicity -- developers can      */
191   /* activate it easily by redefining these macros.                      */
192 #ifndef FT_ERROR
193 #define FT_ERROR( x )  do { } while ( 0 )     /* nothing */
194 #endif
195 
196 #ifndef FT_TRACE
197 #define FT_TRACE( x )   do { } while ( 0 )    /* nothing */
198 #define FT_TRACE1( x )  do { } while ( 0 )    /* nothing */
199 #define FT_TRACE6( x )  do { } while ( 0 )    /* nothing */
200 #define FT_TRACE7( x )  do { } while ( 0 )    /* nothing */
201 #endif
202 
203 #ifndef FT_THROW
204 #define FT_THROW( e )  FT_ERR_CAT( Raster_Err_, e )
205 #endif
206 
207 #define Raster_Err_None          0
208 #define Raster_Err_Not_Ini      -1
209 #define Raster_Err_Overflow     -2
210 #define Raster_Err_Neg_Height   -3
211 #define Raster_Err_Invalid      -4
212 #define Raster_Err_Unsupported  -5
213 
214 #define ft_memset  memset
215 
216 #define FT_DEFINE_RASTER_FUNCS( class_, glyph_format_, raster_new_, \
217                                 raster_reset_, raster_set_mode_,    \
218                                 raster_render_, raster_done_ )      \
219           const FT_Raster_Funcs class_ =                            \
220           {                                                         \
221             glyph_format_,                                          \
222             raster_new_,                                            \
223             raster_reset_,                                          \
224             raster_set_mode_,                                       \
225             raster_render_,                                         \
226             raster_done_                                            \
227          };
228 
229 #else /* !_STANDALONE_ */
230 
231 
232 #include FT_INTERNAL_OBJECTS_H
233 #include FT_INTERNAL_DEBUG_H       /* for FT_TRACE, FT_ERROR, and FT_THROW */
234 
235 #include "rasterrs.h"
236 
237 #define Raster_Err_None         FT_Err_Ok
238 #define Raster_Err_Not_Ini      Raster_Err_Raster_Uninitialized
239 #define Raster_Err_Overflow     Raster_Err_Raster_Overflow
240 #define Raster_Err_Neg_Height   Raster_Err_Raster_Negative_Height
241 #define Raster_Err_Invalid      Raster_Err_Invalid_Outline
242 #define Raster_Err_Unsupported  Raster_Err_Cannot_Render_Glyph
243 
244 
245 #endif /* !_STANDALONE_ */
246 
247 
248 #ifndef FT_MEM_SET
249 #define FT_MEM_SET( d, s, c )  ft_memset( d, s, c )
250 #endif
251 
252 #ifndef FT_MEM_ZERO
253 #define FT_MEM_ZERO( dest, count )  FT_MEM_SET( dest, 0, count )
254 #endif
255 
256   /* FMulDiv means `Fast MulDiv'; it is used in case where `b' is       */
257   /* typically a small value and the result of a*b is known to fit into */
258   /* 32 bits.                                                           */
259 #define FMulDiv( a, b, c )  ( (a) * (b) / (c) )
260 
261   /* On the other hand, SMulDiv means `Slow MulDiv', and is used typically */
262   /* for clipping computations.  It simply uses the FT_MulDiv() function   */
263   /* defined in `ftcalc.h'.                                                */
264 #define SMulDiv           FT_MulDiv
265 #define SMulDiv_No_Round  FT_MulDiv_No_Round
266 
267   /* The rasterizer is a very general purpose component; please leave */
268   /* the following redefinitions there (you never know your target    */
269   /* environment).                                                    */
270 
271 #ifndef TRUE
272 #define TRUE   1
273 #endif
274 
275 #ifndef FALSE
276 #define FALSE  0
277 #endif
278 
279 #ifndef NULL
280 #define NULL  (void*)0
281 #endif
282 
283 #ifndef SUCCESS
284 #define SUCCESS  0
285 #endif
286 
287 #ifndef FAILURE
288 #define FAILURE  1
289 #endif
290 
291 
292 #define MaxBezier  32   /* The maximum number of stacked Bezier curves. */
293                         /* Setting this constant to more than 32 is a   */
294                         /* pure waste of space.                         */
295 
296 #define Pixel_Bits  6   /* fractional bits of *input* coordinates */
297 
298 
299   /*************************************************************************/
300   /*************************************************************************/
301   /**                                                                     **/
302   /**  SIMPLE TYPE DECLARATIONS                                           **/
303   /**                                                                     **/
304   /*************************************************************************/
305   /*************************************************************************/
306 
307   typedef int             Int;
308   typedef unsigned int    UInt;
309   typedef short           Short;
310   typedef unsigned short  UShort, *PUShort;
311   typedef long            Long, *PLong;
312   typedef unsigned long   ULong;
313 
314   typedef unsigned char   Byte, *PByte;
315   typedef char            Bool;
316 
317 
318   typedef union  Alignment_
319   {
320     Long    l;
321     void*   p;
322     void  (*f)(void);
323 
324   } Alignment, *PAlignment;
325 
326 
327   typedef struct  TPoint_
328   {
329     Long  x;
330     Long  y;
331 
332   } TPoint;
333 
334 
335   /* values for the `flags' bit field */
336 #define Flow_Up           0x08U
337 #define Overshoot_Top     0x10U
338 #define Overshoot_Bottom  0x20U
339 
340 
341   /* States of each line, arc, and profile */
342   typedef enum  TStates_
343   {
344     Unknown_State,
345     Ascending_State,
346     Descending_State,
347     Flat_State
348 
349   } TStates;
350 
351 
352   typedef struct TProfile_  TProfile;
353   typedef TProfile*         PProfile;
354 
355   struct  TProfile_
356   {
357     FT_F26Dot6  X;           /* current coordinate during sweep          */
358     PProfile    link;        /* link to next profile (various purposes)  */
359     PLong       offset;      /* start of profile's data in render pool   */
360     UShort      flags;       /* Bit 0-2: drop-out mode                   */
361                              /* Bit 3: profile orientation (up/down)     */
362                              /* Bit 4: is top profile?                   */
363                              /* Bit 5: is bottom profile?                */
364     Long        height;      /* profile's height in scanlines            */
365     Long        start;       /* profile's starting scanline              */
366 
367     Int         countL;      /* number of lines to step before this      */
368                              /* profile becomes drawable                 */
369 
370     PProfile    next;        /* next profile in same contour, used       */
371                              /* during drop-out control                  */
372   };
373 
374   typedef PProfile   TProfileList;
375   typedef PProfile*  PProfileList;
376 
377 
378   /* Simple record used to implement a stack of bands, required */
379   /* by the sub-banding mechanism                               */
380   typedef struct  black_TBand_
381   {
382     Short  y_min;   /* band's minimum */
383     Short  y_max;   /* band's maximum */
384 
385   } black_TBand;
386 
387 
388 #define AlignProfileSize \
389   ( ( sizeof ( TProfile ) + sizeof ( Alignment ) - 1 ) / sizeof ( Long ) )
390 
391 
392 #undef RAS_ARG
393 #undef RAS_ARGS
394 #undef RAS_VAR
395 #undef RAS_VARS
396 
397 #ifdef FT_STATIC_RASTER
398 
399 
400 #define RAS_ARGS       /* void */
401 #define RAS_ARG        /* void */
402 
403 #define RAS_VARS       /* void */
404 #define RAS_VAR        /* void */
405 
406 #define FT_UNUSED_RASTER  do { } while ( 0 )
407 
408 
409 #else /* !FT_STATIC_RASTER */
410 
411 
412 #define RAS_ARGS       black_PWorker  worker,
413 #define RAS_ARG        black_PWorker  worker
414 
415 #define RAS_VARS       worker,
416 #define RAS_VAR        worker
417 
418 #define FT_UNUSED_RASTER  FT_UNUSED( worker )
419 
420 
421 #endif /* !FT_STATIC_RASTER */
422 
423 
424   typedef struct black_TWorker_  black_TWorker, *black_PWorker;
425 
426 
427   /* prototypes used for sweep function dispatch */
428   typedef void
429   Function_Sweep_Init( RAS_ARGS Short*  min,
430                                 Short*  max );
431 
432   typedef void
433   Function_Sweep_Span( RAS_ARGS Short       y,
434                                 FT_F26Dot6  x1,
435                                 FT_F26Dot6  x2,
436                                 PProfile    left,
437                                 PProfile    right );
438 
439   typedef void
440   Function_Sweep_Step( RAS_ARG );
441 
442 
443   /* NOTE: These operations are only valid on 2's complement processors */
444 #undef FLOOR
445 #undef CEILING
446 #undef TRUNC
447 #undef SCALED
448 
449 #define FLOOR( x )    ( (x) & -ras.precision )
450 #define CEILING( x )  ( ( (x) + ras.precision - 1 ) & -ras.precision )
451 #define TRUNC( x )    ( (Long)(x) >> ras.precision_bits )
452 #define FRAC( x )     ( (x) & ( ras.precision - 1 ) )
453 #define SCALED( x )   ( ( (Long)(x) << ras.scale_shift ) - ras.precision_half )
454 
455 #define IS_BOTTOM_OVERSHOOT( x ) \
456           (Bool)( CEILING( x ) - x >= ras.precision_half )
457 #define IS_TOP_OVERSHOOT( x )    \
458           (Bool)( x - FLOOR( x ) >= ras.precision_half )
459 
460   /* The most used variables are positioned at the top of the structure. */
461   /* Thus, their offset can be coded with less opcodes, resulting in a   */
462   /* smaller executable.                                                 */
463 
464   struct  black_TWorker_
465   {
466     Int         precision_bits;     /* precision related variables         */
467     Int         precision;
468     Int         precision_half;
469     Int         precision_shift;
470     Int         precision_step;
471     Int         precision_jitter;
472 
473     Int         scale_shift;        /* == precision_shift   for bitmaps    */
474                                     /* == precision_shift+1 for pixmaps    */
475 
476     PLong       buff;               /* The profiles buffer                 */
477     PLong       sizeBuff;           /* Render pool size                    */
478     PLong       maxBuff;            /* Profiles buffer size                */
479     PLong       top;                /* Current cursor in buffer            */
480 
481     FT_Error    error;
482 
483     Int         numTurns;           /* number of Y-turns in outline        */
484 
485     TPoint*     arc;                /* current Bezier arc pointer          */
486 
487     UShort      bWidth;             /* target bitmap width                 */
488     PByte       bTarget;            /* target bitmap buffer                */
489     PByte       gTarget;            /* target pixmap buffer                */
490 
491     Long        lastX, lastY;
492     Long        minY, maxY;
493 
494     UShort      num_Profs;          /* current number of profiles          */
495 
496     Bool        fresh;              /* signals a fresh new profile which   */
497                                     /* `start' field must be completed     */
498     Bool        joint;              /* signals that the last arc ended     */
499                                     /* exactly on a scanline.  Allows      */
500                                     /* removal of doublets                 */
501     PProfile    cProfile;           /* current profile                     */
502     PProfile    fProfile;           /* head of linked list of profiles     */
503     PProfile    gProfile;           /* contour's first profile in case     */
504                                     /* of impact                           */
505 
506     TStates     state;              /* rendering state                     */
507 
508     FT_Bitmap   target;             /* description of target bit/pixmap    */
509     FT_Outline  outline;
510 
511     Long        traceOfs;           /* current offset in target bitmap     */
512     Long        traceG;             /* current offset in target pixmap     */
513 
514     Short       traceIncr;          /* sweep's increment in target bitmap  */
515 
516     /* dispatch variables */
517 
518     Function_Sweep_Init*  Proc_Sweep_Init;
519     Function_Sweep_Span*  Proc_Sweep_Span;
520     Function_Sweep_Span*  Proc_Sweep_Drop;
521     Function_Sweep_Step*  Proc_Sweep_Step;
522 
523     Byte        dropOutControl;     /* current drop_out control method     */
524 
525     Bool        second_pass;        /* indicates whether a horizontal pass */
526                                     /* should be performed to control      */
527                                     /* drop-out accurately when calling    */
528                                     /* Render_Glyph.                       */
529 
530     TPoint      arcs[3 * MaxBezier + 1]; /* The Bezier stack               */
531 
532     black_TBand  band_stack[16];    /* band stack used for sub-banding     */
533     Int          band_top;          /* band stack top                      */
534 
535   };
536 
537 
538   typedef struct  black_TRaster_
539   {
540     void*          memory;
541 
542   } black_TRaster, *black_PRaster;
543 
544 #ifdef FT_STATIC_RASTER
545 
546   static black_TWorker  cur_ras;
547 #define ras  cur_ras
548 
549 #else /* !FT_STATIC_RASTER */
550 
551 #define ras  (*worker)
552 
553 #endif /* !FT_STATIC_RASTER */
554 
555 
556   /*************************************************************************/
557   /*************************************************************************/
558   /**                                                                     **/
559   /**  PROFILES COMPUTATION                                               **/
560   /**                                                                     **/
561   /*************************************************************************/
562   /*************************************************************************/
563 
564 
565   /*************************************************************************/
566   /*                                                                       */
567   /* <Function>                                                            */
568   /*    Set_High_Precision                                                 */
569   /*                                                                       */
570   /* <Description>                                                         */
571   /*    Set precision variables according to param flag.                   */
572   /*                                                                       */
573   /* <Input>                                                               */
574   /*    High :: Set to True for high precision (typically for ppem < 24),  */
575   /*            false otherwise.                                           */
576   /*                                                                       */
577   static void
Set_High_Precision(RAS_ARGS Int High)578   Set_High_Precision( RAS_ARGS Int  High )
579   {
580     /*
581      * `precision_step' is used in `Bezier_Up' to decide when to split a
582      * given y-monotonous Bezier arc that crosses a scanline before
583      * approximating it as a straight segment.  The default value of 32 (for
584      * low accuracy) corresponds to
585      *
586      *   32 / 64 == 0.5 pixels,
587      *
588      * while for the high accuracy case we have
589      *
590      *   256 / (1 << 12) = 0.0625 pixels.
591      *
592      * `precision_jitter' is an epsilon threshold used in
593      * `Vertical_Sweep_Span' to deal with small imperfections in the Bezier
594      * decomposition (after all, we are working with approximations only);
595      * it avoids switching on additional pixels which would cause artifacts
596      * otherwise.
597      *
598      * The value of `precision_jitter' has been determined heuristically.
599      *
600      */
601 
602     if ( High )
603     {
604       ras.precision_bits   = 12;
605       ras.precision_step   = 256;
606       ras.precision_jitter = 30;
607     }
608     else
609     {
610       ras.precision_bits   = 6;
611       ras.precision_step   = 32;
612       ras.precision_jitter = 2;
613     }
614 
615     FT_TRACE6(( "Set_High_Precision(%s)\n", High ? "true" : "false" ));
616 
617     ras.precision       = 1 << ras.precision_bits;
618     ras.precision_half  = ras.precision / 2;
619     ras.precision_shift = ras.precision_bits - Pixel_Bits;
620   }
621 
622 
623   /*************************************************************************/
624   /*                                                                       */
625   /* <Function>                                                            */
626   /*    New_Profile                                                        */
627   /*                                                                       */
628   /* <Description>                                                         */
629   /*    Create a new profile in the render pool.                           */
630   /*                                                                       */
631   /* <Input>                                                               */
632   /*    aState    :: The state/orientation of the new profile.             */
633   /*                                                                       */
634   /*    overshoot :: Whether the profile's unrounded start position        */
635   /*                 differs by at least a half pixel.                     */
636   /*                                                                       */
637   /* <Return>                                                              */
638   /*   SUCCESS on success.  FAILURE in case of overflow or of incoherent   */
639   /*   profile.                                                            */
640   /*                                                                       */
641   static Bool
New_Profile(RAS_ARGS TStates aState,Bool overshoot)642   New_Profile( RAS_ARGS TStates  aState,
643                         Bool     overshoot )
644   {
645     if ( !ras.fProfile )
646     {
647       ras.cProfile  = (PProfile)ras.top;
648       ras.fProfile  = ras.cProfile;
649       ras.top      += AlignProfileSize;
650     }
651 
652     if ( ras.top >= ras.maxBuff )
653     {
654       ras.error = FT_THROW( Overflow );
655       return FAILURE;
656     }
657 
658     ras.cProfile->flags  = 0;
659     ras.cProfile->start  = 0;
660     ras.cProfile->height = 0;
661     ras.cProfile->offset = ras.top;
662     ras.cProfile->link   = (PProfile)0;
663     ras.cProfile->next   = (PProfile)0;
664     ras.cProfile->flags  = ras.dropOutControl;
665 
666     switch ( aState )
667     {
668     case Ascending_State:
669       ras.cProfile->flags |= Flow_Up;
670       if ( overshoot )
671         ras.cProfile->flags |= Overshoot_Bottom;
672 
673       FT_TRACE6(( "  new ascending profile = %p\n", ras.cProfile ));
674       break;
675 
676     case Descending_State:
677       if ( overshoot )
678         ras.cProfile->flags |= Overshoot_Top;
679       FT_TRACE6(( "  new descending profile = %p\n", ras.cProfile ));
680       break;
681 
682     default:
683       FT_ERROR(( "New_Profile: invalid profile direction\n" ));
684       ras.error = FT_THROW( Invalid );
685       return FAILURE;
686     }
687 
688     if ( !ras.gProfile )
689       ras.gProfile = ras.cProfile;
690 
691     ras.state = aState;
692     ras.fresh = TRUE;
693     ras.joint = FALSE;
694 
695     return SUCCESS;
696   }
697 
698 
699   /*************************************************************************/
700   /*                                                                       */
701   /* <Function>                                                            */
702   /*    End_Profile                                                        */
703   /*                                                                       */
704   /* <Description>                                                         */
705   /*    Finalize the current profile.                                      */
706   /*                                                                       */
707   /* <Input>                                                               */
708   /*    overshoot :: Whether the profile's unrounded end position differs  */
709   /*                 by at least a half pixel.                             */
710   /*                                                                       */
711   /* <Return>                                                              */
712   /*    SUCCESS on success.  FAILURE in case of overflow or incoherency.   */
713   /*                                                                       */
714   static Bool
End_Profile(RAS_ARGS Bool overshoot)715   End_Profile( RAS_ARGS Bool  overshoot )
716   {
717     Long  h;
718 
719 
720     h = (Long)( ras.top - ras.cProfile->offset );
721 
722     if ( h < 0 )
723     {
724       FT_ERROR(( "End_Profile: negative height encountered\n" ));
725       ras.error = FT_THROW( Neg_Height );
726       return FAILURE;
727     }
728 
729     if ( h > 0 )
730     {
731       PProfile  oldProfile;
732 
733 
734       FT_TRACE6(( "  ending profile %p, start = %ld, height = %ld\n",
735                   ras.cProfile, ras.cProfile->start, h ));
736 
737       ras.cProfile->height = h;
738       if ( overshoot )
739       {
740         if ( ras.cProfile->flags & Flow_Up )
741           ras.cProfile->flags |= Overshoot_Top;
742         else
743           ras.cProfile->flags |= Overshoot_Bottom;
744       }
745 
746       oldProfile   = ras.cProfile;
747       ras.cProfile = (PProfile)ras.top;
748 
749       ras.top += AlignProfileSize;
750 
751       ras.cProfile->height = 0;
752       ras.cProfile->offset = ras.top;
753 
754       oldProfile->next = ras.cProfile;
755       ras.num_Profs++;
756     }
757 
758     if ( ras.top >= ras.maxBuff )
759     {
760       FT_TRACE1(( "overflow in End_Profile\n" ));
761       ras.error = FT_THROW( Overflow );
762       return FAILURE;
763     }
764 
765     ras.joint = FALSE;
766 
767     return SUCCESS;
768   }
769 
770 
771   /*************************************************************************/
772   /*                                                                       */
773   /* <Function>                                                            */
774   /*    Insert_Y_Turn                                                      */
775   /*                                                                       */
776   /* <Description>                                                         */
777   /*    Insert a salient into the sorted list placed on top of the render  */
778   /*    pool.                                                              */
779   /*                                                                       */
780   /* <Input>                                                               */
781   /*    New y scanline position.                                           */
782   /*                                                                       */
783   /* <Return>                                                              */
784   /*    SUCCESS on success.  FAILURE in case of overflow.                  */
785   /*                                                                       */
786   static Bool
Insert_Y_Turn(RAS_ARGS Int y)787   Insert_Y_Turn( RAS_ARGS Int  y )
788   {
789     PLong  y_turns;
790     Int    n;
791 
792 
793     n       = ras.numTurns - 1;
794     y_turns = ras.sizeBuff - ras.numTurns;
795 
796     /* look for first y value that is <= */
797     while ( n >= 0 && y < y_turns[n] )
798       n--;
799 
800     /* if it is <, simply insert it, ignore if == */
801     if ( n >= 0 && y > y_turns[n] )
802       while ( n >= 0 )
803       {
804         Int  y2 = (Int)y_turns[n];
805 
806 
807         y_turns[n] = y;
808         y = y2;
809         n--;
810       }
811 
812     if ( n < 0 )
813     {
814       ras.maxBuff--;
815       if ( ras.maxBuff <= ras.top )
816       {
817         ras.error = FT_THROW( Overflow );
818         return FAILURE;
819       }
820       ras.numTurns++;
821       ras.sizeBuff[-ras.numTurns] = y;
822     }
823 
824     return SUCCESS;
825   }
826 
827 
828   /*************************************************************************/
829   /*                                                                       */
830   /* <Function>                                                            */
831   /*    Finalize_Profile_Table                                             */
832   /*                                                                       */
833   /* <Description>                                                         */
834   /*    Adjust all links in the profiles list.                             */
835   /*                                                                       */
836   /* <Return>                                                              */
837   /*    SUCCESS on success.  FAILURE in case of overflow.                  */
838   /*                                                                       */
839   static Bool
Finalize_Profile_Table(RAS_ARG)840   Finalize_Profile_Table( RAS_ARG )
841   {
842     UShort    n;
843     PProfile  p;
844 
845 
846     n = ras.num_Profs;
847     p = ras.fProfile;
848 
849     if ( n > 1 && p )
850     {
851       while ( n > 0 )
852       {
853         Int  bottom, top;
854 
855 
856         if ( n > 1 )
857           p->link = (PProfile)( p->offset + p->height );
858         else
859           p->link = NULL;
860 
861         if ( p->flags & Flow_Up )
862         {
863           bottom = (Int)p->start;
864           top    = (Int)( p->start + p->height - 1 );
865         }
866         else
867         {
868           bottom     = (Int)( p->start - p->height + 1 );
869           top        = (Int)p->start;
870           p->start   = bottom;
871           p->offset += p->height - 1;
872         }
873 
874         if ( Insert_Y_Turn( RAS_VARS bottom )  ||
875              Insert_Y_Turn( RAS_VARS top + 1 ) )
876           return FAILURE;
877 
878         p = p->link;
879         n--;
880       }
881     }
882     else
883       ras.fProfile = NULL;
884 
885     return SUCCESS;
886   }
887 
888 
889   /*************************************************************************/
890   /*                                                                       */
891   /* <Function>                                                            */
892   /*    Split_Conic                                                        */
893   /*                                                                       */
894   /* <Description>                                                         */
895   /*    Subdivide one conic Bezier into two joint sub-arcs in the Bezier   */
896   /*    stack.                                                             */
897   /*                                                                       */
898   /* <Input>                                                               */
899   /*    None (subdivided Bezier is taken from the top of the stack).       */
900   /*                                                                       */
901   /* <Note>                                                                */
902   /*    This routine is the `beef' of this component.  It is  _the_ inner  */
903   /*    loop that should be optimized to hell to get the best performance. */
904   /*                                                                       */
905   static void
Split_Conic(TPoint * base)906   Split_Conic( TPoint*  base )
907   {
908     Long  a, b;
909 
910 
911     base[4].x = base[2].x;
912     b = base[1].x;
913     a = base[3].x = ( base[2].x + b ) / 2;
914     b = base[1].x = ( base[0].x + b ) / 2;
915     base[2].x = ( a + b ) / 2;
916 
917     base[4].y = base[2].y;
918     b = base[1].y;
919     a = base[3].y = ( base[2].y + b ) / 2;
920     b = base[1].y = ( base[0].y + b ) / 2;
921     base[2].y = ( a + b ) / 2;
922 
923     /* hand optimized.  gcc doesn't seem to be too good at common      */
924     /* expression substitution and instruction scheduling ;-)          */
925   }
926 
927 
928   /*************************************************************************/
929   /*                                                                       */
930   /* <Function>                                                            */
931   /*    Split_Cubic                                                        */
932   /*                                                                       */
933   /* <Description>                                                         */
934   /*    Subdivide a third-order Bezier arc into two joint sub-arcs in the  */
935   /*    Bezier stack.                                                      */
936   /*                                                                       */
937   /* <Note>                                                                */
938   /*    This routine is the `beef' of the component.  It is one of _the_   */
939   /*    inner loops that should be optimized like hell to get the best     */
940   /*    performance.                                                       */
941   /*                                                                       */
942   static void
Split_Cubic(TPoint * base)943   Split_Cubic( TPoint*  base )
944   {
945     Long  a, b, c, d;
946 
947 
948     base[6].x = base[3].x;
949     c = base[1].x;
950     d = base[2].x;
951     base[1].x = a = ( base[0].x + c + 1 ) >> 1;
952     base[5].x = b = ( base[3].x + d + 1 ) >> 1;
953     c = ( c + d + 1 ) >> 1;
954     base[2].x = a = ( a + c + 1 ) >> 1;
955     base[4].x = b = ( b + c + 1 ) >> 1;
956     base[3].x = ( a + b + 1 ) >> 1;
957 
958     base[6].y = base[3].y;
959     c = base[1].y;
960     d = base[2].y;
961     base[1].y = a = ( base[0].y + c + 1 ) >> 1;
962     base[5].y = b = ( base[3].y + d + 1 ) >> 1;
963     c = ( c + d + 1 ) >> 1;
964     base[2].y = a = ( a + c + 1 ) >> 1;
965     base[4].y = b = ( b + c + 1 ) >> 1;
966     base[3].y = ( a + b + 1 ) >> 1;
967   }
968 
969 
970   /*************************************************************************/
971   /*                                                                       */
972   /* <Function>                                                            */
973   /*    Line_Up                                                            */
974   /*                                                                       */
975   /* <Description>                                                         */
976   /*    Compute the x-coordinates of an ascending line segment and store   */
977   /*    them in the render pool.                                           */
978   /*                                                                       */
979   /* <Input>                                                               */
980   /*    x1   :: The x-coordinate of the segment's start point.             */
981   /*                                                                       */
982   /*    y1   :: The y-coordinate of the segment's start point.             */
983   /*                                                                       */
984   /*    x2   :: The x-coordinate of the segment's end point.               */
985   /*                                                                       */
986   /*    y2   :: The y-coordinate of the segment's end point.               */
987   /*                                                                       */
988   /*    miny :: A lower vertical clipping bound value.                     */
989   /*                                                                       */
990   /*    maxy :: An upper vertical clipping bound value.                    */
991   /*                                                                       */
992   /* <Return>                                                              */
993   /*    SUCCESS on success, FAILURE on render pool overflow.               */
994   /*                                                                       */
995   static Bool
Line_Up(RAS_ARGS Long x1,Long y1,Long x2,Long y2,Long miny,Long maxy)996   Line_Up( RAS_ARGS Long  x1,
997                     Long  y1,
998                     Long  x2,
999                     Long  y2,
1000                     Long  miny,
1001                     Long  maxy )
1002   {
1003     Long   Dx, Dy;
1004     Int    e1, e2, f1, f2, size;     /* XXX: is `Short' sufficient? */
1005     Long   Ix, Rx, Ax;
1006 
1007     PLong  top;
1008 
1009 
1010     Dx = x2 - x1;
1011     Dy = y2 - y1;
1012 
1013     if ( Dy <= 0 || y2 < miny || y1 > maxy )
1014       return SUCCESS;
1015 
1016     if ( y1 < miny )
1017     {
1018       /* Take care: miny-y1 can be a very large value; we use     */
1019       /*            a slow MulDiv function to avoid clipping bugs */
1020       x1 += SMulDiv( Dx, miny - y1, Dy );
1021       e1  = (Int)TRUNC( miny );
1022       f1  = 0;
1023     }
1024     else
1025     {
1026       e1 = (Int)TRUNC( y1 );
1027       f1 = (Int)FRAC( y1 );
1028     }
1029 
1030     if ( y2 > maxy )
1031     {
1032       /* x2 += FMulDiv( Dx, maxy - y2, Dy );  UNNECESSARY */
1033       e2  = (Int)TRUNC( maxy );
1034       f2  = 0;
1035     }
1036     else
1037     {
1038       e2 = (Int)TRUNC( y2 );
1039       f2 = (Int)FRAC( y2 );
1040     }
1041 
1042     if ( f1 > 0 )
1043     {
1044       if ( e1 == e2 )
1045         return SUCCESS;
1046       else
1047       {
1048         x1 += SMulDiv( Dx, ras.precision - f1, Dy );
1049         e1 += 1;
1050       }
1051     }
1052     else
1053       if ( ras.joint )
1054       {
1055         ras.top--;
1056         ras.joint = FALSE;
1057       }
1058 
1059     ras.joint = (char)( f2 == 0 );
1060 
1061     if ( ras.fresh )
1062     {
1063       ras.cProfile->start = e1;
1064       ras.fresh           = FALSE;
1065     }
1066 
1067     size = e2 - e1 + 1;
1068     if ( ras.top + size >= ras.maxBuff )
1069     {
1070       ras.error = FT_THROW( Overflow );
1071       return FAILURE;
1072     }
1073 
1074     if ( Dx > 0 )
1075     {
1076       Ix = SMulDiv_No_Round( ras.precision, Dx, Dy );
1077       Rx = ( ras.precision * Dx ) % Dy;
1078       Dx = 1;
1079     }
1080     else
1081     {
1082       Ix = -SMulDiv_No_Round( ras.precision, -Dx, Dy );
1083       Rx = ( ras.precision * -Dx ) % Dy;
1084       Dx = -1;
1085     }
1086 
1087     Ax  = -Dy;
1088     top = ras.top;
1089 
1090     while ( size > 0 )
1091     {
1092       *top++ = x1;
1093 
1094       x1 += Ix;
1095       Ax += Rx;
1096       if ( Ax >= 0 )
1097       {
1098         Ax -= Dy;
1099         x1 += Dx;
1100       }
1101       size--;
1102     }
1103 
1104     ras.top = top;
1105     return SUCCESS;
1106   }
1107 
1108 
1109   /*************************************************************************/
1110   /*                                                                       */
1111   /* <Function>                                                            */
1112   /*    Line_Down                                                          */
1113   /*                                                                       */
1114   /* <Description>                                                         */
1115   /*    Compute the x-coordinates of an descending line segment and store  */
1116   /*    them in the render pool.                                           */
1117   /*                                                                       */
1118   /* <Input>                                                               */
1119   /*    x1   :: The x-coordinate of the segment's start point.             */
1120   /*                                                                       */
1121   /*    y1   :: The y-coordinate of the segment's start point.             */
1122   /*                                                                       */
1123   /*    x2   :: The x-coordinate of the segment's end point.               */
1124   /*                                                                       */
1125   /*    y2   :: The y-coordinate of the segment's end point.               */
1126   /*                                                                       */
1127   /*    miny :: A lower vertical clipping bound value.                     */
1128   /*                                                                       */
1129   /*    maxy :: An upper vertical clipping bound value.                    */
1130   /*                                                                       */
1131   /* <Return>                                                              */
1132   /*    SUCCESS on success, FAILURE on render pool overflow.               */
1133   /*                                                                       */
1134   static Bool
Line_Down(RAS_ARGS Long x1,Long y1,Long x2,Long y2,Long miny,Long maxy)1135   Line_Down( RAS_ARGS Long  x1,
1136                       Long  y1,
1137                       Long  x2,
1138                       Long  y2,
1139                       Long  miny,
1140                       Long  maxy )
1141   {
1142     Bool  result, fresh;
1143 
1144 
1145     fresh  = ras.fresh;
1146 
1147     result = Line_Up( RAS_VARS x1, -y1, x2, -y2, -maxy, -miny );
1148 
1149     if ( fresh && !ras.fresh )
1150       ras.cProfile->start = -ras.cProfile->start;
1151 
1152     return result;
1153   }
1154 
1155 
1156   /* A function type describing the functions used to split Bezier arcs */
1157   typedef void  (*TSplitter)( TPoint*  base );
1158 
1159 
1160   /*************************************************************************/
1161   /*                                                                       */
1162   /* <Function>                                                            */
1163   /*    Bezier_Up                                                          */
1164   /*                                                                       */
1165   /* <Description>                                                         */
1166   /*    Compute the x-coordinates of an ascending Bezier arc and store     */
1167   /*    them in the render pool.                                           */
1168   /*                                                                       */
1169   /* <Input>                                                               */
1170   /*    degree   :: The degree of the Bezier arc (either 2 or 3).          */
1171   /*                                                                       */
1172   /*    splitter :: The function to split Bezier arcs.                     */
1173   /*                                                                       */
1174   /*    miny     :: A lower vertical clipping bound value.                 */
1175   /*                                                                       */
1176   /*    maxy     :: An upper vertical clipping bound value.                */
1177   /*                                                                       */
1178   /* <Return>                                                              */
1179   /*    SUCCESS on success, FAILURE on render pool overflow.               */
1180   /*                                                                       */
1181   static Bool
Bezier_Up(RAS_ARGS Int degree,TSplitter splitter,Long miny,Long maxy)1182   Bezier_Up( RAS_ARGS Int        degree,
1183                       TSplitter  splitter,
1184                       Long       miny,
1185                       Long       maxy )
1186   {
1187     Long   y1, y2, e, e2, e0;
1188     Short  f1;
1189 
1190     TPoint*  arc;
1191     TPoint*  start_arc;
1192 
1193     PLong top;
1194 
1195 
1196     arc = ras.arc;
1197     y1  = arc[degree].y;
1198     y2  = arc[0].y;
1199     top = ras.top;
1200 
1201     if ( y2 < miny || y1 > maxy )
1202       goto Fin;
1203 
1204     e2 = FLOOR( y2 );
1205 
1206     if ( e2 > maxy )
1207       e2 = maxy;
1208 
1209     e0 = miny;
1210 
1211     if ( y1 < miny )
1212       e = miny;
1213     else
1214     {
1215       e  = CEILING( y1 );
1216       f1 = (Short)( FRAC( y1 ) );
1217       e0 = e;
1218 
1219       if ( f1 == 0 )
1220       {
1221         if ( ras.joint )
1222         {
1223           top--;
1224           ras.joint = FALSE;
1225         }
1226 
1227         *top++ = arc[degree].x;
1228 
1229         e += ras.precision;
1230       }
1231     }
1232 
1233     if ( ras.fresh )
1234     {
1235       ras.cProfile->start = TRUNC( e0 );
1236       ras.fresh = FALSE;
1237     }
1238 
1239     if ( e2 < e )
1240       goto Fin;
1241 
1242     if ( ( top + TRUNC( e2 - e ) + 1 ) >= ras.maxBuff )
1243     {
1244       ras.top   = top;
1245       ras.error = FT_THROW( Overflow );
1246       return FAILURE;
1247     }
1248 
1249     start_arc = arc;
1250 
1251     while ( arc >= start_arc && e <= e2 )
1252     {
1253       ras.joint = FALSE;
1254 
1255       y2 = arc[0].y;
1256 
1257       if ( y2 > e )
1258       {
1259         y1 = arc[degree].y;
1260         if ( y2 - y1 >= ras.precision_step )
1261         {
1262           splitter( arc );
1263           arc += degree;
1264         }
1265         else
1266         {
1267           *top++ = arc[degree].x + FMulDiv( arc[0].x - arc[degree].x,
1268                                             e - y1, y2 - y1 );
1269           arc -= degree;
1270           e   += ras.precision;
1271         }
1272       }
1273       else
1274       {
1275         if ( y2 == e )
1276         {
1277           ras.joint  = TRUE;
1278           *top++     = arc[0].x;
1279 
1280           e += ras.precision;
1281         }
1282         arc -= degree;
1283       }
1284     }
1285 
1286   Fin:
1287     ras.top  = top;
1288     ras.arc -= degree;
1289     return SUCCESS;
1290   }
1291 
1292 
1293   /*************************************************************************/
1294   /*                                                                       */
1295   /* <Function>                                                            */
1296   /*    Bezier_Down                                                        */
1297   /*                                                                       */
1298   /* <Description>                                                         */
1299   /*    Compute the x-coordinates of an descending Bezier arc and store    */
1300   /*    them in the render pool.                                           */
1301   /*                                                                       */
1302   /* <Input>                                                               */
1303   /*    degree   :: The degree of the Bezier arc (either 2 or 3).          */
1304   /*                                                                       */
1305   /*    splitter :: The function to split Bezier arcs.                     */
1306   /*                                                                       */
1307   /*    miny     :: A lower vertical clipping bound value.                 */
1308   /*                                                                       */
1309   /*    maxy     :: An upper vertical clipping bound value.                */
1310   /*                                                                       */
1311   /* <Return>                                                              */
1312   /*    SUCCESS on success, FAILURE on render pool overflow.               */
1313   /*                                                                       */
1314   static Bool
Bezier_Down(RAS_ARGS Int degree,TSplitter splitter,Long miny,Long maxy)1315   Bezier_Down( RAS_ARGS Int        degree,
1316                         TSplitter  splitter,
1317                         Long       miny,
1318                         Long       maxy )
1319   {
1320     TPoint*  arc = ras.arc;
1321     Bool     result, fresh;
1322 
1323 
1324     arc[0].y = -arc[0].y;
1325     arc[1].y = -arc[1].y;
1326     arc[2].y = -arc[2].y;
1327     if ( degree > 2 )
1328       arc[3].y = -arc[3].y;
1329 
1330     fresh = ras.fresh;
1331 
1332     result = Bezier_Up( RAS_VARS degree, splitter, -maxy, -miny );
1333 
1334     if ( fresh && !ras.fresh )
1335       ras.cProfile->start = -ras.cProfile->start;
1336 
1337     arc[0].y = -arc[0].y;
1338     return result;
1339   }
1340 
1341 
1342   /*************************************************************************/
1343   /*                                                                       */
1344   /* <Function>                                                            */
1345   /*    Line_To                                                            */
1346   /*                                                                       */
1347   /* <Description>                                                         */
1348   /*    Inject a new line segment and adjust the Profiles list.            */
1349   /*                                                                       */
1350   /* <Input>                                                               */
1351   /*   x :: The x-coordinate of the segment's end point (its start point   */
1352   /*        is stored in `lastX').                                         */
1353   /*                                                                       */
1354   /*   y :: The y-coordinate of the segment's end point (its start point   */
1355   /*        is stored in `lastY').                                         */
1356   /*                                                                       */
1357   /* <Return>                                                              */
1358   /*   SUCCESS on success, FAILURE on render pool overflow or incorrect    */
1359   /*   profile.                                                            */
1360   /*                                                                       */
1361   static Bool
Line_To(RAS_ARGS Long x,Long y)1362   Line_To( RAS_ARGS Long  x,
1363                     Long  y )
1364   {
1365     /* First, detect a change of direction */
1366 
1367     switch ( ras.state )
1368     {
1369     case Unknown_State:
1370       if ( y > ras.lastY )
1371       {
1372         if ( New_Profile( RAS_VARS Ascending_State,
1373                                    IS_BOTTOM_OVERSHOOT( ras.lastY ) ) )
1374           return FAILURE;
1375       }
1376       else
1377       {
1378         if ( y < ras.lastY )
1379           if ( New_Profile( RAS_VARS Descending_State,
1380                                      IS_TOP_OVERSHOOT( ras.lastY ) ) )
1381             return FAILURE;
1382       }
1383       break;
1384 
1385     case Ascending_State:
1386       if ( y < ras.lastY )
1387       {
1388         if ( End_Profile( RAS_VARS IS_TOP_OVERSHOOT( ras.lastY ) ) ||
1389              New_Profile( RAS_VARS Descending_State,
1390                                    IS_TOP_OVERSHOOT( ras.lastY ) ) )
1391           return FAILURE;
1392       }
1393       break;
1394 
1395     case Descending_State:
1396       if ( y > ras.lastY )
1397       {
1398         if ( End_Profile( RAS_VARS IS_BOTTOM_OVERSHOOT( ras.lastY ) ) ||
1399              New_Profile( RAS_VARS Ascending_State,
1400                                    IS_BOTTOM_OVERSHOOT( ras.lastY ) ) )
1401           return FAILURE;
1402       }
1403       break;
1404 
1405     default:
1406       ;
1407     }
1408 
1409     /* Then compute the lines */
1410 
1411     switch ( ras.state )
1412     {
1413     case Ascending_State:
1414       if ( Line_Up( RAS_VARS ras.lastX, ras.lastY,
1415                              x, y, ras.minY, ras.maxY ) )
1416         return FAILURE;
1417       break;
1418 
1419     case Descending_State:
1420       if ( Line_Down( RAS_VARS ras.lastX, ras.lastY,
1421                                x, y, ras.minY, ras.maxY ) )
1422         return FAILURE;
1423       break;
1424 
1425     default:
1426       ;
1427     }
1428 
1429     ras.lastX = x;
1430     ras.lastY = y;
1431 
1432     return SUCCESS;
1433   }
1434 
1435 
1436   /*************************************************************************/
1437   /*                                                                       */
1438   /* <Function>                                                            */
1439   /*    Conic_To                                                           */
1440   /*                                                                       */
1441   /* <Description>                                                         */
1442   /*    Inject a new conic arc and adjust the profile list.                */
1443   /*                                                                       */
1444   /* <Input>                                                               */
1445   /*   cx :: The x-coordinate of the arc's new control point.              */
1446   /*                                                                       */
1447   /*   cy :: The y-coordinate of the arc's new control point.              */
1448   /*                                                                       */
1449   /*   x  :: The x-coordinate of the arc's end point (its start point is   */
1450   /*         stored in `lastX').                                           */
1451   /*                                                                       */
1452   /*   y  :: The y-coordinate of the arc's end point (its start point is   */
1453   /*         stored in `lastY').                                           */
1454   /*                                                                       */
1455   /* <Return>                                                              */
1456   /*   SUCCESS on success, FAILURE on render pool overflow or incorrect    */
1457   /*   profile.                                                            */
1458   /*                                                                       */
1459   static Bool
Conic_To(RAS_ARGS Long cx,Long cy,Long x,Long y)1460   Conic_To( RAS_ARGS Long  cx,
1461                      Long  cy,
1462                      Long  x,
1463                      Long  y )
1464   {
1465     Long     y1, y2, y3, x3, ymin, ymax;
1466     TStates  state_bez;
1467 
1468 
1469     ras.arc      = ras.arcs;
1470     ras.arc[2].x = ras.lastX;
1471     ras.arc[2].y = ras.lastY;
1472     ras.arc[1].x = cx;
1473     ras.arc[1].y = cy;
1474     ras.arc[0].x = x;
1475     ras.arc[0].y = y;
1476 
1477     do
1478     {
1479       y1 = ras.arc[2].y;
1480       y2 = ras.arc[1].y;
1481       y3 = ras.arc[0].y;
1482       x3 = ras.arc[0].x;
1483 
1484       /* first, categorize the Bezier arc */
1485 
1486       if ( y1 <= y3 )
1487       {
1488         ymin = y1;
1489         ymax = y3;
1490       }
1491       else
1492       {
1493         ymin = y3;
1494         ymax = y1;
1495       }
1496 
1497       if ( y2 < ymin || y2 > ymax )
1498       {
1499         /* this arc has no given direction, split it! */
1500         Split_Conic( ras.arc );
1501         ras.arc += 2;
1502       }
1503       else if ( y1 == y3 )
1504       {
1505         /* this arc is flat, ignore it and pop it from the Bezier stack */
1506         ras.arc -= 2;
1507       }
1508       else
1509       {
1510         /* the arc is y-monotonous, either ascending or descending */
1511         /* detect a change of direction                            */
1512         state_bez = y1 < y3 ? Ascending_State : Descending_State;
1513         if ( ras.state != state_bez )
1514         {
1515           Bool  o = state_bez == Ascending_State ? IS_BOTTOM_OVERSHOOT( y1 )
1516                                                  : IS_TOP_OVERSHOOT( y1 );
1517 
1518 
1519           /* finalize current profile if any */
1520           if ( ras.state != Unknown_State &&
1521                End_Profile( RAS_VARS o )  )
1522             goto Fail;
1523 
1524           /* create a new profile */
1525           if ( New_Profile( RAS_VARS state_bez, o ) )
1526             goto Fail;
1527         }
1528 
1529         /* now call the appropriate routine */
1530         if ( state_bez == Ascending_State )
1531         {
1532           if ( Bezier_Up( RAS_VARS 2, Split_Conic, ras.minY, ras.maxY ) )
1533             goto Fail;
1534         }
1535         else
1536           if ( Bezier_Down( RAS_VARS 2, Split_Conic, ras.minY, ras.maxY ) )
1537             goto Fail;
1538       }
1539 
1540     } while ( ras.arc >= ras.arcs );
1541 
1542     ras.lastX = x3;
1543     ras.lastY = y3;
1544 
1545     return SUCCESS;
1546 
1547   Fail:
1548     return FAILURE;
1549   }
1550 
1551 
1552   /*************************************************************************/
1553   /*                                                                       */
1554   /* <Function>                                                            */
1555   /*    Cubic_To                                                           */
1556   /*                                                                       */
1557   /* <Description>                                                         */
1558   /*    Inject a new cubic arc and adjust the profile list.                */
1559   /*                                                                       */
1560   /* <Input>                                                               */
1561   /*   cx1 :: The x-coordinate of the arc's first new control point.       */
1562   /*                                                                       */
1563   /*   cy1 :: The y-coordinate of the arc's first new control point.       */
1564   /*                                                                       */
1565   /*   cx2 :: The x-coordinate of the arc's second new control point.      */
1566   /*                                                                       */
1567   /*   cy2 :: The y-coordinate of the arc's second new control point.      */
1568   /*                                                                       */
1569   /*   x   :: The x-coordinate of the arc's end point (its start point is  */
1570   /*          stored in `lastX').                                          */
1571   /*                                                                       */
1572   /*   y   :: The y-coordinate of the arc's end point (its start point is  */
1573   /*          stored in `lastY').                                          */
1574   /*                                                                       */
1575   /* <Return>                                                              */
1576   /*   SUCCESS on success, FAILURE on render pool overflow or incorrect    */
1577   /*   profile.                                                            */
1578   /*                                                                       */
1579   static Bool
Cubic_To(RAS_ARGS Long cx1,Long cy1,Long cx2,Long cy2,Long x,Long y)1580   Cubic_To( RAS_ARGS Long  cx1,
1581                      Long  cy1,
1582                      Long  cx2,
1583                      Long  cy2,
1584                      Long  x,
1585                      Long  y )
1586   {
1587     Long     y1, y2, y3, y4, x4, ymin1, ymax1, ymin2, ymax2;
1588     TStates  state_bez;
1589 
1590 
1591     ras.arc      = ras.arcs;
1592     ras.arc[3].x = ras.lastX;
1593     ras.arc[3].y = ras.lastY;
1594     ras.arc[2].x = cx1;
1595     ras.arc[2].y = cy1;
1596     ras.arc[1].x = cx2;
1597     ras.arc[1].y = cy2;
1598     ras.arc[0].x = x;
1599     ras.arc[0].y = y;
1600 
1601     do
1602     {
1603       y1 = ras.arc[3].y;
1604       y2 = ras.arc[2].y;
1605       y3 = ras.arc[1].y;
1606       y4 = ras.arc[0].y;
1607       x4 = ras.arc[0].x;
1608 
1609       /* first, categorize the Bezier arc */
1610 
1611       if ( y1 <= y4 )
1612       {
1613         ymin1 = y1;
1614         ymax1 = y4;
1615       }
1616       else
1617       {
1618         ymin1 = y4;
1619         ymax1 = y1;
1620       }
1621 
1622       if ( y2 <= y3 )
1623       {
1624         ymin2 = y2;
1625         ymax2 = y3;
1626       }
1627       else
1628       {
1629         ymin2 = y3;
1630         ymax2 = y2;
1631       }
1632 
1633       if ( ymin2 < ymin1 || ymax2 > ymax1 )
1634       {
1635         /* this arc has no given direction, split it! */
1636         Split_Cubic( ras.arc );
1637         ras.arc += 3;
1638       }
1639       else if ( y1 == y4 )
1640       {
1641         /* this arc is flat, ignore it and pop it from the Bezier stack */
1642         ras.arc -= 3;
1643       }
1644       else
1645       {
1646         state_bez = ( y1 <= y4 ) ? Ascending_State : Descending_State;
1647 
1648         /* detect a change of direction */
1649         if ( ras.state != state_bez )
1650         {
1651           Bool  o = state_bez == Ascending_State ? IS_BOTTOM_OVERSHOOT( y1 )
1652                                                  : IS_TOP_OVERSHOOT( y1 );
1653 
1654 
1655           /* finalize current profile if any */
1656           if ( ras.state != Unknown_State &&
1657                End_Profile( RAS_VARS o )  )
1658             goto Fail;
1659 
1660           if ( New_Profile( RAS_VARS state_bez, o ) )
1661             goto Fail;
1662         }
1663 
1664         /* compute intersections */
1665         if ( state_bez == Ascending_State )
1666         {
1667           if ( Bezier_Up( RAS_VARS 3, Split_Cubic, ras.minY, ras.maxY ) )
1668             goto Fail;
1669         }
1670         else
1671           if ( Bezier_Down( RAS_VARS 3, Split_Cubic, ras.minY, ras.maxY ) )
1672             goto Fail;
1673       }
1674 
1675     } while ( ras.arc >= ras.arcs );
1676 
1677     ras.lastX = x4;
1678     ras.lastY = y4;
1679 
1680     return SUCCESS;
1681 
1682   Fail:
1683     return FAILURE;
1684   }
1685 
1686 
1687 #undef  SWAP_
1688 #define SWAP_( x, y )  do                \
1689                        {                 \
1690                          Long  swap = x; \
1691                                          \
1692                                          \
1693                          x = y;          \
1694                          y = swap;       \
1695                        } while ( 0 )
1696 
1697 
1698   /*************************************************************************/
1699   /*                                                                       */
1700   /* <Function>                                                            */
1701   /*    Decompose_Curve                                                    */
1702   /*                                                                       */
1703   /* <Description>                                                         */
1704   /*    Scan the outline arrays in order to emit individual segments and   */
1705   /*    Beziers by calling Line_To() and Bezier_To().  It handles all      */
1706   /*    weird cases, like when the first point is off the curve, or when   */
1707   /*    there are simply no `on' points in the contour!                    */
1708   /*                                                                       */
1709   /* <Input>                                                               */
1710   /*    first   :: The index of the first point in the contour.            */
1711   /*                                                                       */
1712   /*    last    :: The index of the last point in the contour.             */
1713   /*                                                                       */
1714   /*    flipped :: If set, flip the direction of the curve.                */
1715   /*                                                                       */
1716   /* <Return>                                                              */
1717   /*    SUCCESS on success, FAILURE on error.                              */
1718   /*                                                                       */
1719   static Bool
Decompose_Curve(RAS_ARGS UShort first,UShort last,Int flipped)1720   Decompose_Curve( RAS_ARGS UShort  first,
1721                             UShort  last,
1722                             Int     flipped )
1723   {
1724     FT_Vector   v_last;
1725     FT_Vector   v_control;
1726     FT_Vector   v_start;
1727 
1728     FT_Vector*  points;
1729     FT_Vector*  point;
1730     FT_Vector*  limit;
1731     char*       tags;
1732 
1733     UInt        tag;       /* current point's state           */
1734 
1735 
1736     points = ras.outline.points;
1737     limit  = points + last;
1738 
1739     v_start.x = SCALED( points[first].x );
1740     v_start.y = SCALED( points[first].y );
1741     v_last.x  = SCALED( points[last].x );
1742     v_last.y  = SCALED( points[last].y );
1743 
1744     if ( flipped )
1745     {
1746       SWAP_( v_start.x, v_start.y );
1747       SWAP_( v_last.x, v_last.y );
1748     }
1749 
1750     v_control = v_start;
1751 
1752     point = points + first;
1753     tags  = ras.outline.tags + first;
1754 
1755     /* set scan mode if necessary */
1756     if ( tags[0] & FT_CURVE_TAG_HAS_SCANMODE )
1757       ras.dropOutControl = (Byte)tags[0] >> 5;
1758 
1759     tag = FT_CURVE_TAG( tags[0] );
1760 
1761     /* A contour cannot start with a cubic control point! */
1762     if ( tag == FT_CURVE_TAG_CUBIC )
1763       goto Invalid_Outline;
1764 
1765     /* check first point to determine origin */
1766     if ( tag == FT_CURVE_TAG_CONIC )
1767     {
1768       /* first point is conic control.  Yes, this happens. */
1769       if ( FT_CURVE_TAG( ras.outline.tags[last] ) == FT_CURVE_TAG_ON )
1770       {
1771         /* start at last point if it is on the curve */
1772         v_start = v_last;
1773         limit--;
1774       }
1775       else
1776       {
1777         /* if both first and last points are conic,         */
1778         /* start at their middle and record its position    */
1779         /* for closure                                      */
1780         v_start.x = ( v_start.x + v_last.x ) / 2;
1781         v_start.y = ( v_start.y + v_last.y ) / 2;
1782 
1783      /* v_last = v_start; */
1784       }
1785       point--;
1786       tags--;
1787     }
1788 
1789     ras.lastX = v_start.x;
1790     ras.lastY = v_start.y;
1791 
1792     while ( point < limit )
1793     {
1794       point++;
1795       tags++;
1796 
1797       tag = FT_CURVE_TAG( tags[0] );
1798 
1799       switch ( tag )
1800       {
1801       case FT_CURVE_TAG_ON:  /* emit a single line_to */
1802         {
1803           Long  x, y;
1804 
1805 
1806           x = SCALED( point->x );
1807           y = SCALED( point->y );
1808           if ( flipped )
1809             SWAP_( x, y );
1810 
1811           if ( Line_To( RAS_VARS x, y ) )
1812             goto Fail;
1813           continue;
1814         }
1815 
1816       case FT_CURVE_TAG_CONIC:  /* consume conic arcs */
1817         v_control.x = SCALED( point[0].x );
1818         v_control.y = SCALED( point[0].y );
1819 
1820         if ( flipped )
1821           SWAP_( v_control.x, v_control.y );
1822 
1823       Do_Conic:
1824         if ( point < limit )
1825         {
1826           FT_Vector  v_middle;
1827           Long       x, y;
1828 
1829 
1830           point++;
1831           tags++;
1832           tag = FT_CURVE_TAG( tags[0] );
1833 
1834           x = SCALED( point[0].x );
1835           y = SCALED( point[0].y );
1836 
1837           if ( flipped )
1838             SWAP_( x, y );
1839 
1840           if ( tag == FT_CURVE_TAG_ON )
1841           {
1842             if ( Conic_To( RAS_VARS v_control.x, v_control.y, x, y ) )
1843               goto Fail;
1844             continue;
1845           }
1846 
1847           if ( tag != FT_CURVE_TAG_CONIC )
1848             goto Invalid_Outline;
1849 
1850           v_middle.x = ( v_control.x + x ) / 2;
1851           v_middle.y = ( v_control.y + y ) / 2;
1852 
1853           if ( Conic_To( RAS_VARS v_control.x, v_control.y,
1854                                   v_middle.x,  v_middle.y ) )
1855             goto Fail;
1856 
1857           v_control.x = x;
1858           v_control.y = y;
1859 
1860           goto Do_Conic;
1861         }
1862 
1863         if ( Conic_To( RAS_VARS v_control.x, v_control.y,
1864                                 v_start.x,   v_start.y ) )
1865           goto Fail;
1866 
1867         goto Close;
1868 
1869       default:  /* FT_CURVE_TAG_CUBIC */
1870         {
1871           Long  x1, y1, x2, y2, x3, y3;
1872 
1873 
1874           if ( point + 1 > limit                             ||
1875                FT_CURVE_TAG( tags[1] ) != FT_CURVE_TAG_CUBIC )
1876             goto Invalid_Outline;
1877 
1878           point += 2;
1879           tags  += 2;
1880 
1881           x1 = SCALED( point[-2].x );
1882           y1 = SCALED( point[-2].y );
1883           x2 = SCALED( point[-1].x );
1884           y2 = SCALED( point[-1].y );
1885 
1886           if ( flipped )
1887           {
1888             SWAP_( x1, y1 );
1889             SWAP_( x2, y2 );
1890           }
1891 
1892           if ( point <= limit )
1893           {
1894             x3 = SCALED( point[0].x );
1895             y3 = SCALED( point[0].y );
1896 
1897             if ( flipped )
1898               SWAP_( x3, y3 );
1899 
1900             if ( Cubic_To( RAS_VARS x1, y1, x2, y2, x3, y3 ) )
1901               goto Fail;
1902             continue;
1903           }
1904 
1905           if ( Cubic_To( RAS_VARS x1, y1, x2, y2, v_start.x, v_start.y ) )
1906             goto Fail;
1907           goto Close;
1908         }
1909       }
1910     }
1911 
1912     /* close the contour with a line segment */
1913     if ( Line_To( RAS_VARS v_start.x, v_start.y ) )
1914       goto Fail;
1915 
1916   Close:
1917     return SUCCESS;
1918 
1919   Invalid_Outline:
1920     ras.error = FT_THROW( Invalid );
1921 
1922   Fail:
1923     return FAILURE;
1924   }
1925 
1926 
1927   /*************************************************************************/
1928   /*                                                                       */
1929   /* <Function>                                                            */
1930   /*    Convert_Glyph                                                      */
1931   /*                                                                       */
1932   /* <Description>                                                         */
1933   /*    Convert a glyph into a series of segments and arcs and make a      */
1934   /*    profiles list with them.                                           */
1935   /*                                                                       */
1936   /* <Input>                                                               */
1937   /*    flipped :: If set, flip the direction of curve.                    */
1938   /*                                                                       */
1939   /* <Return>                                                              */
1940   /*    SUCCESS on success, FAILURE if any error was encountered during    */
1941   /*    rendering.                                                         */
1942   /*                                                                       */
1943   static Bool
Convert_Glyph(RAS_ARGS Int flipped)1944   Convert_Glyph( RAS_ARGS Int  flipped )
1945   {
1946     Int   i;
1947     UInt  start;
1948 
1949 
1950     ras.fProfile = NULL;
1951     ras.joint    = FALSE;
1952     ras.fresh    = FALSE;
1953 
1954     ras.maxBuff  = ras.sizeBuff - AlignProfileSize;
1955 
1956     ras.numTurns = 0;
1957 
1958     ras.cProfile         = (PProfile)ras.top;
1959     ras.cProfile->offset = ras.top;
1960     ras.num_Profs        = 0;
1961 
1962     start = 0;
1963 
1964     for ( i = 0; i < ras.outline.n_contours; i++ )
1965     {
1966       PProfile  lastProfile;
1967       Bool      o;
1968 
1969 
1970       ras.state    = Unknown_State;
1971       ras.gProfile = NULL;
1972 
1973       if ( Decompose_Curve( RAS_VARS (UShort)start,
1974                                      (UShort)ras.outline.contours[i],
1975                                      flipped ) )
1976         return FAILURE;
1977 
1978       start = (UShort)ras.outline.contours[i] + 1;
1979 
1980       /* we must now check whether the extreme arcs join or not */
1981       if ( FRAC( ras.lastY ) == 0 &&
1982            ras.lastY >= ras.minY  &&
1983            ras.lastY <= ras.maxY  )
1984         if ( ras.gProfile                        &&
1985              ( ras.gProfile->flags & Flow_Up ) ==
1986                ( ras.cProfile->flags & Flow_Up ) )
1987           ras.top--;
1988         /* Note that ras.gProfile can be nil if the contour was too small */
1989         /* to be drawn.                                                   */
1990 
1991       lastProfile = ras.cProfile;
1992       if ( ras.top != ras.cProfile->offset &&
1993            ( ras.cProfile->flags & Flow_Up ) )
1994         o = IS_TOP_OVERSHOOT( ras.lastY );
1995       else
1996         o = IS_BOTTOM_OVERSHOOT( ras.lastY );
1997       if ( End_Profile( RAS_VARS o ) )
1998         return FAILURE;
1999 
2000       /* close the `next profile in contour' linked list */
2001       if ( ras.gProfile )
2002         lastProfile->next = ras.gProfile;
2003     }
2004 
2005     if ( Finalize_Profile_Table( RAS_VAR ) )
2006       return FAILURE;
2007 
2008     return (Bool)( ras.top < ras.maxBuff ? SUCCESS : FAILURE );
2009   }
2010 
2011 
2012   /*************************************************************************/
2013   /*************************************************************************/
2014   /**                                                                     **/
2015   /**  SCAN-LINE SWEEPS AND DRAWING                                       **/
2016   /**                                                                     **/
2017   /*************************************************************************/
2018   /*************************************************************************/
2019 
2020 
2021   /*************************************************************************/
2022   /*                                                                       */
2023   /*  Init_Linked                                                          */
2024   /*                                                                       */
2025   /*    Initializes an empty linked list.                                  */
2026   /*                                                                       */
2027   static void
Init_Linked(TProfileList * l)2028   Init_Linked( TProfileList*  l )
2029   {
2030     *l = NULL;
2031   }
2032 
2033 
2034   /*************************************************************************/
2035   /*                                                                       */
2036   /*  InsNew                                                               */
2037   /*                                                                       */
2038   /*    Inserts a new profile in a linked list.                            */
2039   /*                                                                       */
2040   static void
InsNew(PProfileList list,PProfile profile)2041   InsNew( PProfileList  list,
2042           PProfile      profile )
2043   {
2044     PProfile  *old, current;
2045     Long       x;
2046 
2047 
2048     old     = list;
2049     current = *old;
2050     x       = profile->X;
2051 
2052     while ( current )
2053     {
2054       if ( x < current->X )
2055         break;
2056       old     = &current->link;
2057       current = *old;
2058     }
2059 
2060     profile->link = current;
2061     *old          = profile;
2062   }
2063 
2064 
2065   /*************************************************************************/
2066   /*                                                                       */
2067   /*  DelOld                                                               */
2068   /*                                                                       */
2069   /*    Removes an old profile from a linked list.                         */
2070   /*                                                                       */
2071   static void
DelOld(PProfileList list,PProfile profile)2072   DelOld( PProfileList  list,
2073           PProfile      profile )
2074   {
2075     PProfile  *old, current;
2076 
2077 
2078     old     = list;
2079     current = *old;
2080 
2081     while ( current )
2082     {
2083       if ( current == profile )
2084       {
2085         *old = current->link;
2086         return;
2087       }
2088 
2089       old     = &current->link;
2090       current = *old;
2091     }
2092 
2093     /* we should never get there, unless the profile was not part of */
2094     /* the list.                                                     */
2095   }
2096 
2097 
2098   /*************************************************************************/
2099   /*                                                                       */
2100   /*  Sort                                                                 */
2101   /*                                                                       */
2102   /*    Sorts a trace list.  In 95%, the list is already sorted.  We need  */
2103   /*    an algorithm which is fast in this case.  Bubble sort is enough    */
2104   /*    and simple.                                                        */
2105   /*                                                                       */
2106   static void
Sort(PProfileList list)2107   Sort( PProfileList  list )
2108   {
2109     PProfile  *old, current, next;
2110 
2111 
2112     /* First, set the new X coordinate of each profile */
2113     current = *list;
2114     while ( current )
2115     {
2116       current->X       = *current->offset;
2117       current->offset += current->flags & Flow_Up ? 1 : -1;
2118       current->height--;
2119       current = current->link;
2120     }
2121 
2122     /* Then sort them */
2123     old     = list;
2124     current = *old;
2125 
2126     if ( !current )
2127       return;
2128 
2129     next = current->link;
2130 
2131     while ( next )
2132     {
2133       if ( current->X <= next->X )
2134       {
2135         old     = &current->link;
2136         current = *old;
2137 
2138         if ( !current )
2139           return;
2140       }
2141       else
2142       {
2143         *old          = next;
2144         current->link = next->link;
2145         next->link    = current;
2146 
2147         old     = list;
2148         current = *old;
2149       }
2150 
2151       next = current->link;
2152     }
2153   }
2154 
2155 
2156   /*************************************************************************/
2157   /*                                                                       */
2158   /*  Vertical Sweep Procedure Set                                         */
2159   /*                                                                       */
2160   /*  These four routines are used during the vertical black/white sweep   */
2161   /*  phase by the generic Draw_Sweep() function.                          */
2162   /*                                                                       */
2163   /*************************************************************************/
2164 
2165   static void
Vertical_Sweep_Init(RAS_ARGS Short * min,Short * max)2166   Vertical_Sweep_Init( RAS_ARGS Short*  min,
2167                                 Short*  max )
2168   {
2169     Long  pitch = ras.target.pitch;
2170 
2171     FT_UNUSED( max );
2172 
2173 
2174     ras.traceIncr = (Short)-pitch;
2175     ras.traceOfs  = -*min * pitch;
2176     if ( pitch > 0 )
2177       ras.traceOfs += (Long)( ras.target.rows - 1 ) * pitch;
2178   }
2179 
2180 
2181   static void
Vertical_Sweep_Span(RAS_ARGS Short y,FT_F26Dot6 x1,FT_F26Dot6 x2,PProfile left,PProfile right)2182   Vertical_Sweep_Span( RAS_ARGS Short       y,
2183                                 FT_F26Dot6  x1,
2184                                 FT_F26Dot6  x2,
2185                                 PProfile    left,
2186                                 PProfile    right )
2187   {
2188     Long   e1, e2;
2189     Byte*  target;
2190 
2191     Int  dropOutControl = left->flags & 7;
2192 
2193     FT_UNUSED( y );
2194     FT_UNUSED( left );
2195     FT_UNUSED( right );
2196 
2197 
2198     /* in high-precision mode, we need 12 digits after the comma to */
2199     /* represent multiples of 1/(1<<12) = 1/4096                    */
2200     FT_TRACE7(( "  y=%d x=[%.12f;%.12f], drop-out=%d",
2201                 y,
2202                 x1 / (double)ras.precision,
2203                 x2 / (double)ras.precision,
2204                 dropOutControl ));
2205 
2206     /* Drop-out control */
2207 
2208     e1 = TRUNC( CEILING( x1 ) );
2209 
2210     if ( dropOutControl != 2                             &&
2211          x2 - x1 - ras.precision <= ras.precision_jitter )
2212       e2 = e1;
2213     else
2214       e2 = TRUNC( FLOOR( x2 ) );
2215 
2216     if ( e2 >= 0 && e1 < ras.bWidth )
2217     {
2218       Int   c1, c2;
2219       Byte  f1, f2;
2220 
2221 
2222       if ( e1 < 0 )
2223         e1 = 0;
2224       if ( e2 >= ras.bWidth )
2225         e2 = ras.bWidth - 1;
2226 
2227       FT_TRACE7(( " -> x=[%d;%d]", e1, e2 ));
2228 
2229       c1 = (Short)( e1 >> 3 );
2230       c2 = (Short)( e2 >> 3 );
2231 
2232       f1 = (Byte)  ( 0xFF >> ( e1 & 7 ) );
2233       f2 = (Byte) ~( 0x7F >> ( e2 & 7 ) );
2234 
2235       target = ras.bTarget + ras.traceOfs + c1;
2236       c2 -= c1;
2237 
2238       if ( c2 > 0 )
2239       {
2240         target[0] |= f1;
2241 
2242         /* memset() is slower than the following code on many platforms. */
2243         /* This is due to the fact that, in the vast majority of cases,  */
2244         /* the span length in bytes is relatively small.                 */
2245         c2--;
2246         while ( c2 > 0 )
2247         {
2248           *(++target) = 0xFF;
2249           c2--;
2250         }
2251         target[1] |= f2;
2252       }
2253       else
2254         *target |= ( f1 & f2 );
2255     }
2256 
2257     FT_TRACE7(( "\n" ));
2258   }
2259 
2260 
2261   static void
Vertical_Sweep_Drop(RAS_ARGS Short y,FT_F26Dot6 x1,FT_F26Dot6 x2,PProfile left,PProfile right)2262   Vertical_Sweep_Drop( RAS_ARGS Short       y,
2263                                 FT_F26Dot6  x1,
2264                                 FT_F26Dot6  x2,
2265                                 PProfile    left,
2266                                 PProfile    right )
2267   {
2268     Long   e1, e2, pxl;
2269     Short  c1, f1;
2270 
2271 
2272     FT_TRACE7(( "  y=%d x=[%.12f;%.12f]",
2273                 y,
2274                 x1 / (double)ras.precision,
2275                 x2 / (double)ras.precision ));
2276 
2277     /* Drop-out control */
2278 
2279     /*   e2            x2                    x1           e1   */
2280     /*                                                         */
2281     /*                 ^                     |                 */
2282     /*                 |                     |                 */
2283     /*   +-------------+---------------------+------------+    */
2284     /*                 |                     |                 */
2285     /*                 |                     v                 */
2286     /*                                                         */
2287     /* pixel         contour              contour       pixel  */
2288     /* center                                           center */
2289 
2290     /* drop-out mode    scan conversion rules (as defined in OpenType) */
2291     /* --------------------------------------------------------------- */
2292     /*  0                1, 2, 3                                       */
2293     /*  1                1, 2, 4                                       */
2294     /*  2                1, 2                                          */
2295     /*  3                same as mode 2                                */
2296     /*  4                1, 2, 5                                       */
2297     /*  5                1, 2, 6                                       */
2298     /*  6, 7             same as mode 2                                */
2299 
2300     e1  = CEILING( x1 );
2301     e2  = FLOOR  ( x2 );
2302     pxl = e1;
2303 
2304     if ( e1 > e2 )
2305     {
2306       Int  dropOutControl = left->flags & 7;
2307 
2308 
2309       FT_TRACE7(( ", drop-out=%d", dropOutControl ));
2310 
2311       if ( e1 == e2 + ras.precision )
2312       {
2313         switch ( dropOutControl )
2314         {
2315         case 0: /* simple drop-outs including stubs */
2316           pxl = e2;
2317           break;
2318 
2319         case 4: /* smart drop-outs including stubs */
2320           pxl = FLOOR( ( x1 + x2 - 1 ) / 2 + ras.precision_half );
2321           break;
2322 
2323         case 1: /* simple drop-outs excluding stubs */
2324         case 5: /* smart drop-outs excluding stubs  */
2325 
2326           /* Drop-out Control Rules #4 and #6 */
2327 
2328           /* The specification neither provides an exact definition */
2329           /* of a `stub' nor gives exact rules to exclude them.     */
2330           /*                                                        */
2331           /* Here the constraints we use to recognize a stub.       */
2332           /*                                                        */
2333           /*  upper stub:                                           */
2334           /*                                                        */
2335           /*   - P_Left and P_Right are in the same contour         */
2336           /*   - P_Right is the successor of P_Left in that contour */
2337           /*   - y is the top of P_Left and P_Right                 */
2338           /*                                                        */
2339           /*  lower stub:                                           */
2340           /*                                                        */
2341           /*   - P_Left and P_Right are in the same contour         */
2342           /*   - P_Left is the successor of P_Right in that contour */
2343           /*   - y is the bottom of P_Left                          */
2344           /*                                                        */
2345           /* We draw a stub if the following constraints are met.   */
2346           /*                                                        */
2347           /*   - for an upper or lower stub, there is top or bottom */
2348           /*     overshoot, respectively                            */
2349           /*   - the covered interval is greater or equal to a half */
2350           /*     pixel                                              */
2351 
2352           /* upper stub test */
2353           if ( left->next == right                &&
2354                left->height <= 0                  &&
2355                !( left->flags & Overshoot_Top   &&
2356                   x2 - x1 >= ras.precision_half ) )
2357             goto Exit;
2358 
2359           /* lower stub test */
2360           if ( right->next == left                 &&
2361                left->start == y                    &&
2362                !( left->flags & Overshoot_Bottom &&
2363                   x2 - x1 >= ras.precision_half  ) )
2364             goto Exit;
2365 
2366           if ( dropOutControl == 1 )
2367             pxl = e2;
2368           else
2369             pxl = FLOOR( ( x1 + x2 - 1 ) / 2 + ras.precision_half );
2370           break;
2371 
2372         default: /* modes 2, 3, 6, 7 */
2373           goto Exit;  /* no drop-out control */
2374         }
2375 
2376         /* undocumented but confirmed: If the drop-out would result in a  */
2377         /* pixel outside of the bounding box, use the pixel inside of the */
2378         /* bounding box instead                                           */
2379         if ( pxl < 0 )
2380           pxl = e1;
2381         else if ( TRUNC( pxl ) >= ras.bWidth )
2382           pxl = e2;
2383 
2384         /* check that the other pixel isn't set */
2385         e1 = pxl == e1 ? e2 : e1;
2386 
2387         e1 = TRUNC( e1 );
2388 
2389         c1 = (Short)( e1 >> 3 );
2390         f1 = (Short)( e1 &  7 );
2391 
2392         if ( e1 >= 0 && e1 < ras.bWidth                      &&
2393              ras.bTarget[ras.traceOfs + c1] & ( 0x80 >> f1 ) )
2394           goto Exit;
2395       }
2396       else
2397         goto Exit;
2398     }
2399 
2400     e1 = TRUNC( pxl );
2401 
2402     if ( e1 >= 0 && e1 < ras.bWidth )
2403     {
2404       FT_TRACE7(( " -> x=%d (drop-out)", e1 ));
2405 
2406       c1 = (Short)( e1 >> 3 );
2407       f1 = (Short)( e1 & 7 );
2408 
2409       ras.bTarget[ras.traceOfs + c1] |= (char)( 0x80 >> f1 );
2410     }
2411 
2412   Exit:
2413     FT_TRACE7(( "\n" ));
2414   }
2415 
2416 
2417   static void
Vertical_Sweep_Step(RAS_ARG)2418   Vertical_Sweep_Step( RAS_ARG )
2419   {
2420     ras.traceOfs += ras.traceIncr;
2421   }
2422 
2423 
2424   /***********************************************************************/
2425   /*                                                                     */
2426   /*  Horizontal Sweep Procedure Set                                     */
2427   /*                                                                     */
2428   /*  These four routines are used during the horizontal black/white     */
2429   /*  sweep phase by the generic Draw_Sweep() function.                  */
2430   /*                                                                     */
2431   /***********************************************************************/
2432 
2433   static void
Horizontal_Sweep_Init(RAS_ARGS Short * min,Short * max)2434   Horizontal_Sweep_Init( RAS_ARGS Short*  min,
2435                                   Short*  max )
2436   {
2437     /* nothing, really */
2438     FT_UNUSED_RASTER;
2439     FT_UNUSED( min );
2440     FT_UNUSED( max );
2441   }
2442 
2443 
2444   static void
Horizontal_Sweep_Span(RAS_ARGS Short y,FT_F26Dot6 x1,FT_F26Dot6 x2,PProfile left,PProfile right)2445   Horizontal_Sweep_Span( RAS_ARGS Short       y,
2446                                   FT_F26Dot6  x1,
2447                                   FT_F26Dot6  x2,
2448                                   PProfile    left,
2449                                   PProfile    right )
2450   {
2451     FT_UNUSED( left );
2452     FT_UNUSED( right );
2453 
2454 
2455     if ( x2 - x1 < ras.precision )
2456     {
2457       Long  e1, e2;
2458 
2459 
2460       FT_TRACE7(( "  x=%d y=[%.12f;%.12f]",
2461                   y,
2462                   x1 / (double)ras.precision,
2463                   x2 / (double)ras.precision ));
2464 
2465       e1 = CEILING( x1 );
2466       e2 = FLOOR  ( x2 );
2467 
2468       if ( e1 == e2 )
2469       {
2470         e1 = TRUNC( e1 );
2471 
2472         if ( e1 >= 0 && (ULong)e1 < ras.target.rows )
2473         {
2474           Byte   f1;
2475           PByte  bits;
2476           PByte  p;
2477 
2478 
2479           FT_TRACE7(( " -> y=%d (drop-out)", e1 ));
2480 
2481           bits = ras.bTarget + ( y >> 3 );
2482           f1   = (Byte)( 0x80 >> ( y & 7 ) );
2483           p    = bits - e1 * ras.target.pitch;
2484 
2485           if ( ras.target.pitch > 0 )
2486             p += (Long)( ras.target.rows - 1 ) * ras.target.pitch;
2487 
2488           p[0] |= f1;
2489         }
2490       }
2491 
2492       FT_TRACE7(( "\n" ));
2493     }
2494   }
2495 
2496 
2497   static void
Horizontal_Sweep_Drop(RAS_ARGS Short y,FT_F26Dot6 x1,FT_F26Dot6 x2,PProfile left,PProfile right)2498   Horizontal_Sweep_Drop( RAS_ARGS Short       y,
2499                                   FT_F26Dot6  x1,
2500                                   FT_F26Dot6  x2,
2501                                   PProfile    left,
2502                                   PProfile    right )
2503   {
2504     Long   e1, e2, pxl;
2505     PByte  bits;
2506     Byte   f1;
2507 
2508 
2509     FT_TRACE7(( "  x=%d y=[%.12f;%.12f]",
2510                 y,
2511                 x1 / (double)ras.precision,
2512                 x2 / (double)ras.precision ));
2513 
2514     /* During the horizontal sweep, we only take care of drop-outs */
2515 
2516     /* e1     +       <-- pixel center */
2517     /*        |                        */
2518     /* x1  ---+-->    <-- contour      */
2519     /*        |                        */
2520     /*        |                        */
2521     /* x2  <--+---    <-- contour      */
2522     /*        |                        */
2523     /*        |                        */
2524     /* e2     +       <-- pixel center */
2525 
2526     e1  = CEILING( x1 );
2527     e2  = FLOOR  ( x2 );
2528     pxl = e1;
2529 
2530     if ( e1 > e2 )
2531     {
2532       Int  dropOutControl = left->flags & 7;
2533 
2534 
2535       FT_TRACE7(( ", dropout=%d", dropOutControl ));
2536 
2537       if ( e1 == e2 + ras.precision )
2538       {
2539         switch ( dropOutControl )
2540         {
2541         case 0: /* simple drop-outs including stubs */
2542           pxl = e2;
2543           break;
2544 
2545         case 4: /* smart drop-outs including stubs */
2546           pxl = FLOOR( ( x1 + x2 - 1 ) / 2 + ras.precision_half );
2547           break;
2548 
2549         case 1: /* simple drop-outs excluding stubs */
2550         case 5: /* smart drop-outs excluding stubs  */
2551           /* see Vertical_Sweep_Drop for details */
2552 
2553           /* rightmost stub test */
2554           if ( left->next == right                &&
2555                left->height <= 0                  &&
2556                !( left->flags & Overshoot_Top   &&
2557                   x2 - x1 >= ras.precision_half ) )
2558             goto Exit;
2559 
2560           /* leftmost stub test */
2561           if ( right->next == left                 &&
2562                left->start == y                    &&
2563                !( left->flags & Overshoot_Bottom &&
2564                   x2 - x1 >= ras.precision_half  ) )
2565             goto Exit;
2566 
2567           if ( dropOutControl == 1 )
2568             pxl = e2;
2569           else
2570             pxl = FLOOR( ( x1 + x2 - 1 ) / 2 + ras.precision_half );
2571           break;
2572 
2573         default: /* modes 2, 3, 6, 7 */
2574           goto Exit;  /* no drop-out control */
2575         }
2576 
2577         /* undocumented but confirmed: If the drop-out would result in a  */
2578         /* pixel outside of the bounding box, use the pixel inside of the */
2579         /* bounding box instead                                           */
2580         if ( pxl < 0 )
2581           pxl = e1;
2582         else if ( (ULong)( TRUNC( pxl ) ) >= ras.target.rows )
2583           pxl = e2;
2584 
2585         /* check that the other pixel isn't set */
2586         e1 = pxl == e1 ? e2 : e1;
2587 
2588         e1 = TRUNC( e1 );
2589 
2590         bits = ras.bTarget + ( y >> 3 );
2591         f1   = (Byte)( 0x80 >> ( y & 7 ) );
2592 
2593         bits -= e1 * ras.target.pitch;
2594         if ( ras.target.pitch > 0 )
2595           bits += (Long)( ras.target.rows - 1 ) * ras.target.pitch;
2596 
2597         if ( e1 >= 0                     &&
2598              (ULong)e1 < ras.target.rows &&
2599              *bits & f1                  )
2600           goto Exit;
2601       }
2602       else
2603         goto Exit;
2604     }
2605 
2606     e1 = TRUNC( pxl );
2607 
2608     if ( e1 >= 0 && (ULong)e1 < ras.target.rows )
2609     {
2610       FT_TRACE7(( " -> y=%d (drop-out)", e1 ));
2611 
2612       bits  = ras.bTarget + ( y >> 3 );
2613       f1    = (Byte)( 0x80 >> ( y & 7 ) );
2614       bits -= e1 * ras.target.pitch;
2615 
2616       if ( ras.target.pitch > 0 )
2617         bits += (Long)( ras.target.rows - 1 ) * ras.target.pitch;
2618 
2619       bits[0] |= f1;
2620     }
2621 
2622   Exit:
2623     FT_TRACE7(( "\n" ));
2624   }
2625 
2626 
2627   static void
Horizontal_Sweep_Step(RAS_ARG)2628   Horizontal_Sweep_Step( RAS_ARG )
2629   {
2630     /* Nothing, really */
2631     FT_UNUSED_RASTER;
2632   }
2633 
2634 
2635   /*************************************************************************/
2636   /*                                                                       */
2637   /*  Generic Sweep Drawing routine                                        */
2638   /*                                                                       */
2639   /*************************************************************************/
2640 
2641   static Bool
Draw_Sweep(RAS_ARG)2642   Draw_Sweep( RAS_ARG )
2643   {
2644     Short         y, y_change, y_height;
2645 
2646     PProfile      P, Q, P_Left, P_Right;
2647 
2648     Short         min_Y, max_Y, top, bottom, dropouts;
2649 
2650     Long          x1, x2, xs, e1, e2;
2651 
2652     TProfileList  waiting;
2653     TProfileList  draw_left, draw_right;
2654 
2655 
2656     /* initialize empty linked lists */
2657 
2658     Init_Linked( &waiting );
2659 
2660     Init_Linked( &draw_left  );
2661     Init_Linked( &draw_right );
2662 
2663     /* first, compute min and max Y */
2664 
2665     P     = ras.fProfile;
2666     max_Y = (Short)TRUNC( ras.minY );
2667     min_Y = (Short)TRUNC( ras.maxY );
2668 
2669     while ( P )
2670     {
2671       Q = P->link;
2672 
2673       bottom = (Short)P->start;
2674       top    = (Short)( P->start + P->height - 1 );
2675 
2676       if ( min_Y > bottom )
2677         min_Y = bottom;
2678       if ( max_Y < top )
2679         max_Y = top;
2680 
2681       P->X = 0;
2682       InsNew( &waiting, P );
2683 
2684       P = Q;
2685     }
2686 
2687     /* check the Y-turns */
2688     if ( ras.numTurns == 0 )
2689     {
2690       ras.error = FT_THROW( Invalid );
2691       return FAILURE;
2692     }
2693 
2694     /* now initialize the sweep */
2695 
2696     ras.Proc_Sweep_Init( RAS_VARS &min_Y, &max_Y );
2697 
2698     /* then compute the distance of each profile from min_Y */
2699 
2700     P = waiting;
2701 
2702     while ( P )
2703     {
2704       P->countL = P->start - min_Y;
2705       P = P->link;
2706     }
2707 
2708     /* let's go */
2709 
2710     y        = min_Y;
2711     y_height = 0;
2712 
2713     if ( ras.numTurns > 0                     &&
2714          ras.sizeBuff[-ras.numTurns] == min_Y )
2715       ras.numTurns--;
2716 
2717     while ( ras.numTurns > 0 )
2718     {
2719       /* check waiting list for new activations */
2720 
2721       P = waiting;
2722 
2723       while ( P )
2724       {
2725         Q = P->link;
2726         P->countL -= y_height;
2727         if ( P->countL == 0 )
2728         {
2729           DelOld( &waiting, P );
2730 
2731           if ( P->flags & Flow_Up )
2732             InsNew( &draw_left,  P );
2733           else
2734             InsNew( &draw_right, P );
2735         }
2736 
2737         P = Q;
2738       }
2739 
2740       /* sort the drawing lists */
2741 
2742       Sort( &draw_left );
2743       Sort( &draw_right );
2744 
2745       y_change = (Short)ras.sizeBuff[-ras.numTurns--];
2746       y_height = (Short)( y_change - y );
2747 
2748       while ( y < y_change )
2749       {
2750         /* let's trace */
2751 
2752         dropouts = 0;
2753 
2754         P_Left  = draw_left;
2755         P_Right = draw_right;
2756 
2757         while ( P_Left )
2758         {
2759           x1 = P_Left ->X;
2760           x2 = P_Right->X;
2761 
2762           if ( x1 > x2 )
2763           {
2764             xs = x1;
2765             x1 = x2;
2766             x2 = xs;
2767           }
2768 
2769           e1 = FLOOR( x1 );
2770           e2 = CEILING( x2 );
2771 
2772           if ( x2 - x1 <= ras.precision &&
2773                e1 != x1 && e2 != x2     )
2774           {
2775             if ( e1 > e2 || e2 == e1 + ras.precision )
2776             {
2777               Int  dropOutControl = P_Left->flags & 7;
2778 
2779 
2780               if ( dropOutControl != 2 )
2781               {
2782                 /* a drop-out was detected */
2783 
2784                 P_Left ->X = x1;
2785                 P_Right->X = x2;
2786 
2787                 /* mark profile for drop-out processing */
2788                 P_Left->countL = 1;
2789                 dropouts++;
2790               }
2791 
2792               goto Skip_To_Next;
2793             }
2794           }
2795 
2796           ras.Proc_Sweep_Span( RAS_VARS y, x1, x2, P_Left, P_Right );
2797 
2798         Skip_To_Next:
2799 
2800           P_Left  = P_Left->link;
2801           P_Right = P_Right->link;
2802         }
2803 
2804         /* handle drop-outs _after_ the span drawing --       */
2805         /* drop-out processing has been moved out of the loop */
2806         /* for performance tuning                             */
2807         if ( dropouts > 0 )
2808           goto Scan_DropOuts;
2809 
2810       Next_Line:
2811 
2812         ras.Proc_Sweep_Step( RAS_VAR );
2813 
2814         y++;
2815 
2816         if ( y < y_change )
2817         {
2818           Sort( &draw_left  );
2819           Sort( &draw_right );
2820         }
2821       }
2822 
2823       /* now finalize the profiles that need it */
2824 
2825       P = draw_left;
2826       while ( P )
2827       {
2828         Q = P->link;
2829         if ( P->height == 0 )
2830           DelOld( &draw_left, P );
2831         P = Q;
2832       }
2833 
2834       P = draw_right;
2835       while ( P )
2836       {
2837         Q = P->link;
2838         if ( P->height == 0 )
2839           DelOld( &draw_right, P );
2840         P = Q;
2841       }
2842     }
2843 
2844     /* for gray-scaling, flush the bitmap scanline cache */
2845     while ( y <= max_Y )
2846     {
2847       ras.Proc_Sweep_Step( RAS_VAR );
2848       y++;
2849     }
2850 
2851     return SUCCESS;
2852 
2853   Scan_DropOuts:
2854 
2855     P_Left  = draw_left;
2856     P_Right = draw_right;
2857 
2858     while ( P_Left )
2859     {
2860       if ( P_Left->countL )
2861       {
2862         P_Left->countL = 0;
2863 #if 0
2864         dropouts--;  /* -- this is useful when debugging only */
2865 #endif
2866         ras.Proc_Sweep_Drop( RAS_VARS y,
2867                                       P_Left->X,
2868                                       P_Right->X,
2869                                       P_Left,
2870                                       P_Right );
2871       }
2872 
2873       P_Left  = P_Left->link;
2874       P_Right = P_Right->link;
2875     }
2876 
2877     goto Next_Line;
2878   }
2879 
2880 
2881   /*************************************************************************/
2882   /*                                                                       */
2883   /* <Function>                                                            */
2884   /*    Render_Single_Pass                                                 */
2885   /*                                                                       */
2886   /* <Description>                                                         */
2887   /*    Perform one sweep with sub-banding.                                */
2888   /*                                                                       */
2889   /* <Input>                                                               */
2890   /*    flipped :: If set, flip the direction of the outline.              */
2891   /*                                                                       */
2892   /* <Return>                                                              */
2893   /*    Renderer error code.                                               */
2894   /*                                                                       */
2895   static int
Render_Single_Pass(RAS_ARGS Bool flipped)2896   Render_Single_Pass( RAS_ARGS Bool  flipped )
2897   {
2898     Short  i, j, k;
2899 
2900 
2901     while ( ras.band_top >= 0 )
2902     {
2903       ras.maxY = (Long)ras.band_stack[ras.band_top].y_max * ras.precision;
2904       ras.minY = (Long)ras.band_stack[ras.band_top].y_min * ras.precision;
2905 
2906       ras.top = ras.buff;
2907 
2908       ras.error = Raster_Err_None;
2909 
2910       if ( Convert_Glyph( RAS_VARS flipped ) )
2911       {
2912         if ( ras.error != Raster_Err_Overflow )
2913           return FAILURE;
2914 
2915         ras.error = Raster_Err_None;
2916 
2917         /* sub-banding */
2918 
2919 #ifdef DEBUG_RASTER
2920         ClearBand( RAS_VARS TRUNC( ras.minY ), TRUNC( ras.maxY ) );
2921 #endif
2922 
2923         i = ras.band_stack[ras.band_top].y_min;
2924         j = ras.band_stack[ras.band_top].y_max;
2925 
2926         k = (Short)( ( i + j ) / 2 );
2927 
2928         if ( ras.band_top >= 7 || k < i )
2929         {
2930           ras.band_top = 0;
2931           ras.error    = FT_THROW( Invalid );
2932 
2933           return ras.error;
2934         }
2935 
2936         ras.band_stack[ras.band_top + 1].y_min = k;
2937         ras.band_stack[ras.band_top + 1].y_max = j;
2938 
2939         ras.band_stack[ras.band_top].y_max = (Short)( k - 1 );
2940 
2941         ras.band_top++;
2942       }
2943       else
2944       {
2945         if ( ras.fProfile )
2946           if ( Draw_Sweep( RAS_VAR ) )
2947              return ras.error;
2948         ras.band_top--;
2949       }
2950     }
2951 
2952     return SUCCESS;
2953   }
2954 
2955 
2956   /*************************************************************************/
2957   /*                                                                       */
2958   /* <Function>                                                            */
2959   /*    Render_Glyph                                                       */
2960   /*                                                                       */
2961   /* <Description>                                                         */
2962   /*    Render a glyph in a bitmap.  Sub-banding if needed.                */
2963   /*                                                                       */
2964   /* <Return>                                                              */
2965   /*    FreeType error code.  0 means success.                             */
2966   /*                                                                       */
2967   static FT_Error
Render_Glyph(RAS_ARG)2968   Render_Glyph( RAS_ARG )
2969   {
2970     FT_Error  error;
2971 
2972 
2973     Set_High_Precision( RAS_VARS ras.outline.flags &
2974                                  FT_OUTLINE_HIGH_PRECISION );
2975     ras.scale_shift = ras.precision_shift;
2976 
2977     if ( ras.outline.flags & FT_OUTLINE_IGNORE_DROPOUTS )
2978       ras.dropOutControl = 2;
2979     else
2980     {
2981       if ( ras.outline.flags & FT_OUTLINE_SMART_DROPOUTS )
2982         ras.dropOutControl = 4;
2983       else
2984         ras.dropOutControl = 0;
2985 
2986       if ( !( ras.outline.flags & FT_OUTLINE_INCLUDE_STUBS ) )
2987         ras.dropOutControl += 1;
2988     }
2989 
2990     ras.second_pass = (Bool)( !( ras.outline.flags      &
2991                                  FT_OUTLINE_SINGLE_PASS ) );
2992 
2993     /* Vertical Sweep */
2994     FT_TRACE7(( "Vertical pass (ftraster)\n" ));
2995 
2996     ras.Proc_Sweep_Init = Vertical_Sweep_Init;
2997     ras.Proc_Sweep_Span = Vertical_Sweep_Span;
2998     ras.Proc_Sweep_Drop = Vertical_Sweep_Drop;
2999     ras.Proc_Sweep_Step = Vertical_Sweep_Step;
3000 
3001     ras.band_top            = 0;
3002     ras.band_stack[0].y_min = 0;
3003     ras.band_stack[0].y_max = (Short)( ras.target.rows - 1 );
3004 
3005     ras.bWidth  = (UShort)ras.target.width;
3006     ras.bTarget = (Byte*)ras.target.buffer;
3007 
3008     if ( ( error = Render_Single_Pass( RAS_VARS 0 ) ) != 0 )
3009       return error;
3010 
3011     /* Horizontal Sweep */
3012     if ( ras.second_pass && ras.dropOutControl != 2 )
3013     {
3014       FT_TRACE7(( "Horizontal pass (ftraster)\n" ));
3015 
3016       ras.Proc_Sweep_Init = Horizontal_Sweep_Init;
3017       ras.Proc_Sweep_Span = Horizontal_Sweep_Span;
3018       ras.Proc_Sweep_Drop = Horizontal_Sweep_Drop;
3019       ras.Proc_Sweep_Step = Horizontal_Sweep_Step;
3020 
3021       ras.band_top            = 0;
3022       ras.band_stack[0].y_min = 0;
3023       ras.band_stack[0].y_max = (Short)( ras.target.width - 1 );
3024 
3025       if ( ( error = Render_Single_Pass( RAS_VARS 1 ) ) != 0 )
3026         return error;
3027     }
3028 
3029     return Raster_Err_None;
3030   }
3031 
3032 
3033   static void
ft_black_init(black_PRaster raster)3034   ft_black_init( black_PRaster  raster )
3035   {
3036     FT_UNUSED( raster );
3037   }
3038 
3039 
3040   /**** RASTER OBJECT CREATION: In standalone mode, we simply use *****/
3041   /****                         a static object.                  *****/
3042 
3043 
3044 #ifdef _STANDALONE_
3045 
3046 
3047   static int
ft_black_new(void * memory,FT_Raster * araster)3048   ft_black_new( void*       memory,
3049                 FT_Raster  *araster )
3050   {
3051      static black_TRaster  the_raster;
3052      FT_UNUSED( memory );
3053 
3054 
3055      *araster = (FT_Raster)&the_raster;
3056      FT_MEM_ZERO( &the_raster, sizeof ( the_raster ) );
3057      ft_black_init( &the_raster );
3058 
3059      return 0;
3060   }
3061 
3062 
3063   static void
ft_black_done(FT_Raster raster)3064   ft_black_done( FT_Raster  raster )
3065   {
3066     /* nothing */
3067     FT_UNUSED( raster );
3068   }
3069 
3070 
3071 #else /* !_STANDALONE_ */
3072 
3073 
3074   static int
ft_black_new(FT_Memory memory,black_PRaster * araster)3075   ft_black_new( FT_Memory       memory,
3076                 black_PRaster  *araster )
3077   {
3078     FT_Error       error;
3079     black_PRaster  raster = NULL;
3080 
3081 
3082     *araster = 0;
3083     if ( !FT_NEW( raster ) )
3084     {
3085       raster->memory = memory;
3086       ft_black_init( raster );
3087 
3088       *araster = raster;
3089     }
3090 
3091     return error;
3092   }
3093 
3094 
3095   static void
ft_black_done(black_PRaster raster)3096   ft_black_done( black_PRaster  raster )
3097   {
3098     FT_Memory  memory = (FT_Memory)raster->memory;
3099 
3100 
3101     FT_FREE( raster );
3102   }
3103 
3104 
3105 #endif /* !_STANDALONE_ */
3106 
3107 
3108   static void
ft_black_reset(black_PRaster raster,char * pool_base,Long pool_size)3109   ft_black_reset( black_PRaster  raster,
3110                   char*          pool_base,
3111                   Long           pool_size )
3112   {
3113     FT_UNUSED( raster );
3114     FT_UNUSED( pool_base );
3115     FT_UNUSED( pool_size );
3116   }
3117 
3118 
3119   static int
ft_black_set_mode(black_PRaster raster,ULong mode,const char * palette)3120   ft_black_set_mode( black_PRaster  raster,
3121                      ULong          mode,
3122                      const char*    palette )
3123   {
3124     FT_UNUSED( raster );
3125     FT_UNUSED( mode );
3126     FT_UNUSED( palette );
3127 
3128     return 0;
3129   }
3130 
3131 
3132   static int
ft_black_render(black_PRaster raster,const FT_Raster_Params * params)3133   ft_black_render( black_PRaster            raster,
3134                    const FT_Raster_Params*  params )
3135   {
3136     const FT_Outline*  outline    = (const FT_Outline*)params->source;
3137     const FT_Bitmap*   target_map = params->target;
3138 
3139     black_TWorker  worker[1];
3140 
3141     Long  buffer[FT_MAX( FT_RENDER_POOL_SIZE, 2048 ) / sizeof ( Long )];
3142 
3143 
3144     if ( !raster )
3145       return FT_THROW( Not_Ini );
3146 
3147     if ( !outline )
3148       return FT_THROW( Invalid );
3149 
3150     /* return immediately if the outline is empty */
3151     if ( outline->n_points == 0 || outline->n_contours <= 0 )
3152       return Raster_Err_None;
3153 
3154     if ( !outline->contours || !outline->points )
3155       return FT_THROW( Invalid );
3156 
3157     if ( outline->n_points !=
3158            outline->contours[outline->n_contours - 1] + 1 )
3159       return FT_THROW( Invalid );
3160 
3161     /* this version of the raster does not support direct rendering, sorry */
3162     if ( params->flags & FT_RASTER_FLAG_DIRECT )
3163       return FT_THROW( Unsupported );
3164 
3165     if ( params->flags & FT_RASTER_FLAG_AA )
3166       return FT_THROW( Unsupported );
3167 
3168     if ( !target_map )
3169       return FT_THROW( Invalid );
3170 
3171     /* nothing to do */
3172     if ( !target_map->width || !target_map->rows )
3173       return Raster_Err_None;
3174 
3175     if ( !target_map->buffer )
3176       return FT_THROW( Invalid );
3177 
3178     ras.outline = *outline;
3179     ras.target  = *target_map;
3180 
3181     worker->buff     = buffer;
3182     worker->sizeBuff = (&buffer)[1]; /* Points to right after buffer. */
3183 
3184     return Render_Glyph( RAS_VAR );
3185   }
3186 
3187 
3188   FT_DEFINE_RASTER_FUNCS(
3189     ft_standard_raster,
3190 
3191     FT_GLYPH_FORMAT_OUTLINE,
3192 
3193     (FT_Raster_New_Func)     ft_black_new,
3194     (FT_Raster_Reset_Func)   ft_black_reset,
3195     (FT_Raster_Set_Mode_Func)ft_black_set_mode,
3196     (FT_Raster_Render_Func)  ft_black_render,
3197     (FT_Raster_Done_Func)    ft_black_done )
3198 
3199 
3200 /* END */
3201