~ubuntu-branches/ubuntu/jaunty/ghostscript/jaunty-updates

« back to all changes in this revision

Viewing changes to psi/zgeneric.c

  • Committer: Bazaar Package Importer
  • Author(s): Till Kamppeter
  • Date: 2009-01-20 16:40:45 UTC
  • mfrom: (1.1.10 upstream)
  • Revision ID: james.westby@ubuntu.com-20090120164045-lnfhi0n30o5lwhwa
Tags: 8.64.dfsg.1~svn9377-0ubuntu1
* New upstream release (SVN rev 9377)
   o Fixes many bugs concerning PDF rendering, to make the PDF printing
     workflow correctly working.
   o Fixes long-standing bugs in many drivers, like input paper tray and
     duplex options not working for the built-in PCL 4, 5, 5c, 5e, and
     6/XL drivers, PDF input not working for bjc600, bjc800, and cups
     output devices, several options not working and uninitialized
     memory with cups output device.
   o Merged nearly all patches of the Ubuntu and Debian packages upstream.
   o Fixes LP: #317810, LP: #314439, LP: #314018.
* debian/patches/03_libpaper_support.dpatch,
  debian/patches/11_gs-cjk_font_glyph_handling_fix.dpatch,
  debian/patches/12_gs-cjk_vertical_writing_metrics_fix.dpatch,
  debian/patches/13_gs-cjk_cjkps_examples.dpatch,
  debian/patches/20_bbox_segv_fix.dpatch,
  debian/patches/21_brother_7x0_gdi_fix.dpatch,
  debian/patches/22_epsn_margin_workaround.dpatch,
  debian/patches/24_gs_man_fix.dpatch,
  debian/patches/25_toolbin_insecure_tmp_usage_fix.dpatch,
  debian/patches/26_assorted_script_fixes.dpatch,
  debian/patches/29_gs_css_fix.dpatch,
  debian/patches/30_ps2pdf_man_improvement.dpatch,
  debian/patches/31_fix-gc-sigbus.dpatch,
  debian/patches/34_ftbfs-on-hurd-fix.dpatch,
  debian/patches/35_disable_libcairo.dpatch,
  debian/patches/38_pxl-duplex.dpatch,
  debian/patches/39_pxl-resolution.dpatch,
  debian/patches/42_gs-init-ps-delaybind-fix.dpatch,
  debian/patches/45_bjc600-bjc800-pdf-input.dpatch,
  debian/patches/48_cups-output-device-pdf-duplex-uninitialized-memory-fix.dpatch,
  debian/patches/50_lips4-floating-point-exception.dpatch,
  debian/patches/52_cups-device-logging.dpatch,
  debian/patches/55_pcl-input-slot-fix.dpatch,
  debian/patches/57_pxl-input-slot-fix.dpatch,
  debian/patches/60_pxl-cups-driver-pdf.dpatch,
  debian/patches/62_onebitcmyk-pdf.dpatch,
  debian/patches/65_too-big-temp-files-1.dpatch,
  debian/patches/67_too-big-temp-files-2.dpatch,
  debian/patches/70_take-into-account-data-in-stream-buffer-before-refill.dpatch:
  Removed, applied upstream.
* debian/patches/01_docdir_fix_for_debian.dpatch,
  debian/patches/02_gs_man_fix_debian.dpatch,
  debian/patches/01_docdir-fix-for-debian.dpatch,
  debian/patches/02_docdir-fix-for-debian.dpatch: Renamed patches to
  make merging with Debian easier.
* debian/patches/32_improve-handling-of-media-size-changes-from-gv.dpatch, 
  debian/patches/33_bad-params-to-xinitimage-on-large-bitmaps.dpatch:
  regenerated for new source directory structure.
* debian/rules: Corrected paths to remove cidfmap (it is in Resource/Init/
  in GS 8.64) and to install headers (source paths are psi/ and base/ now).
* debian/rules: Remove all fontmaps, as DeFoMa replaces them.
* debian/local/pdftoraster/pdftoraster.c,
  debian/local/pdftoraster/pdftoraster.convs, debian/rules: Removed
  added pdftoraster filter and use the one which comes with Ghostscript.
