Coverage Report

Created: 2025-12-31 07:31

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/psi/zarith.c
Line
Count
Source
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
/* Arithmetic operators */
18
#include "math_.h"
19
#include "ghost.h"
20
#include "oper.h"
21
#include "store.h"
22
#include "gsstate.h"
23
24
/*
25
 * Many of the procedures in this file are public only so they can be
26
 * called from the FunctionType 4 interpreter (zfunc4.c).
27
 */
28
29
static int mul_64_64_overflowcheck(int64_t abc, int64_t def, int64_t *res);
30
31
/* <num1> <num2> add <sum> */
32
/* We make this into a separate procedure because */
33
/* the interpreter will almost always call it directly. */
34
int
35
zop_add(i_ctx_t *i_ctx_p)
36
651M
{
37
651M
    register os_ptr op = osp;
38
651M
    float result;
39
40
651M
    check_op(2);
41
651M
    switch (r_type(op)) {
42
15
    default:
43
15
        return_op_typecheck(op);
44
65.4M
    case t_real:
45
65.4M
        switch (r_type(op - 1)) {
46
17
        default:
47
17
            return_op_typecheck(op - 1);
48
36.2M
        case t_real:
49
36.2M
            result = op[-1].value.realval + op->value.realval;
50
36.2M
#ifdef HAVE_ISINF
51
36.2M
            if (isinf(result))
52
5
                return_error(gs_error_undefinedresult);
53
36.2M
#endif
54
36.2M
#ifdef HAVE_ISNAN
55
36.2M
            if (isnan(result))
56
0
                return_error(gs_error_undefinedresult);
57
36.2M
#endif
58
36.2M
            op[-1].value.realval = result;
59
36.2M
            break;
60
29.2M
        case t_integer:
61
29.2M
            make_real(op - 1, (double)op[-1].value.intval + op->value.realval);
62
65.4M
        }
63
65.4M
        break;
64
585M
    case t_integer:
65
585M
        switch (r_type(op - 1)) {
66
14
        default:
67
14
            return_op_typecheck(op - 1);
68
60.0M
        case t_real:
69
60.0M
            result = op[-1].value.realval + (double)op->value.intval;
70
60.0M
#ifdef HAVE_ISINF
71
60.0M
            if (isinf(result))
72
3
                return_error(gs_error_undefinedresult);
73
60.0M
#endif
74
60.0M
#ifdef HAVE_ISNAN
75
60.0M
            if (isnan(result))
76
0
                return_error(gs_error_undefinedresult);
77
60.0M
#endif
78
60.0M
            op[-1].value.realval = result;
79
60.0M
            break;
80
525M
        case t_integer: {
81
525M
            if (sizeof(ps_int) != 4 && gs_currentcpsimode(imemory)) {
82
0
                ps_int32 int1 = (ps_int32)op[-1].value.intval;
83
0
                ps_int32 int2 = (ps_int32)op->value.intval;
84
85
0
                if (((int1 += int2) ^ int2) < 0 &&
86
0
                    ((int1 - int2) ^ int2) >= 0
87
0
                    ) {                     /* Overflow, convert to real */
88
0
                    make_real(op - 1, (double)(int1 - int2) + int2);
89
0
                }
90
0
                else {
91
0
                    op[-1].value.intval = (ps_int)int1;
92
0
                }
93
0
            }
94
525M
            else {
95
525M
                ps_int int2 = op->value.intval;
96
97
525M
                if (((op[-1].value.intval += int2) ^ int2) < 0 &&
98
2.85k
                    ((op[-1].value.intval - int2) ^ int2) >= 0
99
525M
                    ) {                     /* Overflow, convert to real */
100
319
                    make_real(op - 1, (double)(op[-1].value.intval - int2) + int2);
101
319
                }
102
525M
            }
103
525M
        }
104
585M
        }
105
651M
    }
106
651M
    return 0;
107
651M
}
108
int
109
zadd(i_ctx_t *i_ctx_p)
110
0
{
111
0
    int code = zop_add(i_ctx_p);
112
113
0
    if (code == 0) {
114
0
        pop(1);
115
0
    }
116
0
    return code;
117
0
}
118
119
/* <num1> <num2> div <real_quotient> */
120
int
121
zdiv(i_ctx_t *i_ctx_p)
122
41.8M
{
123
41.8M
    os_ptr op = osp;
124
41.8M
    os_ptr op1 = op - 1;
125
41.8M
    float result;
126
127
41.8M
    check_op(2);
128
    /* We can't use the non_int_cases macro, */
129
    /* because we have to check explicitly for op == 0. */
130
41.8M
    switch (r_type(op)) {
131
16
        default:
132
16
            return_op_typecheck(op);
133
2.28M
        case t_real:
134
2.28M
            if (op->value.realval == 0)
135
9
                return_error(gs_error_undefinedresult);
136
2.28M
            switch (r_type(op1)) {
137
10
                default:
138
10
                    return_op_typecheck(op1);
139
2.16M
                case t_real:
140
2.16M
                    result = op1->value.realval / op->value.realval;
141
2.16M
#ifdef HAVE_ISINF
142
2.16M
                    if (isinf(result))
143
7
                        return_error(gs_error_undefinedresult);
144
2.16M
#endif
145
2.16M
#ifdef HAVE_ISNAN
146
2.16M
                    if (isnan(result))
147
1
                        return_error(gs_error_undefinedresult);
148
2.16M
#endif
149
2.16M
                    op1->value.realval = result;
150
2.16M
                    break;
151
118k
                case t_integer:
152
118k
                    result = (double)op1->value.intval / op->value.realval;
153
118k
#ifdef HAVE_ISINF
154
118k
                    if (isinf(result))
155
0
                        return_error(gs_error_undefinedresult);
156
118k
#endif
157
118k
#ifdef HAVE_ISNAN
158
118k
                    if (isnan(result))
159
0
                        return_error(gs_error_undefinedresult);
160
118k
#endif
161
2.28M
                    make_real(op1, result);
162
2.28M
            }
163
2.28M
            break;
164
39.5M
        case t_integer:
165
39.5M
            if (op->value.intval == 0)
166
15
                return_error(gs_error_undefinedresult);
167
39.5M
            switch (r_type(op1)) {
168
14
                default:
169
14
                    return_op_typecheck(op1);
170
34.9M
                case t_real:
171
34.9M
                    result = op1->value.realval / (double)op->value.intval;
172
34.9M
#ifdef HAVE_ISINF
173
34.9M
                    if (isinf(result))
174
0
                        return_error(gs_error_undefinedresult);
175
34.9M
#endif
176
34.9M
#ifdef HAVE_ISNAN
177
34.9M
                    if (isnan(result))
178
0
                        return_error(gs_error_undefinedresult);
179
34.9M
#endif
180
34.9M
                    op1->value.realval = result;
181
34.9M
                    break;
182
4.61M
                case t_integer:
183
4.61M
                    result = (double)op1->value.intval / (double)op->value.intval;
184
4.61M
#ifdef HAVE_ISINF
185
4.61M
                    if (isinf(result))
186
0
                        return_error(gs_error_undefinedresult);
187
4.61M
#endif
188
4.61M
#ifdef HAVE_ISNAN
189
4.61M
                    if (isnan(result))
190
0
                        return_error(gs_error_undefinedresult);
191
4.61M
#endif
192
39.5M
                    make_real(op1, result);
193
39.5M
            }
194
41.8M
    }
195
41.8M
    pop(1);
196
41.8M
    return 0;
197
41.8M
}
198
199
/*
200
To detect 64bit x 64bit multiplication overflowing, consider
201
breaking the numbers down into 32bit chunks.
202
203
  abc = (a<<64) + (b<<32) + c
204
      (where a is 0 or -1, and b and c are 32bit unsigned.
205
206
Similarly:
207
208
  def = (d<<64) + (b<<32) + f
209
210
Then:
211
212
  abc.def = ((a<<64) + (b<<32) + c) * ((d<<64) + (e<<32) + f)
213
          = (a<<64).def + (d<<64).abc + (b<<32).(e<<32) +
214
            (b<<32).f + (e<<32).c + cf
215
          = (a.def + d.abc + b.e)<<64 + (b.f + e.c)<<32 + cf
216
217
*/
218
219
static int mul_64_64_overflowcheck(int64_t abc, int64_t def, int64_t *res)
220
13.1M
{
221
13.1M
  uint32_t b = (abc>>32);
222
13.1M
  uint32_t c = (uint32_t)abc;
223
13.1M
  uint32_t e = (def>>32);
224
13.1M
  uint32_t f = (uint32_t)def;
225
13.1M
  uint64_t low, mid, high, bf, ec;
226
227
  /* Low contribution */
228
13.1M
  low = (uint64_t)c * (uint64_t)f;
229
  /* Mid contributions */
230
13.1M
  bf = (uint64_t)b * (uint64_t)f;
231
13.1M
  ec = (uint64_t)e * (uint64_t)c;
232
  /* Top contribution */
233
13.1M
  high = (uint64_t)b * (uint64_t)e;
234
13.1M
  if (abc < 0)
235
382
      high -= def;
236
13.1M
  if (def < 0)
237
54
      high -= abc;
238
  /* How do we check for carries from 64bit unsigned adds?
239
   *  x + y >= (1<<64) == x >= (1<<64) - y
240
   *                   == x >  (1<<64) - y - 1
241
   * if we consider just 64bits, this is:
242
   * x > NOT y
243
   */
244
13.1M
  if (bf > ~ec)
245
18
      high += ((uint64_t)1)<<32;
246
13.1M
  mid = bf + ec;
247
13.1M
  if (low > ~(mid<<32))
248
18
      high += 1;
249
13.1M
  high += (mid>>32);
250
13.1M
  low += (mid<<32);
251
252
13.1M
  *res = low;
253
254
13.1M
  return (int64_t)low < 0 ? high != -1 : high != 0;
255
13.1M
}
256
257
/* <num1> <num2> mul <product> */
258
int
259
zmul(i_ctx_t *i_ctx_p)
260
140M
{
261
140M
    os_ptr op = osp;
262
140M
    float result;
263
264
140M
    switch (r_type(op)) {
265
23
    default:
266
23
        return_op_typecheck(op);
267
67.0M
    case t_real:
268
67.0M
        switch (r_type(op - 1)) {
269
11
        default:
270
11
            return_op_typecheck(op - 1);
271
65.9M
        case t_real:
272
65.9M
            result = op[-1].value.realval * op->value.realval;
273
65.9M
#ifdef HAVE_ISINF
274
65.9M
            if (isinf(result))
275
32
                return_error(gs_error_undefinedresult);
276
65.9M
#endif
277
65.9M
#ifdef HAVE_ISNAN
278
65.9M
            if (isnan(result))
279
0
                return_error(gs_error_undefinedresult);
280
65.9M
#endif
281
65.9M
            op[-1].value.realval = result;
282
65.9M
            break;
283
1.12M
        case t_integer:
284
1.12M
            result = (double)op[-1].value.intval * op->value.realval;
285
1.12M
            make_real(op - 1, result);
286
67.0M
        }
287
67.0M
        break;
288
73.6M
    case t_integer:
289
73.6M
        switch (r_type(op - 1)) {
290
14
        default:
291
14
            return_op_typecheck(op - 1);
292
60.4M
        case t_real:
293
60.4M
            result = op[-1].value.realval * (double)op->value.intval;
294
60.4M
#ifdef HAVE_ISINF
295
60.4M
            if (isinf(result))
296
5
                return_error(gs_error_undefinedresult);
297
60.4M
#endif
298
60.4M
#ifdef HAVE_ISNAN
299
60.4M
            if (isnan(result))
300
0
                return_error(gs_error_undefinedresult);
301
60.4M
#endif
302
60.4M
            op[-1].value.realval = result;
303
60.4M
            break;
304
13.1M
        case t_integer: {
305
13.1M
            if (sizeof(ps_int) != 4 && gs_currentcpsimode(imemory)) {
306
0
                double ab = (double)op[-1].value.intval * op->value.intval;
307
0
                if (ab > (double)MAX_PS_INT32)       /* (double)0x7fffffff */
308
0
                    make_real(op - 1, ab);
309
0
                else if (ab < (double)MIN_PS_INT32) /* (double)(int)0x80000000 */
310
0
                    make_real(op - 1, ab);
311
0
                else
312
0
                    op[-1].value.intval = (ps_int)ab;
313
0
            }
314
13.1M
            else {
315
13.1M
                int64_t result;
316
13.1M
                if (mul_64_64_overflowcheck(op[-1].value.intval, op->value.intval, &result)) {
317
1
                    double ab = (double)op[-1].value.intval * op->value.intval;
318
1
                    make_real(op - 1, ab);
319
13.1M
                } else {
320
13.1M
                    op[-1].value.intval = result;
321
13.1M
                }
322
13.1M
            }
323
13.1M
        }
324
73.6M
        }
325
140M
    }
326
140M
    pop(1);
327
140M
    return 0;
328
140M
}
329
330
/* <num1> <num2> sub <difference> */
331
/* We make this into a separate procedure because */
332
/* the interpreter will almost always call it directly. */
333
int
334
zop_sub(i_ctx_t *i_ctx_p)
335
117M
{
336
117M
    register os_ptr op = osp;
337
338
117M
    check_op(2);
339
117M
    switch (r_type(op)) {
340
100
    default:
341
100
        return_op_typecheck(op);
342
32.2M
    case t_real:
343
32.2M
        switch (r_type(op - 1)) {
344
12
        default:
345
12
            return_op_typecheck(op - 1);
346
374k
        case t_real:
347
374k
            op[-1].value.realval -= op->value.realval;
348
374k
            break;
349
31.9M
        case t_integer:
350
31.9M
            make_real(op - 1, (double)op[-1].value.intval - op->value.realval);
351
32.2M
        }
352
32.2M
        break;
353
85.6M
    case t_integer:
354
85.6M
        switch (r_type(op - 1)) {
355
10
        default:
356
10
            return_op_typecheck(op - 1);
357
89.5k
        case t_real:
358
89.5k
            op[-1].value.realval -= (double)op->value.intval;
359
89.5k
            break;
360
85.5M
        case t_integer: {
361
85.5M
            if (sizeof(ps_int) != 4 && gs_currentcpsimode(imemory)) {
362
0
                ps_int32 int1 = (ps_int)op[-1].value.intval;
363
0
                ps_int32 int2 = (ps_int)op->value.intval;
364
0
                ps_int32 int3;
365
366
0
                if ((int1 ^ (int3 = int1 - int2)) < 0 &&
367
0
                    (int1 ^ int2) < 0
368
0
                    ) {                     /* Overflow, convert to real */
369
0
                    make_real(op - 1, (float)int1 - op->value.intval);
370
0
                }
371
0
                else {
372
0
                    op[-1].value.intval = (ps_int)int3;
373
0
                }
374
0
            }
375
85.5M
            else {
376
85.5M
                ps_int int1 = op[-1].value.intval;
377
378
85.5M
                if ((int1 ^ (op[-1].value.intval = int1 - op->value.intval)) < 0 &&
379
163k
                    (int1 ^ op->value.intval) < 0
380
85.5M
                    ) {                     /* Overflow, convert to real */
381
0
                    make_real(op - 1, (float)int1 - op->value.intval);
382
0
                }
383
85.5M
            }
384
85.5M
        }
385
85.6M
        }
386
117M
    }
387
117M
    return 0;
388
117M
}
389
int
390
zsub(i_ctx_t *i_ctx_p)
391
0
{
392
0
    int code = zop_sub(i_ctx_p);
393
394
0
    if (code == 0) {
395
0
        pop(1);
396
0
    }
397
0
    return code;
398
0
}
399
400
/* <num1> <num2> idiv <int_quotient> */
401
int
402
zidiv(i_ctx_t *i_ctx_p)
403
48.3M
{
404
48.3M
    os_ptr op = osp;
405
406
48.3M
    check_op(2);
407
48.3M
    check_type(*op, t_integer);
408
48.3M
    check_type(op[-1], t_integer);
409
48.3M
    if (sizeof(ps_int) && gs_currentcpsimode(imemory)) {
410
0
        int tmpval;
411
0
        if ((op->value.intval == 0) || (op[-1].value.intval == (ps_int)MIN_PS_INT32 && op->value.intval == -1)) {
412
            /* Anomalous boundary case: -MININT / -1, fail. */
413
0
            return_error(gs_error_undefinedresult);
414
0
        }
415
0
        tmpval = (int)op[-1].value.intval / op->value.intval;
416
0
        op[-1].value.intval = (int64_t)tmpval;
417
0
    }
418
48.3M
    else {
419
48.3M
        if ((op->value.intval == 0) || (op[-1].value.intval == MIN_PS_INT && op->value.intval == -1)) {
420
            /* Anomalous boundary case: -MININT / -1, fail. */
421
7
            return_error(gs_error_undefinedresult);
422
7
        }
423
48.3M
        op[-1].value.intval /= op->value.intval;
424
48.3M
    }
425
48.3M
    pop(1);
426
48.3M
    return 0;
427
48.3M
}
428
429
/* <int1> <int2> mod <remainder> */
430
int
431
zmod(i_ctx_t *i_ctx_p)
432
21.8M
{
433
21.8M
    os_ptr op = osp;
434
435
21.8M
    check_op(2);
436
21.8M
    check_type(*op, t_integer);
437
21.8M
    check_type(op[-1], t_integer);
438
21.8M
    if (op->value.intval == 0 || op[-1].value.intval == MIN_PS_INT)
439
11
        return_error(gs_error_undefinedresult);
440
21.8M
    op[-1].value.intval %= op->value.intval;
441
21.8M
    pop(1);
442
21.8M
    return 0;
443
21.8M
}
444
445
/* <num1> neg <num2> */
446
int
447
zneg(i_ctx_t *i_ctx_p)
448
5.63M
{
449
5.63M
    os_ptr op = osp;
450
451
5.63M
    check_op(1);
452
5.63M
    switch (r_type(op)) {
453
14
        default:
454
14
            return_op_typecheck(op);
455
2.34M
        case t_real:
456
2.34M
            op->value.realval = -op->value.realval;
457
2.34M
            break;
458
3.29M
        case t_integer:
459
3.29M
            if (sizeof(ps_int) != 32 && gs_currentcpsimode(imemory)) {
460
0
                if (((unsigned int)op->value.intval) == MIN_PS_INT32)
461
0
                    make_real(op, -(float)(ps_uint32)MIN_PS_INT32);
462
0
                else
463
0
                    op->value.intval = -op->value.intval;
464
0
            }
465
3.29M
            else {
466
3.29M
                if (op->value.intval == MIN_PS_INT)
467
3.29M
                    make_real(op, -(float)MIN_PS_INT);
468
3.29M
                else
469
3.29M
                    op->value.intval = -op->value.intval;
470
3.29M
            }
471
5.63M
    }
472
5.63M
    return 0;
473
5.63M
}
474
475
/* <num1> abs <num2> */
476
int
477
zabs(i_ctx_t *i_ctx_p)
478
24.7M
{
479
24.7M
    os_ptr op = osp;
480
481
24.7M
    check_op(1);
482
24.7M
    switch (r_type(op)) {
483
107
        default:
484
107
            return_op_typecheck(op);
485
3.42M
        case t_real:
486
3.42M
            if (op->value.realval >= 0)
487
1.82M
                return 0;
488
1.60M
            break;
489
21.3M
        case t_integer:
490
21.3M
            if (op->value.intval >= 0)
491
21.3M
                return 0;
492
163
            break;
493
24.7M
    }
494
1.60M
    return zneg(i_ctx_p);
495
24.7M
}
496
497
/* <num1> ceiling <num2> */
498
int
499
zceiling(i_ctx_t *i_ctx_p)
500
260
{
501
260
    os_ptr op = osp;
502
503
260
    check_op(1);
504
246
    switch (r_type(op)) {
505
12
        default:
506
12
            return_op_typecheck(op);
507
147
        case t_real:
508
147
            op->value.realval = ceil(op->value.realval);
509
234
        case t_integer:;
510
246
    }
511
234
    return 0;
512
246
}
513
514
/* <num1> floor <num2> */
515
int
516
zfloor(i_ctx_t *i_ctx_p)
517
294
{
518
294
    os_ptr op = osp;
519
520
294
    check_op(1);
521
280
    switch (r_type(op)) {
522
10
        default:
523
10
            return_op_typecheck(op);
524
130
        case t_real:
525
130
            op->value.realval = floor(op->value.realval);
526
270
        case t_integer:;
527
280
    }
528
270
    return 0;
529
280
}
530
531
/* <num1> round <num2> */
532
int
533
zround(i_ctx_t *i_ctx_p)
534
6.15k
{
535
6.15k
    os_ptr op = osp;
536
537
6.15k
    check_op(1);
538
6.14k
    switch (r_type(op)) {
539
18
        default:
540
18
            return_op_typecheck(op);
541
1.12k
        case t_real:
542
1.12k
            op->value.realval = floor(op->value.realval + 0.5);
543
6.12k
        case t_integer:;
544
6.14k
    }
545
6.12k
    return 0;
546
6.14k
}
547
548
/* <num1> truncate <num2> */
549
int
550
ztruncate(i_ctx_t *i_ctx_p)
551
2.44M
{
552
2.44M
    os_ptr op = osp;
553
554
2.44M
    check_op(1);
555
2.44M
    switch (r_type(op)) {
556
12
        default:
557
12
            return_op_typecheck(op);
558
1.60k
        case t_real:
559
1.60k
            op->value.realval =
560
1.60k
                (op->value.realval < 0.0 ?
561
19
                 ceil(op->value.realval) :
562
1.60k
                 floor(op->value.realval));
563
2.44M
        case t_integer:;
564
2.44M
    }
565
2.44M
    return 0;
566
2.44M
}
567
568
/* Non-standard operators */
569
570
/* <int1> <int2> .bitadd <sum> */
571
static int
572
zbitadd(i_ctx_t *i_ctx_p)
573
0
{
574
0
    os_ptr op = osp;
575
576
0
    check_op(2);
577
0
    check_type(*op, t_integer);
578
0
    check_type(op[-1], t_integer);
579
0
    op[-1].value.intval += op->value.intval;
580
0
    pop(1);
581
0
    return 0;
582
0
}
583
584
/* ------ Initialization table ------ */
585
586
const op_def zarith_op_defs[] =
587
{
588
    {"1abs", zabs},
589
    {"2add", zadd},
590
    {"2.bitadd", zbitadd},
591
    {"1ceiling", zceiling},
592
    {"2div", zdiv},
593
    {"2idiv", zidiv},
594
    {"1floor", zfloor},
595
    {"2mod", zmod},
596
    {"2mul", zmul},
597
    {"1neg", zneg},
598
    {"1round", zround},
599
    {"2sub", zsub},
600
    {"1truncate", ztruncate},
601
    op_def_end(0)
602
};