1 /***************************************************************************/
5 /* CID-keyed Type1 parser (body). */
7 /* Copyright 1996-2001, 2002, 2003, 2004, 2005, 2006, 2007 by */
8 /* David Turner, Robert Wilhelm, and Werner Lemberg. */
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. */
16 /***************************************************************************/
20 #include FT_INTERNAL_DEBUG_H
21 #include FT_INTERNAL_CALC_H
22 #include FT_INTERNAL_OBJECTS_H
23 #include FT_INTERNAL_STREAM_H
30 /*************************************************************************/
32 /* The macro FT_COMPONENT is used in trace mode. It is an implicit */
33 /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */
34 /* messages during execution. */
37 #define FT_COMPONENT trace_cidparse
40 /*************************************************************************/
41 /*************************************************************************/
42 /*************************************************************************/
44 /***** INPUT STREAM PARSER *****/
46 /*************************************************************************/
47 /*************************************************************************/
48 /*************************************************************************/
51 FT_LOCAL_DEF( FT_Error
)
52 cid_parser_new( CID_Parser
* parser
,
58 FT_ULong base_offset
, offset
, ps_len
;
63 FT_MEM_ZERO( parser
, sizeof ( *parser
) );
64 psaux
->ps_parser_funcs
->init( &parser
->root
, 0, 0, memory
);
66 parser
->stream
= stream
;
68 base_offset
= FT_STREAM_POS();
70 /* first of all, check the font format in the header */
71 if ( FT_FRAME_ENTER( 31 ) )
74 if ( ft_strncmp( (char *)stream
->cursor
,
75 "%!PS-Adobe-3.0 Resource-CIDFont", 31 ) )
77 FT_TRACE2(( "[not a valid CID-keyed font]\n" ));
78 error
= CID_Err_Unknown_File_Format
;
86 /* now, read the rest of the file until we find */
87 /* `StartData' or `/sfnts' */
89 FT_Byte buffer
[256 + 10];
90 FT_Int read_len
= 256 + 10;
94 for ( offset
= (FT_ULong
)FT_STREAM_POS(); ; offset
+= 256 )
99 stream_len
= stream
->size
- FT_STREAM_POS();
100 if ( stream_len
== 0 )
103 read_len
= FT_MIN( read_len
, stream_len
);
104 if ( FT_STREAM_READ( p
, read_len
) )
107 if ( read_len
< 256 )
110 limit
= p
+ read_len
- 10;
112 for ( p
= buffer
; p
< limit
; p
++ )
114 if ( p
[0] == 'S' && ft_strncmp( (char*)p
, "StartData", 9 ) == 0 )
116 /* save offset of binary data after `StartData' */
117 offset
+= p
- buffer
+ 10;
120 else if ( p
[1] == 's' && ft_strncmp( (char*)p
, "/sfnts", 6 ) == 0 )
122 offset
+= p
- buffer
+ 7;
127 FT_MEM_MOVE( buffer
, p
, 10 );
134 /* We have found the start of the binary data or the `/sfnts' token. */
135 /* Now rewind and extract the frame corresponding to this PostScript */
138 ps_len
= offset
- base_offset
;
139 if ( FT_STREAM_SEEK( base_offset
) ||
140 FT_FRAME_EXTRACT( ps_len
, parser
->postscript
) )
143 parser
->data_offset
= offset
;
144 parser
->postscript_len
= ps_len
;
145 parser
->root
.base
= parser
->postscript
;
146 parser
->root
.cursor
= parser
->postscript
;
147 parser
->root
.limit
= parser
->root
.cursor
+ ps_len
;
148 parser
->num_dict
= -1;
150 /* Finally, we check whether `StartData' or `/sfnts' was real -- */
151 /* it could be in a comment or string. We also get the arguments */
152 /* of `StartData' to find out whether the data is represented in */
153 /* binary or hex format. */
155 arg1
= parser
->root
.cursor
;
156 cid_parser_skip_PS_token( parser
);
157 cid_parser_skip_spaces ( parser
);
158 arg2
= parser
->root
.cursor
;
159 cid_parser_skip_PS_token( parser
);
160 cid_parser_skip_spaces ( parser
);
162 limit
= parser
->root
.limit
;
163 cur
= parser
->root
.cursor
;
165 while ( cur
< limit
)
167 if ( parser
->root
.error
)
170 if ( cur
[0] == 'S' && ft_strncmp( (char*)cur
, "StartData", 9 ) == 0 )
172 if ( ft_strncmp( (char*)arg1
, "(Hex)", 5 ) == 0 )
173 parser
->binary_length
= ft_atol( (const char *)arg2
);
175 limit
= parser
->root
.limit
;
176 cur
= parser
->root
.cursor
;
179 else if ( cur
[1] == 's' && ft_strncmp( (char*)cur
, "/sfnts", 6 ) == 0 )
181 FT_TRACE2(( "cid_parser_new: cannot handle Type 11 fonts\n" ));
182 error
= CID_Err_Unknown_File_Format
;
186 cid_parser_skip_PS_token( parser
);
187 cid_parser_skip_spaces ( parser
);
190 cur
= parser
->root
.cursor
;
193 /* we haven't found the correct `StartData'; go back and continue */
195 FT_FRAME_RELEASE( parser
->postscript
);
196 if ( !FT_STREAM_SEEK( offset
) )
205 cid_parser_done( CID_Parser
* parser
)
207 /* always free the private dictionary */
208 if ( parser
->postscript
)
210 FT_Stream stream
= parser
->stream
;
213 FT_FRAME_RELEASE( parser
->postscript
);
215 parser
->root
.funcs
.done( &parser
->root
);