Coverage Report

Created: 2025-06-10 07:27

/src/ghostpdl/psi/iparam.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-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
17
/* Interpreter implementations of parameter dictionaries */
18
#include "memory_.h"
19
#include "string_.h"
20
#include "ghost.h"
21
#include "ierrors.h"
22
#include "oper.h"   /* for check_type */
23
#include "opcheck.h"
24
#include "ialloc.h"
25
#include "idict.h"
26
#include "imemory.h"    /* for iutil.h */
27
#include "iname.h"
28
#include "istack.h"
29
#include "iparam.h"
30
#include "iutil.h"    /* for num_params */
31
#include "ivmspace.h"
32
#include "store.h"
33
#include "gsstruct.h"           /* for st_bytes */
34
35
/* ================ Utilities ================ */
36
37
/* Convert a key to a ref. */
38
static int
39
ref_param_key(const iparam_list * plist, gs_param_name pkey, ref * pkref)
40
1.12G
{
41
1.12G
    if (plist->int_keys) {
42
0
        long key;
43
44
0
        if (sscanf(pkey, "%ld", &key) != 1)
45
0
            return_error(gs_error_rangecheck);
46
0
        make_int(pkref, key);
47
0
        return 0;
48
0
    } else
49
1.12G
        return name_ref(plist->memory, (const byte *)pkey, strlen(pkey), pkref, 0);
50
1.12G
}
51
52
/* Fill in a gs_param_key_t from a name or int ref. */
53
static int
54
ref_to_key(const ref * pref, gs_param_key_t * key, iparam_list *plist)
55
3.26M
{
56
3.26M
    if (r_has_type(pref, t_name)) {
57
3.26M
        ref nref;
58
59
3.26M
        name_string_ref(plist->memory, pref, &nref);
60
3.26M
        key->data = nref.value.const_bytes;
61
3.26M
        key->size = r_size(&nref);
62
3.26M
        key->persistent = false; /* names may be freed */
63
3.26M
    } else if (r_has_type(pref, t_integer)) {
64
0
        char istr[sizeof(long) * 8 / 3 + 2];
65
0
        int len;
66
0
        byte *buf;
67
68
0
        gs_snprintf(istr, sizeof(istr), "%"PRIpsint, pref->value.intval);
69
0
        len = strlen(istr);
70
        /* GC will take care of freeing this: */
71
0
        buf = gs_alloc_string(plist->memory, len, "ref_to_key");
72
0
        if (!buf)
73
0
            return_error(gs_error_VMerror);
74
0
        key->data = buf;
75
0
        key->size = len;
76
0
        key->persistent = true;
77
0
    } else
78
0
        return_error(gs_error_typecheck);
79
3.26M
    return 0;
80
3.26M
}
81
82
/* ================ Writing parameters to refs ================ */
83
84
/* Forward references */
85
static int array_new_indexed_plist_write(dict_param_list *plist,
86
                                          ref *parray, const ref *pwanted,
87
                                          gs_ref_memory_t *imem);
88
89
/* ---------------- Generic writing procedures ---------------- */
90
91
static param_proc_begin_xmit_collection(ref_param_begin_write_collection);
92
static param_proc_end_xmit_collection(ref_param_end_write_collection);
93
static param_proc_xmit_typed(ref_param_write_typed);
94
static param_proc_next_key(ref_param_get_next_key);
95
static param_proc_requested(ref_param_requested);
96
static const gs_param_list_procs ref_write_procs =
97
{
98
    ref_param_write_typed,
99
    ref_param_begin_write_collection,
100
    ref_param_end_write_collection,
101
    ref_param_get_next_key,
102
    NULL,     /* request */
103
    ref_param_requested
104
};
105
static int ref_array_param_requested(const iparam_list *, gs_param_name,
106
                                      ref *, uint, client_name_t);
107
static int ref_param_write(iparam_list *, gs_param_name, const ref *);
108
static int ref_param_write_string_value(ref *, const gs_param_string *,
109
                                         gs_ref_memory_t *);
110
static int ref_param_write_name_value(const gs_memory_t *mem, ref *, const gs_param_string *);
111
static int
112
ref_param_make_int(ref *pe, const void *pvalue, uint i, gs_ref_memory_t *imem)
113
36.3M
{
114
36.3M
    make_tav(pe, t_integer, imemory_new_mask(imem), intval,
115
36.3M
             ((const gs_param_int_array *)pvalue)->data[i]);
116
36.3M
    return 0;
117
36.3M
}
118
static int
119
ref_param_make_float(ref *pe, const void *pvalue, uint i, gs_ref_memory_t *imem)
120
14.6M
{
121
14.6M
    make_tav(pe, t_real, imemory_new_mask(imem), realval,
122
14.6M
             ((const gs_param_float_array *)pvalue)->data[i]);
123
14.6M
    return 0;
124
14.6M
}
125
static int
126
ref_param_make_string(ref *pe, const void *pvalue, uint i, gs_ref_memory_t *imem)
127
0
{
128
0
    return ref_param_write_string_value(pe,
129
0
                         &((const gs_param_string_array *)pvalue)->data[i],
130
0
                                        imem);
131
0
}
132
static int
133
ref_param_make_name(ref * pe, const void *pvalue, uint i, gs_ref_memory_t *imem)
134
3.63M
{
135
3.63M
    return ref_param_write_name_value((const gs_memory_t *)imem, pe,
136
3.63M
                         &((const gs_param_string_array *)pvalue)->data[i]);
137
3.63M
}
138
static int
139
ref_param_write_typed_array(gs_param_list * plist, gs_param_name pkey,
140
                            void *pvalue, uint count,
141
                            int (*make)(ref *, const void *, uint,
142
                                        gs_ref_memory_t *))
143
50.0M
{
144
50.0M
    iparam_list *const iplist = (iparam_list *) plist;
145
50.0M
    ref value;
146
50.0M
    uint i;
147
50.0M
    ref *pe;
148
50.0M
    int code;
149
150
50.0M
    if ((code = ref_array_param_requested(iplist, pkey, &value, count,
151
50.0M
                                       "ref_param_write_typed_array")) <= 0)
152
31.0M
        return code;
153
73.6M
    for (i = 0, pe = value.value.refs; i < count; ++i, ++pe)
154
54.6M
        if ((code = (*make) (pe, pvalue, i, iplist->ref_memory)) < 0)
155
0
            return code;
156
18.9M
    return ref_param_write(iplist, pkey, &value);
157
18.9M
}
158
static int
159
ref_param_begin_write_collection(gs_param_list * plist, gs_param_name pkey,
160
                                 gs_param_dict * pvalue,
161
                                 gs_param_collection_type_t coll_type)
