Coverage Report

Created: 2026-02-14 07:09

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/base/gsmatrix.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
/* Matrix operators for Ghostscript library */
18
#include "math_.h"
19
#include "memory_.h"
20
#include "gx.h"
21
#include "gserrors.h"
22
#include "gxfarith.h"
23
#include "gxfixed.h"
24
#include "gxmatrix.h"
25
#include "stream.h"
26
27
/* The identity matrix */
28
static const gs_matrix gs_identity_matrix =
29
{identity_matrix_body};
30
31
/* ------ Matrix creation ------ */
32
33
/* Create an identity matrix */
34
void
35
gs_make_identity(gs_matrix * pmat)
36
90.1M
{
37
90.1M
    *pmat = gs_identity_matrix;
38
90.1M
}
39
40
/* Create a translation matrix */
41
int
42
gs_make_translation(double dx, double dy, gs_matrix * pmat)
43
136k
{
44
136k
    *pmat = gs_identity_matrix;
45
136k
    pmat->tx = dx;
46
136k
    pmat->ty = dy;
47
136k
    return 0;
48
136k
}
49
50
/* Create a scaling matrix */
51
int
52
gs_make_scaling(double sx, double sy, gs_matrix * pmat)
53
8.18M
{
54
8.18M
    *pmat = gs_identity_matrix;
55
8.18M
    pmat->xx = sx;
56
8.18M
    pmat->yy = sy;
57
8.18M
    return 0;
58
8.18M
}
59
60
/* Create a rotation matrix. */
61
/* The angle is in degrees. */
62
int
63
gs_make_rotation(double ang, gs_matrix * pmat)
64
1.58M
{
65
1.58M
    gs_sincos_t sincos;
66
67
1.58M
    gs_sincos_degrees(ang, &sincos);
68
1.58M
    pmat->yy = pmat->xx = sincos.cos;
69
1.58M
    pmat->xy = sincos.sin;
70
1.58M
    pmat->yx = -sincos.sin;
71
1.58M
    pmat->tx = pmat->ty = 0.0;
72
1.58M
    return 0;
73
1.58M
}
74
75
/* ------ Matrix arithmetic ------ */
76
77
/* Multiply two matrices.  We should check for floating exceptions, */
78
/* but for the moment it's just too awkward. */
79
/* Since this is used heavily, we check for shortcuts. */
80
int
81
gs_matrix_multiply(const gs_matrix * pm1, const gs_matrix * pm2, gs_matrix * pmr)
82
43.2M
{
83
43.2M
    double xx1 = pm1->xx, yy1 = pm1->yy;
84
43.2M
    double tx1 = pm1->tx, ty1 = pm1->ty;
85
43.2M
    double xx2 = pm2->xx, yy2 = pm2->yy;
86
43.2M
    double xy2 = pm2->xy, yx2 = pm2->yx;
87
88
43.2M
    if (is_xxyy(pm1)) {
89
41.1M
        pmr->tx = tx1 * xx2 + pm2->tx;
90
41.1M
        pmr->ty = ty1 * yy2 + pm2->ty;
91
41.1M
        if (is_fzero(xy2))
92
40.0M
            pmr->xy = 0;
93
1.11M
        else
94
1.11M
            pmr->xy = xx1 * xy2,
95
1.11M
                pmr->ty += tx1 * xy2;
96
41.1M
        pmr->xx = xx1 * xx2;
97
41.1M
        if (is_fzero(yx2))
98
39.9M
            pmr->yx = 0;
99
1.23M
        else
100
1.23M
            pmr->yx = yy1 * yx2,
101
1.23M
                pmr->tx += ty1 * yx2;
102
41.1M
        pmr->yy = yy1 * yy2;
103
41.1M
    } else {
104
2.09M
        double xy1 = pm1->xy, yx1 = pm1->yx;
105
106
2.09M
        pmr->xx = xx1 * xx2 + xy1 * yx2;
107
2.09M
        pmr->xy = xx1 * xy2 + xy1 * yy2;
108
2.09M
        pmr->yy = yx1 * xy2 + yy1 * yy2;
109
2.09M
        pmr->yx = yx1 * xx2 + yy1 * yx2;
110
2.09M
        pmr->tx = tx1 * xx2 + ty1 * yx2 + pm2->tx;
111
2.09M
        pmr->ty = tx1 * xy2 + ty1 * yy2 + pm2->ty;
112
2.09M
    }
113
43.2M
    return 0;
114
43.2M
}
115
int
116
gs_matrix_multiply_double(const gs_matrix_double * pm1, const gs_matrix * pm2, gs_matrix_double * pmr)
117
1.53M
{
118
1.53M
    double xx1 = pm1->xx, yy1 = pm1->yy;
119
1.53M
    double tx1 = pm1->tx, ty1 = pm1->ty;
120
1.53M
    double xx2 = pm2->xx, yy2 = pm2->yy;
121
1.53M
    double xy2 = pm2->xy, yx2 = pm2->yx;
122
123
1.53M
    if (is_xxyy(pm1)) {
124
1.36M
        pmr->tx = tx1 * xx2 + pm2->tx;
125
1.36M
        pmr->ty = ty1 * yy2 + pm2->ty;
126
1.36M
        if (is_fzero(xy2))
127
962k
            pmr->xy = 0;
128
407k
        else
129
407k
            pmr->xy = xx1 * xy2,
130
407k
                pmr->ty += tx1 * xy2;
131
1.36M
        pmr->xx = xx1 * xx2;
132
1.36M
        if (is_fzero(yx2))
133
938k
            pmr->yx = 0;
134
431k
        else
135
431k
            pmr->yx = yy1 * yx2,
136
431k
                pmr->tx += ty1 * yx2;
137
1.36M
        pmr->yy = yy1 * yy2;
138
1.36M
    } else {
139
169k
        double xy1 = pm1->xy, yx1 = pm1->yx;
140
141
169k
        pmr->xx = xx1 * xx2 + xy1 * yx2;
142
169k
        pmr->xy = xx1 * xy2 + xy1 * yy2;
143
169k
        pmr->yy = yx1 * xy2 + yy1 * yy2;
144
169k
        pmr->yx = yx1 * xx2 + yy1 * yx2;
145
169k
        pmr->tx = tx1 * xx2 + ty1 * yx2 + pm2->tx;
146
169k
        pmr->ty = tx1 * xy2 + ty1 * yy2 + pm2->ty;
147
169k
    }
148
1.53M
    return 0;
149
1.53M
}
150
151
/* Invert a matrix.  Return gs_error_undefinedresult if not invertible. */
152
int
153
gs_matrix_invert(const gs_matrix * pm, gs_matrix * pmr)
154
27.1M
{       /* We have to be careful about fetch/store order, */
155
    /* because pm might be the same as pmr. */
156
27.1M
    if (is_xxyy(pm)) {
157
17.0M
        if (is_fzero(pm->xx) || is_fzero(pm->yy))
158
27.3k
            return_error(gs_error_undefinedresult);
159
17.0M
        pmr->tx = -(pmr->xx = 1.0 / pm->xx) * pm->tx;
160
17.0M
        pmr->xy = 0.0;
161
17.0M
        pmr->yx = 0.0;
162
17.0M
        pmr->ty = -(pmr->yy = 1.0 / pm->yy) * pm->ty;
163
17.0M
    } else {
164
10.0M
        float mxx = pm->xx, myy = pm->yy, mxy = pm->xy, myx = pm->yx;
165
10.0M
        float mtx = pm->tx, mty = pm->ty;
166
        /* we declare det as double since on at least some computer (i.e. peeves)
167
           declaring it as a float results in different values for pmr depending
168
           on whether or not optimization is turned on.  I believe this is caused
169
           by the compiler keeping the det value in an internal register when
170
           optimization is enable.  As evidence of this if you add a debugging
171
           statement to print out det the optimized code acts the same as the
172
           unoptimized code.  declearing det as double does not change the CET 10-09.ps
173
           output. */
174
10.0M
        double det = (float)(mxx * myy) - (float)(mxy * myx);
175
176
        /*
177
         * We are doing the math as floats instead of doubles to reproduce
178
         * the results in page 1 of CET 10-09.ps
179
         */
180
10.0M
        if (det == 0)
181
9.84k
            return_error(gs_error_undefinedresult);
182
10.0M
        pmr->xx = myy / det;
183
10.0M
        pmr->xy = -mxy / det;
184
10.0M
        pmr->yx = -myx / det;
185
10.0M
        pmr->yy = mxx / det;
186
10.0M
        pmr->tx = (((float)(mty * myx) - (float)(mtx * myy))) / det;
187
10.0M
        pmr->ty = (((float)(mtx * mxy) - (float)(mty * mxx))) / det;
188
10.0M
    }
189
27.0M
    return 0;
190
27.1M
}
191
int
192
gs_matrix_invert_to_double(const gs_matrix * pm, gs_matrix_double * pmr)
193
1.54M
{       /* We have to be careful about fetch/store order, */
194
    /* because pm might be the same as pmr. */
195
1.54M
    if (is_xxyy(pm)) {
196
1.37M
        if (is_fzero(pm->xx) || is_fzero(pm->yy))
197
142
            return_error(gs_error_undefinedresult);
198
1.36M
        pmr->tx = -(pmr->xx = 1.0 / pm->xx) * pm->tx;
199
1.36M
        pmr->xy = 0.0;
200
1.36M
        pmr->yx = 0.0;
201
1.36M
        pmr->ty = -(pmr->yy = 1.0 / pm->yy) * pm->ty;
202
1.36M
    } else {
203
169k
        double mxx = pm->xx, myy = pm->yy, mxy = pm->xy, myx = pm->yx;
204
169k
        double mtx = pm->tx, mty = pm->ty;
205
169k
        double det = (mxx * myy) - (mxy * myx);
206
207
        /*
208
         * We are doing the math as floats instead of doubles to reproduce
209
         * the results in page 1 of CET 10-09.ps
210
         */
211
169k
        if (det == 0)
212
5
            return_error(gs_error_undefinedresult);
213
169k
        pmr->xx = myy / det;
214
169k
        pmr->xy = -mxy / det;
215
169k
        pmr->yx = -myx / det;
216
169k
        pmr->yy = mxx / det;
217
169k
        pmr->tx = (((mty * myx) - (mtx * myy))) / det;
218
169k
        pmr->ty = (((mtx * mxy) - (mty * mxx))) / det;
219
169k
    }
220
1.53M
    return 0;
221
1.54M
}
222
223
/* Translate a matrix, possibly in place. */
224
int
225
gs_matrix_translate(const gs_matrix * pm, double dx, double dy, gs_matrix * pmr)
226
85.4k
{
227
85.4k
    gs_point trans;
228
85.4k
    int code = gs_distance_transform(dx, dy, pm, &trans);
229
230
85.4k
    if (code < 0)
231
0
        return code;
232
85.4k
    if (pmr != pm)
233
85.4k
        *pmr = *pm;
234
85.4k
    pmr->tx += trans.x;
235
85.4k
    pmr->ty += trans.y;
236
85.4k
    return 0;
237
85.4k
}
238
239
/* Scale a matrix, possibly in place. */
240
int
241
gs_matrix_scale(const gs_matrix * pm, double sx, double sy, gs_matrix * pmr)
242
4.00M
{
243
4.00M
    pmr->xx = pm->xx * sx;
244
4.00M
    pmr->xy = pm->xy * sx;
245
4.00M
    pmr->yx = pm->yx * sy;
246
4.00M
    pmr->yy = pm->yy * sy;
247
4.00M
    if (pmr != pm) {
248
240k
        pmr->tx = pm->tx;
249
240k
        pmr->ty = pm->ty;
250
240k
    }
251
4.00M
    return 0;
252
4.00M
}
253
254
/* Rotate a matrix, possibly in place.  The angle is in degrees. */
255
int
256
gs_matrix_rotate(const gs_matrix * pm, double ang, gs_matrix * pmr)
257
1.47M
{
258
1.47M
    double mxx, mxy;
259
1.47M
    gs_sincos_t sincos;
260
261
1.47M
    gs_sincos_degrees(ang, &sincos);
262
1.47M
    mxx = pm->xx, mxy = pm->xy;
263
1.47M
    pmr->xx = sincos.cos * mxx + sincos.sin * pm->yx;
264
1.47M
    pmr->xy = sincos.cos * mxy + sincos.sin * pm->yy;
265
1.47M
    pmr->yx = sincos.cos * pm->yx - sincos.sin * mxx;
266
1.47M
    pmr->yy = sincos.cos * pm->yy - sincos.sin * mxy;
267
1.47M
    if (pmr != pm) {
268
0
        pmr->tx = pm->tx;
269
0
        pmr->ty = pm->ty;
270
0
    }
271
1.47M
    return 0;
272
1.47M
}
273
274
/* ------ Coordinate transformations (floating point) ------ */
275
276
/* Note that all the transformation routines take separate */
277
/* x and y arguments, but return their result in a point. */
278
279
/* Transform a point. */
280
int
281
gs_point_transform(double x, double y, const gs_matrix * pmat,
282
                   gs_point * ppt)
