Coverage Report

Created: 2025-12-31 07:31

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/pdf/pdf_cmap.c
Line
Count
Source
1
/* Copyright (C) 2020-2025 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
#include "strmio.h"
17
#include "stream.h"
18
#include "scanchar.h"
19
20
#include "pdf_int.h"
21
#include "pdf_cmap.h"
22
23
#include "pdf_stack.h"
24
#include "pdf_dict.h"
25
#include "pdf_file.h"
26
#include "pdf_fontps.h"
27
#include "pdf_deref.h"
28
29
static int pdfi_free_cmap_contents(pdf_cmap *cmap);
30
31
static int cmap_usecmap_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
32
394
{
33
394
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
34
394
    pdf_name *n = NULL;
35
394
    pdf_cmap *upcmap = NULL;
36
394
    int code = 0;
37
38
394
    if (pdf_ps_stack_count(s) < 1)
39
3
        return_error(gs_error_stackunderflow);
40
41
    /* If we've already got some definitions, ignore the usecmap op */
42
391
    if (pdficmap->code_space.num_ranges == 0) {
43
391
        byte *nstr = NULL;
44
391
        int len = s->cur[0].size;
45
46
391
        if (pdf_ps_obj_has_type(&(s->cur[0]), PDF_PS_OBJ_NAME)) {
47
391
            nstr = s->cur[0].val.name;
48
391
        }
49
0
        else if (pdf_ps_obj_has_type(&(s->cur[0]), PDF_PS_OBJ_STRING)) {
50
0
            nstr = s->cur[0].val.string;
51
0
        }
52
0
        else {
53
0
            code = gs_note_error(gs_error_typecheck);
54
0
        }
55
391
        if (code >= 0)
56
391
            code = pdfi_name_alloc(pdficmap->ctx, nstr, len, (pdf_obj **)&n);
57
391
        if (code >= 0) {
58
391
            pdfi_countup(n);
59
391
            code = pdfi_read_cmap(pdficmap->ctx, (pdf_obj *)n, &upcmap);
60
391
            if (code >= 0) {
61
367
                gx_code_space_range_t * ranges =
62
367
                         (gx_code_space_range_t *)gs_alloc_byte_array(mem, upcmap->code_space.num_ranges,
63
367
                          sizeof(gx_code_space_range_t), "cmap_usecmap_func(ranges)");
64
367
                if (ranges != NULL) {
65
367
                    int i;
66
367
                    memcpy(&pdficmap->code_space, &upcmap->code_space, sizeof(pdficmap->code_space));
67
851
                    for (i = 0; i < upcmap->code_space.num_ranges; i++) {
68
484
                        memcpy(&(ranges[i]), &(upcmap->code_space.ranges[i]), sizeof(ranges[i]));
69
484
                    }
70
367
                    pdficmap->code_space.ranges = ranges;
71
367
                    memcpy(&pdficmap->cmap_range, &upcmap->cmap_range, sizeof(pdficmap->cmap_range));
72
367
                    memcpy(&pdficmap->notdef_cmap_range, &upcmap->notdef_cmap_range, sizeof(pdficmap->notdef_cmap_range));
73
                    /* Once we've assumed control of these, NULL out entries for the sub-cmap. */
74
367
                    upcmap->cmap_range.ranges = NULL;
75
367
                    upcmap->notdef_cmap_range.ranges = NULL;
76
                    /* But we keep the subcmap itself because we rely on its storage */
77
367
                    pdficmap->next = upcmap;
78
367
                    pdfi_countup(upcmap);
79
367
                }
80
367
            }
81
391
        }
82
391
    }
83
391
    pdfi_countdown(upcmap);
84
391
    pdfi_countdown(n);
85
391
    if (code < 0) {
86
24
        (void)pdf_ps_stack_pop(s, 1);
87
24
        return code;
88
24
    }
89
367
    return pdf_ps_stack_pop(s, 1);
90
391
}
91
92
#if 0 /* no longer used */
93
static int cmap_pushmark_func(gs_memory_t *mem, pdf_ps_ctx_t *stack, pdf_cmap *pdficmap)
94
{
95
    return pdf_ps_stack_push_mark(stack);
96
}
97
#endif
98
99
100
static int cmap_endcodespacerange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
101
100k
{
102
100k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
103
100k
    int i, numranges, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
104
100k
    gx_code_space_t *code_space = &pdficmap->code_space;
105
100k
    int nr = code_space->num_ranges;
106
100k
    gx_code_space_range_t *gcsr = code_space->ranges;
107
108
    /* increment to_pop to cover the mark object */
109
100k
    numranges = to_pop++;
110
100k
    while (numranges % 2) numranges--;
111
112
    /* The following hard limit is an implementation limit */
113
100k
    if (numranges < 0 || numranges + 1 > PDF_PS_STACK_MAX) {
114
510
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "cmap_endcodespacerange_func", NULL);
115
510
        return_error(gs_error_syntaxerror);
116
510
    }
117
118
    /* The following "soft" limit is according to the spec */
119
99.6k
    if (numranges > 200) {
120
0
        int code;
121
0
        if ((code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_syntaxerror), NULL, W_PDF_LIMITCHECK_TYPE0_CMAP, "cmap_endcodespacerange_func", NULL)) < 0) {
122
0
            (void)pdf_ps_stack_pop(s, to_pop);
123
0
            return code;
124
0
        }
125
0
    }
126
127
99.6k
    if (numranges > 0
128
99.6k
     && pdf_ps_obj_has_type(&(s->cur[0]), PDF_PS_OBJ_STRING)  && s->cur[0].size <= MAX_CMAP_CODE_SIZE
129
99.6k
     && pdf_ps_obj_has_type(&(s->cur[-1]), PDF_PS_OBJ_STRING) && s->cur[-1].size <= MAX_CMAP_CODE_SIZE) {
130
131
99.5k
        code_space->num_ranges += numranges >> 1;
132
133
99.5k
        code_space->ranges = (gx_code_space_range_t *)gs_alloc_byte_array(mem, code_space->num_ranges,
134
99.5k
                          sizeof(gx_code_space_range_t), "cmap_endcodespacerange_func(ranges)");
135
99.5k
        if (code_space->ranges != NULL) {
136
99.5k
            if (nr > 0) {
137
85
                memcpy(code_space->ranges, gcsr, nr * sizeof(gx_code_space_range_t));
138
85
                gs_free_object(mem, gcsr, "cmap_endcodespacerange_func(gcsr");
139
85
            }
140
141
200k
            for (i = nr; i < code_space->num_ranges; i++) {
142
100k
                int si = i - nr;
143
100k
                int s1 = s->cur[-((si * 2) + 1)].size < MAX_CMAP_CODE_SIZE ? s->cur[-((si * 2) + 1)].size : MAX_CMAP_CODE_SIZE;
144
100k
                int s2 = s->cur[-(si * 2)].size < MAX_CMAP_CODE_SIZE ? s->cur[-(si * 2)].size : MAX_CMAP_CODE_SIZE;
145
146
100k
                memcpy(code_space->ranges[i].first, s->cur[-((si * 2) + 1)].val.string, s1);
147
100k
                memcpy(code_space->ranges[i].last, s->cur[-(si * 2)].val.string, s2);
148
100k
                code_space->ranges[i].size = s->cur[-(si * 2)].size;
149
100k
            }
150
99.5k
        }
151
0
        else {
152
0
            (void)pdf_ps_stack_pop(s, to_pop);
153
0
            return_error(gs_error_VMerror);
154
0
        }
155
99.5k
    }
156
99.6k
    return pdf_ps_stack_pop(s, to_pop);