162
5.99M
{
163
5.99M
    iparam_list *const iplist = (iparam_list *) plist;
164
5.99M
    gs_ref_memory_t *imem = iplist->ref_memory;
165
5.99M
    dict_param_list *dlist = (dict_param_list *)
166
5.99M
        gs_alloc_bytes(plist->memory, size_of(dict_param_list),
167
5.99M
                       "ref_param_begin_write_collection");
168
5.99M
    int code;
169
170
5.99M
    if (dlist == 0)
171
0
        return_error(gs_error_VMerror);
172
5.99M
    if (coll_type != gs_param_collection_array) {
173
5.99M
        ref dref;
174
175
5.99M
        code = dict_alloc(imem, pvalue->size, &dref);
176
5.99M
        if (code >= 0) {
177
5.99M
            code = dict_param_list_write(dlist, &dref, NULL, imem);
178
5.99M
            dlist->int_keys = coll_type == gs_param_collection_dict_int_keys;
179
5.99M
        }
180
5.99M
    } else {
181
0
        ref aref;
182
183
0
        code = gs_alloc_ref_array(imem, &aref, a_all, pvalue->size,
184
0
                                  "ref_param_begin_write_collection");
185
0
        if (code >= 0)
186
0
            code = array_new_indexed_plist_write(dlist, &aref, NULL, imem);
187
0
    }
188
5.99M
    if (code < 0)
189
0
        gs_free_object(plist->memory, dlist, "ref_param_begin_write_collection");
190
5.99M
    else
191
5.99M
        pvalue->list = (gs_param_list *) dlist;
192
5.99M
    return code;
193
5.99M
}
194
static int
195
ref_param_end_write_collection(gs_param_list * plist, gs_param_name pkey,
196
                               gs_param_dict * pvalue)
197
5.99M
{
198
5.99M
    iparam_list *const iplist = (iparam_list *) plist;
199
5.99M
    int code = ref_param_write(iplist, pkey,
200
5.99M
                               &((dict_param_list *) pvalue->list)->dict);
201
202
5.99M
    gs_free_object(plist->memory, pvalue->list, "ref_param_end_write_collection");
203
5.99M
    pvalue->list = 0;
204
5.99M
    return code;
205
5.99M
}
206
static int
207
ref_param_write_typed(gs_param_list * plist, gs_param_name pkey,
208
                      gs_param_typed_value * pvalue)
209
642M
{
210
642M
    iparam_list *const iplist = (iparam_list *) plist;
211
642M
    ref value;
212
642M
    int code = 0;
213
214
642M
    switch (pvalue->type) {
215
11.8M
        case gs_param_type_null:
216
11.8M
            make_null(&value);
217
11.8M
            break;
218
197M
        case gs_param_type_bool:
219
197M
            make_bool(&value, pvalue->value.b);
220
197M
            break;
221
169M
        case gs_param_type_int:
222
169M
            make_int(&value, pvalue->value.i);
223
169M
            break;
224
21.0M
        case gs_param_type_long:
225
            /* FIXME: Rangecheck? */
226
21.0M
            make_int(&value, pvalue->value.l);
227
21.0M
            break;
228
18.4M
        case gs_param_type_size_t:
229
            /* FIXME: Rangecheck? */
230
18.4M
            make_int(&value, pvalue->value.z);
231
18.4M
            break;
232
7.07M
        case gs_param_type_i64:
233
            /* FIXME: Rangecheck? */
234
7.07M
            make_int(&value, pvalue->value.i64);
235
7.07M
            break;
236
37.4M
        case gs_param_type_float:
237
37.4M
            make_real(&value, pvalue->value.f);
238
37.4M
            break;
239
100M
        case gs_param_type_string:
240
100M
            if (!ref_param_requested(plist, pkey))
241
80.2M
                return 0;
242
20.6M
            code = ref_param_write_string_value(&value, &pvalue->value.s,
243
20.6M
                                                iplist->ref_memory);
244
20.6M
            break;
245
28.6M
        case gs_param_type_name:
246
28.6M
            if (!ref_param_requested(plist, pkey))
247
18.7M
                return 0;
248
9.83M
            code = ref_param_write_name_value(iplist->memory, &value, &pvalue->value.n);
249
9.83M
            break;
250
12.4M
        case gs_param_type_int_array:
251
12.4M
            return ref_param_write_typed_array(plist, pkey, &pvalue->value.ia,
252
12.4M
                                               pvalue->value.ia.size,
253
12.4M
                                               ref_param_make_int);
254
22.4M
        case gs_param_type_float_array:
255
22.4M
            return ref_param_write_typed_array(plist, pkey, &pvalue->value.fa,
256
22.4M
                                               pvalue->value.fa.size,
257
22.4M
                                               ref_param_make_float);
258
1.19M
        case gs_param_type_string_array:
259
1.19M
            return ref_param_write_typed_array(plist, pkey, &pvalue->value.sa,
260
1.19M
                                               pvalue->value.sa.size,
261
1.19M
                                               ref_param_make_string);
262
13.9M
        case gs_param_type_name_array:
263
13.9M
            return ref_param_write_typed_array(plist, pkey, &pvalue->value.na,
264
13.9M
                                               pvalue->value.na.size,
265
13.9M
                                               ref_param_make_name);
266
0
        case gs_param_type_dict:
267
0
        case gs_param_type_dict_int_keys:
268
0
        case gs_param_type_array:
269
0
            return ref_param_begin_write_collection(plist, pkey,
270
0
                                                    &pvalue->value.d,
271
0
              (gs_param_collection_type_t)(pvalue->type - gs_param_type_dict));
272
0
        default:
273
0
            return_error(gs_error_typecheck);
274
642M
    }
275
493M
    if (code < 0)
276
0
        return code;
277
493M
    return ref_param_write(iplist, pkey, &value);
278
493M
}
279
280
/* Check whether a given parameter was requested. */
281
static int
282
ref_param_requested(const gs_param_list * plist, gs_param_name pkey)
283
704M
{
284
704M
    const iparam_list *const ciplist = (const iparam_list *)plist;
285
704M
    ref kref;
286
704M
    ref *ignore_value;
287
288
704M
    if (!r_has_type(&ciplist->u.w.wanted, t_dictionary))
289
148M
        return -1;
290
556M
    if (ref_param_key(ciplist, pkey, &kref) < 0)
291
0
        return -1;   /* catch it later */
292
556M
    return (dict_find(&ciplist->u.w.wanted, &kref, &ignore_value) > 0);
293
556M
}
294
295
/* Check whether an array parameter is wanted, and allocate it if so. */
296
/* Return <0 on error, 0 if not wanted, 1 if wanted. */
297
static int
298
ref_array_param_requested(const iparam_list *iplist, gs_param_name pkey,
299
                          ref *pvalue, uint size, client_name_t cname)
300
50.0M
{
301
50.0M
    int code;
302
303
50.0M
    if (!ref_param_requested((const gs_param_list *)iplist, pkey))
304
31.0M
        return 0;
305
18.9M
    code = gs_alloc_ref_array(iplist->ref_memory, pvalue, a_all, size, cname);
306
18.9M
    return (code < 0 ? code : 1);
307
50.0M
}
308
309
/* ---------------- Internal routines ---------------- */
310
311
/* Prepare to write a string value. */
312
static int
313
ref_param_write_string_value(ref * pref, const gs_param_string * pvalue,
314
                             gs_ref_memory_t *imem)
315
20.6M
{
316
20.6M
    const byte *pdata = pvalue->data;
317
20.6M
    uint n = pvalue->size;
318
319
20.6M
    if (pvalue->persistent)
320
20.6M
        make_const_string(pref, a_readonly | avm_foreign, n, pdata);
321
10.6M
    else {
322
10.6M
        byte *pstr = gs_alloc_string((gs_memory_t *)imem, n,
323
10.6M
                                     "ref_param_write_string");
324
325
10.6M
        if (pstr == 0)
326
0
            return_error(gs_error_VMerror);
327
10.6M
        if (n > 0)
328
6.67M
            memcpy(pstr, pdata, n);
329
10.6M
        make_string(pref, a_readonly | imemory_space(imem), n, pstr);
330
10.6M
    }
331
20.6M
    return 0;
332
20.6M
}
333
334
/* Prepare to write a name value. */
335
static int
336
ref_param_write_name_value(const gs_memory_t *mem, ref * pref, const gs_param_string * pvalue)
337
13.4M
{
338
13.4M
    return name_ref(mem, pvalue->data, pvalue->size, pref,
339
13.4M
                    (pvalue->persistent ? 0 : 1));
340
13.4M
}
341
342
/* Generic routine for writing a ref parameter. */
343
static int
344
ref_param_write(iparam_list * plist, gs_param_name pkey, const ref * pvalue)
345
518M
{
346
518M
    ref kref;
347
518M
    int code;
348
349
518M
    if (!ref_param_requested((gs_param_list *) plist, pkey))
350
347M
        return 0;
351
171M
    code = ref_param_key(plist, pkey, &kref);
352
171M
    if (code < 0)
353
0
        return code;
354
171M
    return (*plist->u.w.write) (plist, &kref, pvalue);
355
171M
}
356
357
/* ---------------- Implementations ---------------- */
358
359
/* Initialize for writing parameters. */
360
static void
361
ref_param_write_init(iparam_list * plist, const ref * pwanted,
362
                     gs_ref_memory_t *imem)
363
19.0M
{
364
19.0M
    gs_param_list_init((gs_param_list *)plist, &ref_write_procs,
365
19.0M
                       (gs_memory_t *)imem);
366
19.0M
    plist->ref_memory = imem;
367
19.0M
    if (pwanted == 0)
368
19.0M
        make_null(&plist->u.w.wanted);
369
10.5M
    else
370
10.5M
        plist->u.w.wanted = *pwanted;
371
19.0M
    plist->results = 0;
372
19.0M
    plist->int_keys = false;
373
19.0M
}
374
375
/* Implementation for getting parameters to a stack. */
376
static int
377
stack_param_write(iparam_list * plist, const ref * pkey, const ref * pvalue)
378
152M
{
379
152M
    stack_param_list *const splist = (stack_param_list *) plist;
380
152M
    ref_stack_t *pstack = splist->pstack;
381
152M
    s_ptr p = pstack->p;
382
383
152M
    if (pstack->top - p < 2) {
384
458
        int code = ref_stack_push(pstack, 2);
385
458
        ref *o;
386
387
458
        if (code < 0)
388
0
            return code;
389
458
        o = ref_stack_index(pstack, 1);
390
458
        if (o == NULL)
391
0
            return_error(gs_error_stackunderflow);
392
458
        else
393
458
            *o = *pkey;
394
458
        p = pstack->p;
395
152M
    } else {
396
152M
        pstack->p = p += 2;
397
152M
        p[-1] = *pkey;
398
152M
    }
399
152M
    *p = *pvalue;
400
152M
    splist->count++;
401
152M
    return 0;
402
152M
}
403
404
/* Implementation for enumerating parameters on a stack */
405
static int      /* ret 0 ok, 1 if EOF, or -ve err */
406
stack_param_enumerate(iparam_list * plist, gs_param_enumerator_t * penum,
407
                      gs_param_key_t * key, ref_type * type)
408
0
{
409
0
    int code;
410
0
    stack_param_list *const splist = (stack_param_list *) plist;
411
0
    int index = penum->intval;
412
0
    ref *stack_element;
413
414
0
    do {
415
0
        if (index >= splist->count*2)
416
0
            return 1;
417
0
        stack_element =
418
0
            ref_stack_index(splist->pstack, index + 1 + splist->skip);
419
0
        if (!stack_element)
420
0
            return 1;
421
0
        index += 2;
422
0
    } while (!r_has_type(stack_element, t_name));
423
0
    *type = r_type(stack_element);
424
0
    code = ref_to_key(stack_element, key, plist);
425
0
    penum->intval = index;
426
0
    return code;
427
0
}
428
429
int
430
stack_param_list_write(stack_param_list * plist, ref_stack_t * pstack,
431
                       const ref * pwanted, gs_ref_memory_t *imem)
432
12.9M
{
433
12.9M
    plist->u.w.write = stack_param_write;
434
12.9M
    ref_param_write_init((iparam_list *) plist, pwanted, imem);
435
12.9M
    plist->enumerate = stack_param_enumerate;
436
12.9M
    plist->pstack = pstack;
437
12.9M
    plist->skip = 0;
438
12.9M
    plist->count = 0;
439
12.9M
    return 0;
440
12.9M
}
441
442
/* Implementation for getting parameters to a dictionary. */
443
static int
444
dict_param_write(iparam_list * plist, const ref * pkey, const ref * pvalue)
445
18.7M
{
446
18.7M
    int code =
447
18.7M
        dict_put(&((dict_param_list *) plist)->dict, pkey, pvalue, NULL);
448
449
18.7M
    return min(code, 0);
450
18.7M
}
451
452
/* Implementation for enumerating parameters in a dictionary */
453
static int      /* ret 0 ok, 1 if EOF, or -ve err */
454
dict_param_enumerate(iparam_list * plist, gs_param_enumerator_t * penum,
455
                     gs_param_key_t * key, ref_type * type)