283
2.56G
{
284
    /*
285
     * The float casts are there to reproduce results in CET 10-01.ps
286
     * page 4.
287
     */
288
2.56G
    ppt->x = (float)(x * pmat->xx) + pmat->tx;
289
2.56G
    ppt->y = (float)(y * pmat->yy) + pmat->ty;
290
2.56G
    if (!is_fzero(pmat->yx))
291
132M
        ppt->x += (float)(y * pmat->yx);
292
2.56G
    if (!is_fzero(pmat->xy))
293
1.95G
        ppt->y += (float)(x * pmat->xy);
294
2.56G
    return 0;
295
2.56G
}
296
297
/* Inverse-transform a point. */
298
/* Return gs_error_undefinedresult if the matrix is not invertible. */
299
int
300
gs_point_transform_inverse(double x, double y, const gs_matrix * pmat,
301
                           gs_point * ppt)
302
176M
{
303
176M
    if (is_xxyy(pmat)) {
304
168M
        if (is_fzero(pmat->xx) || is_fzero(pmat->yy))
305
8.61k
            return_error(gs_error_undefinedresult);
306
168M
        ppt->x = (x - pmat->tx) / pmat->xx;
307
168M
        ppt->y = (y - pmat->ty) / pmat->yy;
308
168M
        return 0;
309
168M
    } else if (is_xyyx(pmat)) {
310
1.34M
        if (is_fzero(pmat->xy) || is_fzero(pmat->yx))
311
593
            return_error(gs_error_undefinedresult);
312
1.34M
        ppt->x = (y - pmat->ty) / pmat->xy;
313
1.34M
        ppt->y = (x - pmat->tx) / pmat->yx;
314
1.34M
        return 0;
315
5.96M
    } else {     /* There are faster ways to do this, */
316
        /* but we won't implement one unless we have to. */
317
5.96M
        gs_matrix imat;
318
5.96M
        int code = gs_matrix_invert(pmat, &imat);
319
320
5.96M
        if (code < 0)
321
601
            return code;
322
5.96M
        return gs_point_transform(x, y, &imat, ppt);
323
5.96M
    }
324
176M
}
325
326
/* Transform a distance. */
327
int
328
gs_distance_transform(double dx, double dy, const gs_matrix * pmat,
329
                      gs_point * pdpt)
