Coverage Report

Created: 2025-06-10 07:27

/src/ghostpdl/psi/zarray.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-2023 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
/* Array operators */
18
#include "memory_.h"
19
#include "ghost.h"
20
#include "ialloc.h"
21
#include "ipacked.h"
22
#include "oper.h"
23
#include "store.h"
24
25
/* The generic operators (copy, get, put, getinterval, putinterval, */
26
/* length, and forall) are implemented in zgeneric.c. */
27
28
/* <int> array <array> */
29
int
30
zarray(i_ctx_t *i_ctx_p)
31
48.7M
{
32
48.7M
    os_ptr op = osp;
33
48.7M
    uint size;
34
48.7M
    int code;
35
36
48.7M
    check_op(1);
37
48.7M
    check_type(*op, t_integer);
38
48.7M
    if (op->value.intval < 0)
39
3
        return_error(gs_error_rangecheck);
40
48.7M
    if (op->value.intval > max_array_size)
41
0
        return_error(gs_error_limitcheck);
42
48.7M
    size = op->value.intval;
43
48.7M
    code = ialloc_ref_array((ref *)op, a_all, size, "array");
44
48.7M
    if (code < 0)
45
0
        return code;
46
48.7M
    refset_null(op->value.refs, size);
47
48.7M
    return 0;
48
48.7M
}
49
50
/* <array> aload <obj_0> ... <obj_n-1> <array> */
51
static int
52
zaload(i_ctx_t *i_ctx_p)
53
13.4M
{
54
13.4M
    os_ptr op = osp;
55
13.4M
    ref aref;
56
13.4M
    uint asize;
57
58
13.4M
    check_op(1);
59
13.4M
    ref_assign(&aref, op);
60
13.4M
    if (!r_is_array(&aref))
61
13.4M
        return_op_typecheck(op);
62
13.4M
    check_read(aref);
63
13.4M
    asize = r_size(&aref);
64
13.4M
    if (asize > ostop - op) { /* Use the slow, general algorithm. */
65
155
        int code = ref_stack_push(&o_stack, asize);
66
155
        uint i;
67
155
        const ref_packed *packed = aref.value.packed;
68
69
155
        if (code < 0)
70
0
            return code;
71
5.46M
        for (i = asize; i > 0; i--, packed = packed_next(packed)) {
72
5.46M
            ref *o = ref_stack_index(&o_stack, i);
73
5.46M
            if (o == NULL)
74
0
                continue;
75
5.46M
            packed_get(imemory, packed, o);
76
5.46M
        }
77
155
        *osp = aref;
78
155
        return 0;
79
155
    }
80
13.4M
    if (r_has_type(&aref, t_array))
81
13.3M
        memcpy(op, aref.value.refs, asize * sizeof(ref));
82
119k
    else {
83
119k
        uint i;
84
119k
        const ref_packed *packed = aref.value.packed;
85
119k
        os_ptr pdest = op;
86
87
2.45M
        for (i = 0; i < asize; i++, pdest++, packed = packed_next(packed))
88
2.33M
            packed_get(imemory, packed, pdest);
89
119k
    }
90
13.4M
    push(asize);
91
13.4M
    ref_assign(op, &aref);
92
13.4M
    return 0;
93
13.4M
}
94
95
/* <obj_0> ... <obj_n-1> <array> astore <array> */
96
static int
97
zastore(i_ctx_t *i_ctx_p)
98
47.0M
{
99
47.0M
    os_ptr op = osp;
100
47.0M
    uint size;
101
47.0M
    int code;
102
103
47.0M
    check_op(1);
104
47.0M
    if (!r_is_array(op))
105
47.0M
        return_op_typecheck(op);
106
47.0M
    size = r_size(op);
107
    /* Amazingly, the following is valid: 0 array noaccess astore */
108
47.0M
    if (size == 0)
109
269k
        return 0;
110
46.7M
    if (!r_has_type_attrs(op, t_array, a_write))
111
0
        return_error(gs_error_invalidaccess);
112
46.7M
    if (size > op - osbot) {
113
        /* The store operation might involve other stack segments. */
114
12.3k
        ref arr, *o;
115
116
12.3k
        if (size >= ref_stack_count(&o_stack))
117
1
            return_error(gs_error_stackunderflow);
118
12.3k
        arr = *op;
119
12.3k
        code = ref_stack_store(&o_stack, &arr, size, 1, 0, true, idmemory,
120
12.3k
                               "astore");
121
12.3k
        if (code < 0)
122
0
            return code;
123
12.3k
        ref_stack_pop(&o_stack, size);
124
12.3k
        o = ref_stack_index(&o_stack, 0);
125
12.3k
        if (o == NULL)
126
0
            return_error(gs_error_stackunderflow);
127
12.3k
        *o = arr;
128
46.7M
    } else {
129
46.7M
        code = refcpy_to_old(op, 0, op - size, size, idmemory, "astore");
130
46.7M
        if (code < 0)
131
0
            return code;
132
46.7M
        op[-(int)size] = *op;
133
46.7M
        pop(size);
134
46.7M
    }
135
46.7M
    return 0;
136
46.7M
}
137
138
/* ------ Initialization procedure ------ */
139
140
const op_def zarray_op_defs[] =
141
{
142
    {"1aload", zaload},
143
    {"1array", zarray},
144
    {"1astore", zastore},
145
    op_def_end(0)
146
};