456
4.21M
{
457
4.21M
    ref elt[2];
458
4.21M
    int code;
459
4.21M
    dict_param_list *const pdlist = (dict_param_list *) plist;
460
4.21M
    int index =
461
4.21M
    (penum->intval != 0 ? penum->intval : dict_first(&pdlist->dict));
462
463
4.21M
    index = dict_next(&pdlist->dict, index, elt);
464
4.21M
    if (index < 0)
465
940k
        return 1;
466
3.26M
    *type = r_type(&elt[0]);
467
3.26M
    code = ref_to_key(&elt[0], key, plist);
468
3.26M
    penum->intval = index;
469
3.26M
    return code;
470
4.21M
}
471
472
int
473
dict_param_list_write(dict_param_list *plist, ref *pdict, const ref *pwanted,
474
                      gs_ref_memory_t *imem)
475
6.06M
{
476
6.06M
    check_dict_write(*pdict);
477
6.06M
    plist->u.w.write = dict_param_write;
478
6.06M
    plist->enumerate = dict_param_enumerate;
479
6.06M
    ref_param_write_init((iparam_list *) plist, pwanted, imem);
480
6.06M
    plist->dict = *pdict;
481
6.06M
    return 0;
482
6.06M
}
483
484
/* Implementation for getting parameters to an indexed array. */
485
/* Note that this is now internal, since it only handles "new" arrays. */
486
static int
487
array_new_indexed_param_write(iparam_list * iplist, const ref * pkey,
488
                          const ref * pvalue)
489
0
{
490
0
    const ref *const arr = &((dict_param_list *)iplist)->dict;
491
0
    ref *eltp;
492
493
0
    if (!r_has_type(pkey, t_integer))
494
0
        return_error(gs_error_typecheck);
495
0
    check_int_ltu(*pkey, r_size(arr));
496
0
    store_check_dest(arr, pvalue);
497
0
    eltp = arr->value.refs + pkey->value.intval;
498
    /* ref_assign_new(eltp, pvalue); */
499
0
    ref_assign(eltp, pvalue);
500
0
    r_set_attrs(eltp, imemory_new_mask(iplist->ref_memory));
501
0
    return 0;
502
0
}
503
static int
504
array_new_indexed_plist_write(dict_param_list * plist, ref * parray,
505
                              const ref * pwanted, gs_ref_memory_t *imem)
506
0
{
507
0
    check_array(*parray);
508
0
    check_write(*parray);
509
0
    plist->u.w.write = array_new_indexed_param_write;
510
0
    ref_param_write_init((iparam_list *) plist, pwanted, imem);
511
0
    plist->dict = *parray;
512
0
    plist->int_keys = true;
513
0
    return 0;
514
0
}
515
516
/* ================ Reading refs to parameters ================ */
517
518
/* ---------------- Generic reading procedures ---------------- */
519
520
static param_proc_begin_xmit_collection(ref_param_begin_read_collection);
521
static param_proc_end_xmit_collection(ref_param_end_read_collection);
522
static param_proc_xmit_typed(ref_param_read_typed);
523
524
/*static param_proc_next_key(ref_param_get_next_key); already dec'ld above */
525
static param_proc_get_policy(ref_param_read_get_policy);
526
static param_proc_signal_error(ref_param_read_signal_error);
527
static param_proc_commit(ref_param_read_commit);
528
static const gs_param_list_procs ref_read_procs =
529
{
530
    ref_param_read_typed,
531
    ref_param_begin_read_collection,
532
    ref_param_end_read_collection,
533
    ref_param_get_next_key,
534
    NULL,     /* request */
535
    NULL,     /* requested */
536
    ref_param_read_get_policy,
537
    ref_param_read_signal_error,
538
    ref_param_read_commit,
539
    NULL
540
};
541
static int ref_param_read(iparam_list *, gs_param_name,
542
                           iparam_loc *, int);
543
static int ref_param_read_string_value(gs_memory_t *mem,
544
                                        const iparam_loc *,
545
                                        gs_param_string *);
546
static int ref_param_read_array(iparam_list *, gs_param_name,
547
                                 iparam_loc *);
548
549
#define iparam_note_error(loc, code)\
550
0
  gs_note_error(*(loc).presult = code)
551
#define iparam_check_type(loc, typ)\
552
0
  if ( !r_has_type((loc).pvalue, typ) )\
553
0
    return iparam_note_error(loc, gs_error_typecheck)
554
#define iparam_check_read(loc)\
555
16.1M
  if ( !r_has_attr((loc).pvalue, a_read) )\
556
16.1M
    return iparam_note_error(loc, gs_error_invalidaccess)
557
558
static int
559
ref_param_read_int_array(gs_param_list * plist, gs_param_name pkey,
560
                         gs_param_int_array * pvalue)
561
1.92M
{
562
1.92M
    iparam_list *const iplist = (iparam_list *) plist;
563
1.92M
    iparam_loc loc;
564
1.92M
    int code = ref_param_read_array(iplist, pkey, &loc);
565
1.92M
    int *piv;
566
1.92M
    uint size;
567
1.92M
    long i;
568
569
1.92M
    if (code != 0)
570
0
        return code;
571
1.92M
    size = r_size(loc.pvalue);
572
1.92M
    piv = (int *)gs_alloc_byte_array(plist->memory, size, sizeof(int),
573
1.92M
                                     "ref_param_read_int_array");
574
575
1.92M
    if (piv == 0)
576
0
        return_error(gs_error_VMerror);
577
9.77M
    for (i = 0; i < size; i++) {
578
7.84M
        ref elt;
579
580
7.84M
        array_get(plist->memory, loc.pvalue, i, &elt);
581
7.84M
        if (!r_has_type(&elt, t_integer)) {
582
0
            code = gs_note_error(gs_error_typecheck);
583
0
            break;
584
0
        }
585
7.84M
        piv[i] = (int)elt.value.intval;
586
7.84M
    }
587
1.92M
    if (code < 0) {
588
0
        gs_free_object(plist->memory, piv, "ref_param_read_int_array");
589
0
        return (*loc.presult = code);
590
0
    }
591
1.92M
    pvalue->data = piv;
592
1.92M
    pvalue->size = size;
593
1.92M
    pvalue->persistent = true;
594
1.92M
    return 0;
595
1.92M
}
596
static int
597
ref_param_read_float_array(gs_param_list * plist, gs_param_name pkey,
598
                           gs_param_float_array * pvalue)