* debian/ghostscript.links: s/8.63/8.64/

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/* Copyright (C) 2001-2006 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, modified
 
8
   or distributed except as expressly authorized under the terms of that
 
9
   license.  Refer to licensing information at http://www.artifex.com/
 
10
   or contact Artifex Software, Inc.,  7 Mt. Lassen Drive - Suite A-134,
 
11
   San Rafael, CA  94903, U.S.A., +1(415)492-9861, for further information.
 
12
*/
 
13
 
 
14
/* $Id: zgeneric.c 9043 2008-08-28 22:48:19Z giles $ */
 
15
/* Array/string/dictionary generic operators for PostScript */
 
16
#include "memory_.h"
 
17
#include "ghost.h"
 
18
#include "gsstruct.h"           /* for st_bytes */
 
19
#include "oper.h"
 
20
#include "dstack.h"             /* for systemdict */
 
21
#include "estack.h"             /* for forall */
 
22
#include "iddict.h"
 
23
#include "iname.h"
 
24
#include "ipacked.h"
 
25
#include "ivmspace.h"
 
26
#include "store.h"
 
27
 
 
28
/* This file implements copy, get, put, getinterval, putinterval, */
 
29
/* length, and forall, which apply generically to */
 
30
/* arrays, strings, and dictionaries.  (Copy also has a special */
 
31
/* meaning for copying the top N elements of the stack.) */
 
32
 
 
33
/* See the comment in opdef.h for an invariant which allows */
 
34
/* more efficient implementation of forall. */
 
35
 
 
36
/* Forward references */
 
37
static int zcopy_integer(i_ctx_t *);
 
38
static int zcopy_interval(i_ctx_t *);
 
39
static int copy_interval(i_ctx_t *, os_ptr, uint, os_ptr, client_name_t);
 
40
 
 
41
/* <various1> <various2> copy <various> */
 
42
/* <obj1> ... <objn> <int> copy <obj1> ... <objn> <obj1> ... <objn> */
 
43
/* Note that this implements copy for arrays and strings, */
 
44
/* but not for dictionaries (see zcopy_dict in zdict.c). */
 
45
int
 
46
zcopy(i_ctx_t *i_ctx_p)
 
47
{
 
48
    os_ptr op = osp;
 
49
    int type = r_type(op);
 
50
 
 
51
    if (type == t_integer)
 
52
        return zcopy_integer(i_ctx_p);
 
53
    check_op(2);
 
54
    switch (type) {
 
55
        case t_array:
 
56
        case t_string:
 
57
            return zcopy_interval(i_ctx_p);
 
58
        case t_dictionary:
 
59
            return zcopy_dict(i_ctx_p);
 
60
        default:
 
61
            return_op_typecheck(op);
 
62
    }
 
63
}
 
64
 
 
65
/* <obj1> ... <objn> <int> copy <obj1> ... <objn> <obj1> ... <objn> */
 
66
static int
 
67
zcopy_integer(i_ctx_t *i_ctx_p)
 
68
{
 
69
    os_ptr op = osp;
 
70
    os_ptr op1 = op - 1;
 
71
    int count, i;
 
72
    int code;
 
73
 
 
74
    if ((ulong) op->value.intval > (ulong)(op - osbot)) {
 
75
        /* There might be enough elements in other blocks. */
 
76
        check_type(*op, t_integer);
 
77
        if (op->value.intval >= (int)ref_stack_count(&o_stack)) 
 
78
            return_error(e_stackunderflow);
 
79
        if (op->value.intval < 0) 
 
80
            return_error(e_rangecheck);
 
81
        check_int_ltu(*op, ref_stack_count(&o_stack));
 
82
        count = op->value.intval;
 
83
    } else if (op1 + (count = op->value.intval) <= ostop) {
 
84
        /* Fast case. */
 
85
        memcpy((char *)op, (char *)(op - count), count * sizeof(ref));
 
86
        push(count - 1);
 
87
        return 0;
 
88
    }
 
89
    /* Do it the slow, general way. */
 
90
    code = ref_stack_push(&o_stack, count - 1);
 
91
    if (code < 0)
 
92
        return code;
 
93
    for (i = 0; i < count; i++)
 
94
        *ref_stack_index(&o_stack, i) =
 
95
            *ref_stack_index(&o_stack, i + count);
 
96
    return 0;
 
97
}
 