330
182M
{
331
182M
    pdpt->x = dx * pmat->xx;
332
182M
    pdpt->y = dy * pmat->yy;
333
182M
    if (!is_fzero(pmat->yx))
334
53.7M
        pdpt->x += dy * pmat->yx;
335
182M
    if (!is_fzero(pmat->xy))
336
53.0M
        pdpt->y += dx * pmat->xy;
337
182M
    return 0;
338
182M
}
339
340
/* Inverse-transform a distance. */
341
/* Return gs_error_undefinedresult if the matrix is not invertible. */
342
int
343
gs_distance_transform_inverse(double dx, double dy,
344
                              const gs_matrix * pmat, gs_point * pdpt)
345
51.9M
{
346
51.9M
    if (is_xxyy(pmat)) {
347
20.1M
        if (is_fzero(pmat->xx) || is_fzero(pmat->yy))
348
60.4k
            return_error(gs_error_undefinedresult);
349
20.0M
        pdpt->x = dx / pmat->xx;
350
20.0M
        pdpt->y = dy / pmat->yy;
351
31.7M
    } else if (is_xyyx(pmat)) {
352
365k
        if (is_fzero(pmat->xy) || is_fzero(pmat->yx))
353
4.40k
            return_error(gs_error_undefinedresult);
354
361k
        pdpt->x = dy / pmat->xy;
355
361k
        pdpt->y = dx / pmat->yx;
356
31.4M
    } else {
357
31.4M
        double det = pmat->xx * pmat->yy - pmat->xy * pmat->yx;
358
359
31.4M
        if (det == 0)
360
343k
            return_error(gs_error_undefinedresult);
361
31.0M
        pdpt->x = (dx * pmat->yy - dy * pmat->yx) / det;
362
31.0M
        pdpt->y = (dy * pmat->xx - dx * pmat->xy) / det;
363
31.0M
    }
364
51.5M
    return 0;
365
51.9M
}
366
367
/* Compute the bounding box of 4 points. */
368
int
369
gs_points_bbox(const gs_point pts[4], gs_rect * pbox)
370
21.6M
{
371
21.6M
#define assign_min_max(vmin, vmax, v0, v1)\
372
86.4M
  if ( v0 < v1 ) vmin = v0, vmax = v1; else vmin = v1, vmax = v0
373
21.6M
#define assign_min_max_4(vmin, vmax, v0, v1, v2, v3)\
374
43.2M
  { double min01, max01, min23, max23;\
375
43.2M
    assign_min_max(min01, max01, v0, v1);\
376
43.2M
    assign_min_max(min23, max23, v2, v3);\
377
43.2M
    vmin = min(min01, min23);\
378
43.2M
    vmax = max(max01, max23);\
379
43.2M
  }
380
21.6M
    assign_min_max_4(pbox->p.x, pbox->q.x,
381
21.6M
                     pts[0].x, pts[1].x, pts[2].x, pts[3].x);
382
21.6M
    assign_min_max_4(pbox->p.y, pbox->q.y,
383
21.6M
                     pts[0].y, pts[1].y, pts[2].y, pts[3].y);
384
21.6M
#undef assign_min_max
385
21.6M
#undef assign_min_max_4
386
21.6M
    return 0;
387
21.6M
}
388
389
/* Transform or inverse-transform a bounding box. */
390
/* Return gs_error_undefinedresult if the matrix is not invertible. */
391
static int
392
bbox_transform_either_only(const gs_rect * pbox_in, const gs_matrix * pmat,
393
                           gs_point pts[4],
394
     int (*point_xform) (double, double, const gs_matrix *, gs_point *))