599
1.64M
{
600
1.64M
    iparam_list *const iplist = (iparam_list *) plist;
601
1.64M
    iparam_loc loc;
602
1.64M
    ref aref, elt;
603
1.64M
    int code = ref_param_read_array(iplist, pkey, &loc);
604
1.64M
    float *pfv;
605
1.64M
    uint size;
606
1.64M
    long i;
607
608
1.64M
    if (code != 0)
609
0
        return code;
610
1.64M
    size = r_size(loc.pvalue);
611
1.64M
    pfv = (float *)gs_alloc_byte_array(plist->memory, size, sizeof(float),
612
1.64M
                                       "ref_param_read_float_array");
613
614
1.64M
    if (pfv == 0)
615
0
        return_error(gs_error_VMerror);
616
1.64M
    aref = *loc.pvalue;
617
1.64M
    loc.pvalue = &elt;
618
5.25M
    for (i = 0; code >= 0 && i < size; i++) {
619
3.60M
        array_get(plist->memory, &aref, i, &elt);
620
3.60M
        code = float_param(&elt, pfv + i);
621
3.60M
    }
622
1.64M
    if (code < 0) {
623
0
        gs_free_object(plist->memory, pfv, "ref_read_float_array_param");
624
0
        return (*loc.presult = code);
625
0
    }
626
1.64M
    pvalue->data = pfv;
627
1.64M
    pvalue->size = size;
628
1.64M
    pvalue->persistent = true;
629
1.64M
    return 0;
630
1.64M
}
631
static int
632
ref_param_read_string_array(gs_param_list * plist, gs_param_name pkey,
633
                            gs_param_string_array * pvalue)
634
190k
{
635
190k
    iparam_list *const iplist = (iparam_list *) plist;
636
190k
    iparam_loc loc;
637
190k
    ref aref;
638
190k
    int code = ref_param_read_array(iplist, pkey, &loc);
639
190k
    gs_param_string *psv;
640
190k
    uint size;
641
190k
    long i;
642
643
190k
    if (code != 0)
644
0
        return code;
645
190k
    size = r_size(loc.pvalue);
646
190k
    psv = (gs_param_string *)
647
190k
        gs_alloc_byte_array(plist->memory, size, sizeof(gs_param_string),
648
190k
                            "ref_param_read_string_array");
649
190k
    if (psv == 0)
650
0
        return_error(gs_error_VMerror);
651
190k
    aref = *loc.pvalue;
652
190k
    if (r_has_type(&aref, t_array)) {
653
2.60M
        for (i = 0; code >= 0 && i < size; i++) {
654
2.41M
            loc.pvalue = aref.value.refs + i;
655
2.41M
            code = ref_param_read_string_value(plist->memory, &loc, psv + i);
656
2.41M
        }
657
190k
    } else {
658
0
        ref elt;
659
660
0
        loc.pvalue = &elt;
661
0
        for (i = 0; code >= 0 && i < size; i++) {
662
0
            array_get(plist->memory, &aref, i, &elt);
663
0
            code = ref_param_read_string_value(plist->memory, &loc, psv + i);
664
0
        }
665
0
    }
666
190k
    if (code < 0) {
667
0
        gs_free_object(plist->memory, psv, "ref_param_read_string_array");
668
0
        return (*loc.presult = code);
669
0
    }
670
190k
    pvalue->data = psv;
671
190k
    pvalue->size = size;
672
190k
    pvalue->persistent = true;
673
190k
    return 0;
674
190k
}
675
static int
676
ref_param_begin_read_collection(gs_param_list * plist, gs_param_name pkey,
677
                                gs_param_dict * pvalue,
678
                                gs_param_collection_type_t coll_type)
679
4.59M
{
680
4.59M
    iparam_list *const iplist = (iparam_list *) plist;
681
4.59M
    iparam_loc loc;
682
4.59M
    bool int_keys = coll_type != 0;
683
4.59M
    int code = ref_param_read(iplist, pkey, &loc, -1);
684
4.59M
    dict_param_list *dlist;
685
686
4.59M
    if (code != 0)
687
3.65M
        return code;
688
940k
    dlist = (dict_param_list *)
689
940k
        gs_alloc_bytes(plist->memory, size_of(dict_param_list),
690
940k
                       "ref_param_begin_read_collection");
691
940k
    if (dlist == 0)
692
0
        return_error(gs_error_VMerror);
693
940k
    if (r_has_type(loc.pvalue, t_dictionary)) {
694
940k
        code = dict_param_list_read(dlist, loc.pvalue, NULL, false,
695
940k
                                    iplist->ref_memory);
696
940k
        dlist->int_keys = int_keys;
697
940k
        if (code >= 0)
698
940k
            pvalue->size = dict_length(loc.pvalue);
699
940k
    } else if (int_keys && r_is_array(loc.pvalue)) {
700
0
        code = array_indexed_param_list_read(dlist, loc.pvalue, NULL, false,
701
0
                                             iplist->ref_memory);
702
0
        if (code >= 0)
703
0
            pvalue->size = r_size(loc.pvalue);
704
0
    } else
705
0
        code = gs_note_error(gs_error_typecheck);
706
940k
    if (code < 0) {
707
0
        gs_free_object(plist->memory, dlist, "ref_param_begin_write_collection");
708
0
        return iparam_note_error(loc, code);
709
0
    }
710
940k
    pvalue->list = (gs_param_list *) dlist;
711
940k
    return 0;
712
940k
}
713
static int
714
ref_param_end_read_collection(gs_param_list * plist, gs_param_name pkey,
715
                              gs_param_dict * pvalue)
716
940k
{
717
940k
    iparam_list_release((dict_param_list *) pvalue->list);
718
940k
    gs_free_object(plist->memory, pvalue->list,
719
940k
                   "ref_param_end_read_collection");
720
940k
    return 0;
721
940k
}
722
static int
723
ref_param_read_typed(gs_param_list * plist, gs_param_name pkey,
724
                     gs_param_typed_value * pvalue)
