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