395
21.6M
{
396
21.6M
    int code;
397
398
21.6M
    if ((code = (*point_xform) (pbox_in->p.x, pbox_in->p.y, pmat, &pts[0])) < 0 ||
399
21.6M
        (code = (*point_xform) (pbox_in->p.x, pbox_in->q.y, pmat, &pts[1])) < 0 ||
400
21.6M
        (code = (*point_xform) (pbox_in->q.x, pbox_in->p.y, pmat, &pts[2])) < 0 ||
401
21.6M
     (code = (*point_xform) (pbox_in->q.x, pbox_in->q.y, pmat, &pts[3])) < 0
402
21.6M
        )
403
21.6M
        DO_NOTHING;
404
21.6M
    return code;
405
21.6M
}
406
407
static int
408
bbox_transform_either(const gs_rect * pbox_in, const gs_matrix * pmat,
409
                      gs_rect * pbox_out,
410
     int (*point_xform) (double, double, const gs_matrix *, gs_point *))
411
21.6M
{
412
21.6M
    int code;
413
414
    /*
415
     * In principle, we could transform only one point and two
416
     * distance vectors; however, because of rounding, we will only
417
     * get fully consistent results if we transform all 4 points.
418
     * We must compute the max and min after transforming,
419
     * since a rotation may be involved.
420
     */
421
21.6M
    gs_point pts[4];
422
423
21.6M
    if ((code = bbox_transform_either_only(pbox_in, pmat, pts, point_xform)) < 0)
424
936
        return code;
425
21.6M
    return gs_points_bbox(pts, pbox_out);
426
21.6M
}
427
int
428
gs_bbox_transform(const gs_rect * pbox_in, const gs_matrix * pmat,
429
                  gs_rect * pbox_out)
