From 4447b2c84e66699b52d0f207ad2604276b0e1a40 Mon Sep 17 00:00:00 2001 From: Werner Lemberg Date: Thu, 6 Jun 2013 21:28:36 +0200 Subject: [PATCH] [cff] Add early exit feature for width-only calls. This is for `FT_Get_Advance'. There are 7 places where the spec says the width can be defined: hstem/hstemhm vstem/vstemhm cntrmask/hintmask hmoveto vmoveto rmoveto endchar * src/cff/cf2intrp.c (cf2_doStems): Exit early for width-only calls, if possible. (cf2_interpT2CharString) , , , , , , : Exit early for width-only calls. --- ChangeLog | 23 +++++++++++++++ src/cff/cf2intrp.c | 73 ++++++++++++++++++++++++++++++++++------------ 2 files changed, 78 insertions(+), 18 deletions(-) diff --git a/ChangeLog b/ChangeLog index 269d91744..afb006001 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,26 @@ +2013-06-06 Dave Arnold + + [cff] Add early exit feature for width-only calls. + + This is for `FT_Get_Advance'. + + There are 7 places where the spec says the width can be defined: + + hstem/hstemhm + vstem/vstemhm + cntrmask/hintmask + hmoveto + vmoveto + rmoveto + endchar + + * src/cff/cf2intrp.c (cf2_doStems): Exit early for width-only calls, + if possible. + + (cf2_interpT2CharString) , , + , , , + , : Exit early for width-only calls. + 2013-06-06 Werner Lemberg Next round of compiler fixes. diff --git a/src/cff/cf2intrp.c b/src/cff/cf2intrp.c index 71a15fe03..5b73e6049 100644 --- a/src/cff/cf2intrp.c +++ b/src/cff/cf2intrp.c @@ -292,6 +292,12 @@ /* variable accumulates delta values from operand stack */ CF2_Fixed position = hintOffset; + if ( hasWidthArg && ! *haveWidth ) + *width = cf2_stack_getReal( opStack, 0 ) + + cf2_getNominalWidthX( font->decoder ); + + if ( font->decoder->width_only ) + goto exit; for ( i = hasWidthArg ? 1 : 0; i < count; i += 2 ) { @@ -311,13 +317,11 @@ cf2_arrstack_push( stemHintArray, &stemhint ); /* defer error check */ } - if ( hasWidthArg && ! *haveWidth ) - *width = cf2_stack_getReal( opStack, 0 ) + - cf2_getNominalWidthX( font->decoder ); - - *haveWidth = TRUE; - cf2_stack_clear( opStack ); + + exit: + /* cf2_doStems must define a width (may be default) */ + *haveWidth = TRUE; } @@ -507,6 +511,9 @@ * What we implement here uses the first validly specified width, but * does not detect errors for specifying more than one width. * + * If one of the above operators occurs without explicitly specifying + * a width, we assume the default width. + * */ haveWidth = FALSE; *width = cf2_getDefaultWidthX( decoder ); @@ -595,6 +602,10 @@ width, &haveWidth, 0 ); + + if ( font->decoder->width_only ) + goto exit; + break; case cf2_cmdVSTEMHM: @@ -612,19 +623,28 @@ width, &haveWidth, 0 ); + + if ( font->decoder->width_only ) + goto exit; + break; case cf2_cmdVMOVETO: FT_TRACE4(( " vmoveto\n" )); + if ( cf2_stack_count( opStack ) > 1 && !haveWidth ) + *width = cf2_stack_getReal( opStack, 0 ) + nominalWidthX; + + /* width is defined or default after this */ + haveWidth = TRUE; + + if ( font->decoder->width_only ) + goto exit; + curY += cf2_stack_popFixed( opStack ); cf2_glyphpath_moveTo( &glyphPath, curX, curY ); - if ( cf2_stack_count( opStack ) > 0 && !haveWidth ) - *width = cf2_stack_popFixed( opStack ) + nominalWidthX; - - haveWidth = TRUE; break; case cf2_cmdRLINETO: @@ -1048,8 +1068,12 @@ *width = cf2_stack_getReal( opStack, 0 ) + nominalWidthX; } + /* width is defined or default after this */ haveWidth = TRUE; + if ( font->decoder->width_only ) + goto exit; + /* close path if still open */ cf2_glyphpath_closeOpenPath( &glyphPath ); @@ -1133,6 +1157,9 @@ &haveWidth, 0 ); + if ( font->decoder->width_only ) + goto exit; + if ( op1 == cf2_cmdHINTMASK ) { /* consume the hint mask bytes which follow the operator */ @@ -1183,28 +1210,38 @@ case cf2_cmdRMOVETO: FT_TRACE4(( " rmoveto\n" )); + if ( cf2_stack_count( opStack ) > 2 && !haveWidth ) + *width = cf2_stack_getReal( opStack, 0 ) + nominalWidthX; + + /* width is defined or default after this */ + haveWidth = TRUE; + + if ( font->decoder->width_only ) + goto exit; + curY += cf2_stack_popFixed( opStack ); curX += cf2_stack_popFixed( opStack ); cf2_glyphpath_moveTo( &glyphPath, curX, curY ); - if ( cf2_stack_count( opStack ) > 0 && !haveWidth ) - *width = cf2_stack_popFixed( opStack ) + nominalWidthX; - - haveWidth = TRUE; break; case cf2_cmdHMOVETO: FT_TRACE4(( " hmoveto\n" )); + if ( cf2_stack_count( opStack ) > 1 && !haveWidth ) + *width = cf2_stack_getReal( opStack, 0 ) + nominalWidthX; + + /* width is defined or default after this */ + haveWidth = TRUE; + + if ( font->decoder->width_only ) + goto exit; + curX += cf2_stack_popFixed( opStack ); cf2_glyphpath_moveTo( &glyphPath, curX, curY ); - if ( cf2_stack_count( opStack ) > 0 && !haveWidth ) - *width = cf2_stack_popFixed( opStack ) + nominalWidthX; - - haveWidth = TRUE; break; case cf2_cmdRLINECURVE: