Coverage Report

Created: 2025-06-10 07:27

/src/ghostpdl/pdf/pdf_cmap.c
Line
Count
Source (jump to first uncovered line)
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
291
{
33
291
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
34
291
    pdf_name *n = NULL;
35
291
    pdf_cmap *upcmap = NULL;
36
291
    int code = 0;
37
38
291
    if (pdf_ps_stack_count(s) < 1)
39
2
        return_error(gs_error_stackunderflow);
40
41
    /* If we've already got some definitions, ignore the usecmap op */
42
289
    if (pdficmap->code_space.num_ranges == 0) {
43
289
        byte *nstr = NULL;
44
289
        int len = s->cur[0].size;
45
46
289
        if (pdf_ps_obj_has_type(&(s->cur[0]), PDF_PS_OBJ_NAME)) {
47
289
            nstr = s->cur[0].val.name;
48
289
        }
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
289
        if (code >= 0)
56
289
            code = pdfi_name_alloc(pdficmap->ctx, nstr, len, (pdf_obj **)&n);
57
289
        if (code >= 0) {
58
289
            pdfi_countup(n);
59
289
            code = pdfi_read_cmap(pdficmap->ctx, (pdf_obj *)n, &upcmap);
60
289
            if (code >= 0) {
61
269
                gx_code_space_range_t * ranges =
62
269
                         (gx_code_space_range_t *)gs_alloc_byte_array(mem, upcmap->code_space.num_ranges,
63
269
                          sizeof(gx_code_space_range_t), "cmap_usecmap_func(ranges)");
64
269
                if (ranges != NULL) {
65
269
                    int i;
66
269
                    memcpy(&pdficmap->code_space, &upcmap->code_space, sizeof(pdficmap->code_space));
67
646
                    for (i = 0; i < upcmap->code_space.num_ranges; i++) {
68
377
                        memcpy(&(ranges[i]), &(upcmap->code_space.ranges[i]), sizeof(ranges[i]));
69
377
                    }
70
269
                    pdficmap->code_space.ranges = ranges;
71
269
                    memcpy(&pdficmap->cmap_range, &upcmap->cmap_range, sizeof(pdficmap->cmap_range));
72
269
                    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
269
                    upcmap->cmap_range.ranges = NULL;
75
269
                    upcmap->notdef_cmap_range.ranges = NULL;
76
                    /* But we keep the subcmap itself because we rely on its storage */
77
269
                    pdficmap->next = upcmap;
78
269
                    pdfi_countup(upcmap);
79
269
                }
80
269
            }
81
289
        }
82
289
    }
83
289
    pdfi_countdown(upcmap);
84
289
    pdfi_countdown(n);
85
289
    if (code < 0) {
86
20
        (void)pdf_ps_stack_pop(s, 1);
87
20
        return code;
88
20
    }
89
269
    return pdf_ps_stack_pop(s, 1);
90
289
}
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
77.1k
{
102
77.1k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
103
77.1k
    int i, numranges, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
104
77.1k
    gx_code_space_t *code_space = &pdficmap->code_space;
105
77.1k
    int nr = code_space->num_ranges;
106
77.1k
    gx_code_space_range_t *gcsr = code_space->ranges;
107
108
    /* increment to_pop to cover the mark object */
109
77.1k
    numranges = to_pop++;
110
77.2k
    while (numranges % 2) numranges--;
111
112
    /* The following hard limit is an implementation limit */
113
77.1k
    if (numranges < 0 || numranges + 1 > PDF_PS_STACK_MAX) {
114
407
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "cmap_endcodespacerange_func", NULL);
115
407
        return_error(gs_error_syntaxerror);
116
407
    }
117
118
    /* The following "soft" limit is according to the spec */
119
76.7k
    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
76.7k
    if (numranges > 0
128
76.7k
     && pdf_ps_obj_has_type(&(s->cur[0]), PDF_PS_OBJ_STRING)  && s->cur[0].size <= MAX_CMAP_CODE_SIZE
129
76.7k
     && pdf_ps_obj_has_type(&(s->cur[-1]), PDF_PS_OBJ_STRING) && s->cur[-1].size <= MAX_CMAP_CODE_SIZE) {
130
131
76.6k
        code_space->num_ranges += numranges >> 1;
132
133
76.6k
        code_space->ranges = (gx_code_space_range_t *)gs_alloc_byte_array(mem, code_space->num_ranges,
134
76.6k
                          sizeof(gx_code_space_range_t), "cmap_endcodespacerange_func(ranges)");
135
76.6k
        if (code_space->ranges != NULL) {
136
76.6k
            if (nr > 0) {
137
45
                memcpy(code_space->ranges, gcsr, nr * sizeof(gx_code_space_range_t));
138
45
                gs_free_object(mem, gcsr, "cmap_endcodespacerange_func(gcsr");
139
45
            }
140
141
154k
            for (i = nr; i < code_space->num_ranges; i++) {
142
77.5k
                int si = i - nr;
143
77.5k
                int s1 = s->cur[-((si * 2) + 1)].size < MAX_CMAP_CODE_SIZE ? s->cur[-((si * 2) + 1)].size : MAX_CMAP_CODE_SIZE;
144
77.5k
                int s2 = s->cur[-(si * 2)].size < MAX_CMAP_CODE_SIZE ? s->cur[-(si * 2)].size : MAX_CMAP_CODE_SIZE;
145
146
77.5k
                memcpy(code_space->ranges[i].first, s->cur[-((si * 2) + 1)].val.string, s1);
147
77.5k
                memcpy(code_space->ranges[i].last, s->cur[-(si * 2)].val.string, s2);
148
77.5k
                code_space->ranges[i].size = s->cur[-(si * 2)].size;
149
77.5k
            }
150
76.6k
        }
151
0
        else {
152
0
            (void)pdf_ps_stack_pop(s, to_pop);
153
0
            return_error(gs_error_VMerror);
154
0
        }
155
76.6k
    }
156
76.7k
    return pdf_ps_stack_pop(s, to_pop);
157
76.7k
}
158
159
static int cmap_insert_map(pdfi_cmap_range_t *cmap_range, pdfi_cmap_range_map_t *pdfir)
160
14.4M
{
161
14.4M
    if (cmap_range->ranges == NULL) {
162
74.6k
        cmap_range->ranges = cmap_range->ranges_tail = pdfir;
163
74.6k
    }
164
14.3M
    else {
165
14.3M
        cmap_range->ranges_tail->next = pdfir;
166
14.3M
        cmap_range->ranges_tail = pdfir;
167
14.3M
    }
168
14.4M
    cmap_range->numrangemaps++;
169
14.4M
    return 0;
170
14.4M
}
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
130k
{
174
130k
    int ncodemaps, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
175
130k
    unsigned int i, j;
176
130k
    pdfi_cmap_range_map_t *pdfir;
177
130k
    pdf_ps_stack_object_t *stobj;
178
179
    /* increment to_pop to cover the mark object */
180
130k
    ncodemaps = to_pop++;
181
    /* mapping should have 3 objects on the stack:
182
     * startcode, endcode and basecid
183
     */
184
131k
    while (ncodemaps % 3) ncodemaps--;
185
186
    /* The following hard limit is an implementation limit */
187
130k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
188
12
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "general_endcidrange_func", NULL);
189
12
        return_error(gs_error_syntaxerror);
190
12
    }
191
192
    /* The following "soft" limit is according to the spec */
193
130k
    if (ncodemaps > 300) {
194
2
        int code;
195
2
        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
2
    }
200
201
130k
    stobj = &s->cur[-ncodemaps] + 1;
202
203
11.3M
    for (i = 0; i < ncodemaps; i += 3) {
204
11.1M
        int preflen, valuelen;
205
206
11.1M
        if (pdf_ps_obj_has_type(&(stobj[i + 2]), PDF_PS_OBJ_INTEGER)
207
11.1M
        &&  pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_STRING)
208
11.1M
        &&  pdf_ps_obj_has_type(&(stobj[i]), PDF_PS_OBJ_STRING)
209
11.1M
        &&  pdf_ps_obj_size(&(stobj[i])) > 0
210
11.1M
        &&  pdf_ps_obj_size(&(stobj[i])) == pdf_ps_obj_size(&(stobj[i + 1]))){
211
212
11.1M
            uint cidbase = stobj[i + 2].val.i;
213
214
            /* First, find the length of the prefix */
215
22.9M
            for (preflen = 0; preflen < stobj[i].size; preflen++) {
216
22.3M
                if(stobj[i].val.string[preflen] != stobj[i + 1].val.string[preflen]) {
217
10.5M
                    break;
218
10.5M
                }
219
22.3M
            }
220
221
11.1M
            if (preflen == stobj[i].size) {
222
579k
                preflen = 1;
223
579k
            }
224
225
11.1M
            if (preflen > MAX_CMAP_CODE_SIZE || stobj[i].size - preflen > MAX_CMAP_CODE_SIZE || stobj[i + 1].size - preflen > MAX_CMAP_CODE_SIZE
226
11.1M
                || ((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
11.1M
            for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
234
11.1M
                DO_NOTHING;
235
236
11.1M
            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
11.1M
            j = sizeof(pdfi_cmap_range_map_t) + 2 * (stobj[i].size - preflen) + valuelen;
243
244
11.1M
            pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
245
11.1M
            if (pdfir != NULL) {
246
11.1M
                gx_cmap_lookup_range_t *gxr = &pdfir->range;
247
11.1M
                pdfir->next = NULL;
248
11.1M
                gxr->num_entries = 1;
249
11.1M
                gxr->keys.data = (byte *)&(pdfir[1]);
250
11.1M
                gxr->values.data = gxr->keys.data + 2 * (stobj[i].size - preflen);
251
252
11.1M
                gxr->cmap = NULL;
253
11.1M
                gxr->font_index = 0;
254
11.1M
                gxr->key_is_range = true;
255
11.1M
                gxr->value_type = cmap_range == &(pdficmap->cmap_range) ? CODE_VALUE_CID : CODE_VALUE_NOTDEF;
256
11.1M
                gxr->key_prefix_size = preflen;
257
11.1M
                gxr->key_size = stobj[i].size - gxr->key_prefix_size;
258
11.1M
                memcpy(gxr->key_prefix, stobj[i].val.string, gxr->key_prefix_size);
259
260
11.1M
                memcpy(gxr->keys.data, stobj[i].val.string + gxr->key_prefix_size, stobj[i].size - gxr->key_prefix_size);
261
11.1M
                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
11.1M
                gxr->keys.size = (stobj[i].size - gxr->key_prefix_size) + (stobj[i + 1].size - gxr->key_prefix_size);
264
33.5M
                for (j = 0; j < valuelen; j++) {
265
22.3M
                    gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
266
22.3M
                }
267
11.1M
                gxr->value_size = valuelen; /* I'm not sure.... */
268
11.1M
                gxr->values.size = valuelen;
269
11.1M
                if (cmap_insert_map(cmap_range, pdfir) < 0) break;
270
11.1M
            }
271
0
            else {
272
0
                (void)pdf_ps_stack_pop(s, to_pop);
273
0
                return_error(gs_error_VMerror);
274
0
            }
275
11.1M
        }
276
11.1M
    }
277
130k
    return pdf_ps_stack_pop(s, to_pop);
278
130k
}
279
280
static int cmap_endcidrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
281
130k
{
282
130k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
283
130k
    return general_endcidrange_func(mem, s, pdficmap, &pdficmap->cmap_range);
284
130k
}
285
286
static int cmap_endnotdefrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
287
370
{
288
370
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
289
370
    return general_endcidrange_func(mem, s, pdficmap, &pdficmap->notdef_cmap_range);
290
370
}
291
292
static int cmap_endfbrange_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
293
20.4k
{
294
20.4k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
295
20.4k
    int ncodemaps, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
296
20.4k
    unsigned int i, j, k;
297
20.4k
    pdfi_cmap_range_map_t *pdfir;
298
20.4k
    pdf_ps_stack_object_t *stobj;
299
300
    /* increment to_pop to cover the mark object */
301
20.4k
    ncodemaps = to_pop++;
302
    /* mapping should have 3 objects on the stack
303
     */
304
21.2k
    while (ncodemaps % 3) ncodemaps--;
305
306
    /* The following hard limit is an implementation limit */
307
20.4k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
308
29
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "cmap_endfbrange_func", NULL);
309
29
        return_error(gs_error_syntaxerror);
310
29
    }
311
    /* The following "soft" limit is according to the spec */
312
313
20.4k
    if (ncodemaps > 300) {
314
165
        int code;
315
165
        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
165
    }
320
321
20.4k
    stobj = &s->cur[-ncodemaps] + 1;
322
474k
    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
454k
        if (pdf_ps_obj_has_type(&(stobj[i + 2]), PDF_PS_OBJ_STRING)) {
327
441k
            pdf_ps_stack_object_t *arr;
328
441k
            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
441k
            if (arr == NULL) {
330
0
                (void)pdf_ps_stack_pop(s, to_pop);
331
0
                return_error(gs_error_VMerror);
332
0
            }
333
441k
            else {
334
441k
                memcpy(arr, &(stobj[i + 2]), sizeof(pdf_ps_stack_object_t));
335
441k
                pdf_ps_make_array(&(stobj[i + 2]), arr, 1);
336
441k
            }
337
441k
        }
338
454k
    }
339
340
20.4k
    stobj = &s->cur[-ncodemaps] + 1;
341
342
472k
    for (i = 0; i < ncodemaps; i += 3) {
343
452k
        int preflen, valuelen;
344
345
452k
        if (pdf_ps_obj_has_type(&(stobj[i + 2]), PDF_PS_OBJ_ARRAY)
346
452k
        &&  pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_STRING)
347
452k
        &&  pdf_ps_obj_has_type(&(stobj[i]), PDF_PS_OBJ_STRING)
348
452k
        &&  pdf_ps_obj_size(&(stobj[i + 1])) > 0
349
452k
        &&  pdf_ps_obj_size(&(stobj[i])) > 0){
350
351
437k
            uint cidbase;
352
437k
            int m, size;
353
354
437k
            if (stobj[i + 2].size < 1)
355
25
                continue;
356
357
437k
            else if (stobj[i + 2].size == 1) {
358
436k
                if (stobj[i + 2].val.arr[0].type != PDF_PS_OBJ_STRING)
359
0
                     continue;
360
361
436k
                size = stobj[i + 2].val.arr[0].size;
362
363
436k
                cidbase = 0;
364
1.31M
                for (m = 0; m < size; m++) {
365
876k
                    cidbase |= stobj[i + 2].val.arr[0].val.string[size - m - 1] << (8 * m);
366
876k
                }
367
368
                /* First, find the length of the prefix */
369
916k
                for (preflen = 0; preflen < stobj[i].size; preflen++) {
370
734k
                    if(stobj[i].val.string[preflen] != stobj[i + 1].val.string[preflen]) {
371
254k
                        break;
372
254k
                    }
373
734k
                }
374
375
436k
                if (preflen == stobj[i].size) {
376
181k
                    preflen = 1;
377
181k
                }
378
379
436k
                if (preflen > MAX_CMAP_CODE_SIZE || stobj[i].size - preflen > MAX_CMAP_CODE_SIZE || stobj[i + 1].size - preflen > MAX_CMAP_CODE_SIZE
380
436k
                    || ((int64_t)stobj[i].size) - preflen < 0 || ((int64_t)stobj[i + 1].size) - preflen < 0) {
381
79
                    (void)pdf_ps_stack_pop(s, to_pop);
382
79
                    return_error(gs_error_syntaxerror);
383
79
                }
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
491k
                for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
388
436k
                    DO_NOTHING;
389
390
436k
                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
436k
                j = sizeof(pdfi_cmap_range_map_t) + 2 * (stobj[i].size - preflen) + valuelen;
397
398
436k
                pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
399
436k
                if (pdfir != NULL) {
400
436k
                    gx_cmap_lookup_range_t *gxr = &pdfir->range;
401
436k
                    pdfir->next = NULL;
402
436k
                    gxr->num_entries = 1;
403
436k
                    gxr->keys.data = (byte *)&(pdfir[1]);
404
436k
                    gxr->values.data = gxr->keys.data + 2 * (stobj[i].size - preflen);
405
406
436k
                    gxr->cmap = NULL;
407
436k
                    gxr->font_index = 0;
408
436k
                    gxr->key_is_range = true;
409
436k
                    gxr->value_type = CODE_VALUE_CID;
410
436k
                    gxr->key_prefix_size = preflen;
411
436k
                    gxr->key_size = stobj[i].size - gxr->key_prefix_size;
412
436k
                    memcpy(gxr->key_prefix, stobj[i].val.string, gxr->key_prefix_size);
413
414
436k
                    memcpy(gxr->keys.data, stobj[i].val.string + gxr->key_prefix_size, stobj[i].size - gxr->key_prefix_size);
415
436k
                    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
436k
                    gxr->keys.size = (stobj[i].size - gxr->key_prefix_size) + (stobj[i + 1].size - gxr->key_prefix_size);
418
1.31M
                    for (j = 0; j < valuelen; j++) {
419
879k
                        gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
420
879k
                    }
421
436k
                    gxr->value_size = valuelen; /* I'm not sure.... */
422
436k
                    gxr->values.size = valuelen;
423
436k
                    if (cmap_insert_map(&pdficmap->cmap_range, pdfir) < 0) break;
424
436k
                }
425
0
                else {
426
0
                    (void)pdf_ps_stack_pop(s, to_pop);
427
0
                    return_error(gs_error_VMerror);
428
0
                }
429
436k
            }
430
1.48k
            else {
431
1.48k
                int m, size, keysize;
432
1.48k
                uint codelo = 0, codehi = 0;
433
434
1.48k
                size = stobj[i].size;
435
4.07k
                for (m = 0; m < size; m++) {
436
2.58k
                    codelo |= stobj[i].val.string[size - m - 1] << (8 * m);
437
2.58k
                }
438
1.48k
                size = stobj[i + 1].size;
439
4.07k
                for (m = 0; m < size; m++) {
440
2.58k
                    codehi |= stobj[i + 1].val.string[size - m - 1] << (8 * m);
441
2.58k
                }
442
443
1.48k
                if (codehi <= codelo || stobj[i + 2].size < (codehi - codelo))
444
24
                    continue;
445
446
106k
                for (k = codelo; k <= codehi; k++) {
447
104k
                    uint cidbase;
448
104k
                    int ind = k - codelo;
449
450
104k
                    if (stobj[i + 2].val.arr[ind].type != PDF_PS_OBJ_STRING)
451
8
                         continue;
452
453
104k
                    size = stobj[i + 2].val.arr[ind].size;
454
455
104k
                    cidbase = 0;
456
314k
                    for (m = 0; m < size; m++) {
457
209k
                        cidbase |= stobj[i + 2].val.arr[ind].val.string[size - m - 1] << (8 * m);
458
209k
                    }
459
                    /* Find how many bytes we need for the cidbase value */
460
                    /* We always store at least two bytes for the cidbase value */
461
104k
                    for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
462
104k
                        DO_NOTHING;
463
464
104k
                    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
104k
                    for (keysize = 16; keysize < 32 && (cidbase >> keysize) > 0; keysize += 1)
469
104k
                        DO_NOTHING;
470
471
104k
                    keysize = ((keysize + 7) & ~7) >> 3;
472
104k
                    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
104k
                    preflen = keysize > 4 ? 4 : keysize;
477
104k
                    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
104k
                    j = sizeof(pdfi_cmap_range_map_t) + keysize + valuelen;
484
485
104k
                    pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
486
104k
                    if (pdfir != NULL) {
487
104k
                        gx_cmap_lookup_range_t *gxr = &pdfir->range;
488
104k
                        pdfir->next = NULL;
489
104k
                        gxr->num_entries = 1;
490
104k
                        gxr->keys.data = (byte *)&(pdfir[1]);
491
104k
                        gxr->values.data = gxr->keys.data + keysize;
492
493
104k
                        gxr->cmap = NULL;
494
104k
                        gxr->font_index = 0;
495
104k
                        gxr->key_is_range = false;
496
104k
                        gxr->value_type = CODE_VALUE_CID;
497
104k
                        gxr->key_prefix_size = preflen;
498
104k
                        gxr->key_size = keysize;
499
314k
                        for (j = 0; j < preflen; j++) {
500
209k
                            gxr->key_prefix[j] = (k >> ((preflen - 1 - j) * 8)) & 255;
501
209k
                        }
502
503
104k
                        for (j = preflen; j < preflen + keysize; j++) {
504
0
                            gxr->keys.data[j] = (k >> (((preflen + keysize) - 1 - j) * 8)) & 255;
505
0
                        }
506
507
104k
                        gxr->keys.size = keysize;
508
314k
                        for (j = 0; j < valuelen; j++) {
509
209k
                            gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
510
209k
                        }
511
104k
                        gxr->value_size = valuelen; /* I'm not sure.... */
512
104k
                        gxr->values.size = valuelen;
513
104k
                        if (cmap_insert_map(&pdficmap->cmap_range, pdfir) < 0) break;
514
104k
                    }
515
0
                    else {
516
0
                        (void)pdf_ps_stack_pop(s, to_pop);
517
0
                        return_error(gs_error_VMerror);
518
0
                    }
519
104k
                }
520
1.46k
            }
521
437k
        }
522
452k
    }
523
20.3k
    return pdf_ps_stack_pop(s, to_pop);
524
20.4k
}
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
46.6k
{
528
46.6k
    int ncodemaps, to_pop = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
529
46.6k
    int i, j;
530
46.6k
    pdfi_cmap_range_map_t *pdfir;
531
46.6k
    pdf_ps_stack_object_t *stobj;
532
533
    /* increment to_pop to cover the mark object */
534
46.6k
    ncodemaps = to_pop++;
535
    /* mapping should have 2 objects on the stack:
536
     * startcode, endcode and basecid
537
     */
538
47.0k
    while (ncodemaps % 2) ncodemaps--;
539
540
46.6k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
541
15
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "general_endcidchar_func", NULL);
542
15
        return_error(gs_error_syntaxerror);
543
15
    }
544
545
46.6k
    if (ncodemaps > 200) {
546
277
        int code;
547
277
        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
277
    }
552
553
46.6k
    stobj = &s->cur[-ncodemaps] + 1;
554
555
2.80M
    for (i = 0; i < ncodemaps; i += 2) {
556
2.75M
        int preflen = 1, valuelen;
557
558
2.75M
        if (pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_INTEGER)
559
2.75M
        &&  pdf_ps_obj_has_type(&(stobj[i]), PDF_PS_OBJ_STRING) && stobj[i].size > 0) {
560
2.72M
            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
3.75M
            for (valuelen = 16; valuelen < 32 && (cidbase >> valuelen) > 0; valuelen += 1)
565
2.72M
                DO_NOTHING;
566
567
2.72M
            preflen = stobj[i].size > 4 ? 4 : stobj[i].size;
568
569
2.72M
            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
2.72M
            j = sizeof(pdfi_cmap_range_map_t) + (stobj[i].size - preflen) + valuelen;
576
577
2.72M
            pdfir = (pdfi_cmap_range_map_t *)gs_alloc_bytes(mem, j, "cmap_endcidrange_func(pdfi_cmap_range_map_t)");
578
2.72M
            if (pdfir != NULL) {
579
2.72M
                gx_cmap_lookup_range_t *gxr = &pdfir->range;
580
2.72M
                pdfir->next = NULL;
581
2.72M
                gxr->num_entries = 1;
582
2.72M
                gxr->keys.data = (byte *)&(pdfir[1]);
583
2.72M
                gxr->values.data = gxr->keys.data + (stobj[i].size - preflen);
584
585
2.72M
                gxr->cmap = NULL;
586
2.72M
                gxr->font_index = 0;
587
2.72M
                gxr->key_is_range = false;
588
2.72M
                gxr->value_type = cmap_range == &(pdficmap->cmap_range) ? CODE_VALUE_CID : CODE_VALUE_NOTDEF;
589
2.72M
                gxr->key_prefix_size = preflen;
590
2.72M
                gxr->key_size = stobj[i].size - gxr->key_prefix_size;
591
2.72M
                memcpy(gxr->key_prefix, stobj[i].val.string, gxr->key_prefix_size);
592
593
2.72M
                memcpy(gxr->keys.data, stobj[i].val.string + gxr->key_prefix_size, stobj[i].size - gxr->key_prefix_size);
594
595
2.72M
                gxr->keys.size = stobj[i].size - gxr->key_prefix_size;
596
8.32M
                for (j = 0; j < valuelen; j++) {
597
5.59M
                    gxr->values.data[j] = (cidbase >> ((valuelen - 1 - j) * 8)) & 255;
598
5.59M
                }
599
2.72M
                gxr->value_size = valuelen; /* I'm not sure.... */
600
2.72M
                gxr->values.size = valuelen;
601
2.72M
                if (cmap_insert_map(cmap_range, pdfir) < 0) break;
602
2.72M
            }
603
0
            else {
604
0
                (void)pdf_ps_stack_pop(s, to_pop);
605
0
                return_error(gs_error_VMerror);
606
0
            }
607
2.72M
        }
608
2.75M
    }
609
46.6k
    return pdf_ps_stack_pop(s, to_pop);
610
46.6k
}
611
612
static int cmap_endcidchar_func(gs_memory_t *mem, pdf_ps_ctx_t *s, byte *buf, byte *bufend)
613
8.90k
{
614
8.90k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
615
8.90k
    return general_endcidchar_func(mem, s, pdficmap, &pdficmap->cmap_range);
616
8.90k
}
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
37.9k
{
626
37.9k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
627
37.9k
    int ncodemaps = pdf_ps_stack_count_to_mark(s, PDF_PS_OBJ_MARK);
628
37.9k
    pdf_ps_stack_object_t *stobj;
629
37.9k
    int i, j;
630
631
    /* The following hard limit is an implementation limit */
632
37.9k
    if (ncodemaps < 0 || ncodemaps + 1 > PDF_PS_STACK_MAX) {
633
145
        pdfi_set_error(s->pdfi_ctx, 0, NULL, E_PDF_BAD_TYPE0_CMAP, "cmap_endbfchar_func", NULL);
634
145
        return_error(gs_error_syntaxerror);
635
145
    }
636
637
    /* The following "soft" limit is according to the spec */
638
37.7k
    if (ncodemaps > 200) {
639
280
        int code;
640
280
        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
280
    }
645
646
37.7k
    stobj = &s->cur[-ncodemaps] + 1;
647
648
1.98M
    for (i = 0; i < ncodemaps; i += 2) {
649
1.94M
        if (pdf_ps_obj_has_type(&(stobj[i + 1]), PDF_PS_OBJ_STRING)) {
650
1.94M
            byte *c = stobj[i + 1].val.string;
651
1.94M
            int l = stobj[i + 1].size;
652
1.94M
            unsigned int v = 0;
653
654
6.22M
            for (j = 0; j < l; j++) {
655
4.27M
                v += c[l - j - 1] << (8 * j);
656
4.27M
            }
657
1.94M
            pdf_ps_make_int(&(stobj[i + 1]), v);
658
1.94M
        }
659
4.83k
        else {
660
4.83k
            continue;
661
4.83k
        }
662
1.94M
    }
663
37.7k
    return general_endcidchar_func(mem, s, pdficmap, &pdficmap->cmap_range);
664
37.7k
}
665
666
1.93M
#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
478k
{
670
478k
    int code = 0, code2 = 0;
671
478k
    pdf_cmap *pdficmap = (pdf_cmap *)s->client_data;
672
673
478k
    if (pdf_ps_stack_count(s) < 2) {
674
42.6k
        return pdf_ps_stack_pop(s, 1);
675
42.6k
    }
676
677
436k
    if (pdf_ps_obj_has_type(&s->cur[-1], PDF_PS_OBJ_NAME)) {
678
435k
        if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("Registry"))) {
679
42.7k
            if (pdficmap->csi_reg.data != NULL)
680
17
               gs_free_object(mem, pdficmap->csi_reg.data, "cmap_def_func(Registry)");
681
682
42.7k
            pdficmap->csi_reg.data = gs_alloc_bytes(mem, s->cur[0].size + 1, "cmap_def_func(Registry)");
683
42.7k
            if (pdficmap->csi_reg.data != NULL) {
684
42.7k
                pdficmap->csi_reg.size = s->cur[0].size;
685
42.7k
                if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_STRING)) {
686
42.7k
                    memcpy(pdficmap->csi_reg.data, s->cur[0].val.string, s->cur[0].size);
687
42.7k
                }
688
4
                else {
689
4
                    memcpy(pdficmap->csi_reg.data, s->cur[0].val.name, s->cur[0].size);
690
4
                }
691
42.7k
                pdficmap->csi_reg.data[pdficmap->csi_reg.size] = '\0';
692
42.7k
            }
693
0
            else {
694
0
                code = gs_note_error(gs_error_VMerror);
695
0
            }
696
42.7k
        }
697
393k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("Ordering"))) {
698
42.7k
            if (pdficmap->csi_ord.data != NULL)
699
17
               gs_free_object(mem, pdficmap->csi_ord.data, "cmap_def_func(Ordering)");
700
701
42.7k
            pdficmap->csi_ord.data = gs_alloc_bytes(mem, s->cur[0].size + 1, "cmap_def_func(Ordering)");
702
42.7k
            if (pdficmap->csi_ord.data != NULL) {
703
42.7k
                pdficmap->csi_ord.size = s->cur[0].size;
704
42.7k
                if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_STRING))
705
42.7k
                    memcpy(pdficmap->csi_ord.data, s->cur[0].val.string, s->cur[0].size);
706
4
                else
707
4
                    memcpy(pdficmap->csi_ord.data, s->cur[0].val.name, s->cur[0].size);
708
42.7k
                pdficmap->csi_ord.data[pdficmap->csi_ord.size] = '\0';
709
42.7k
            }
710
0
            else {
711
0
                code = gs_note_error(gs_error_VMerror);
712
0
            }
713
42.7k
        }
714
350k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("Supplement"))) {
715
73.5k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
716
73.5k
                pdficmap->csi_supplement = s->cur[0].val.i;
717
73.5k
            }
718
0
            else {
719
0
                pdficmap->csi_supplement = 0;
720
0
            }
721
73.5k
        }
722
276k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("CMapName"))) {
723
77.2k
            if (pdficmap->name.data != NULL)
724
34
               gs_free_object(mem, pdficmap->name.data, "cmap_def_func(CMapName)");
725
726
77.2k
            pdficmap->name.data = gs_alloc_bytes(mem, s->cur[0].size + 1, "cmap_def_func(CMapName)");
727
77.2k
            if (pdficmap->name.data != NULL) {
728
77.2k
                pdficmap->name.size = s->cur[0].size;
729
77.2k
                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
77.2k
                else
732
77.2k
                    memcpy(pdficmap->name.data, s->cur[0].val.name, s->cur[0].size);
733
77.2k
                pdficmap->name.data[pdficmap->name.size] = '\0';
734
77.2k
            }
735
0
            else {
736
0
                code = gs_note_error(gs_error_VMerror);
737
0
            }
738
77.2k
        }
739
199k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("CMapVersion"))) {
740
40.9k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
741
305
                pdficmap->vers = (float)s->cur[0].val.i;
742
305
            }
743
40.6k
            else if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_FLOAT)){
744
40.6k
                pdficmap->vers = s->cur[0].val.f;
745
40.6k
            }
746
0
            else {
747
0
                pdficmap->vers = (float)0;
748
0
            }
749
40.9k
        }
750
158k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("CMapType"))) {
751
75.7k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
752
75.7k
                pdficmap->cmaptype = s->cur[0].val.i;
753
75.7k
            }
754
0
            else {
755
0
                pdficmap->cmaptype = 1;
756
0
            }
757
75.7k
        }
758
82.7k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("XUID"))) {
759
40.6k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_ARRAY)) {
760
40.6k
                int len = s->cur->size;
761
762
40.6k
                if (pdficmap->uid.xvalues != NULL)
763
0
                   gs_free_object(mem, pdficmap->uid.xvalues, "cmap_def_func(XUID)");
764
765
40.6k
                pdficmap->uid.xvalues = (long *)gs_alloc_bytes(mem, len * sizeof(*pdficmap->uid.xvalues), "cmap_def_func(XUID)");
766
40.6k
                if (pdficmap->uid.xvalues != NULL) {
767
40.6k
                     int i;
768
40.6k
                     pdf_ps_stack_object_t *a = s->cur->val.arr;
769
40.6k
                     pdficmap->uid.id = -len;
770
202k
                     for (i = 0; i < len; i++) {
771
162k
                         if (pdf_ps_obj_has_type(&a[i], PDF_PS_OBJ_INTEGER)) {
772
162k
                             pdficmap->uid.xvalues[i] = (long)a[i].val.i;
773
162k
                         }
774
0
                         else {
775
0
                             pdficmap->uid.xvalues[i] = 0;
776
0
                         }
777
162k
                     }
778
40.6k
                }
779
0
                else {
780
0
                    code = gs_note_error(gs_error_VMerror);
781
0
                }
782
40.6k
            }
783
40.6k
        }
784
42.1k
        else if (!memcmp(s->cur[-1].val.name, CMAP_NAME_AND_LEN("WMode"))) {
785
41.2k
            if (pdf_ps_obj_has_type(&s->cur[0], PDF_PS_OBJ_INTEGER)) {
786
41.2k
                if (s->cur[0].val.i != 0) {
787
278
                    if (s->cur[0].val.i != 1) {
788
9
                        code = pdfi_set_warning_stop(s->pdfi_ctx, gs_note_error(gs_error_rangecheck), NULL, W_PDF_BAD_WMODE, "cmap_def_func", NULL);
789
9
                    }
790
278
                    pdficmap->wmode = 1;
791
278
                }
792
41.0k
                else
793
41.0k
                    pdficmap->wmode = 0;
794
41.2k
            }
795
0
            else {
796
0
                pdficmap->wmode = 0;
797
0
            }
798
41.2k
        }
799
435k
    }
800
801
436k
    code2 = pdf_ps_stack_pop(s, 2);
802
436k
    if (code < 0)
803
0
        return code;
804
436k
    else
805
436k
        return code2;
806
436k
}
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
40.7k
{
844
40.7k
    int code = 0;
845
40.7k
    stream *s;
846
40.7k
    char fname[gp_file_name_sizeof];
847
40.7k
    const char *path_pfx = "CMap/";
848
40.7k
    fname[0] = '\0';
849
850
40.7k
    if (strlen(path_pfx) + cmap_name->size >= gp_file_name_sizeof)
851
6
        return_error(gs_error_rangecheck);
852
853
40.7k
    strncat(fname, path_pfx, strlen(path_pfx));
854
40.7k
    strncat(fname, (char *)cmap_name->data, cmap_name->size);
855
40.7k
    code = pdfi_open_resource_file(ctx, (const char *)fname, (const int)strlen(fname), &s);
856
40.7k
    if (code >= 0) {
857
40.3k
        sfseek(s, 0, SEEK_END);
858
40.3k
        *buflen = sftell(s);
859
40.3k
        sfseek(s, 0, SEEK_SET);
860
40.3k
        *buf = gs_alloc_bytes(ctx->memory, *buflen, "pdf_cmap_open_file(buf)");
861
40.3k
        if (*buf != NULL) {
862
40.3k
            sfread(*buf, 1, *buflen, s);
863
40.3k
        }
864
0
        else {
865
0
            code = gs_note_error(gs_error_VMerror);
866
0
        }
867
40.3k
        sfclose(s);
868
40.3k
    }
869
40.7k
    return code;
870
40.7k
}
871
872
static int
873
pdfi_make_gs_cmap(gs_memory_t *mem, pdf_cmap *pdficmap)
874
78.5k
{
875
78.5k
    int code = 0, i;
876
78.5k
    gs_cmap_adobe1_t *pgscmap = 0;
877
78.5k
    gx_cmap_lookup_range_t *lookups, *ndlookups = NULL;
878
78.5k
    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
78.5k
    if ((code = gs_cmap_adobe1_alloc(&pgscmap, pdficmap->wmode,
884
78.5k
                                     pdficmap->name.data,
885
78.5k
                                     pdficmap->name.size,
886
78.5k
                                     1,
887
78.5k
                                     0, 0, 0, 0, 0, mem)) >= 0) {
888
78.5k
        gs_free_object(mem, pgscmap->code_space.ranges, "empty ranges");
889
890
78.5k
        lookups = gs_alloc_struct_array(mem, pdficmap->cmap_range.numrangemaps,
891
78.5k
                               gx_cmap_lookup_range_t,
892
78.5k
                               &st_cmap_lookup_range_element,
893
78.5k
                               "pdfi_make_gs_cmap(lookup ranges)");
894
78.5k
        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
78.5k
        if (pdficmap->notdef_cmap_range.numrangemaps > 0){
900
405
            ndlookups = gs_alloc_struct_array(mem, pdficmap->notdef_cmap_range.numrangemaps,
901
405
                               gx_cmap_lookup_range_t,
902
405
                               &st_cmap_lookup_range_element,
903
405
                               "pdfi_make_gs_cmap(notdef lookup ranges)");
904
405
            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
405
        }
911
78.5k
        pgscmap->def.lookup = lookups;
912
78.5k
        pgscmap->def.num_lookup = pdficmap->cmap_range.numrangemaps;
913
78.5k
        pgscmap->notdef.lookup = ndlookups;
914
78.5k
        pgscmap->notdef.num_lookup = pdficmap->notdef_cmap_range.numrangemaps;
915
916
78.5k
        pgscmap->CIDSystemInfo[0].Registry.data = pdficmap->csi_reg.data;
917
78.5k
        pgscmap->CIDSystemInfo[0].Registry.size = pdficmap->csi_reg.size;
918
78.5k
        pgscmap->CIDSystemInfo[0].Ordering.data = pdficmap->csi_ord.data;
919
78.5k
        pgscmap->CIDSystemInfo[0].Ordering.size = pdficmap->csi_ord.size;
920
78.5k
        pgscmap->CIDSystemInfo[0].Supplement = pdficmap->csi_supplement;
921
78.5k
        memcpy(&pgscmap->code_space, &pdficmap->code_space, sizeof(pgscmap->code_space));
922
78.5k
        memcpy(&pgscmap->uid, &pdficmap->uid, sizeof(pdficmap->uid));
923
78.5k
        l = pdficmap->cmap_range.ranges;
924
14.4M
        for (i = 0; i < pdficmap->cmap_range.numrangemaps && l != NULL; i++) {
925
14.3M
            memcpy(&lookups[i], &l->range, sizeof(gx_cmap_lookup_range_t));
926
14.3M
            l = l->next;
927
14.3M
        }
928
929
78.5k
        l = pdficmap->notdef_cmap_range.ranges;
930
78.9k
        for (i = 0; i < pdficmap->notdef_cmap_range.numrangemaps && l != NULL; i++) {
931
405
            memcpy(&ndlookups[i], &l->range, sizeof(gx_cmap_lookup_range_t));
932
405
            l = l->next;
933
405
        }
934
935
78.5k
        pdficmap->gscmap = pgscmap;
936
78.5k
    }
937
938
78.5k
done:
939
78.5k
    return code;
940
78.5k
}
941
942
int
943
pdfi_read_cmap(pdf_context *ctx, pdf_obj *cmap, pdf_cmap **pcmap)
944
79.8k
{
945
79.8k
    int code = 0;
946
79.8k
    pdf_cmap pdficm[3] = {0};
947
79.8k
    pdf_cmap *pdfi_cmap = &(pdficm[1]);
948
79.8k
    byte *buf = NULL;
949
79.8k
    int64_t buflen = 0;
950
79.8k
    pdf_ps_ctx_t cmap_ctx;
951
952
79.8k
    pdfi_cmap->ctx = ctx;
953
79.8k
    switch (pdfi_type_of(cmap)) {
954
40.7k
        case PDF_NAME:
955
40.7k
        {
956
40.7k
            gs_string cmname;
957
40.7k
            pdf_name *cmapn = (pdf_name *)cmap;
958
40.7k
            cmname.data = cmapn->data;
959
40.7k
            cmname.size = cmapn->length;
960
40.7k
            code = pdf_cmap_open_file(ctx, &cmname, &buf, &buflen);
961
40.7k
            if (code < 0)
962
340
                goto error_out;
963
40.3k
            break;
964
40.7k
        }
965
40.3k
        case PDF_STREAM:
966
39.0k
        {
967
39.0k
            pdf_obj *ucmap;
968
39.0k
            pdf_cmap *upcmap = NULL;
969
39.0k
            pdf_dict *cmap_dict = NULL;
970
971
39.0k
            code = pdfi_dict_from_obj(ctx, cmap, &cmap_dict);
972
39.0k
            if (code < 0)
973
0
                goto error_out;
974
975
39.0k
            code = pdfi_dict_knownget(ctx, cmap_dict, "UseCMap", &ucmap);
976
39.0k
            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
39.0k
            code = pdfi_stream_to_buffer(ctx, (pdf_stream *)cmap, &buf, &buflen);
1008
39.0k
            if (code < 0) {
1009
188
                goto error_out;
1010
188
            }
1011
38.9k
            break;
1012
39.0k
        }
1013
38.9k
        default:
1014
61
            code = gs_note_error(gs_error_typecheck);
1015
61
            goto error_out;
1016
79.8k
    }
1017
79.2k
    pdfi_cmap->ctx = ctx;
1018
79.2k
    pdfi_cmap->buf = buf;
1019
79.2k
    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
79.2k
    pdfi_cmap->cmaptype = 1;
1025
1026
79.2k
    pdfi_pscript_stack_init(ctx, cmap_oper_list, (void *)pdfi_cmap, &cmap_ctx);
1027
1028
79.2k
    code = pdfi_pscript_interpret(&cmap_ctx, buf, buflen);
1029
79.2k
    pdfi_pscript_stack_finit(&cmap_ctx);
1030
79.2k
    if (code < 0) goto error_out;
1031
1032
78.5k
    code = pdfi_make_gs_cmap(ctx->memory, pdfi_cmap);
1033
1034
78.5k
    if (code >= 0) {
1035
78.5k
        *pcmap = (pdf_cmap *)gs_alloc_bytes(ctx->memory, sizeof(pdf_cmap), "pdfi_read_cmap(*pcmap)");
1036
78.5k
        if (*pcmap != NULL) {
1037
78.5k
            pdfi_cmap->type = PDF_CMAP;
1038
78.5k
            pdfi_cmap->ctx = ctx;
1039
78.5k
            pdfi_cmap->refcnt = 1;
1040
78.5k
            pdfi_cmap->object_num = cmap->object_num;
1041
78.5k
            pdfi_cmap->generation_num = cmap->generation_num;
1042
78.5k
            pdfi_cmap->indirect_num = cmap->indirect_num;
1043
78.5k
            pdfi_cmap->indirect_gen = cmap->indirect_gen;
1044
78.5k
            memcpy(*pcmap, pdfi_cmap, sizeof(pdf_cmap));
1045
78.5k
            pdfi_cmap = *pcmap;
1046
            /* object_num can be zero if the dictionary was defined inline */
1047
78.5k
            if (pdfi_cmap->object_num != 0) {
1048
38.1k
                code = replace_cache_entry(ctx, (pdf_obj *)pdfi_cmap);
1049
38.1k
            }
1050
78.5k
        } else {
1051
0
            code = gs_note_error(gs_error_VMerror);
1052
0
            goto error_out;
1053
0
        }
1054
78.5k
    }
1055
0
    else {
1056
0
        goto error_out;
1057
0
    }
1058
78.5k
    return 0;
1059
1060
1.36k
error_out:
1061
1.36k
    pdfi_free_cmap_contents(pdfi_cmap);
1062
1.36k
    memset(pdfi_cmap, 0x00, sizeof(pdf_cmap));
1063
1.36k
    return code;
1064
78.5k
}
1065
1066
static int pdfi_free_cmap_contents(pdf_cmap *cmap)
1067
79.8k
{
1068
79.8k
    pdfi_cmap_range_map_t *pdfir;
1069
79.8k
    gs_cmap_adobe1_t *pgscmap = cmap->gscmap;
1070
1071
79.8k
    if (pgscmap != NULL) {
1072
78.5k
        gs_free_object(OBJ_MEMORY(cmap), pgscmap->def.lookup, "pdfi_free_cmap(def.lookup)");
1073
78.5k
        gs_free_object(OBJ_MEMORY(cmap), pgscmap->notdef.lookup, "pdfi_free_cmap(notdef.lookup)");
1074
78.5k
        (void)gs_cmap_free((gs_cmap_t *)pgscmap, OBJ_MEMORY(cmap));
1075
78.5k
    }
1076
79.8k
    gs_free_object(OBJ_MEMORY(cmap), cmap->code_space.ranges, "pdfi_free_cmap(code_space.ranges");
1077
79.8k
    pdfir = cmap->cmap_range.ranges;
1078
14.5M
    while (pdfir != NULL) {
1079
14.4M
        pdfi_cmap_range_map_t *pdfir2 = pdfir->next;
1080
14.4M
        gs_free_object(OBJ_MEMORY(cmap), pdfir, "pdfi_free_cmap(cmap_range.ranges");
1081
14.4M
        pdfir = pdfir2;
1082
14.4M
    }
1083
79.8k
    pdfir = cmap->notdef_cmap_range.ranges;
1084
80.2k
    while (pdfir != NULL) {
1085
370
        pdfi_cmap_range_map_t *pdfir2 = pdfir->next;
1086
370
        gs_free_object(OBJ_MEMORY(cmap), pdfir, "pdfi_free_cmap(cmap_range.ranges");
1087
370
        pdfir = pdfir2;
1088
370
    }
1089
79.8k
    gs_free_object(OBJ_MEMORY(cmap), cmap->csi_reg.data, "pdfi_free_cmap(csi_reg.data");
1090
79.8k
    gs_free_object(OBJ_MEMORY(cmap), cmap->csi_ord.data, "pdfi_free_cmap(csi_ord.data");
1091
79.8k
    gs_free_object(OBJ_MEMORY(cmap), cmap->name.data, "pdfi_free_cmap(name.data");
1092
79.8k
    gs_free_object(OBJ_MEMORY(cmap), cmap->uid.xvalues, "pdfi_free_cmap(xuid.xvalues");
1093
79.8k
    pdfi_countdown(cmap->next);
1094
79.8k
    gs_free_object(OBJ_MEMORY(cmap), cmap->buf, "pdfi_free_cmap(cmap->buf");
1095
1096
79.8k
    return 0;
1097
79.8k
}
1098
1099
int pdfi_free_cmap(pdf_obj *cmapo)
1100
78.5k
{
1101
78.5k
    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
78.5k
    pdfi_free_cmap_contents(cmap);
1122
78.5k
    gs_free_object(OBJ_MEMORY(cmap), cmap, "pdfi_free_cmap(cmap");
1123
78.5k
    return 0;
1124
78.5k
}