98
 
 
99
/* <array1> <array2> copy <subarray2> */
 
100
/* <string1> <string2> copy <substring2> */
 
101
static int
 
102
zcopy_interval(i_ctx_t *i_ctx_p)
 
103
{
 
104
    os_ptr op = osp;
 
105
    os_ptr op1 = op - 1;
 
106
    int code = copy_interval(i_ctx_p, op, 0, op1, "copy");
 
107
 
 
108
    if (code < 0)
 
109
        return code;
 
110
    r_set_size(op, r_size(op1));
 
111
    *op1 = *op;
 
112
    pop(1);
 
113
    return 0;
 
114
}
 
115
 
 
116
/* <array|dict|name|packedarray|string> length <int> */
 
117
static int
 
118
zlength(i_ctx_t *i_ctx_p)
 
119
{
 
120
    os_ptr op = osp;
 
121
    switch (r_type(op)) {
 
122
        case t_array:
 
123
        case t_string:
 
124
        case t_mixedarray:
 
125
        case t_shortarray:
 
126
            check_read(*op);
 
127
            make_int(op, r_size(op));
 
128
            return 0;
 
129
        case t_dictionary:
 
130
            check_dict_read(*op);
 
131
            make_int(op, dict_length(op));
 
132
            return 0;
 
133
        case t_name: {
 
134
            ref str;
 
135
 
 
136
            name_string_ref(imemory, op, &str);
 
137
            make_int(op, r_size(&str));
 
138
            return 0;
 
139
        }
 
140
        case t_astruct:
 
141
            if (gs_object_type(imemory, op->value.pstruct) != &st_bytes)
 
142
                return_error(e_typecheck);
 
143
            check_read(*op);
 
144
            make_int(op, gs_object_size(imemory, op->value.pstruct));
 
145
            return 0;
 
146
        default:
 
147
            return_op_typecheck(op);
 
148
    }
 
149
}
 
150
 
 
151
/* <array|packedarray|string> <index> get <obj> */
 
152
/* <dict> <key> get <obj> */
 
153
static int
 
154
zget(i_ctx_t *i_ctx_p)
 
155
{
 
156
    int code;
 
157
    os_ptr op = osp;
 
158
    os_ptr op1 = op - 1;
 
159
    ref *pvalue;
 
160
 
 
161
    switch (r_type(op1)) {
 
162
        case t_dictionary:
 
163
            check_dict_read(*op1);
 
164
            if (dict_find(op1, op, &pvalue) <= 0)
 
165
                return_error(e_undefined);
 
166
            op[-1] = *pvalue;
 
167
            break;
 
168
        case t_string:
 
169
            check_read(*op1);
 
170
            check_int_ltu(*op, r_size(op1));
 
171
            make_int(op1, op1->value.bytes[(uint) op->value.intval]);
 
172
            break;
 
173
        case t_array:
 
174
        case t_mixedarray:
 
175
        case t_shortarray:
 
176
            check_type(*op, t_integer);
 
177
            check_read(*op1);
 
178
            code = array_get(imemory, op1, op->value.intval, op1);
 
179
            if (code < 0) 
 
180
                return code;
 
181
            break;
 
182
        case t__invalid:
 
183
            return_error(e_stackunderflow);
 
184
        default:
 
185
            return_error(e_typecheck); 
 
186
    }
 
187
    pop(1);
 
188
    return 0;
 
189
}
 
190
 
 
191
/* <array> <index> <obj> put - */
 
192
/* <dict> <key> <value> put - */
 
193
/* <string> <index> <int> put - */
 
194
static int
 
195
zput(i_ctx_t *i_ctx_p)
 