430
15.2M
{
431
15.2M
    return bbox_transform_either(pbox_in, pmat, pbox_out,
432
15.2M
                                 gs_point_transform);
433
15.2M
}
434
int
435
gs_bbox_transform_only(const gs_rect * pbox_in, const gs_matrix * pmat,
436
                       gs_point points[4])
437
22.5k
{
438
22.5k
    return bbox_transform_either_only(pbox_in, pmat, points,
439
22.5k
                                      gs_point_transform);
440
22.5k
}
441
int
442
gs_bbox_transform_inverse(const gs_rect * pbox_in, const gs_matrix * pmat,
443
                          gs_rect * pbox_out)
444
6.30M
{
445
6.30M
    int code = bbox_transform_either(pbox_in, pmat, pbox_out,
446
6.30M
                                 gs_point_transform_inverse);
447
448
6.30M
    return code;
449
6.30M
}
450
451
/* ------ Coordinate transformations (to fixed point) ------ */
452
453
38.1M
#define f_fits_in_fixed(f) f_fits_in_bits(f, fixed_int_bits)
454
455
/* Make a gs_matrix_fixed from a gs_matrix. */
456
int
457
gs_matrix_fixed_from_matrix(gs_matrix_fixed *pfmat, const gs_matrix *pmat)
458
17.7M
{
459
17.7M
    *(gs_matrix *)pfmat = *pmat;
460
17.7M
    if (f_fits_in_fixed(pmat->tx) && f_fits_in_fixed(pmat->ty)) {
461
17.7M
        pfmat->tx = fixed2float(pfmat->tx_fixed = float2fixed(pmat->tx));
462
17.7M
        pfmat->ty = fixed2float(pfmat->ty_fixed = float2fixed(pmat->ty));
463
17.7M
        pfmat->txy_fixed_valid = true;
464
17.7M
    } else {
465
0
        pfmat->txy_fixed_valid = false;
466
0
    }
467
17.7M
    return 0;
468
17.7M
}
469
470
/* Transform a point with a fixed-point result. */
471
int
472
gs_point_transform2fixed(const gs_matrix_fixed * pmat,
473
                         double x, double y, gs_fixed_point * ppt)
