1 /***************************************************************************/
2 /*                                                                         */
3 /*  cidparse.c                                                             */
4 /*                                                                         */
5 /*    CID-keyed Type1 parser (body).                                       */
6 /*                                                                         */
7 /*  Copyright 1996-2007, 2009, 2013, 2014 by                               */
8 /*  David Turner, Robert Wilhelm, and Werner Lemberg.                      */
9 /*                                                                         */
10 /*  This file is part of the FreeType project, and may only be used,       */
11 /*  modified, and distributed under the terms of the FreeType project      */
12 /*  license, LICENSE.TXT.  By continuing to use, modify, or distribute     */
13 /*  this file you indicate that you have read the license and              */
14 /*  understand and accept it fully.                                        */
15 /*                                                                         */
16 /***************************************************************************/
17 
18 
19 #include <ft2build.h>
20 #include FT_INTERNAL_DEBUG_H
21 #include FT_INTERNAL_OBJECTS_H
22 #include FT_INTERNAL_STREAM_H
23 
24 #include "cidparse.h"
25 
26 #include "ciderrs.h"
27 
28 
29   /*************************************************************************/
30   /*                                                                       */
31   /* The macro FT_COMPONENT is used in trace mode.  It is an implicit      */
32   /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log  */
33   /* messages during execution.                                            */
34   /*                                                                       */
35 #undef  FT_COMPONENT
36 #define FT_COMPONENT  trace_cidparse
37 
38 
39   /*************************************************************************/
40   /*************************************************************************/
41   /*************************************************************************/
42   /*****                                                               *****/
43   /*****                    INPUT STREAM PARSER                        *****/
44   /*****                                                               *****/
45   /*************************************************************************/
46   /*************************************************************************/
47   /*************************************************************************/
48 
49 
50   FT_LOCAL_DEF( FT_Error )
cid_parser_new(CID_Parser * parser,FT_Stream stream,FT_Memory memory,PSAux_Service psaux)51   cid_parser_new( CID_Parser*    parser,
52                   FT_Stream      stream,
53                   FT_Memory      memory,
54                   PSAux_Service  psaux )
55   {
56     FT_Error  error;
57     FT_ULong  base_offset, offset, ps_len;
58     FT_Byte   *cur, *limit;
59     FT_Byte   *arg1, *arg2;
60 
61 
62     FT_MEM_ZERO( parser, sizeof ( *parser ) );
63     psaux->ps_parser_funcs->init( &parser->root, 0, 0, memory );
64 
65     parser->stream = stream;
66 
67     base_offset = FT_STREAM_POS();
68 
69     /* first of all, check the font format in the header */
70     if ( FT_FRAME_ENTER( 31 ) )
71       goto Exit;
72 
73     if ( ft_strncmp( (char *)stream->cursor,
74                      "%!PS-Adobe-3.0 Resource-CIDFont", 31 ) )
75     {
76       FT_TRACE2(( "  not a CID-keyed font\n" ));
77       error = FT_THROW( Unknown_File_Format );
78     }
79 
80     FT_FRAME_EXIT();
81     if ( error )
82       goto Exit;
83 
84   Again:
85     /* now, read the rest of the file until we find */
86     /* `StartData' or `/sfnts'                      */
87     {
88       FT_Byte   buffer[256 + 10];
89       FT_Long   read_len = 256 + 10; /* same as signed FT_Stream->size */
90       FT_Byte*  p        = buffer;
91 
92 
93       for ( offset = FT_STREAM_POS(); ; offset += 256 )
94       {
95         FT_Long  stream_len; /* same as signed FT_Stream->size */
96 
97 
98         stream_len = stream->size - FT_STREAM_POS();
99         if ( stream_len == 0 )
100         {
101           FT_TRACE2(( "cid_parser_new: no `StartData' keyword found\n" ));
102           error = FT_THROW( Invalid_File_Format );
103           goto Exit;
104         }
105 
106         read_len = FT_MIN( read_len, stream_len );
107         if ( FT_STREAM_READ( p, read_len ) )
108           goto Exit;
109 
110         if ( read_len < 256 )
111           p[read_len]  = '\0';
112 
113         limit = p + read_len - 10;
114 
115         for ( p = buffer; p < limit; p++ )
116         {
117           if ( p[0] == 'S' && ft_strncmp( (char*)p, "StartData", 9 ) == 0 )
118           {
119             /* save offset of binary data after `StartData' */
120             offset += (FT_ULong)( p - buffer + 10 );
121             goto Found;
122           }
123           else if ( p[1] == 's' && ft_strncmp( (char*)p, "/sfnts", 6 ) == 0 )
124           {
125             offset += (FT_ULong)( p - buffer + 7 );
126             goto Found;
127           }
128         }
129 
130         FT_MEM_MOVE( buffer, p, 10 );
131         read_len = 256;
132         p = buffer + 10;
133       }
134     }
135 
136   Found:
137     /* We have found the start of the binary data or the `/sfnts' token. */
138     /* Now rewind and extract the frame corresponding to this PostScript */
139     /* section.                                                          */
140 
141     ps_len = offset - base_offset;
142     if ( FT_STREAM_SEEK( base_offset )                  ||
143          FT_FRAME_EXTRACT( ps_len, parser->postscript ) )
144       goto Exit;
145 
146     parser->data_offset    = offset;
147     parser->postscript_len = ps_len;
148     parser->root.base      = parser->postscript;
149     parser->root.cursor    = parser->postscript;
150     parser->root.limit     = parser->root.cursor + ps_len;
151     parser->num_dict       = -1;
152 
153     /* Finally, we check whether `StartData' or `/sfnts' was real --  */
154     /* it could be in a comment or string.  We also get the arguments */
155     /* of `StartData' to find out whether the data is represented in  */
156     /* binary or hex format.                                          */
157 
158     arg1 = parser->root.cursor;
159     cid_parser_skip_PS_token( parser );
160     cid_parser_skip_spaces  ( parser );
161     arg2 = parser->root.cursor;
162     cid_parser_skip_PS_token( parser );
163     cid_parser_skip_spaces  ( parser );
164 
165     limit = parser->root.limit;
166     cur   = parser->root.cursor;
167 
168     while ( cur < limit )
169     {
170       if ( parser->root.error )
171       {
172         error = parser->root.error;
173         goto Exit;
174       }
175 
176       if ( cur[0] == 'S' && ft_strncmp( (char*)cur, "StartData", 9 ) == 0 )
177       {
178         if ( ft_strncmp( (char*)arg1, "(Hex)", 5 ) == 0 )
179           parser->binary_length = ft_atol( (const char *)arg2 );
180 
181         goto Exit;
182       }
183       else if ( cur[1] == 's' && ft_strncmp( (char*)cur, "/sfnts", 6 ) == 0 )
184       {
185         FT_TRACE2(( "cid_parser_new: cannot handle Type 11 fonts\n" ));
186         error = FT_THROW( Unknown_File_Format );
187         goto Exit;
188       }
189 
190       cid_parser_skip_PS_token( parser );
191       cid_parser_skip_spaces  ( parser );
192       arg1 = arg2;
193       arg2 = cur;
194       cur  = parser->root.cursor;
195     }
196 
197     /* we haven't found the correct `StartData'; go back and continue */
198     /* searching                                                      */
199     FT_FRAME_RELEASE( parser->postscript );
200     if ( !FT_STREAM_SEEK( offset ) )
201       goto Again;
202 
203   Exit:
204     return error;
205   }
206 
207 
208   FT_LOCAL_DEF( void )
cid_parser_done(CID_Parser * parser)209   cid_parser_done( CID_Parser*  parser )
210   {
211     /* always free the private dictionary */
212     if ( parser->postscript )
213     {
214       FT_Stream  stream = parser->stream;
215 
216 
217       FT_FRAME_RELEASE( parser->postscript );
218     }
219     parser->root.funcs.done( &parser->root );
220   }
221 
222 
223 /* END */
224