196
{
 
197
    os_ptr op = osp;
 
198
    os_ptr op1 = op - 1;
 
199
    os_ptr op2 = op1 - 1;
 
200
    byte *sdata;
 
201
    uint ssize;
 
202
 
 
203
    switch (r_type(op2)) {
 
204
        case t_dictionary:
 
205
            if (i_ctx_p->in_superexec == 0)
 
206
                check_dict_write(*op2);
 
207
            {
 
208
                int code = idict_put(op2, op1, op);
 
209
 
 
210
                if (code < 0)
 
211
                    return code;        /* error */
 
212
            }
 
213
            break;
 
214
        case t_array:
 
215
            check_write(*op2);
 
216
            check_int_ltu(*op1, r_size(op2));
 
217
            store_check_dest(op2, op);
 
218
            {
 
219
                ref *eltp = op2->value.refs + (uint) op1->value.intval;
 
220
 
 
221
                ref_assign_old(op2, eltp, op, "put");
 
222
            }
 
223
            break;
 
224
        case t_mixedarray:      /* packed arrays are read-only */
 
225
        case t_shortarray:
 
226
            return_error(e_invalidaccess);
 
227
        case t_string:
 
228
            sdata = op2->value.bytes;
 
229
            ssize = r_size(op2);
 
230
str:        check_write(*op2);
 
231
            check_int_ltu(*op1, ssize);
 
232
            check_int_leu(*op, 0xff);
 
233
            sdata[(uint)op1->value.intval] = (byte)op->value.intval;
 
234
            break;
 
235
        case t_astruct:
 
236
            if (gs_object_type(imemory, op2->value.pstruct) != &st_bytes)
 
237
                return_error(e_typecheck);
 
238
            sdata = r_ptr(op2, byte);
 
239
            ssize = gs_object_size(imemory, op2->value.pstruct);
 
240
            goto str;
 
241
        default:
 
242
            return_op_typecheck(op2);
 
243
    }
 
244
    pop(3);
 
245
    return 0;
 
246
}
 
247
 
 
248
/* <array> <index> <obj> .forceput - */
 
249
/* <dict> <key> <value> .forceput - */
 
250
/*
 
251
 * This forces a "put" even if the object is not writable, and (if the
 
252
 * object is systemdict or the save level is 0) even if the value is in
 
253
 * local VM.  It is meant to be used only for replacing the value of
 
254
 * FontDirectory in systemdict when switching between local and global VM,
 
255
 * and a few similar applications.  After initialization, this operator
 
256
 * should no longer be accessible by name.
 
257
 */
 
258
static int
 
259
zforceput(i_ctx_t *i_ctx_p)
 
260
{
 
261
    os_ptr op = osp;
 
262
    os_ptr op1 = op - 1;
 
263
    os_ptr op2 = op - 2;
 
264
    int code;
 
265
 
 
266
    switch (r_type(op2)) {
 
267
    case t_array:
 
268
        check_int_ltu(*op1, r_size(op2));
 
269
        if (r_space(op2) > r_space(op)) {
 
270
            if (imemory_save_level(iimemory))
 
271
                return_error(e_invalidaccess);
 
272
        }
 
273
        {
 
274
            ref *eltp = op2->value.refs + (uint) op1->value.intval;
 
275
 
 
276
            ref_assign_old(op2, eltp, op, "put");
 
277
        }
 
278
        break;
 
279
    case t_dictionary:
 
280
        if (op2->value.pdict == systemdict->value.pdict ||
 
281
            !imemory_save_level(iimemory)
 
282
            ) {
 
283
            uint space = r_space(op2);
 
284
 
 
285
            r_set_space(op2, avm_local);
 
286
            code = idict_put(op2, op1, op);
 
287
            r_set_space(op2, space);
 
288
        } else
 
289
            code = idict_put(op2, op1, op);
 
290
        if (code < 0)
 
291
            return code;
 
292
        break;
 
293
    default:
 
294
        return_error(e_typecheck);
 
295
    }
 
296
    pop(3);
 
297
    return 0;
 
298
}
 
299
 
 
300
/* <seq:array|packedarray|string> <index> <count> getinterval <subseq> */
 
301
static int
 
302
zgetinterval(i_ctx_t *i_ctx_p)
 