474
10.9M
{
475
10.9M
    fixed px, py, t;
476
10.9M
    double xtemp, ytemp;
477
10.9M
    int code;
478
479
10.9M
    if (!pmat->txy_fixed_valid) { /* The translation is out of range.  Do the */
480
        /* computation in floating point, and convert to */
481
        /* fixed at the end. */
482
272k
        gs_point fpt;
483
484
272k
        gs_point_transform(x, y, (const gs_matrix *)pmat, &fpt);
485
272k
        if (!(f_fits_in_fixed(fpt.x) && f_fits_in_fixed(fpt.y)))
486
12
            return_error(gs_error_limitcheck);
487
272k
        ppt->x = float2fixed(fpt.x);
488
272k
        ppt->y = float2fixed(fpt.y);
489
272k
        return 0;
490
272k
    }
491
10.6M
    if (!is_fzero(pmat->xy)) { /* Hope for 90 degree rotation */
492
8.72k
        if ((code = CHECK_DFMUL2FIXED_VARS(px, y, pmat->yx, xtemp)) < 0 ||
493
8.72k
            (code = CHECK_DFMUL2FIXED_VARS(py, x, pmat->xy, ytemp)) < 0
494
8.72k
            )
495
6
            return code;
496
8.72k
        FINISH_DFMUL2FIXED_VARS(px, xtemp);
497
8.72k
        FINISH_DFMUL2FIXED_VARS(py, ytemp);
498
8.72k
        if (!is_fzero(pmat->xx)) {
499
4.63k
            if ((code = CHECK_DFMUL2FIXED_VARS(t, x, pmat->xx, xtemp)) < 0)
500
0
                return code;
501
4.63k
            FINISH_DFMUL2FIXED_VARS(t, xtemp);
502
4.63k
            if ((code = CHECK_SET_FIXED_SUM(px, px, t)) < 0)
503
0
                return code;
504
4.63k
        }
505
8.72k
        if (!is_fzero(pmat->yy)) {
506
4.63k
            if ((code = CHECK_DFMUL2FIXED_VARS(t, y, pmat->yy, ytemp)) < 0)
507
0
                return code;
508
4.63k
            FINISH_DFMUL2FIXED_VARS(t, ytemp);
509
4.63k
            if ((code = CHECK_SET_FIXED_SUM(py, py, t)) < 0)
510
0
                return code;
511
4.63k
        }
512
10.6M
    } else {
513
10.6M
        if ((code = CHECK_DFMUL2FIXED_VARS(px, x, pmat->xx, xtemp)) < 0 ||
514
10.6M
            (code = CHECK_DFMUL2FIXED_VARS(py, y, pmat->yy, ytemp)) < 0
515
10.6M
            )
516
7.93k
            return code;
517
10.6M
        FINISH_DFMUL2FIXED_VARS(px, xtemp);
518
10.6M
        FINISH_DFMUL2FIXED_VARS(py, ytemp);
519
10.6M
        if (!is_fzero(pmat->yx)) {
520
0
            if ((code = CHECK_DFMUL2FIXED_VARS(t, y, pmat->yx, ytemp)) < 0)
521
0
                return code;
522
0
            FINISH_DFMUL2FIXED_VARS(t, ytemp);
523
0
            if ((code = CHECK_SET_FIXED_SUM(px, px, t)) < 0)
524
0
                return code;
525
0
        }
526
10.6M
    }
527
10.6M
    if (((code = CHECK_SET_FIXED_SUM(ppt->x, px, pmat->tx_fixed)) < 0) ||
528
10.6M
        ((code = CHECK_SET_FIXED_SUM(ppt->y, py, pmat->ty_fixed)) < 0) )
529
153
        return code;
530
10.6M
    return 0;
531
10.6M
}
532
533
#if PRECISE_CURRENTPOINT
534
/* Transform a point with a fixed-point result. */
535
/* Used for the best precision of the current point,
536
   see comment in clamp_point_aux. */