725
387M
{
726
387M
    iparam_list *const iplist = (iparam_list *) plist;
727
387M
    iparam_loc loc;
728
387M
    ref elt;
729
387M
    int code = ref_param_read(iplist, pkey, &loc, -1);
730
731
387M
    if (code != 0)
732
340M
        return code;
733
47.2M
    switch (r_type(loc.pvalue)) {
734
5.07M
        case t_array:
735
5.07M
        case t_mixedarray:
736
5.07M
        case t_shortarray:
737
5.07M
            iparam_check_read(loc);
738
5.07M
            if (r_size(loc.pvalue) <= 0) {
739
                /* 0-length array; can't get type info */
740
1.30M
                pvalue->type = gs_param_type_array;
741
1.30M
                pvalue->value.d.list = 0;
742
1.30M
                pvalue->value.d.size = 0;
743
1.30M
                return 0;
744
1.30M
            }
745
            /*
746
             * We have to guess at the array type.  First we guess based
747
             * on the type of the first element of the array.  If that
748
             * fails, we try again with more general types.
749
             */
750
3.76M
            array_get(plist->memory, loc.pvalue, 0, &elt);
751
3.76M
            switch (r_type(&elt)) {
752
1.92M
                case t_integer:
753
1.92M
                    pvalue->type = gs_param_type_int_array;
754
1.92M
                    code = ref_param_read_int_array(plist, pkey,
755
1.92M
                                                    &pvalue->value.ia);
756
1.92M
                    if (code != gs_error_typecheck)
757
1.92M
                        return code;
758
                    /* This might be a float array.  Fall through. */
759
0
                    *loc.presult = 0;  /* reset error */
760
1.64M
                case t_real:
761
1.64M
                    pvalue->type = gs_param_type_float_array;
762
1.64M
                    return ref_param_read_float_array(plist, pkey,
763
1.64M
                                                      &pvalue->value.fa);
764
38
                case t_string:
765
38
                    pvalue->type = gs_param_type_string_array;
766
38
                    return ref_param_read_string_array(plist, pkey,
767
38
                                                       &pvalue->value.sa);
768
190k
                case t_name:
769
190k
                    pvalue->type = gs_param_type_name_array;
770
190k
                    return ref_param_read_string_array(plist, pkey,
771
190k
                                                       &pvalue->value.na);
772
0
                default:
773
0
                    break;
774
3.76M
            }
775
0
            return gs_note_error(gs_error_typecheck);
776
12.9M
        case t_boolean:
777
12.9M
            pvalue->type = gs_param_type_bool;
778
12.9M
            pvalue->value.b = loc.pvalue->value.boolval;
779
12.9M
            return 0;
780
1
        case t_dictionary:
781
1
            code = ref_param_begin_read_collection(plist, pkey,
782
1
                            &pvalue->value.d, gs_param_collection_dict_any);
783
1
            if (code < 0)
784
0
                return code;
785
1
            pvalue->type = gs_param_type_dict;
786
787
            /* fixup new dict's type & int_keys field if contents have int keys */
788
1
            {
789
1
                gs_param_enumerator_t enumr;
790
1
                gs_param_key_t key;
791
1
                ref_type keytype;
792
1
                dict_param_list *dlist = (dict_param_list *) pvalue->value.d.list;
793
794
1
                param_init_enumerator(&enumr);
795
1
                if (!(*(dlist->enumerate))
796
1
                    ((iparam_list *) dlist, &enumr, &key, &keytype)
797
1
                    && keytype == t_integer) {
798
0
                    dlist->int_keys = 1;
799
0
                    pvalue->type = gs_param_type_dict_int_keys;
800
0
                }
801
1
            }
802
1
            return 0;
803
17.3M
        case t_integer:
804
17.3M
            pvalue->type = gs_param_type_i64;
805
17.3M
            pvalue->value.i64 = loc.pvalue->value.intval;
806
17.3M
            return 0;
807
2.28M
        case t_name:
808
2.28M
            pvalue->type = gs_param_type_name;
809
2.28M
            return ref_param_read_string_value(plist->memory, &loc, &pvalue->value.n);
810
385k
        case t_null:
811
385k
            pvalue->type = gs_param_type_null;
812
385k
            return 0;
813
1.95M
        case t_real:
814
1.95M
            pvalue->value.f = loc.pvalue->value.realval;
815
1.95M
            pvalue->type = gs_param_type_float;
816
1.95M
            return 0;
817
7.26M
        case t_string:
818
7.26M
        case t_astruct:
819
7.26M
            pvalue->type = gs_param_type_string;
820
7.26M
            return ref_param_read_string_value(plist->memory, &loc, &pvalue->value.s);
821
0
        default:
822
0
            break;
823
47.2M
    }
824
0
    return gs_note_error(gs_error_typecheck);
825
47.2M
}
826
827
static int
828
ref_param_read_get_policy(gs_param_list * plist, gs_param_name pkey)
829
685
{
830
685
    iparam_list *const iplist = (iparam_list *) plist;
831
685
    ref *pvalue;
832
833
685
    if (!(r_has_type(&iplist->u.r.policies, t_dictionary) &&
834
685
          dict_find_string(&iplist->u.r.policies, pkey, &pvalue) > 0 &&
835
685
          r_has_type(pvalue, t_integer))
836
685
        )
837
685
        return gs_param_policy_ignore;
838
0
    return (int)pvalue->value.intval;
839
685
}
840
static int
841
ref_param_read_signal_error(gs_param_list * plist, gs_param_name pkey, int code)
842
685
{
843
685
    iparam_list *const iplist = (iparam_list *) plist;
844
685
    iparam_loc loc = {0};
845
846
685
    ref_param_read(iplist, pkey, &loc, -1);
847
685
    if (loc.presult)
848
345
        *loc.presult = code;
849
685
    switch (ref_param_read_get_policy(plist, pkey)) {
850
685
        case gs_param_policy_ignore:
851
685
            return 0;
852
0
        case gs_param_policy_consult_user:
853
0
            return_error(gs_error_configurationerror);
854
0
        default:
855
0
            return code;
856
685
    }
857
685
}
858
static int
859
ref_param_read_commit(gs_param_list * plist)
860
1.55M
{
861
1.55M
    iparam_list *const iplist = (iparam_list *) plist;
862
1.55M
    int i;
863
1.55M
    int ecode = 0;
864
865
1.55M
    if (!iplist->u.r.require_all)
866
374k
        return 0;
867
    /* Check to make sure that all parameters were actually read. */
868
18.4M
    for (i = 0; i < iplist->count; ++i)
869
17.2M
        if (iplist->results[i] == 0)
870
1.18M
            iplist->results[i] = ecode = gs_note_error(gs_error_undefined);
871
1.18M
    return ecode;
872
1.55M
}
873
static int
874
ref_param_get_next_key(gs_param_list * plist, gs_param_enumerator_t * penum,
875
                       gs_param_key_t * key)