303
{
 
304
    os_ptr op = osp;
 
305
    os_ptr op1 = op - 1;
 
306
    os_ptr op2 = op1 - 1;
 
307
    uint index;
 
308
    uint count;
 
309
 
 
310
    switch (r_type(op2)) {
 
311
        default:
 
312
            return_op_typecheck(op2);
 
313
        case t_array:
 
314
        case t_string:
 
315
        case t_mixedarray:
 
316
        case t_shortarray:;
 
317
    }
 
318
    check_read(*op2);
 
319
    check_int_leu(*op1, r_size(op2));
 
320
    index = op1->value.intval;
 
321
    check_int_leu(*op, r_size(op2) - index);
 
322
    count = op->value.intval;
 
323
    switch (r_type(op2)) {
 
324
        case t_array:
 
325
            op2->value.refs += index;
 
326
            break;
 
327
        case t_string:
 
328
            op2->value.bytes += index;
 
329
            break;
 
330
        case t_mixedarray: {
 
331
            const ref_packed *packed = op2->value.packed;
 
332
 
 
333
            for (; index--;)
 
334
                packed = packed_next(packed);
 
335
            op2->value.packed = packed;
 
336
            break;
 
337
        }
 
338
        case t_shortarray:
 
339
            op2->value.packed += index;
 
340
            break;
 
341
    }
 
342
    r_set_size(op2, count);
 
343
    pop(2);
 
344
    return 0;
 
345
}
 
346
 
 
347
/* <array1> <index> <array2|packedarray2> putinterval - */
 
348
/* <string1> <index> <string2> putinterval - */
 
349
/* <bytestring1> <index> <string2> putinterval - */
 
350
static int
 
351
zputinterval(i_ctx_t *i_ctx_p)
 
352
{
 
353
    os_ptr op = osp;
 
354
    os_ptr opindex = op - 1;
 
355
    os_ptr opto = opindex - 1;
 
356
    int code;
 
357
 
 
358
    switch (r_type(opto)) {
 
359
        default:
 
360
            return_error(e_typecheck);
 
361
        case t__invalid:
 
362
            if (r_type(op) != t_array && r_type(op) != t_string && r_type(op) != t__invalid)
 
363
                return_error(e_typecheck); /* to match Distiller */
 
364
            else
 
365
                return_error(e_stackunderflow);
 
366
        case t_mixedarray:
 
367
        case t_shortarray:
 
368
            return_error(e_invalidaccess);
 
369
        case t_array:
 
370
        case t_string:
 
371
            check_write(*opto);
 
372
            check_int_leu(*opindex, r_size(opto));
 
373
            code = copy_interval(i_ctx_p, opto, (uint)(opindex->value.intval),
 
374
                                 op, "putinterval");
 
375
            break;
 
376
        case t_astruct: {
 
377
            uint dsize, ssize, index;
 
378
 
 
379
            check_write(*opto);
 
380
            if (gs_object_type(imemory, opto->value.pstruct) != &st_bytes)
 
381
                return_error(e_typecheck);
 
382
            dsize = gs_object_size(imemory, opto->value.pstruct);
 
383
            check_int_leu(*opindex, dsize);
 
384
            index = (uint)opindex->value.intval;
 
385
            check_read_type(*op, t_string);
 
386
            ssize = r_size(op);
 
387
            if (ssize > dsize - index)
 
388
                return_error(e_rangecheck);
 
389
            memcpy(r_ptr(opto, byte) + index, op->value.const_bytes, ssize);
 
390
            code = 0;
 
391
            break;
 
392
        }
 
393
    }
 
394
    if (code >= 0)
 
395
        pop(3);
 
396
    return code;
 
397
}
 
398
 
 
399
/* <array|packedarray|string> <<element> proc> forall - */
 
400
/* <dict> <<key> <value> proc> forall - */
 
401
static int
 
402
    array_continue(i_ctx_t *),
 
403
    dict_continue(i_ctx_t *),
 
404
    string_continue(i_ctx_t *),
 
405
    packedarray_continue(i_ctx_t *);
 
406
static int forall_cleanup(i_ctx_t *);
 
407
static int
 
408
zforall(i_ctx_t *i_ctx_p)
 