537
int
538
gs_point_transform2fixed_rounding(const gs_matrix_fixed * pmat,
539
                         double x, double y, gs_fixed_point * ppt)
540
1.03M
{
541
1.03M
    gs_point fpt;
542
543
1.03M
    gs_point_transform(x, y, (const gs_matrix *)pmat, &fpt);
544
1.03M
    if (!(f_fits_in_fixed(fpt.x) && f_fits_in_fixed(fpt.y)))
545
95
        return_error(gs_error_limitcheck);
546
1.03M
    ppt->x = float2fixed_rounded(fpt.x);
547
1.03M
    ppt->y = float2fixed_rounded(fpt.y);
548
1.03M
    return 0;
549
1.03M
}
550
#endif
551
552
/* Transform a distance with a fixed-point result. */
553
int
554
gs_distance_transform2fixed(const gs_matrix_fixed * pmat,
555
                            double dx, double dy, gs_fixed_point * ppt)
556
52.0M
{
557
52.0M
    fixed px, py, t;
558
52.0M
    double xtemp, ytemp;
559
52.0M
    int code;
560
561
52.0M
    if ((code = CHECK_DFMUL2FIXED_VARS(px, dx, pmat->xx, xtemp)) < 0 ||
562
51.7M
        (code = CHECK_DFMUL2FIXED_VARS(py, dy, pmat->yy, ytemp)) < 0
563
52.0M
        )
564
349k
        return code;
565
51.7M
    FINISH_DFMUL2FIXED_VARS(px, xtemp);
566
51.7M
    FINISH_DFMUL2FIXED_VARS(py, ytemp);
567
51.7M
    if (!is_fzero(pmat->yx)) {
568
2.25M
        if ((code = CHECK_DFMUL2FIXED_VARS(t, dy, pmat->yx, ytemp)) < 0)
569
19.7k
            return code;
570
2.23M
        FINISH_DFMUL2FIXED_VARS(t, ytemp);
571
2.23M
        if ((code = CHECK_SET_FIXED_SUM(px, px, t)) < 0)
572
16
            return code;
573
2.23M
    }
574
51.7M
    if (!is_fzero(pmat->xy)) {
575
2.14M
        if ((code = CHECK_DFMUL2FIXED_VARS(t, dx, pmat->xy, xtemp)) < 0)
576
11.7k
            return code;
577
2.13M
        FINISH_DFMUL2FIXED_VARS(t, xtemp);
578
2.13M
        if ((code = CHECK_SET_FIXED_SUM(py, py, t)) < 0)
579
0
            return code;
580
2.13M
    }
581
51.6M
    ppt->x = px;
582
51.6M
    ppt->y = py;
583
51.6M
    return 0;
584
51.7M
}
585
586
/* ------ Serialization ------ */
587
588
/*
589
 * For maximum conciseness in band lists, we write a matrix as a control
590
 * byte followed by 0 to 6 values.  The control byte has the format
591
 * AABBCD00.  AA and BB control (xx,yy) and (xy,yx) as follows:
592
 *  00 = values are (0.0, 0.0)
593
 *  01 = values are (V, V) [1 value follows]
594
 *  10 = values are (V, -V) [1 value follows]
595
 *  11 = values are (U, V) [2 values follow]
596
 * C and D control tx and ty as follows:
597
 *  0 = value is 0.0
598
 *  1 = value follows
599
 * The following code is the only place that knows this representation.
600
 */