876
4.21M
{
877
4.21M
    ref_type keytype;   /* result not needed here */
878
4.21M
    iparam_list *const pilist = (iparam_list *) plist;
879
880
4.21M
    return (*pilist->enumerate) (pilist, penum, key, &keytype);
881
4.21M
}
882
883
/* ---------------- Internal routines ---------------- */
884
885
/* Read a string value. */
886
static int
887
ref_param_read_string_value(gs_memory_t *mem, const iparam_loc * ploc, gs_param_string * pvalue)
888
11.9M
{
889
11.9M
    const ref *pref = ploc->pvalue;
890
891
11.9M
    switch (r_type(pref)) {
892
4.67M
        case t_name: {
893
4.67M
            ref nref;
894
895
4.67M
            name_string_ref(mem, pref, &nref);
896
4.67M
            pvalue->data = nref.value.const_bytes;
897
4.67M
            pvalue->size = r_size(&nref);
898
4.67M
            pvalue->persistent = true;
899
4.67M
        }
900
4.67M
            break;
901
7.28M
        case t_string:
902
7.28M
            iparam_check_read(*ploc);
903
7.28M
            pvalue->data = pref->value.const_bytes;
904
7.28M
            pvalue->size = r_size(pref);
905
7.28M
            pvalue->persistent = false;
906
7.28M
            break;
907
0
        case t_astruct:
908
            /* Note: technically, instead of the "mem" argument, we
909
               should be using the plists's ref_memory. However, in a
910
               simple call to .putdeviceparams, they are identical. */
911
0
            iparam_check_read(*ploc);
912
0
            if (gs_object_type(mem, pref->value.pstruct) != &st_bytes)
913
0
                return iparam_note_error(*ploc, gs_error_typecheck);
914
0
            pvalue->data = r_ptr(pref, byte);
915
0
            pvalue->size = gs_object_size(mem, pref->value.pstruct);
916
0
            pvalue->persistent = false;
917
0
            break;
918
0
        default:
919
0
            return iparam_note_error(*ploc, gs_error_typecheck);
920
11.9M
    }
921
11.9M
    return 0;
922
11.9M
}
923
924
/* Read an array (or packed array) parameter. */
925
static int
926
ref_param_read_array(iparam_list * plist, gs_param_name pkey, iparam_loc * ploc)
927
3.76M
{
928
3.76M
    int code = ref_param_read(plist, pkey, ploc, -1);
929
930
3.76M
    if (code != 0)
931
0
        return code;
932
3.76M
    if (!r_is_array(ploc->pvalue))
933
0
        return iparam_note_error(*ploc, gs_error_typecheck);
934
3.76M
    iparam_check_read(*ploc);
935
3.76M
    return 0;
936
3.76M
}
937
938
/* Generic routine for reading a ref parameter. */
939
static int
940
ref_param_read(iparam_list * plist, gs_param_name pkey, iparam_loc * ploc,
941
               int type)
942
395M
{
943
395M
    iparam_list *const iplist = (iparam_list *) plist;
944
395M
    ref kref;
945
395M
    int code = ref_param_key(plist, pkey, &kref);
946
947
395M
    if (code < 0)
948
0
        return code;
949
395M
    code = (*plist->u.r.read) (iplist, &kref, ploc);
950
395M
    if (code != 0)
951
343M
        return code;
952
51.9M
    if (type >= 0)
953
0
        iparam_check_type(*ploc, type);
954
51.9M
    return 0;
955
51.9M
}
956
957
/* ---------------- Implementations ---------------- */
958
959
/* Implementation for putting parameters from an empty collection. */
960
static int
961
empty_param_read(iparam_list * plist, const ref * pkey, iparam_loc * ploc)
962
0
{
963
0
    return 1;
964
0
}
965
966
/* Initialize for reading parameters. */
967
static int
968
ref_param_read_init(iparam_list * plist, uint count, const ref * ppolicies,
969
                    bool require_all, gs_ref_memory_t *imem)
970
6.90M
{
971
6.90M
    gs_param_list_init((gs_param_list *)plist, &ref_read_procs,
972
6.90M
                       (gs_memory_t *)imem);
973
6.90M
    plist->ref_memory = imem;
974
6.90M
    if (ppolicies == 0)
975
6.90M
        make_null(&plist->u.r.policies);
976
1.57M
    else
977
1.57M
        plist->u.r.policies = *ppolicies;
978
6.90M
    plist->u.r.require_all = require_all;
979
6.90M
    plist->count = count;
980
6.90M
    plist->results = (int *)
981
6.90M
        gs_alloc_byte_array(plist->memory, count, sizeof(int),
982
6.90M
                            "ref_param_read_init");
983
984
6.90M
    if (plist->results == 0)
985
0
        return_error(gs_error_VMerror);
986
6.90M
    memset(plist->results, 0, count * sizeof(int));
987
988
6.90M
    plist->int_keys = false;
989
6.90M
    return 0;
990
6.90M
}
991
992
/* Implementation for putting parameters from an indexed array. */
993
static int
994
array_indexed_param_read(iparam_list * plist, const ref * pkey, iparam_loc * ploc)
995
0
{
996
0
    ref *const arr = &((dict_param_list *) plist)->dict;
997
998
0
    check_type(*pkey, t_integer);
999
0
    if (pkey->value.intval < 0 || pkey->value.intval >= r_size(arr))
1000
0
        return 1;
1001
0
    ploc->pvalue = arr->value.refs + pkey->value.intval;
1002
0
    ploc->presult = &plist->results[pkey->value.intval];
1003
0
    *ploc->presult = 1;
1004
0
    return 0;
1005
0
}
1006
int
1007
array_indexed_param_list_read(dict_param_list * plist, const ref * parray,
1008
                              const ref * ppolicies, bool require_all,
1009
                              gs_ref_memory_t *ref_memory)