409
{
 
410
    os_ptr op = osp;
 
411
    os_ptr obj = op - 1;
 
412
    es_ptr ep = esp;
 
413
    es_ptr cproc = ep + 4;
 
414
 
 
415
    check_estack(6);
 
416
    check_proc(*op);
 
417
    switch (r_type(obj)) {
 
418
        default:
 
419
            return_op_typecheck(obj);
 
420
        case t_array:
 
421
            check_read(*obj);
 
422
            make_op_estack(cproc, array_continue);
 
423
            break;
 
424
        case t_dictionary:
 
425
            check_dict_read(*obj);
 
426
            make_int(cproc, dict_first(obj));
 
427
            ++cproc;
 
428
            make_op_estack(cproc, dict_continue);
 
429
            break;
 
430
        case t_string:
 
431
            check_read(*obj);
 
432
            make_op_estack(cproc, string_continue);
 
433
            break;
 
434
        case t_mixedarray:
 
435
        case t_shortarray:
 
436
            check_read(*obj);
 
437
            make_op_estack(cproc, packedarray_continue);
 
438
            break;
 
439
    }
 
440
    /*
 
441
     * Push:
 
442
     *   - a mark;
 
443
     *   - the composite object;
 
444
     *   - the procedure;
 
445
     *   - the iteration index (only for dictionaries, done above);
 
446
     * and invoke the continuation operator.
 
447
     */
 
448
    make_mark_estack(ep + 1, es_for, forall_cleanup);
 
449
    ep[2] = *obj;
 
450
    ep[3] = *op;
 
451
    esp = cproc - 1;
 
452
    pop(2);
 
453
    return (*real_opproc(cproc))(i_ctx_p);
 
454
}
 
455
/* Continuation operator for arrays */
 
456
static int
 
457
array_continue(i_ctx_t *i_ctx_p)
 
458
{
 
459
    os_ptr op = osp;
 
460
    es_ptr obj = esp - 1;
 
461
 
 
462
    if (r_size(obj)) {          /* continue */
 
463
        push(1);
 
464
        r_dec_size(obj, 1);
 
465
        *op = *obj->value.refs;
 
466
        obj->value.refs++;
 
467
        esp += 2;
 
468
        *esp = obj[1];
 
469
        return o_push_estack;
 
470
    } else {                    /* done */
 
471
        esp -= 3;               /* pop mark, object, proc */
 
472
        return o_pop_estack;
 
473
    }
 
474
}
 
475
/* Continuation operator for dictionaries */
 
476
static int
 
477
dict_continue(i_ctx_t *i_ctx_p)
 
478
{
 
479
    os_ptr op = osp;
 
480
    es_ptr obj = esp - 2;
 
481
    int index = (int)esp->value.intval;
 
482
 
 
483
    push(2);                    /* make room for key and value */
 
484
    if ((index = dict_next(obj, index, op - 1)) >= 0) { /* continue */
 
485
        esp->value.intval = index;
 
486
        esp += 2;
 
487
        *esp = obj[1];
 
488
        return o_push_estack;
 
489
    } else {                    /* done */
 
490
        pop(2);                 /* undo push */
 
491
        esp -= 4;               /* pop mark, object, proc, index */
 
492
        return o_pop_estack;
 
493
    }
 
494
}
 
495
/* Continuation operator for strings */
 
496
static int
 
497
string_continue(i_ctx_t *i_ctx_p)
 
498
{
 
499
    os_ptr op = osp;
 
500
    es_ptr obj = esp - 1;
 
501
 
 
502
    if (r_size(obj)) {          /* continue */
 
503
        r_dec_size(obj, 1);
 
504
        push(1);
 
505
        make_int(op, *obj->value.bytes);
 
506
        obj->value.bytes++;
 
507
        esp += 2;
 
508
        *esp = obj[1];
 
509
        return o_push_estack;
 
510
    } else {                    /* done */
 
511
        esp -= 3;               /* pop mark, object, proc */
 
512
        return o_pop_estack;
 
513
    }
 
514
}
 
515
/* Continuation operator for packed arrays */
 
516
static int
 
517
packedarray_continue(i_ctx_t *i_ctx_p)
 