601
602
/* Put a matrix on a stream. */
603
int
604
sput_matrix(stream *s, const gs_matrix *pmat)
605
6.56M
{
606
6.56M
    byte buf[1 + 6 * sizeof(float)];
607
6.56M
    byte *cp = buf + 1;
608
6.56M
    byte b = 0;
609
6.56M
    float coeff[6];
610
6.56M
    int i;
611
6.56M
    uint ignore;
612
613
6.56M
    coeff[0] = pmat->xx;
614
6.56M
    coeff[1] = pmat->xy;
615
6.56M
    coeff[2] = pmat->yx;
616
6.56M
    coeff[3] = pmat->yy;
617
6.56M
    coeff[4] = pmat->tx;
618
6.56M
    coeff[5] = pmat->ty;
619
19.6M
    for (i = 0; i < 4; i += 2) {
620
13.1M
        float u = coeff[i], v = coeff[i ^ 3];
621
622
13.1M
        b <<= 2;
623
13.1M
        if (u != 0 || v != 0) {
624
7.56M
            memcpy(cp, &u, sizeof(float));
625
7.56M
            cp += sizeof(float);
626
627
7.56M
            if (v == u)
628
550k
                b += 1;
629
7.01M
            else if (v == -u)
630
3.64M
                b += 2;
631
3.37M
            else {
632
3.37M
                b += 3;
633
3.37M
                memcpy(cp, &v, sizeof(float));
634
3.37M
                cp += sizeof(float);
635
3.37M
            }
636
7.56M
        }
637
13.1M
    }
638
19.6M
    for (; i < 6; ++i) {
639
13.1M
        float v = coeff[i];
640
641
13.1M
        b <<= 1;
642
13.1M
        if (v != 0) {
643
12.0M
            ++b;
644
12.0M
            memcpy(cp, &v, sizeof(float));
645
12.0M
            cp += sizeof(float);
646
12.0M
        }
647
13.1M
    }
648
6.56M
    buf[0] = b << 2;
649
6.56M
    return sputs(s, buf, cp - buf, &ignore);
650
6.56M
}
651
652
/* Get a matrix from a stream. */
653
int
654
sget_matrix(stream *s, gs_matrix *pmat)
655
9.87M
{
656
9.87M
    int b = sgetc(s);
657
9.87M
    float coeff[6];
658
9.87M
    int i;
659
9.87M
    int status;
660
9.87M
    uint nread;
661
662
9.87M
    if (b < 0)
663
0
        return b;
664
29.6M
    for (i = 0; i < 4; i += 2, b <<= 2)
665
19.7M
        if (!(b & 0xc0))
666
9.41M
            coeff[i] = coeff[i ^ 3] = 0.0;
667
10.3M
        else {
668
10.3M
            float value;
669
670
10.3M
            status = sgets(s, (byte *)&value, sizeof(value), &nread);
671
10.3M
            if (status < 0 && status != EOFC)
672
0
                return_error(gs_error_ioerror);
673
10.3M
            coeff[i] = value;
674
10.3M
            switch ((b >> 6) & 3) {
675
1.15M
                case 1:
676
1.15M
                    coeff[i ^ 3] = value;
677
1.15M
                    break;
678
6.42M
                case 2:
679
6.42M
                    coeff[i ^ 3] = -value;
680
6.42M
                    break;
681
2.76M
                case 3:
682
2.76M
                    status = sgets(s, (byte *)&coeff[i ^ 3],
683
2.76M
                                   sizeof(coeff[0]), &nread);
684
2.76M
                    if (status < 0 && status != EOFC)
685
0
                        return_error(gs_error_ioerror);
686
10.3M
            }
687
10.3M
        }
688
29.6M
    for (; i < 6; ++i, b <<= 1)
689
19.7M
        if (b & 0x80) {
690
16.1M
            status = sgets(s, (byte *)&coeff[i], sizeof(coeff[0]), &nread);
691
16.1M
            if (status < 0 && status != EOFC)
692
0
                return_error(gs_error_ioerror);
693
16.1M
        } else
694
3.61M
            coeff[i] = 0.0;
695
9.87M
    pmat->xx = coeff[0];
696
9.87M
    pmat->xy = coeff[1];
697
9.87M
    pmat->yx = coeff[2];
698
9.87M
    pmat->yy = coeff[3];
699
9.87M
    pmat->tx = coeff[4];
700
9.87M
    pmat->ty = coeff[5];
701
9.87M
    return 0;
702
9.87M
}
703
704
/* Compare two matrices */
705
int
706
10.5M
gs_matrix_compare(const gs_matrix *pmat1, const gs_matrix *pmat2) {
707
10.5M
  if (pmat1->xx != pmat2->xx)
708
255k
    return(1);
709
10.2M
  if (pmat1->xy != pmat2->xy)
710
2.75k
    return(1);
711
10.2M
  if (pmat1->yx != pmat2->yx)
712
128
    return(1);
713
10.2M
  if (pmat1->yy != pmat2->yy)
714
8.30k
    return(1);
715
10.2M
  if (pmat1->tx != pmat2->tx)
716
8.11M
    return(1);
717
2.17M
  if (pmat1->ty != pmat2->ty)
718
71.9k
    return(1);
719
2.09M
  return(0);
720
2.17M
}