1010
0
{
1011
0
    iparam_list *const iplist = (iparam_list *) plist;
1012
0
    int code;
1013
1014
0
    check_read_type(*parray, t_array);
1015
0
    plist->u.r.read = array_indexed_param_read;
1016
0
    plist->dict = *parray;
1017
0
    code = ref_param_read_init(iplist, r_size(parray), ppolicies,
1018
0
                               require_all, ref_memory);
1019
0
    plist->int_keys = true;
1020
0
    return code;
1021
0
}
1022
1023
/* Implementation for putting parameters from an array. */
1024
static int
1025
array_param_read(iparam_list * plist, const ref * pkey, iparam_loc * ploc)
1026
0
{
1027
0
    ref *bot = ((array_param_list *) plist)->bot;
1028
0
    ref *ptr = bot;
1029
0
    ref *top = ((array_param_list *) plist)->top;
1030
1031
0
    for (; ptr < top; ptr += 2) {
1032
0
        if (r_has_type(ptr, t_name) && name_eq(ptr, pkey)) {
1033
0
            ploc->pvalue = ptr + 1;
1034
0
            ploc->presult = &plist->results[ptr - bot];
1035
0
            *ploc->presult = 1;
1036
0
            return 0;
1037
0
        }
1038
0
    }
1039
0
    return 1;
1040
0
}
1041
1042
/* Implementation for enumerating parameters in an array */
1043
static int      /* ret 0 ok, 1 if EOF, or -ve err */
1044
array_param_enumerate(iparam_list * plist, gs_param_enumerator_t * penum,
1045
                      gs_param_key_t * key, ref_type * type)
1046
0
{
1047
0
    int index = penum->intval;
1048
0
    ref *bot = ((array_param_list *) plist)->bot;
1049
0
    ref *ptr = bot + index;
1050
0
    ref *top = ((array_param_list *) plist)->top;
1051
1052
0
    for (; ptr < top; ptr += 2) {
1053
0
        index += 2;
1054
1055
0
        if (r_has_type(ptr, t_name)) {
1056
0
            int code = ref_to_key(ptr, key, plist);
1057
1058
0
            *type = r_type(ptr);
1059
0
            penum->intval = index;
1060
0
            return code;
1061
0
        }
1062
0
    }
1063
0
    return 1;
1064
0
}
1065
1066
int
1067
array_param_list_read(array_param_list * plist, ref * bot, uint count,
1068
                      const ref * ppolicies, bool require_all,
1069
                      gs_ref_memory_t *imem)
1070
0
{
1071
0
    iparam_list *const iplist = (iparam_list *) plist;
1072
1073
0
    if (count & 1)
1074
0
        return_error(gs_error_rangecheck);
1075
0
    plist->u.r.read = array_param_read;
1076
0
    plist->enumerate = array_param_enumerate;
1077
0
    plist->bot = bot;
1078
0
    plist->top = bot + count;
1079
0
    return ref_param_read_init(iplist, count, ppolicies, require_all, imem);
1080
0
}
1081
1082
/* Implementation for putting parameters from a stack. */
1083
static int
1084
stack_param_read(iparam_list * plist, const ref * pkey, iparam_loc * ploc)
1085
281M
{
1086
281M
    stack_param_list *const splist = (stack_param_list *) plist;
1087
281M
    ref_stack_t *pstack = splist->pstack;
1088
1089
    /* This implementation is slow, but it probably doesn't matter. */
1090
281M
    uint index = splist->skip + 1;
1091
281M
    uint count = splist->count;
1092
1093
4.78G
    for (; count; count--, index += 2) {
1094
4.53G
        const ref *p = ref_stack_index(pstack, index);
1095
1096
4.53G
        if (p == NULL)
1097
0
            continue;
1098
1099
4.53G
        if (r_has_type(p, t_name) && name_eq(p, pkey)) {
1100
31.8M
            ploc->pvalue = ref_stack_index(pstack, index - 1);
1101
31.8M
            ploc->presult = &plist->results[count - 1];
1102
31.8M
            *ploc->presult = 1;
1103
31.8M
            return 0;
1104
31.8M
        }
1105
4.53G
    }
1106
249M
    return 1;
1107
281M
}
1108
int
1109
stack_param_list_read(stack_param_list * plist, ref_stack_t * pstack,
1110
                      uint skip, const ref * ppolicies, bool require_all,
1111
                      gs_ref_memory_t *imem)
1112
1.57M
{
1113
1.57M
    iparam_list *const iplist = (iparam_list *) plist;
1114
1.57M
    uint count = ref_stack_counttomark(pstack);
1115
1116
1.57M
    if (count == 0)
1117
0
        return_error(gs_error_unmatchedmark);
1118
1.57M
    count -= skip + 1;
1119
1.57M
    if (count & 1)
1120
0
        return_error(gs_error_rangecheck);
1121
1.57M
    plist->u.r.read = stack_param_read;
1122
1.57M
    plist->enumerate = stack_param_enumerate;
1123
1.57M
    plist->pstack = pstack;
1124
1.57M
    plist->skip = skip;
1125
1.57M
    return ref_param_read_init(iplist, count >> 1, ppolicies, require_all, imem);
1126
1.57M
}
1127
1128
/* Implementation for putting parameters from a dictionary. */
1129
static int
1130
dict_param_read(iparam_list * plist, const ref * pkey, iparam_loc * ploc)
1131
114M
{
1132
114M
    ref const *spdict = &((dict_param_list *) plist)->dict;
1133
114M
    int code = dict_find(spdict, pkey, &ploc->pvalue);
1134
1135
114M
    if (code != 1)
1136
94.2M
        return 1;
1137
20.0M
    ploc->presult =
1138
20.0M
        &plist->results[dict_value_index(spdict, ploc->pvalue)];
1139
20.0M
    *ploc->presult = 1;
1140
20.0M
    return 0;
1141
114M
}
1142
int
1143
dict_param_list_read(dict_param_list * plist, const ref * pdict,
1144
                     const ref * ppolicies, bool require_all,
1145
                     gs_ref_memory_t *imem)
1146
5.32M
{
1147
5.32M
    iparam_list *const iplist = (iparam_list *) plist;
1148
5.32M
    uint count;
1149
1150
5.32M
    if (pdict == 0) {
1151
0
        plist->u.r.read = empty_param_read;
1152
0
        count = 0;
1153
5.32M
    } else {
1154
5.32M
        check_dict_read(*pdict);
1155
5.32M
        plist->u.r.read = dict_param_read;
1156
5.32M
        plist->dict = *pdict;
1157
5.32M
        count = dict_max_index(pdict) + 1;
1158
5.32M
    }
1159
5.32M
    plist->enumerate = dict_param_enumerate;
1160
5.32M
    return ref_param_read_init(iplist, count, ppolicies, require_all, imem);
1161
5.32M
}