157
99.6k
}
158
159
static int cmap_insert_map(pdfi_cmap_range_t *cmap_range, pdfi_cmap_range_map_t *pdfir)
160
17.9M
{
161
17.9M
    if (cmap_range->ranges == NULL) {
162
96.6k
        cmap_range->ranges = cmap_range->ranges_tail = pdfir;
163
96.6k
    }
164
17.8M
    else {
165
17.8M
        cmap_range->ranges_tail->next = pdfir;
166
17.8M
        cmap_range->ranges_tail = pdfir;
167
17.8M
    }
168
17.9M
    cmap_range->numrangemaps++;
169
17.9M
    return 0;
170
17.9M
}
171
172
static int general_endcidrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, pdf_cmap *pdficmap, pdfi_cmap_range_t *cmap_range)
173
163k
{
174
163k
    int ncodemaps, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
175
163k
    unsigned int i, j;
176
163k
    pdfi_cmap_range_map_t *pdfir;
177
163k
    pdf_ps_stack_object_t *stobj;
178
179
    /* increment to_pop to cover the mark object */
180
163k
    ncodemaps = to_pop++;
181
    /* mapping should have 3 objects on the stack:
182
     * startcode, endcode and basecid
183
     */
184
163k
    while (ncodemaps % 3) ncodemaps--;
185
186
    /* The following hard limit is an implementation limit */
187
163k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
188
11
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "general_endcidrange_func", NULL);
189
11
        return_error(gs_error_syntaxerror);
190
11
    }
191
192
    /* The following "soft" limit is according to the spec */
193
163k
    if (ncodemaps > 300) {
194
3
        int code;
195
3
        if ((code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_syntaxerror), NULL, W_PDF_LIMITCHECK_TYPE0_CMAP, "general_endcidrange_func", NULL)) < 0) {
196
0
            (void)pdf_ps_stack_pop(s, to_pop);
197
0
            return code;
198
0
        }
199
3
    }
200
201
163k
    stobj = &s->cur[-ncodemaps] + 1;
202
203
14.0M
    for (i = 0; i < ncodemaps; i += 3) {
204
13.8M
        int preflen, valuelen;
205
206
13.8M
        if (pdf_ps_obj_has_type(&(stobj[i + 2]), PDF_PS_OBJ_INTEGER)
207
13.8M
        &&  pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_STRING)
208
13.8M
        &&  pdf_ps_obj_has_type(&(stobj[i]), PDF_PS_OBJ_STRING)
209
13.8M
        &&  pdf_ps_obj_size(&(stobj[i])) > 0
210
13.8M
        &&  pdf_ps_obj_size(&(stobj[i])) == pdf_ps_obj_size(&(stobj[i + 1]))){
211
212
13.8M
            uint cidbase = stobj[i + 2].val.i;
213
214
            /* First, find the length of the prefix */
215
28.3M
            for (preflen = 0; preflen < stobj[i].size; preflen++) {
216
27.7M
                if(stobj[i].val.string[preflen] != stobj[i + 1].val.string[preflen]) {
217
13.2M
                    break;
218
13.2M
                }
219
27.7M
            }
220
221
13.8M
            if (preflen == stobj[i].size) {
222
609k
                preflen = 1;
223
609k
            }
224
225
13.8M
            if (preflen > MAX_CMAP_CODE_SIZE || stobj[i].size - preflen > MAX_CMAP_CODE_SIZE || stobj[i + 1].size - preflen > MAX_CMAP_CODE_SIZE
226
13.8M
                || ((int64_t)stobj[i].size) - preflen < 0 || ((int64_t)stobj[i + 1].size) - preflen < 0) {
227
0
                (void)pdf_ps_stack_pop(s, to_pop);
228
0
                return_error(gs_error_syntaxerror);
229
0
            }
230
231
            /* Find how many bytes we need for the cidbase value */
232
            /* We always store at least two bytes for the cidbase value */
233
13.8M
            for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
234
13.8M
                DO_NOTHING;
235
236
13.8M
            valuelen = ((valuelen + 7) & ~7) >> 3;
237
238
            /* The prefix is already directly in the gx_cmap_lookup_range_t
239
             * We need to store the lower and upper character codes, after lopping the prefix
240
             * off them. The upper and lower codes must be the same number of bytes.
241
             */
242
13.8M
            j = sizeof(pdfi_cmap_range_map_t) + 2 * (stobj[i].size - preflen) + valuelen;
243
244
13.8M
            pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
245
13.8M
            if (pdfir != NULL) {
246
13.8M
                gx_cmap_lookup_range_t *gxr = &pdfir->range;
247
13.8M
                pdfir->next = NULL;
248
13.8M
                gxr->num_entries = 1;
249
13.8M
                gxr->keys.data = (byte *)&(pdfir[1]);
250
13.8M
                gxr->values.data = gxr->keys.data + 2 * (stobj[i].size - preflen);
251
252
13.8M
                gxr->cmap = NULL;
253
13.8M
                gxr->font_index = 0;
254
13.8M
                gxr->key_is_range = true;
255
13.8M
                gxr->value_type = cmap_range == &(pdficmap->cmap_range) ? CODE_VALUE_CID : CODE_VALUE_NOTDEF;
256
13.8M
                gxr->key_prefix_size = preflen;
257
13.8M
                gxr->key_size = stobj[i].size - gxr->key_prefix_size;
258
13.8M
                memcpy(gxr->key_prefix, stobj[i].val.string, gxr->key_prefix_size);
259
260
13.8M
                memcpy(gxr->keys.data, stobj[i].val.string + gxr->key_prefix_size, stobj[i].size - gxr->key_prefix_size);
261
13.8M
                memcpy(gxr->keys.data + (stobj[i].size - gxr->key_prefix_size), stobj[i + 1].val.string + gxr->key_prefix_size, stobj[i + 1].size - gxr->key_prefix_size);
262
263
13.8M
                gxr->keys.size = (stobj[i].size - gxr->key_prefix_size) + (stobj[i + 1].size - gxr->key_prefix_size);
264
41.6M
                for (j = 0; j < valuelen; j++) {
265
27.7M
                    gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
266
27.7M
                }
267
13.8M
                gxr->value_size = valuelen; /* I'm not sure.... */
268
13.8M
                gxr->values.size = valuelen;
269
13.8M
                if (cmap_insert_map(cmap_range, pdfir) < 0) break;
270
13.8M
            }
271
0
            else {
272
0
                (void)pdf_ps_stack_pop(s, to_pop);
273
0
                return_error(gs_error_VMerror);
274
0
            }
275
13.8M
        }
276
13.8M
    }
277
163k
    return pdf_ps_stack_pop(s, to_pop);
278
163k
}
279
280
static int cmap_endcidrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
281
162k
{
282
162k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
283
162k
    return general_endcidrange_func(mem, s, pdficmap, &pdficmap->cmap_range);
284
162k
}
285
286
static int cmap_endnotdefrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
287
392
{
288
392
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
289
392
    return general_endcidrange_func(mem, s, pdficmap, &pdficmap->notdef_cmap_range);
290
392
}
291
292
static int cmap_endfbrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
293
25.7k
{
294
25.7k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
295
25.7k
    int ncodemaps, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
296
25.7k
    unsigned int i, j, k;
297
25.7k
    pdfi_cmap_range_map_t *pdfir;
298
25.7k
    pdf_ps_stack_object_t *stobj;
299
300
    /* increment to_pop to cover the mark object */
301
25.7k
    ncodemaps = to_pop++;
302
    /* mapping should have 3 objects on the stack
303
     */
304
26.8k
    while (ncodemaps % 3) ncodemaps--;
305
306
    /* The following hard limit is an implementation limit */
307
25.7k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
308
30
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "cmap_endfbrange_func", NULL);
309
30
        return_error(gs_error_syntaxerror);
310
30
    }
311
    /* The following "soft" limit is according to the spec */
312
313
25.7k
    if (ncodemaps > 300) {
314
302
        int code;
315
302
        if ((code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_syntaxerror), NULL, W_PDF_LIMITCHECK_TYPE0_CMAP, "cmap_endfbrange_func", NULL)) < 0) {
316
0
            (void)pdf_ps_stack_pop(s, to_pop);
317
0
            return code;
318
0
        }
319
302
    }
320
321
25.7k
    stobj = &s->cur[-ncodemaps] + 1;
322
637k
    for (i = 0; i < ncodemaps; i += 3) {
323
        /* Lazy: to make the loop below simpler, put single
324
           values into a one element array
325
         */
326
611k
        if (pdf_ps_obj_has_type(&(stobj[i + 2]), PDF_PS_OBJ_STRING)) {
327
591k
            pdf_ps_stack_object_t *arr;
328
591k
            arr = (pdf_ps_stack_object_t *) gs_alloc_bytes(mem, sizeof(pdf_ps_stack_object_t), "cmap_endfbrange_func(pdf_ps_stack_object_t");
329
591k
            if (arr == NULL) {
330
0
                (void)pdf_ps_stack_pop(s, to_pop);
331
0
                return_error(gs_error_VMerror);
332
0
            }
333
591k
            else {
334
591k
                memcpy(arr, &(stobj[i + 2]), sizeof(pdf_ps_stack_object_t));
335
591k
                pdf_ps_make_array(&(stobj[i + 2]), arr, 1);
336
591k
            }
337
591k
        }
338
611k
    }
339
340
25.7k
    stobj = &s->cur[-ncodemaps] + 1;
341
342
634k
    for (i = 0; i < ncodemaps; i += 3) {
343
608k
        int preflen, valuelen;
344
345
608k
        if (pdf_ps_obj_has_type(&(stobj[i + 2]), PDF_PS_OBJ_ARRAY)
346
594k
        &&  pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_STRING)
347
586k
        &&  pdf_ps_obj_has_type(&(stobj[i]), PDF_PS_OBJ_STRING)
348
582k
        &&  pdf_ps_obj_size(&(stobj[i + 1])) > 0
349
582k
        &&  pdf_ps_obj_size(&(stobj[i])) > 0){
350
351
582k
            uint cidbase;
352
582k
            int m, size;
353
354
582k
            if (stobj[i + 2].size < 1)
355
32
                continue;
356
357
582k
            else if (stobj[i + 2].size == 1) {
358
580k
                if (stobj[i + 2].val.arr[0].type != PDF_PS_OBJ_STRING)
359
0
                     continue;
360
361
580k
                size = stobj[i + 2].val.arr[0].size;
362
363
580k
                cidbase = 0;
364
1.74M
                for (m = 0; m < size; m++) {
365
1.16M
                    cidbase |= stobj[i + 2].val.arr[0].val.string[size - m - 1] << (8 * m);
366
1.16M
                }
367
368
                /* First, find the length of the prefix */
369
1.21M
                for (preflen = 0; preflen < stobj[i].size; preflen++) {
370
988k
                    if(stobj[i].val.string[preflen] != stobj[i + 1].val.string[preflen]) {
371
350k
                        break;
372
350k
                    }
373
988k
                }
374
375
580k
                if (preflen == stobj[i].size) {
376
230k
                    preflen = 1;
377
230k
                }
378
379
580k
                if (preflen > MAX_CMAP_CODE_SIZE || stobj[i].size - preflen > MAX_CMAP_CODE_SIZE || stobj[i + 1].size - preflen > MAX_CMAP_CODE_SIZE
380
580k
                    || ((int64_t)stobj[i].size) - preflen < 0 || ((int64_t)stobj[i + 1].size) - preflen < 0) {
381
98
                    (void)pdf_ps_stack_pop(s, to_pop);
382
98
                    return_error(gs_error_syntaxerror);
383
98
                }
384
385
                /* Find how many bytes we need for the cidbase value */
386
                /* We always store at least two bytes for the cidbase value */
387
639k
                for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
388
580k
                    DO_NOTHING;
389
390
580k
                valuelen = ((valuelen + 7) & ~7) >> 3;
391
392
                /* The prefix is already directly in the gx_cmap_lookup_range_t
393
                 * We need to store the lower and upper character codes, after lopping the prefix
394
                 * off them. The upper and lower codes must be the same number of bytes.
395
                 */
396
580k
                j = sizeof(pdfi_cmap_range_map_t) + 2 * (stobj[i].size - preflen) + valuelen;
397
398
580k
                pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
399
580k
                if (pdfir != NULL) {
400
580k
                    gx_cmap_lookup_range_t *gxr = &pdfir->range;
401
580k
                    pdfir->next = NULL;
402
580k
                    gxr->num_entries = 1;
403
580k
                    gxr->keys.data = (byte *)&(pdfir[1]);
404
580k
                    gxr->values.data = gxr->keys.data + 2 * (stobj[i].size - preflen);
405
406
580k
                    gxr->cmap = NULL;
407
580k
                    gxr->font_index = 0;
408
580k
                    gxr->key_is_range = true;
409
580k
                    gxr->value_type = CODE_VALUE_CID;
410
580k
                    gxr->key_prefix_size = preflen;
411
580k
                    gxr->key_size = stobj[i].size - gxr->key_prefix_size;
412
580k
                    memcpy(gxr->key_prefix, stobj[i].val.string, gxr->key_prefix_size);
413
414
580k
                    memcpy(gxr->keys.data, stobj[i].val.string + gxr->key_prefix_size, stobj[i].size - gxr->key_prefix_size);
415
580k
                    memcpy(gxr->keys.data + (stobj[i].size - gxr->key_prefix_size), stobj[i + 1].val.string + gxr->key_prefix_size, stobj[i + 1].size - gxr->key_prefix_size);
416
417
580k
                    gxr->keys.size = (stobj[i].size - gxr->key_prefix_size) + (stobj[i + 1].size - gxr->key_prefix_size);
418
1.74M
                    for (j = 0; j < valuelen; j++) {
419
1.16M
                        gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
420
1.16M
                    }
421
580k
                    gxr->value_size = valuelen; /* I'm not sure.... */
422
580k
                    gxr->values.size = valuelen;
423
580k
                    if (cmap_insert_map(&pdficmap->cmap_range, pdfir) < 0) break;
424
580k
                }
425
0
                else {
426
0
                    (void)pdf_ps_stack_pop(s, to_pop);
427
0
                    return_error(gs_error_VMerror);
428
0
                }
429
580k
            }
430
1.67k
            else {
431
1.67k
                int m, size, keysize;
432
1.67k
                uint codelo = 0, codehi = 0;
433
434
1.67k
                size = stobj[i].size;
435
4.66k
                for (m = 0; m < size; m++) {
436
2.98k
                    codelo |= stobj[i].val.string[size - m - 1] << (8 * m);
437
2.98k
                }
438
1.67k
                size = stobj[i + 1].size;
439
4.66k
                for (m = 0; m < size; m++) {
440
2.98k
                    codehi |= stobj[i + 1].val.string[size - m - 1] << (8 * m);
441
2.98k
                }
442
443
1.67k
                if (codehi <= codelo || stobj[i + 2].size < (codehi - codelo))
444
31
                    continue;
445
446
107k
                for (k = codelo; k <= codehi; k++) {
447
106k
                    uint cidbase;
448
106k
                    int ind = k - codelo;
449
450
106k
                    if (stobj[i + 2].val.arr[ind].type != PDF_PS_OBJ_STRING)
451
11
                         continue;
452
453
106k
                    size = stobj[i + 2].val.arr[ind].size;
454
455
106k
                    cidbase = 0;
456
318k
                    for (m = 0; m < size; m++) {
457
212k
                        cidbase |= stobj[i + 2].val.arr[ind].val.string[size - m - 1] << (8 * m);
458
212k
                    }
459
                    /* Find how many bytes we need for the cidbase value */
460
                    /* We always store at least two bytes for the cidbase value */
461
106k
                    for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
462
106k
                        DO_NOTHING;
463
464
106k
                    valuelen = ((valuelen + 7) & ~7) >> 3;
465
466
                    /* Find how many bytes we need for the cidbase value */
467
                    /* We always store at least two bytes for the cidbase value */
468
106k
                    for (keysize = 16; keysize < 32 && (cidbase >> keysize) > 0; keysize += 1)
469
106k
                        DO_NOTHING;
470
471
106k
                    keysize = ((keysize + 7) & ~7) >> 3;
472
106k
                    if (keysize > MAX_CMAP_CODE_SIZE * 2) {
473
0
                        (void)pdf_ps_stack_pop(s, to_pop);
474
0
                        return_error(gs_error_syntaxerror);
475
0
                    }
476
106k
                    preflen = keysize > 4 ? 4 : keysize;
477
106k
                    keysize -= preflen;
478
479
                    /* The prefix is already directly in the gx_cmap_lookup_range_t
480
                     * We need to store the lower and upper character codes, after lopping the prefix
481
                     * off them. The upper and lower codes must be the same number of bytes.
482
                     */
483
106k
                    j = sizeof(pdfi_cmap_range_map_t) + keysize + valuelen;
484
485
106k
                    pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
486
106k
                    if (pdfir != NULL) {
487
106k
                        gx_cmap_lookup_range_t *gxr = &pdfir->range;
488
106k
                        pdfir->next = NULL;
489
106k
                        gxr->num_entries = 1;
490
106k
                        gxr->keys.data = (byte *)&(pdfir[1]);
491
106k
                        gxr->values.data = gxr->keys.data + keysize;
492
493
106k
                        gxr->cmap = NULL;
494
106k
                        gxr->font_index = 0;
495
106k
                        gxr->key_is_range = false;
496
106k
                        gxr->value_type = CODE_VALUE_CID;
497
106k
                        gxr->key_prefix_size = preflen;
498
106k
                        gxr->key_size = keysize;
499
318k
                        for (j = 0; j < preflen; j++) {
500
212k
                            gxr->key_prefix[j] = (k >> ((preflen - 1 - j) * 8)) & 255;
501
212k
                        }
502
503
106k
                        for (j = preflen; j < preflen + keysize; j++) {
504
0
                            gxr->keys.data[j] = (k >> (((preflen + keysize) - 1 - j) * 8)) & 255;
505
0
                        }
506
507
106k
                        gxr->keys.size = keysize;
508
318k
                        for (j = 0; j < valuelen; j++) {
509
212k
                            gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
510
212k
                        }
511
106k
                        gxr->value_size = valuelen; /* I'm not sure.... */
512
106k
                        gxr->values.size = valuelen;
513
106k
                        if (cmap_insert_map(&pdficmap->cmap_range, pdfir) < 0) break;
514
106k
                    }
515
0
                    else {
516
0
                        (void)pdf_ps_stack_pop(s, to_pop);
517
0
                        return_error(gs_error_VMerror);
518
0
                    }
519
106k
                }
520
1.64k
            }
521
582k
        }
522
608k
    }
523
25.6k
    return pdf_ps_stack_pop(s, to_pop);
524
25.7k
}
525
526
static int general_endcidchar_func(gs_memory_t *mem, pdf_ps_ctx_t *s, pdf_cmap *pdficmap, pdfi_cmap_range_t *cmap_range)
527
60.3k
{
528
60.3k
    int ncodemaps, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
529
60.3k
    int i, j;
530
60.3k
    pdfi_cmap_range_map_t *pdfir;
531
60.3k
    pdf_ps_stack_object_t *stobj;
532
533
    /* increment to_pop to cover the mark object */
534
60.3k
    ncodemaps = to_pop++;
535
    /* mapping should have 2 objects on the stack:
536
     * startcode, endcode and basecid
537
     */
538
60.7k
    while (ncodemaps % 2) ncodemaps--;
539
540
60.3k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
541
19
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "general_endcidchar_func", NULL);
542
19
        return_error(gs_error_syntaxerror);
543
19
    }
544
545
60.3k
    if (ncodemaps > 200) {
546
393
        int code;
547
393
        if ((code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_syntaxerror), NULL, W_PDF_LIMITCHECK_TYPE0_CMAP, "general_endcidchar_func", NULL)) < 0) {
548
0
            (void)pdf_ps_stack_pop(s, to_pop);
549
0
            return code;
550
0
        }
551
393
    }
552
553
60.3k
    stobj = &s->cur[-ncodemaps] + 1;
554
555
3.49M
    for (i = 0; i < ncodemaps; i += 2) {
556
3.43M
        int preflen = 1, valuelen;
557
558
3.43M
        if (pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_INTEGER)
559
3.41M
        &&  pdf_ps_obj_has_type(&(stobj[i]), PDF_PS_OBJ_STRING) && stobj[i].size > 0) {
560
3.38M
            uint cidbase = stobj[i + 1].val.i;
561
562
            /* Find how many bytes we need for the cidbase value */
563
            /* We always store at least two bytes for the cidbase value */
564
4.47M
            for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
565
3.38M
                DO_NOTHING;
566
567
3.38M
            preflen = stobj[i].size > 4 ? 4 : stobj[i].size;
568
569
3.38M
            valuelen = ((valuelen + 7) & ~7) >> 3;
570
571
            /* The prefix is already directly in the gx_cmap_lookup_range_t
572
             * We need to store the lower and upper character codes, after lopping the prefix
573
             * off them. The upper and lower codes must be the same number of bytes.
574
             */
575
3.38M
            j = sizeof(pdfi_cmap_range_map_t) + (stobj[i].size - preflen) + valuelen;
576
577
3.38M
            pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
578
3.38M
            if (pdfir != NULL) {
579
3.38M
                gx_cmap_lookup_range_t *gxr = &pdfir->range;
580
3.38M
                pdfir->next = NULL;
581
3.38M
                gxr->num_entries = 1;
582
3.38M
                gxr->keys.data = (byte *)&(pdfir[1]);
583
3.38M
                gxr->values.data = gxr->keys.data + (stobj[i].size - preflen);
584
585
3.38M
                gxr->cmap = NULL;
586
3.38M
                gxr->font_index = 0;
587
3.38M
                gxr->key_is_range = false;
588
3.38M
                gxr->value_type = cmap_range == &(pdficmap->cmap_range) ? CODE_VALUE_CID : CODE_VALUE_NOTDEF;
589
3.38M
                gxr->key_prefix_size = preflen;
590
3.38M
                gxr->key_size = stobj[i].size - gxr->key_prefix_size;
591
3.38M
                memcpy(gxr->key_prefix, stobj[i].val.string, gxr->key_prefix_size);
592
593
3.38M
                memcpy(gxr->keys.data, stobj[i].val.string + gxr->key_prefix_size, stobj[i].size - gxr->key_prefix_size);
594
595
3.38M
                gxr->keys.size = stobj[i].size - gxr->key_prefix_size;
596
10.3M
                for (j = 0; j < valuelen; j++) {
597
6.92M
                    gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
598
6.92M
                }
599
3.38M
                gxr->value_size = valuelen; /* I'm not sure.... */
600
3.38M
                gxr->values.size = valuelen;
601
3.38M
                if (cmap_insert_map(cmap_range, pdfir) < 0) break;
602
3.38M
            }
603
0
            else {
604
0
                (void)pdf_ps_stack_pop(s, to_pop);
605
0
                return_error(gs_error_VMerror);
606
0
            }
607
3.38M
        }
608
3.43M
    }
609
60.3k
    return pdf_ps_stack_pop(s, to_pop);
610
60.3k
}
611
612
static int cmap_endcidchar_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
613
10.1k
{
614
10.1k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
615
10.1k
    return general_endcidchar_func(mem, s, pdficmap, &pdficmap->cmap_range);
616
10.1k
}
617
618
static int cmap_endnotdefchar_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
619
0
{
620
0
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
621
0
    return general_endcidchar_func(mem, s, pdficmap, &pdficmap->notdef_cmap_range);
622
0
}
623
624
static int cmap_endbfchar_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
625
50.3k
{
626
50.3k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
627
50.3k
    int ncodemaps = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
628
50.3k
    pdf_ps_stack_object_t *stobj;
629
50.3k
    int i, j;
630
631
    /* The following hard limit is an implementation limit */
632
50.3k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
633
170
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "cmap_endbfchar_func", NULL);
634
170
        return_error(gs_error_syntaxerror);
635
170
    }
636
637
    /* The following "soft" limit is according to the spec */
638
50.1k
    if (ncodemaps > 200) {
639
389
        int code;
640
389
        if ((code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_syntaxerror), NULL, W_PDF_LIMITCHECK_TYPE0_CMAP, "cmap_endbfchar_func", NULL)) < 0) {
641
0
            (void)pdf_ps_stack_pop(s, ncodemaps);
642
0
            return code;
643
0
        }
644
389
    }
645
646
50.1k
    stobj = &s->cur[-ncodemaps] + 1;
647
648
2.56M
    for (i = 0; i < ncodemaps; i += 2) {
649
2.51M
        if (pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_STRING)) {
650
2.50M
            byte *c = stobj[i + 1].val.string;
651
2.50M
            int l = stobj[i + 1].size;
652
2.50M
            unsigned int v = 0;
653
654
7.94M
            for (j = 0; j < l; j++) {
655
5.43M
                v += c[l - j - 1] << (8 * j);
656
5.43M
            }
657
2.50M
            pdf_ps_make_int(&(stobj[i + 1]), v);
658
2.50M
        }
659
6.06k
        else {
660
6.06k
            continue;
661
6.06k
        }
662
2.51M
    }
663
50.1k
    return general_endcidchar_func(mem, s, pdficmap, &pdficmap->cmap_range);
664
50.1k
}
665
666
2.47M
#define CMAP_NAME_AND_LEN(s) PDF_PS_OPER_NAME_AND_LEN(s)
667
668
static int cmap_def_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
669
612k
{
670
612k
    int code = 0, code2 = 0;
671
612k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
672
673
612k
    if (pdf_ps_stack_count(s) < 2) {
674
53.9k
        return pdf_ps_stack_pop(s, 1);
675
53.9k
    }
676
677
558k
    if (pdf_ps_obj_has_type(&s->cur[-1], PDF_PS_OBJ_NAME)) {
678
558k
        if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("Registry"))) {
679
54.1k
            if (pdficmap->csi_reg.data != NULL)
680
23
               gs_free_object(mem, pdficmap->csi_reg.data, "cmap_def_func(Registry)");
681
682
54.1k
            pdficmap->csi_reg.data = gs_alloc_bytes(mem, (size_t)s->cur[0].size + 1, "cmap_def_func(Registry)");
683
54.1k
            if (pdficmap->csi_reg.data != NULL) {
684
54.1k
                pdficmap->csi_reg.size = s->cur[0].size;
685
54.1k
                if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_STRING)) {
686
54.1k
                    memcpy(pdficmap->csi_reg.data, s->cur[0].val.string, s->cur[0].size);
687
54.1k
                }
688
5
                else {
689
5
                    memcpy(pdficmap->csi_reg.data, s->cur[0].val.name, s->cur[0].size);
690
5
                }
691
54.1k
                pdficmap->csi_reg.data[pdficmap->csi_reg.size] = '\0';
692
54.1k
            }
693
0
            else {
694
0
                code = gs_note_error(gs_error_VMerror);
695
0
            }
696
54.1k
        }
697
503k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("Ordering"))) {
698
54.0k
            if (pdficmap->csi_ord.data != NULL)
699
23
               gs_free_object(mem, pdficmap->csi_ord.data, "cmap_def_func(Ordering)");
700
701
54.0k
            pdficmap->csi_ord.data = gs_alloc_bytes(mem, (size_t)s->cur[0].size + 1, "cmap_def_func(Ordering)");
702
54.0k
            if (pdficmap->csi_ord.data != NULL) {
703
54.0k
                pdficmap->csi_ord.size = s->cur[0].size;
704
54.0k
                if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_STRING))
705
54.0k
                    memcpy(pdficmap->csi_ord.data, s->cur[0].val.string, s->cur[0].size);
706
5
                else
707
5
                    memcpy(pdficmap->csi_ord.data, s->cur[0].val.name, s->cur[0].size);
708
54.0k
                pdficmap->csi_ord.data[pdficmap->csi_ord.size] = '\0';
709
54.0k
            }
710
0
            else {
711
0
                code = gs_note_error(gs_error_VMerror);
712
0
            }
713
54.0k
        }
714
449k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("Supplement"))) {
715
95.7k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
716
95.7k
                pdficmap->csi_supplement = s->cur[0].val.i;
717
95.7k
            }
718
0
            else {
719
0
                pdficmap->csi_supplement = 0;
720
0
            }
721
95.7k
        }
722
354k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("CMapName"))) {
723
100k
            if (pdficmap->name.data != NULL)
724
67
               gs_free_object(mem, pdficmap->name.data, "cmap_def_func(CMapName)");
725
726
100k
            pdficmap->name.data = gs_alloc_bytes(mem, (size_t)s->cur[0].size + 1, "cmap_def_func(CMapName)");
727
100k
            if (pdficmap->name.data != NULL) {
728
100k
                pdficmap->name.size = s->cur[0].size;
729
100k
                if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_STRING))
730
0
                    memcpy(pdficmap->name.data, s->cur[0].val.string, s->cur[0].size);
731
100k
                else
732
100k
                    memcpy(pdficmap->name.data, s->cur[0].val.name, s->cur[0].size);
733
100k
                pdficmap->name.data[pdficmap->name.size] = '\0';
734
100k
            }
735
0
            else {
736
0
                code = gs_note_error(gs_error_VMerror);
737
0
            }
738
100k
        }
739
253k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("CMapVersion"))) {
740
51.5k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
741
365
                pdficmap->vers = (float)s->cur[0].val.i;
742
365
            }
743
51.2k
            else if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_FLOAT)){
744
51.2k
                pdficmap->vers = s->cur[0].val.f;
745
51.2k
            }
746
0
            else {
747
0
                pdficmap->vers = (float)0;
748
0
            }
749
51.5k
        }
750
202k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("CMapType"))) {
751
98.2k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
752
98.2k
                pdficmap->cmaptype = s->cur[0].val.i;
753
98.2k
            }
754
0
            else {
755
0
                pdficmap->cmaptype = 1;
756
0
            }
757
98.2k
        }
758
104k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("XUID"))) {
759
51.1k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_ARRAY)) {
760
51.1k
                int len = s->cur->size;
761
762
51.1k
                if (pdficmap->uid.xvalues != NULL)
763
0
                   gs_free_object(mem, pdficmap->uid.xvalues, "cmap_def_func(XUID)");
764
765
51.1k
                pdficmap->uid.xvalues = (long *)gs_alloc_bytes(mem, (size_t)len * sizeof(*pdficmap->uid.xvalues), "cmap_def_func(XUID)");
766
51.1k
                if (pdficmap->uid.xvalues != NULL) {
767
51.1k
                     int i;
768
51.1k
                     pdf_ps_stack_object_t *a = s->cur->val.arr;
769
51.1k
                     pdficmap->uid.id = -len;
770
255k
                     for (i = 0; i < len; i++) {
771
204k
                         if (pdf_ps_obj_has_type(&a[i], PDF_PS_OBJ_INTEGER)) {
772
204k
                             pdficmap->uid.xvalues[i] = (long)a[i].val.i;
773
204k
                         }
774
0
                         else {
775
0
                             pdficmap->uid.xvalues[i] = 0;
776
0
                         }
777
204k
                     }
778
51.1k
                }
779
0
                else {
780
0
                    code = gs_note_error(gs_error_VMerror);
781
0
                }
782
51.1k
            }
783
51.1k
        }
784
52.9k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("WMode"))) {
785
51.9k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
786
51.9k
                if (s->cur[0].val.i != 0) {
787
379
                    if (s->cur[0].val.i != 1) {
788
12
                        code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_rangecheck), NULL, W_PDF_BAD_WMODE, "cmap_def_func", NULL);
789
12
                    }
790
379
                    pdficmap->wmode = 1;
791
379
                }
792
51.5k
                else
793
51.5k
                    pdficmap->wmode = 0;
794
51.9k
            }
795
0
            else {
796
0
                pdficmap->wmode = 0;
797
0
            }
798
51.9k
        }
799
558k
    }
800
801
558k
    code2 = pdf_ps_stack_pop(s, 2);
802
558k
    if (code < 0)
803
0
        return code;
804
558k
    else
805
558k
        return code2;