518
{
 
519
    os_ptr op = osp;
 
520
    es_ptr obj = esp - 1;
 
521
 
 
522
    if (r_size(obj)) {          /* continue */
 
523
        const ref_packed *packed = obj->value.packed;
 
524
 
 
525
        r_dec_size(obj, 1);
 
526
        push(1);
 
527
        packed_get(imemory, packed, op);
 
528
        obj->value.packed = packed_next(packed);
 
529
        esp += 2;
 
530
        *esp = obj[1];
 
531
        return o_push_estack;
 
532
    } else {                    /* done */
 
533
        esp -= 3;               /* pop mark, object, proc */
 
534
        return o_pop_estack;
 
535
    }
 
536
}
 
537
/* Vacuous cleanup procedure */
 
538
static int
 
539
forall_cleanup(i_ctx_t *i_ctx_p)
 
540
{
 
541
    return 0;
 
542
}
 
543
 
 
544
/* ------ Initialization procedure ------ */
 
545
 
 
546
const op_def zgeneric_op_defs[] =
 
547
{
 
548
    {"1copy", zcopy},
 
549
    {"2forall", zforall},
 
550
    {"3.forceput", zforceput},
 
551
    {"2get", zget},
 
552
    {"3getinterval", zgetinterval},
 
553
    {"1length", zlength},
 
554
    {"3put", zput},
 
555
    {"3putinterval", zputinterval},
 
556
                /* Internal operators */
 
557
    {"0%array_continue", array_continue},
 
558
    {"0%dict_continue", dict_continue},
 
559
    {"0%packedarray_continue", packedarray_continue},
 
560
    {"0%string_continue", string_continue},
 
561
    op_def_end(0)
 
562
};
 
563
 
 
564
/* ------ Shared routines ------ */
 
565
 
 
566
/* Copy an interval from one operand to another. */
 
567
/* This is used by both putinterval and string/array copy. */
 
568
/* The destination is known to be an array or string, */
 
569
/* and the starting index is known to be less than or equal to */
 
570
/* its length; nothing else has been checked. */
 
571
static int
 
572
copy_interval(i_ctx_t *i_ctx_p /* for ref_assign_old */, os_ptr prto,
 
573
              uint index, os_ptr prfrom, client_name_t cname)
 
574
{
 
575
    int fromtype = r_type(prfrom);
 
576
    uint fromsize = r_size(prfrom);
 
577
 
 
578
    if (!(fromtype == r_type(prto) ||
 
579
          ((fromtype == t_shortarray || fromtype == t_mixedarray) &&
 
580
           r_type(prto) == t_array))
 
581
        )
 
582
        return_op_typecheck(prfrom);
 
583
    check_read(*prfrom);
 
584
    check_write(*prto);
 
585
    if (fromsize > r_size(prto) - index)
 
586
        return_error(e_rangecheck);
 
587
    switch (fromtype) {
 
588
        case t_array:
 
589
            {                   /* We have to worry about aliasing, */
 
590
                /* but refcpy_to_old takes care of it for us. */
 
591
                return refcpy_to_old(prto, index, prfrom->value.refs,
 
592
                                     fromsize, idmemory, cname);
 
593
            }
 
594
        case t_string:
 
595
            {   /* memmove takes care of aliasing. */
 
596
                memmove(prto->value.bytes + index, prfrom->value.bytes,
 
597
                        fromsize);
 
598
            }
 
599
            break;
 
600
        case t_mixedarray:
 
601
        case t_shortarray:
 
602
            {   /* We don't have to worry about aliasing, because */
 
603
                /* packed arrays are read-only and hence the destination */
 
604
                /* can't be a packed array. */
 
605
                uint i;
 
606
                const ref_packed *packed = prfrom->value.packed;
 
607
                ref *pdest = prto->value.refs + index;
 
608
                ref elt;
 
609
 
 
610
                for (i = 0; i < fromsize; i++, pdest++) {
 
611
                    packed_get(imemory, packed, &elt);
 
612
                    ref_assign_old(prto, pdest, &elt, cname);
 
613
                    packed = packed_next(packed);
 
614
                }
 
615
            }
 
616
            break;
 
617
    }
 
618
    return 0;
 
619
}