806
558k
}
807
808
static pdf_ps_oper_list_t cmap_oper_list[] =
809
{
810
  {PDF_PS_OPER_NAME_AND_LEN("usecmap"), cmap_usecmap_func},
811
  {PDF_PS_OPER_NAME_AND_LEN("usefont"), ps_pdf_null_oper_func},
812
  {PDF_PS_OPER_NAME_AND_LEN("beginusematrix"), ps_pdf_null_oper_func},
813
  {PDF_PS_OPER_NAME_AND_LEN("endusematrix"), ps_pdf_null_oper_func},
814
  {PDF_PS_OPER_NAME_AND_LEN("begincodespacerange"), pdf_ps_pop_and_pushmark_func},
815
  {PDF_PS_OPER_NAME_AND_LEN("endcodespacerange"), cmap_endcodespacerange_func},
816
  {PDF_PS_OPER_NAME_AND_LEN("begincmap"), ps_pdf_null_oper_func},
817
  {PDF_PS_OPER_NAME_AND_LEN("beginbfchar"), pdf_ps_pop_and_pushmark_func},
818
  {PDF_PS_OPER_NAME_AND_LEN("endbfchar"), cmap_endbfchar_func},
819
  {PDF_PS_OPER_NAME_AND_LEN("beginbfrange"), pdf_ps_pop_and_pushmark_func},
820
  {PDF_PS_OPER_NAME_AND_LEN("endbfrange"), cmap_endfbrange_func},
821
  {PDF_PS_OPER_NAME_AND_LEN("begincidchar"), pdf_ps_pop_and_pushmark_func},
822
  {PDF_PS_OPER_NAME_AND_LEN("endcidchar"), cmap_endcidchar_func},
823
  {PDF_PS_OPER_NAME_AND_LEN("begincidrange"), pdf_ps_pop_and_pushmark_func},
824
  {PDF_PS_OPER_NAME_AND_LEN("endcidrange"), cmap_endcidrange_func},
825
  {PDF_PS_OPER_NAME_AND_LEN("beginnotdefchar"), pdf_ps_pop_and_pushmark_func},
826
  {PDF_PS_OPER_NAME_AND_LEN("endnotdefchar"), cmap_endnotdefchar_func},
827
  {PDF_PS_OPER_NAME_AND_LEN("beginnotdefrange"), pdf_ps_pop_and_pushmark_func},
828
  {PDF_PS_OPER_NAME_AND_LEN("endnotdefrange"), cmap_endnotdefrange_func},
829
  {PDF_PS_OPER_NAME_AND_LEN("findresource"), clear_stack_oper_func},
830
  {PDF_PS_OPER_NAME_AND_LEN("dict"), pdf_ps_pop_oper_func},
831
  {PDF_PS_OPER_NAME_AND_LEN("begin"), ps_pdf_null_oper_func},
832
  {PDF_PS_OPER_NAME_AND_LEN("end"), ps_pdf_null_oper_func},
833
  {PDF_PS_OPER_NAME_AND_LEN("pop"), ps_pdf_null_oper_func},
834
  {PDF_PS_OPER_NAME_AND_LEN("def"), cmap_def_func},
835
  {PDF_PS_OPER_NAME_AND_LEN("dup"), ps_pdf_null_oper_func},
836
  {PDF_PS_OPER_NAME_AND_LEN("defineresource"), clear_stack_oper_func},
837
  {PDF_PS_OPER_NAME_AND_LEN("beginrearrangedfont"), clear_stack_oper_func}, /* we should never see this */
838
  {NULL, 0, NULL}
839
};
840
841
static int
842
pdf_cmap_open_file(pdf_context *ctx, gs_string *cmap_name, byte **buf, int64_t *buflen)
843
51.3k
{
844
51.3k
    int code = 0;
845
51.3k
    stream *s;
846
51.3k
    char fname[gp_file_name_sizeof];
847
51.3k
    const char *path_pfx = "CMap/";
848
51.3k
    fname[0] = '\0';
849
850
51.3k
    if (strlen(path_pfx) + cmap_name->size >= gp_file_name_sizeof)
851
9
        return_error(gs_error_rangecheck);
852
853
51.2k
    strncat(fname, path_pfx, strlen(path_pfx));
854
51.2k
    strncat(fname, (char *)cmap_name->data, cmap_name->size);
855
51.2k
    code = pdfi_open_resource_file(ctx, (const char *)fname, (const int)strlen(fname), &s);
856
51.2k
    if (code >= 0) {
857
50.8k
        sfseek(s, 0, SEEK_END);
858
50.8k
        *buflen = sftell(s);
859
50.8k
        sfseek(s, 0, SEEK_SET);
860
50.8k
        *buf = gs_alloc_bytes(ctx->memory, *buflen, "pdf_cmap_open_file(buf)");
861
50.8k
        if (*buf != NULL) {
862
50.8k
            sfread(*buf, 1, *buflen, s);
863
50.8k
        }
864
0
        else {
865
0
            code = gs_note_error(gs_error_VMerror);
866
0
        }
867
50.8k
        sfclose(s);
868
50.8k
    }
869
51.2k
    return code;
870
51.3k
}
871
872
static int
873
pdfi_make_gs_cmap(gs_memory_t *mem, pdf_cmap *pdficmap)
874
101k
{
875
101k
    int code = 0, i;
876
101k
    gs_cmap_adobe1_t *pgscmap = 0;
877
101k
    gx_cmap_lookup_range_t *lookups, *ndlookups = NULL;
878
101k
    pdfi_cmap_range_map_t *l;
879
    /* FIXME: We have to use gs_cmap_adobe1_alloc() to get the cmap procs
880
       but even if it gets passed num_ranges = 0 it still allocates
881
       a zero length array. Change gs_cmap_adobe1_alloc to work better
882
     */
883
101k
    if ((code = gs_cmap_adobe1_alloc(&pgscmap, pdficmap->wmode,
884
101k
                                     pdficmap->name.data,
885
101k
                                     pdficmap->name.size,
886
101k
                                     1,
887
101k
                                     0, 0, 0, 0, 0, mem)) >= 0) {
888
101k
        gs_free_object(mem, pgscmap->code_space.ranges, "empty ranges");
889
890
101k
        lookups = gs_alloc_struct_array(mem, pdficmap->cmap_range.numrangemaps,
891
101k
                               gx_cmap_lookup_range_t,
892
101k
                               &st_cmap_lookup_range_element,
893
101k
                               "pdfi_make_gs_cmap(lookup ranges)");
894
101k
        if (lookups == NULL) {
895
0
            gs_free_object(mem, pgscmap, "pdfi_make_gs_cmap(pgscmap)");
896
0
            code = gs_note_error(gs_error_VMerror);
897
0
            goto done;
898
0
        }
899
101k
        if (pdficmap->notdef_cmap_range.numrangemaps > 0){
900
430
            ndlookups = gs_alloc_struct_array(mem, pdficmap->notdef_cmap_range.numrangemaps,
901
430
                               gx_cmap_lookup_range_t,
902
430
                               &st_cmap_lookup_range_element,
903
430
                               "pdfi_make_gs_cmap(notdef lookup ranges)");
904
430
            if (ndlookups == NULL) {
905
0
                gs_free_object(mem, lookups, "pdfi_make_gs_cmap(lookups)");
906
0
                gs_free_object(mem, pgscmap, "pdfi_make_gs_cmap(pgscmap)");
907
0
                code = gs_note_error(gs_error_VMerror);
908
0
                goto done;
909
0
            }
910
430
        }
911
101k
        pgscmap->def.lookup = lookups;
912
101k
        pgscmap->def.num_lookup = pdficmap->cmap_range.numrangemaps;
913
101k
        pgscmap->notdef.lookup = ndlookups;
914
101k
        pgscmap->notdef.num_lookup = pdficmap->notdef_cmap_range.numrangemaps;
915
916
101k
        pgscmap->CIDSystemInfo[0].Registry.data = pdficmap->csi_reg.data;
917
101k
        pgscmap->CIDSystemInfo[0].Registry.size = pdficmap->csi_reg.size;
918
101k
        pgscmap->CIDSystemInfo[0].Ordering.data = pdficmap->csi_ord.data;
919
101k
        pgscmap->CIDSystemInfo[0].Ordering.size = pdficmap->csi_ord.size;
920
101k
        pgscmap->CIDSystemInfo[0].Supplement = pdficmap->csi_supplement;
921
101k
        memcpy(&pgscmap->code_space, &pdficmap->code_space, sizeof(pgscmap->code_space));
922
101k
        memcpy(&pgscmap->uid, &pdficmap->uid, sizeof(pdficmap->uid));
923
101k
        l = pdficmap->cmap_range.ranges;
924
18.0M
        for (i = 0; i < pdficmap->cmap_range.numrangemaps && l != NULL; i++) {
925
17.9M
            memcpy(&lookups[i], &l->range, sizeof(gx_cmap_lookup_range_t));
926
17.9M
            l = l->next;
927
17.9M
        }
928
929
101k
        l = pdficmap->notdef_cmap_range.ranges;
930
102k
        for (i = 0; i < pdficmap->notdef_cmap_range.numrangemaps && l != NULL; i++) {
931
430
            memcpy(&ndlookups[i], &l->range, sizeof(gx_cmap_lookup_range_t));
932
430
            l = l->next;
933
430
        }
934
935
101k
        pdficmap->gscmap = pgscmap;
936
101k
    }
937
938
101k
done:
939
101k
    return code;
940
101k
}
941
942
int
943
pdfi_read_cmap(pdf_context *ctx, pdf_obj *cmap, pdf_cmap **pcmap)
944
103k
{
945
103k
    int code = 0;
946
103k
    pdf_cmap pdficm[3] = {0};
947
103k
    pdf_cmap *pdfi_cmap = &(pdficm[1]);
948
103k
    byte *buf = NULL;
949
103k
    int64_t buflen = 0;
950
103k
    pdf_ps_ctx_t cmap_ctx;
951
952
103k
    pdfi_cmap->ctx = ctx;
953
103k
    switch (pdfi_type_of(cmap)) {
954
51.3k
        case PDF_NAME:
955
51.3k
        {
956
51.3k
            gs_string cmname;
957
51.3k
            pdf_name *cmapn = (pdf_name *)cmap;
958
51.3k
            cmname.data = cmapn->data;
959
51.3k
            cmname.size = cmapn->length;
960
51.3k
            code = pdf_cmap_open_file(ctx, &cmname, &buf, &buflen);
961
51.3k
            if (code < 0)
962
436
                goto error_out;
963
50.8k
            break;
964
51.3k
        }
965
52.1k
        case PDF_STREAM:
966
52.1k
        {
967
52.1k
            pdf_obj *ucmap;
968
52.1k
            pdf_cmap *upcmap = NULL;
969
52.1k
            pdf_dict *cmap_dict = NULL;
970
971
52.1k
            code = pdfi_dict_from_obj(ctx, cmap, &cmap_dict);
972
52.1k
            if (code < 0)
973
0
                goto error_out;
974
975
52.1k
            code = pdfi_dict_knownget(ctx, cmap_dict, "UseCMap", &ucmap);
976
52.1k
            if (code > 0) {
977
0
                code = pdfi_read_cmap(ctx, ucmap, &upcmap);
978
0
                pdfi_countdown(ucmap);
979
0
                if (code >= 0) {
980
0
                    gx_code_space_range_t * ranges =
981
0
                         (gx_code_space_range_t *)gs_alloc_byte_array(ctx->memory, upcmap->code_space.num_ranges,
982
0
                          sizeof(gx_code_space_range_t), "cmap_usecmap_func(ranges)");
983
0
                    if (ranges != NULL) {
984
0
                        int i;
985
0
                        memcpy(&pdfi_cmap->code_space, &upcmap->code_space, sizeof(pdfi_cmap->code_space));
986
0
                        for (i = 0; i < upcmap->code_space.num_ranges; i++) {
987
0
                            memcpy(&(ranges[i]), &(upcmap->code_space.ranges[i]), sizeof(ranges[i]));
988
0
                        }
989
0
                        pdfi_cmap->code_space.ranges = ranges;
990
0
                        memcpy(&pdfi_cmap->cmap_range, &upcmap->cmap_range, sizeof(pdfi_cmap->cmap_range));
991
0
                        memcpy(&pdfi_cmap->notdef_cmap_range, &upcmap->notdef_cmap_range, sizeof(pdfi_cmap->notdef_cmap_range));
992
                        /* Once we've assumed control of these, NULL out entries for the sub-cmap. */
993
0
                        upcmap->cmap_range.ranges = NULL;
994
0
                        upcmap->notdef_cmap_range.ranges = NULL;
995
                        /* But we keep the subcmap itself because we rely on its storage */
996
0
                        pdfi_cmap->next = upcmap;
997
0
                    } else {
998
0
                        code = gs_note_error(gs_error_VMerror);
999
0
                        goto error_out;
1000
0
                    }
1001
0
                }
1002
0
                else {
1003
0
                    pdfi_countdown(upcmap);
1004
0
                }
1005
0
            }
1006
1007
52.1k
            code = pdfi_stream_to_buffer(ctx, (pdf_stream *)cmap, &buf, &buflen);
1008
52.1k
            if (code < 0) {
1009
236
                goto error_out;
1010
236
            }
1011
51.9k
            break;
1012
52.1k
        }
1013
51.9k
        default:
1014
84
            code = gs_note_error(gs_error_typecheck);
1015
84
            goto error_out;
1016
103k
    }
1017
102k
    pdfi_cmap->ctx = ctx;
1018
102k
    pdfi_cmap->buf = buf;
1019
102k
    pdfi_cmap->buflen = buflen;
1020
1021
    /* In case of technically invalid CMap files which do not contain a CMapType, See Bug #690737.
1022
     * This makes sure we clean up the CMap contents in pdfi_free_cmap() below.
1023
     */
1024
102k
    pdfi_cmap->cmaptype = 1;
1025
1026
102k
    pdfi_pscript_stack_init(ctx, cmap_oper_list, (void *)pdfi_cmap, &cmap_ctx);
1027
1028
102k
    code = pdfi_pscript_interpret(&cmap_ctx, buf, buflen);
1029
102k
    pdfi_pscript_stack_finit(&cmap_ctx);
1030
102k
    if (code < 0) goto error_out;
1031
1032
101k
    code = pdfi_make_gs_cmap(ctx->memory, pdfi_cmap);
1033
1034
101k
    if (code >= 0) {
1035
101k
        *pcmap = (pdf_cmap *)gs_alloc_bytes(ctx->memory, sizeof(pdf_cmap), "pdfi_read_cmap(*pcmap)");
1036
101k
        if (*pcmap != NULL) {
1037
101k
            pdfi_cmap->type = PDF_CMAP;
1038
101k
            pdfi_cmap->ctx = ctx;
1039
101k
            pdfi_cmap->refcnt = 1;
1040
101k
            pdfi_cmap->object_num = cmap->object_num;
1041
101k
            pdfi_cmap->generation_num = cmap->generation_num;
1042
101k
            pdfi_cmap->indirect_num = cmap->indirect_num;
1043
101k
            pdfi_cmap->indirect_gen = cmap->indirect_gen;
1044
101k
            memcpy(*pcmap, pdfi_cmap, sizeof(pdf_cmap));
1045
101k
            pdfi_cmap = *pcmap;
1046
            /* object_num can be zero if the dictionary was defined inline */
1047
101k
            if (pdfi_cmap->object_num != 0) {
1048
50.9k
                code = replace_cache_entry(ctx, (pdf_obj *)pdfi_cmap);
1049
50.9k
            }
1050
101k
        } else {
1051
0
            code = gs_note_error(gs_error_VMerror);
1052
0
            goto error_out;
1053
0
        }
1054
101k
    }
1055
0
    else {
1056
0
        goto error_out;
1057
0
    }
1058
101k
    return 0;
1059
1060
1.71k
error_out:
1061
1.71k
    pdfi_free_cmap_contents(pdfi_cmap);
1062
1.71k
    memset(pdfi_cmap, 0x00, sizeof(pdf_cmap));
1063
1.71k
    return code;
1064
101k
}
1065
1066
static int pdfi_free_cmap_contents(pdf_cmap *cmap)
1067
103k
{
1068
103k
    pdfi_cmap_range_map_t *pdfir;
1069
103k
    gs_cmap_adobe1_t *pgscmap = cmap->gscmap;
1070
1071
103k
    if (pgscmap != NULL) {
1072
101k
        gs_free_object(OBJ_MEMORY(cmap), pgscmap->def.lookup, "pdfi_free_cmap(def.lookup)");
1073
101k
        gs_free_object(OBJ_MEMORY(cmap), pgscmap->notdef.lookup, "pdfi_free_cmap(notdef.lookup)");
1074
101k
        (void)gs_cmap_free((gs_cmap_t *)pgscmap, OBJ_MEMORY(cmap));
1075
101k
    }
1076
103k
    gs_free_object(OBJ_MEMORY(cmap), cmap->code_space.ranges, "pdfi_free_cmap(code_space.ranges");
1077
103k
    pdfir = cmap->cmap_range.ranges;
1078
18.0M
    while (pdfir != NULL) {
1079
17.9M
        pdfi_cmap_range_map_t *pdfir2 = pdfir->next;
1080
17.9M
        gs_free_object(OBJ_MEMORY(cmap), pdfir, "pdfi_free_cmap(cmap_range.ranges");
1081
17.9M
        pdfir = pdfir2;
1082
17.9M
    }
1083
103k
    pdfir = cmap->notdef_cmap_range.ranges;
1084
103k
    while (pdfir != NULL) {
1085
392
        pdfi_cmap_range_map_t *pdfir2 = pdfir->next;
1086
392
        gs_free_object(OBJ_MEMORY(cmap), pdfir, "pdfi_free_cmap(cmap_range.ranges");
1087
392
        pdfir = pdfir2;
1088
392
    }
1089
103k
    gs_free_object(OBJ_MEMORY(cmap), cmap->csi_reg.data, "pdfi_free_cmap(csi_reg.data");
1090
103k
    gs_free_object(OBJ_MEMORY(cmap), cmap->csi_ord.data, "pdfi_free_cmap(csi_ord.data");
1091
103k
    gs_free_object(OBJ_MEMORY(cmap), cmap->name.data, "pdfi_free_cmap(name.data");
1092
103k
    gs_free_object(OBJ_MEMORY(cmap), cmap->uid.xvalues, "pdfi_free_cmap(xuid.xvalues");
1093
103k
    pdfi_countdown(cmap->next);
1094
103k
    gs_free_object(OBJ_MEMORY(cmap), cmap->buf, "pdfi_free_cmap(cmap->buf");
1095
1096
103k
    return 0;
1097
103k
}
1098
1099
int pdfi_free_cmap(pdf_obj *cmapo)
1100
101k
{
1101
101k
    pdf_cmap *cmap = (pdf_cmap *)cmapo;
1102
    /*
1103
     * Note there is some inconsistency in the various specifications regarding CMapType; the
1104
     * Adobe tech note 5014 specifically says it only documents CMaps with a CmapType of 0, the
1105
     * PLRM says that CMapType can be 0 or 1, and the two are equivalent, the PDF Reference Manual
1106
     * doesn't say, it just refers to tech note 5014 but the example has a CMapType of 1. The PDF
1107
     * Reference does describe ToUnicode CMaps which have a CMapType of 2.
1108
     */
1109
    /* Well it seems we have PDF files which use CMapType 2 CMaps as values for a /Encoding, which is
1110
     * I believe incorrect, as these are ToUnicode CMaps....
1111
     * There's nothing for it, we'll just haev to free all CMaps for now. Note for Chris when implementing
1112
     * ToUnicode CMaps, we'll obviously have to rely on the context to know whether a CMap is an Encoding
1113
     * or a ToUnicode, we cna't use the CmMapType, just as you suspected. :-(
1114
     * See bug #696449 633_R00728_E.pdf
1115
     */
1116
    /*
1117
     * For now, we represent ToUnicode CMaps (CMapType 2) in the same data structures as regular CMaps
1118
     * (CMapType 0/1) so there is no reason (yet!) to differentiate between the two.
1119
     */
1120
1121
101k
    pdfi_free_cmap_contents(cmap);
1122
    gs_free_object(OBJ_MEMORY(cmap), cmap, "pdfi_free_cmap(cmap");
1123
101k
    return 0;
1124
101k
}