Coverage Report

Created: 2026-02-26 07:15

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/xpdf-4.06/splash/Splash.cc
Line
Count
Source
1
//========================================================================
2
//
3
// Splash.cc
4
//
5
// Copyright 2003-2020 Glyph & Cog, LLC
6
//
7
//========================================================================
8
9
#include <aconf.h>
10
11
#include <stdlib.h>
12
#include <string.h>
13
#include <limits.h>
14
#include <math.h>
15
#include "gmem.h"
16
#include "gmempp.h"
17
#include "GString.h"
18
#include "SplashErrorCodes.h"
19
#include "SplashMath.h"
20
#include "SplashBitmap.h"
21
#include "SplashState.h"
22
#include "SplashPath.h"
23
#include "SplashXPath.h"
24
#include "SplashXPathScanner.h"
25
#include "SplashPattern.h"
26
#include "SplashScreen.h"
27
#include "SplashFont.h"
28
#include "SplashGlyphBitmap.h"
29
#include "Splash.h"
30
31
// the MSVC math.h doesn't define this
32
#ifndef M_PI
33
#define M_PI 3.14159265358979323846
34
#endif
35
36
//------------------------------------------------------------------------
37
38
// distance of Bezier control point from center for circle approximation
39
// = (4 * (sqrt(2) - 1) / 3) * r
40
1.96M
#define bezierCircle ((SplashCoord)0.55228475)
41
16.7k
#define bezierCircle2 ((SplashCoord)(0.5 * 0.55228475))
42
43
// Divide a 16-bit value (in [0, 255*255]) by 255, returning an 8-bit result.
44
100k
static inline Guchar div255(int x) {
45
100k
  return (Guchar)((x + (x >> 8) + 0x80) >> 8);
46
100k
}
47
48
// Clip x to lie in [0, 255].
49
3.92k
static inline Guchar clip255(int x) {
50
3.92k
  return x < 0 ? 0 : x > 255 ? 255 : (Guchar)x;
51
3.92k
}
52
53
// Used by drawImage and fillImageMask to divide the target
54
// quadrilateral into sections.
55
struct ImageSection {
56
  int y0, y1;       // actual y range
57
  int ia0, ia1;       // vertex indices for edge A
58
  int ib0, ib1;       // vertex indices for edge B
59
  SplashCoord xa0, ya0, xa1, ya1; // edge A
60
  SplashCoord dxdya;      // slope of edge A
61
  SplashCoord xb0, yb0, xb1, yb1; // edge B
62
  SplashCoord dxdyb;      // slope of edge B
63
};
64
65
//------------------------------------------------------------------------
66
// SplashPipe
67
//------------------------------------------------------------------------
68
69
#define splashPipeMaxStages 9
70
71
struct SplashPipe {
72
  // source pattern
73
  SplashPattern *pattern;
74
75
  // source alpha and color
76
  Guchar aInput;
77
  SplashColor cSrcVal;
78
79
  // source overprint mask
80
  //~ this is a kludge - this pointer should be passed as an arg to the
81
  //~   pipeRun function, but that would require passing in a lot of
82
  //~   null pointers, since it's rarely used
83
  Guint *srcOverprintMaskPtr;
84
85
  // alpha is shape
86
  GBool alphaIsShape;
87
88
  // special cases and result color
89
  GBool noTransparency;
90
  GBool alphaOnly;
91
  SplashPipeResultColorCtrl resultColorCtrl;
92
93
  // non-isolated group correction
94
  // (this is only used when Splash::composite() is called to composite
95
  // a non-isolated group onto the backdrop)
96
  GBool nonIsolatedGroup;
97
98
  // the "run" function
99
  void (Splash::*run)(SplashPipe *pipe, int x0, int x1, int y,
100
          Guchar *shapPetr, Guchar *alphaPtr,
101
          SplashColorPtr cSrcPtr);
102
};
103
104
SplashPipeResultColorCtrl Splash::pipeResultColorNoAlphaBlend[] = {
105
  splashPipeResultColorNoAlphaBlendMono,
106
  splashPipeResultColorNoAlphaBlendMono,
107
  splashPipeResultColorNoAlphaBlendRGB,
108
  splashPipeResultColorNoAlphaBlendRGB
109
#if SPLASH_CMYK
110
  ,
111
  splashPipeResultColorNoAlphaBlendCMYK
112
#endif
113
};
114
115
SplashPipeResultColorCtrl Splash::pipeResultColorAlphaNoBlend[] = {
116
  splashPipeResultColorAlphaNoBlendMono,
117
  splashPipeResultColorAlphaNoBlendMono,
118
  splashPipeResultColorAlphaNoBlendRGB,
119
  splashPipeResultColorAlphaNoBlendRGB
120
#if SPLASH_CMYK
121
  ,
122
  splashPipeResultColorAlphaNoBlendCMYK
123
#endif
124
};
125
126
SplashPipeResultColorCtrl Splash::pipeResultColorAlphaBlend[] = {
127
  splashPipeResultColorAlphaBlendMono,
128
  splashPipeResultColorAlphaBlendMono,
129
  splashPipeResultColorAlphaBlendRGB,
130
  splashPipeResultColorAlphaBlendRGB
131
#if SPLASH_CMYK
132
  ,
133
  splashPipeResultColorAlphaBlendCMYK
134
#endif
135
};
136
137
//------------------------------------------------------------------------
138
// modified region
139
//------------------------------------------------------------------------
140
141
223k
void Splash::clearModRegion() {
142
223k
  modXMin = bitmap->width;
143
223k
  modYMin = bitmap->height;
144
223k
  modXMax = -1;
145
223k
  modYMax = -1;
146
223k
}
147
148
508k
inline void Splash::updateModX(int x) {
149
508k
  if (x < modXMin) {
150
212k
    modXMin = x;
151
212k
  }
152
508k
  if (x > modXMax) {
153
213k
    modXMax = x;
154
213k
  }
155
508k
}
156
157
462k
inline void Splash::updateModY(int y) {
158
462k
  if (y < modYMin) {
159
212k
    modYMin = y;
160
212k
  }
161
462k
  if (y > modYMax) {
162
213k
    modYMax = y;
163
213k
  }
164
462k
}
165
166
//------------------------------------------------------------------------
167
// pipeline
168
//------------------------------------------------------------------------
169
170
inline void Splash::pipeInit(SplashPipe *pipe, SplashPattern *pattern,
171
           Guchar aInput, GBool usesShape, GBool usesAlpha,
172
           GBool nonIsolatedGroup, GBool usesSrcOverprint,
173
38.5k
           GBool alphaIsShape) {
174
38.5k
  SplashColorMode mode;
175
176
38.5k
  mode = bitmap->mode;
177
178
38.5k
  pipe->pattern = NULL;
179
180
  // source color
181
38.5k
  if (pattern && pattern->isStatic()) {
182
33.2k
    pattern->getColor(0, 0, pipe->cSrcVal);
183
33.2k
    pipe->pattern = NULL;
184
33.2k
  } else {
185
5.28k
    pipe->pattern = pattern;
186
5.28k
  }
187
188
  // source alpha
189
38.5k
  pipe->aInput = aInput;
190
191
  // source overprint mask
192
38.5k
  pipe->srcOverprintMaskPtr = NULL;
193
194
  // alpha is shape
195
38.5k
  pipe->alphaIsShape = alphaIsShape;
196
197
  // special cases
198
38.5k
  pipe->noTransparency = aInput == 255 &&
199
38.1k
                         !state->softMask &&
200
37.5k
                         !usesShape &&
201
37.5k
                         !usesAlpha &&
202
0
                         !state->inNonIsolatedGroup &&
203
0
       !state->inKnockoutGroup &&
204
0
                         !nonIsolatedGroup &&
205
0
                         state->overprintMask == 0xffffffff;
206
38.5k
  pipe->alphaOnly = aInput == 255 &&
207
38.1k
                    !state->softMask &&
208
37.5k
                    !usesShape &&
209
37.5k
                    usesAlpha &&
210
37.5k
                    !state->inNonIsolatedGroup &&
211
17.8k
        !state->inKnockoutGroup &&
212
17.8k
                    !nonIsolatedGroup &&
213
13.9k
                    state->overprintMask == 0xffffffff;
214
215
  // result color
216
38.5k
  if (pipe->noTransparency) {
217
    // the !state->blendFunc case is handled separately in pipeRun
218
0
    pipe->resultColorCtrl = pipeResultColorNoAlphaBlend[mode];
219
38.5k
  } else if (!state->blendFunc) {
220
37.9k
    pipe->resultColorCtrl = pipeResultColorAlphaNoBlend[mode];
221
37.9k
  } else {
222
563
    pipe->resultColorCtrl = pipeResultColorAlphaBlend[mode];
223
563
  }
224
225
  // non-isolated group correction
226
38.5k
  pipe->nonIsolatedGroup = nonIsolatedGroup;
227
228
  // select the 'run' function
229
38.5k
  pipe->run = &Splash::pipeRun;
230
38.5k
  if (shapeBitmap || overprintMaskBitmap || usesSrcOverprint) {
231
    // use Splash::pipeRun
232
38.5k
  } else if (!pipe->pattern && pipe->noTransparency && !state->blendFunc) {
233
0
    if (mode == splashModeMono1 && !bitmap->alpha) {
234
0
      pipe->run = &Splash::pipeRunSimpleMono1;
235
0
    } else if (mode == splashModeMono8 && bitmap->alpha) {
236
0
      pipe->run = &Splash::pipeRunSimpleMono8;
237
0
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
238
0
      pipe->run = &Splash::pipeRunSimpleRGB8;
239
0
    } else if (mode == splashModeBGR8 && bitmap->alpha) {
240
0
      pipe->run = &Splash::pipeRunSimpleBGR8;
241
0
#if SPLASH_CMYK
242
0
    } else if (mode == splashModeCMYK8 && bitmap->alpha) {
243
0
      pipe->run = &Splash::pipeRunSimpleCMYK8;
244
0
#endif
245
0
    }
246
38.5k
  } else if (!pipe->pattern && pipe->alphaOnly && !state->blendFunc) {
247
13.7k
    if (mode == splashModeMono1 && !bitmap->alpha) {
248
159
      pipe->run = &Splash::pipeRunShapeMono1;
249
13.6k
    } else if (mode == splashModeMono8 && bitmap->alpha) {
250
483
      pipe->run = &Splash::pipeRunShapeMono8;
251
13.1k
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
252
12.7k
      pipe->run = &Splash::pipeRunShapeRGB8;
253
12.7k
    } else if (mode == splashModeBGR8 && bitmap->alpha) {
254
0
      pipe->run = &Splash::pipeRunShapeBGR8;
255
0
#if SPLASH_CMYK
256
359
    } else if (mode == splashModeCMYK8 && bitmap->alpha) {
257
0
      pipe->run = &Splash::pipeRunShapeCMYK8;
258
0
#endif
259
359
    } else if (mode == splashModeMono8 && !bitmap->alpha) {
260
      // this is used when drawing soft-masked images
261
359
      pipe->run = &Splash::pipeRunShapeNoAlphaMono8;
262
359
    }
263
24.7k
  } else if (!pipe->pattern && !pipe->noTransparency && !state->softMask &&
264
24.1k
       usesAlpha &&
265
24.1k
       !(state->inNonIsolatedGroup && (alpha0Bitmap ||
266
19.7k
               parent->bitmap->alpha)) &&
267
4.44k
       !state->inKnockoutGroup &&
268
4.44k
       !state->blendFunc && !pipe->nonIsolatedGroup) {
269
336
    if (mode == splashModeMono1 && !bitmap->alpha) {
270
0
      pipe->run = &Splash::pipeRunAAMono1;
271
336
    } else if (mode == splashModeMono8 && bitmap->alpha) {
272
0
      pipe->run = &Splash::pipeRunAAMono8;
273
336
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
274
336
      pipe->run = &Splash::pipeRunAARGB8;
275
336
    } else if (mode == splashModeBGR8 && bitmap->alpha) {
276
0
      pipe->run = &Splash::pipeRunAABGR8;
277
0
#if SPLASH_CMYK
278
0
    } else if (mode == splashModeCMYK8 && bitmap->alpha) {
279
0
      pipe->run = &Splash::pipeRunAACMYK8;
280
0
#endif
281
0
    }
282
24.4k
  } else if (!pipe->pattern &&
283
24.4k
             aInput == 255 &&
284
24.3k
             state->softMask &&
285
584
             usesAlpha &&
286
584
             !state->inNonIsolatedGroup &&
287
579
             !state->inKnockoutGroup &&
288
579
             !nonIsolatedGroup &&
289
536
             state->overprintMask == 0xffffffff &&
290
536
             !state->blendFunc) {
291
430
    if (mode == splashModeMono8 && bitmap->alpha) {
292
139
      pipe->run = &Splash::pipeRunSoftMaskMono8;
293
291
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
294
291
      pipe->run = &Splash::pipeRunSoftMaskRGB8;
295
291
    } else if (mode == splashModeBGR8 && bitmap->alpha) {
296
0
      pipe->run = &Splash::pipeRunSoftMaskBGR8;
297
0
#if SPLASH_CMYK
298
0
    } else if (mode == splashModeCMYK8 && bitmap->alpha) {
299
0
      pipe->run = &Splash::pipeRunSoftMaskCMYK8;
300
0
#endif
301
0
    }
302
24.0k
  } else if (!pipe->pattern && !pipe->noTransparency && !state->softMask &&
303
23.8k
       usesAlpha &&
304
23.8k
       state->inNonIsolatedGroup && (alpha0Bitmap ||
305
19.7k
             parent->bitmap->alpha) &&
306
19.7k
       !state->inKnockoutGroup &&
307
19.7k
       !state->blendFunc && !pipe->nonIsolatedGroup) {
308
19.6k
    if (mode == splashModeMono8 && bitmap->alpha) {
309
0
      pipe->run = &Splash::pipeRunNonIsoMono8;
310
19.6k
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
311
19.6k
      pipe->run = &Splash::pipeRunNonIsoRGB8;
312
19.6k
    } else if (mode == splashModeBGR8 && bitmap->alpha) {
313
0
      pipe->run = &Splash::pipeRunNonIsoBGR8;
314
0
#if SPLASH_CMYK
315
0
    } else if (mode == splashModeCMYK8 && bitmap->alpha) {
316
0
      pipe->run = &Splash::pipeRunNonIsoCMYK8;
317
0
#endif
318
0
    }
319
19.6k
  }
320
38.5k
}
321
322
// general case
323
void Splash::pipeRun(SplashPipe *pipe, int x0, int x1, int y,
324
         Guchar *shapePtr, Guchar *alphaPtr,
325
1.89k
         SplashColorPtr cSrcPtr) {
326
1.89k
  Guchar *shapePtr2, *alphaPtr2;
327
1.89k
  Guchar shape, alpha, fSrc, aSrc, fDest, aDest;
328
1.89k
  Guchar alphaI, alphaIm1, alpha0, fResult, aResult;
329
1.89k
  Guchar m1, m2, m3, m4;
330
1.89k
  SplashColor cSrc, cDest, c0, cBlend;
331
1.89k
  Guchar shapeVal, alphaVal, cResult0, cResult1, cResult2, cResult3;
332
1.89k
  int cSrcStride, shapeStride, alphaStride, x, lastX, t;
333
1.89k
  SplashColorPtr cBlendIn, destColorPtr;
334
1.89k
  Guchar destColorMask;
335
1.89k
  Guchar *destShapePtr, *destAlphaPtr;
336
1.89k
  SplashColorPtr color0Ptr;
337
1.89k
  Guchar color0Mask;
338
1.89k
  Guchar *alpha0Ptr;
339
1.89k
  SplashColorPtr softMaskPtr;
340
1.89k
  Guint overprintMask;
341
1.89k
  Guint *overprintMaskPtr;
342
1.89k
#if SPLASH_CMYK
343
1.89k
  Guchar aPrev;
344
1.89k
  SplashColor cSrc2, cBlendIn2;
345
1.89k
#endif
346
1.89k
  int i;
347
348
1.89k
  if (cSrcPtr && !pipe->pattern) {
349
1.57k
    cSrcStride = bitmapComps;
350
1.57k
  } else {
351
314
    cSrcPtr = pipe->cSrcVal;
352
314
    cSrcStride = 0;
353
314
  }
354
1.89k
  if (shapePtr) {
355
0
    shapePtr2 = shapePtr;
356
0
    shapeStride = 1;
357
1.89k
  } else {
358
1.89k
    shapeVal = 0xff;
359
1.89k
    shapePtr2 = &shapeVal;
360
1.89k
    shapeStride = 0;
361
1.89k
  }
362
1.89k
  if (alphaPtr) {
363
1.89k
    alphaPtr2 = alphaPtr;
364
1.89k
    alphaStride = 1;
365
1.89k
  } else {
366
0
    alphaVal = 0xff;
367
0
    alphaPtr2 = &alphaVal;
368
0
    alphaStride = 0;
369
0
  }
370
371
1.89k
  if (shapePtr) {
372
0
    for (; x0 <= x1; ++x0) {
373
0
      if (*shapePtr2) {
374
0
  break;
375
0
      }
376
0
      cSrcPtr += cSrcStride;
377
0
      ++shapePtr2;
378
0
      ++alphaPtr2;
379
0
      if (pipe->srcOverprintMaskPtr) {
380
0
  ++pipe->srcOverprintMaskPtr;
381
0
      }
382
0
    }
383
1.89k
  } else if (alphaPtr) {
384
2.08k
    for (; x0 <= x1; ++x0) {
385
1.89k
      if (*alphaPtr2) {
386
1.70k
  break;
387
1.70k
      }
388
193
      cSrcPtr += cSrcStride;
389
193
      ++alphaPtr2;
390
193
      if (pipe->srcOverprintMaskPtr) {
391
0
  ++pipe->srcOverprintMaskPtr;
392
0
      }
393
193
    }
394
1.89k
  }
395
1.89k
  if (x0 > x1) {
396
193
    return;
397
193
  }
398
1.70k
  updateModX(x0);
399
1.70k
  updateModY(y);
400
1.70k
  lastX = x0;
401
402
1.70k
  useBitmapRow(y);
403
404
1.70k
  if (bitmap->mode == splashModeMono1) {
405
0
    destColorPtr = &bitmap->data[y * bitmap->rowSize + (x0 >> 3)];
406
0
    destColorMask = (Guchar)(0x80 >> (x0 & 7));
407
1.70k
  } else {
408
1.70k
    destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * bitmapComps];
409
1.70k
    destColorMask = 0; // make gcc happy
410
1.70k
  }
411
1.70k
  if (shapeBitmap) {
412
0
    destShapePtr = &shapeBitmap->alpha[y * shapeBitmap->alphaRowSize + x0];
413
1.70k
  } else {
414
1.70k
    destShapePtr = NULL;
415
1.70k
  }
416
1.70k
  if (bitmap->alpha) {
417
1.70k
    destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
418
1.70k
  } else {
419
0
    destAlphaPtr = NULL;
420
0
  }
421
1.70k
  if (state->softMask) {
422
113
    softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
423
1.58k
  } else {
424
1.58k
    softMaskPtr = NULL;
425
1.58k
  }
426
1.70k
  if (state->inKnockoutGroup) {
427
0
    if (bitmap->mode == splashModeMono1) {
428
0
      color0Ptr =
429
0
          &parent->bitmap->data[(parentOffsetY + y) * parent->bitmap->rowSize +
430
0
        ((parentOffsetX + x0) >> 3)];
431
0
      color0Mask = (Guchar)(0x80 >> ((parentOffsetX + x0) & 7));
432
0
    } else {
433
0
      color0Ptr =
434
0
          &parent->bitmap->data[(parentOffsetY + y) * parent->bitmap->rowSize +
435
0
        (parentOffsetX + x0) * bitmapComps];
436
0
      color0Mask = 0; // make gcc happy
437
0
    }
438
1.70k
  } else {
439
1.70k
    color0Ptr = NULL;
440
1.70k
    color0Mask = 0; // make gcc happy
441
1.70k
  }
442
1.70k
  if (alpha0Bitmap) {
443
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
444
1.70k
  } else if (state->inNonIsolatedGroup && parent->bitmap->alpha) {
445
115
    alpha0Ptr = &parent->bitmap->alpha[
446
115
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
447
115
        + (parentOffsetX + x0)];
448
1.58k
  } else {
449
1.58k
    alpha0Ptr = NULL;
450
1.58k
  }
451
1.70k
  if (overprintMaskBitmap) {
452
0
    overprintMaskPtr = overprintMaskBitmap + y * bitmap->width + x0;
453
1.70k
  } else {
454
1.70k
    overprintMaskPtr = NULL;
455
1.70k
  }
456
457
  // c0 is only used for non-iso knockout groups, but in some other
458
  // cases we use m2 * c0[i], with m2=0 -- so initialize c0[i]=0 here
459
  // to avoid valgrind warnings
460
6.80k
  for (i = 0; i < bitmapComps; ++i) {
461
5.10k
    c0[i] = 0;
462
5.10k
  }
463
464
3.40k
  for (x = x0; x <= x1; ++x) {
465
466
    //----- shape and alpha
467
468
1.70k
    shape = *shapePtr2;
469
1.70k
    alpha = *alphaPtr2;
470
1.70k
    if (!(shapePtr ? shape : alpha)) {
471
0
      if (bitmap->mode == splashModeMono1) {
472
0
  destColorPtr += destColorMask & 1;
473
0
  destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
474
0
      } else {
475
0
  destColorPtr += bitmapComps;
476
0
      }
477
0
      if (destShapePtr) {
478
0
  ++destShapePtr;
479
0
      }
480
0
      if (destAlphaPtr) {
481
0
  ++destAlphaPtr;
482
0
      }
483
0
      if (softMaskPtr) {
484
0
  ++softMaskPtr;
485
0
      }
486
0
      if (color0Ptr) {
487
0
  if (bitmap->mode == splashModeMono1) {
488
0
    color0Ptr += color0Mask & 1;
489
0
    color0Mask = (Guchar)((color0Mask << 7) | (color0Mask >> 1));
490
0
  } else {
491
0
    color0Ptr += bitmapComps;
492
0
  }
493
0
      }
494
0
      if (alpha0Ptr) {
495
0
  ++alpha0Ptr;
496
0
      }
497
0
      cSrcPtr += cSrcStride;
498
0
      shapePtr2 += shapeStride;
499
0
      alphaPtr2 += alphaStride;
500
0
      if (pipe->srcOverprintMaskPtr) {
501
0
  ++pipe->srcOverprintMaskPtr;
502
0
      }
503
0
      if (overprintMaskPtr) {
504
0
  ++overprintMaskPtr;
505
0
      }
506
0
      continue;
507
0
    }
508
1.70k
    lastX = x;
509
510
    //----- source color
511
512
    // static pattern: handled in pipeInit
513
    // fixed color: handled in pipeInit
514
515
    // dynamic pattern
516
1.70k
    if (pipe->pattern) {
517
0
      pipe->pattern->getColor(x, y, pipe->cSrcVal);
518
0
    }
519
520
1.70k
    cResult0 = cResult1 = cResult2 = cResult3 = 0; // make gcc happy
521
522
1.70k
    if (pipe->noTransparency && !state->blendFunc) {
523
524
      //----- handle overprint
525
526
0
      if (overprintMaskPtr) {
527
0
  *overprintMaskPtr++ = 0xffffffff;
528
0
      }
529
530
      //----- result color
531
532
0
      switch (bitmap->mode) {
533
0
      case splashModeMono1:
534
0
      case splashModeMono8:
535
0
  cResult0 = state->grayTransfer[cSrcPtr[0]];
536
0
  break;
537
0
      case splashModeRGB8:
538
0
      case splashModeBGR8:
539
0
  cResult0 = state->rgbTransferR[cSrcPtr[0]];
540
0
  cResult1 = state->rgbTransferG[cSrcPtr[1]];
541
0
  cResult2 = state->rgbTransferB[cSrcPtr[2]];
542
0
  break;
543
0
#if SPLASH_CMYK
544
0
      case splashModeCMYK8:
545
0
  cResult0 = state->cmykTransferC[cSrcPtr[0]];
546
0
  cResult1 = state->cmykTransferM[cSrcPtr[1]];
547
0
  cResult2 = state->cmykTransferY[cSrcPtr[2]];
548
0
  cResult3 = state->cmykTransferK[cSrcPtr[3]];
549
0
  break;
550
0
#endif
551
0
      }
552
0
      fResult = 255;
553
0
      aResult = 255;
554
555
1.70k
    } else { // if (noTransparency && !blendFunc)
556
557
      //----- read destination color
558
559
1.70k
      switch (bitmap->mode) {
560
0
      case splashModeMono1:
561
0
  cDest[0] = (*destColorPtr & destColorMask) ? 0xff : 0x00;
562
0
  break;
563
0
      case splashModeMono8:
564
0
  cDest[0] = *destColorPtr;
565
0
  break;
566
1.70k
      case splashModeRGB8:
567
1.70k
  cDest[0] = destColorPtr[0];
568
1.70k
  cDest[1] = destColorPtr[1];
569
1.70k
  cDest[2] = destColorPtr[2];
570
1.70k
  break;
571
0
      case splashModeBGR8:
572
0
  cDest[0] = destColorPtr[2];
573
0
  cDest[1] = destColorPtr[1];
574
0
  cDest[2] = destColorPtr[0];
575
0
  break;
576
0
#if SPLASH_CMYK
577
0
      case splashModeCMYK8:
578
0
  cDest[0] = destColorPtr[0];
579
0
  cDest[1] = destColorPtr[1];
580
0
  cDest[2] = destColorPtr[2];
581
0
  cDest[3] = destColorPtr[3];
582
0
  break;
583
1.70k
#endif
584
1.70k
      }
585
586
      //----- read backdrop color
587
      //      (only used for non-iso knockout groups)
588
589
1.70k
      if (color0Ptr && alpha0Ptr) {
590
0
  switch (bitmap->mode) {
591
0
  case splashModeMono1:
592
0
    c0[0] = (*color0Ptr & color0Mask) ? 0xff : 0x00;
593
0
    color0Ptr += color0Mask & 1;
594
0
    color0Mask = (Guchar)((color0Mask << 7) | (color0Mask >> 1));
595
0
    break;
596
0
  case splashModeMono8:
597
0
    c0[0] = *color0Ptr++;
598
0
    break;
599
0
  case splashModeRGB8:
600
0
    c0[0] = color0Ptr[0];
601
0
    c0[1] = color0Ptr[1];
602
0
    c0[2] = color0Ptr[2];
603
0
    color0Ptr += 3;
604
0
    break;
605
0
  case splashModeBGR8:
606
0
    c0[2] = color0Ptr[0];
607
0
    c0[1] = color0Ptr[1];
608
0
    c0[0] = color0Ptr[2];
609
0
    color0Ptr += 3;
610
0
    break;
611
0
#if SPLASH_CMYK
612
0
  case splashModeCMYK8:
613
0
    c0[0] = color0Ptr[0];
614
0
    c0[1] = color0Ptr[1];
615
0
    c0[2] = color0Ptr[2];
616
0
    c0[3] = color0Ptr[3];
617
0
    color0Ptr += 4;
618
0
    break;
619
0
#endif
620
0
  }
621
0
  cBlendIn = c0;
622
1.70k
      } else {
623
1.70k
  cBlendIn = cDest;
624
1.70k
      }
625
626
      //----- read destination shape and alpha
627
628
1.70k
      if (destShapePtr) {
629
0
  fDest = *destShapePtr;
630
1.70k
      } else {
631
1.70k
  fDest = 0xff;
632
1.70k
      }
633
634
1.70k
      if (destAlphaPtr) {
635
1.70k
  aDest = *destAlphaPtr;
636
1.70k
      } else {
637
0
  aDest = 0xff;
638
0
      }
639
640
      //----- read source color; handle overprint
641
642
1.70k
      if (pipe->srcOverprintMaskPtr) {
643
0
  overprintMask = *pipe->srcOverprintMaskPtr++;
644
1.70k
      } else {
645
1.70k
  overprintMask = state->overprintMask;
646
1.70k
      }
647
1.70k
      if (overprintMaskPtr) {
648
0
  *overprintMaskPtr++ |= overprintMask;
649
0
      }
650
651
1.70k
      switch (bitmap->mode) {
652
0
      case splashModeMono1:
653
0
      case splashModeMono8:
654
0
  cSrc[0] = state->grayTransfer[cSrcPtr[0]];
655
0
  break;
656
1.70k
      case splashModeRGB8:
657
1.70k
      case splashModeBGR8:
658
1.70k
  cSrc[0] = state->rgbTransferR[cSrcPtr[0]];
659
1.70k
  cSrc[1] = state->rgbTransferG[cSrcPtr[1]];
660
1.70k
  cSrc[2] = state->rgbTransferB[cSrcPtr[2]];
661
1.70k
  break;
662
0
#if SPLASH_CMYK
663
0
      case splashModeCMYK8:
664
0
  if (alpha0Ptr) {   // non-isolated group
665
0
    if (color0Ptr) { //   non-isolated, knockout group
666
0
      aPrev = *alpha0Ptr;
667
0
    } else {         //   non-isolated, non-knockout group
668
0
      aPrev = (Guchar)(*alpha0Ptr + aDest - div255(*alpha0Ptr * aDest));
669
0
    }
670
0
  } else {           // isolated group
671
0
    if (color0Ptr) { //   isolated, knockout group
672
0
      aPrev = 0;
673
0
    } else {         //   isolated, non-knockout group
674
0
      aPrev = aDest;
675
0
    }
676
0
  }
677
0
  if (overprintMask & 0x01) {
678
0
    cSrc[0] = state->cmykTransferC[cSrcPtr[0]];
679
0
  } else {
680
0
    cSrc[0] = div255(aPrev * cDest[0]);
681
0
  }
682
0
  if (overprintMask & 0x02) {
683
0
    cSrc[1] = state->cmykTransferM[cSrcPtr[1]];
684
0
  } else {
685
0
    cSrc[1] = div255(aPrev * cDest[1]);
686
0
  }
687
0
  if (overprintMask & 0x04) {
688
0
    cSrc[2] = state->cmykTransferY[cSrcPtr[2]];
689
0
  } else {
690
0
    cSrc[2] = div255(aPrev * cDest[2]);
691
0
  }
692
0
  if (overprintMask & 0x08) {
693
0
    cSrc[3] = state->cmykTransferK[cSrcPtr[3]];
694
0
  } else {
695
0
    cSrc[3] = div255(aPrev * cDest[3]);
696
0
  }
697
0
  break;
698
1.70k
#endif
699
1.70k
      }
700
701
      //----- source shape and alpha
702
703
1.70k
      fSrc = shape;
704
1.70k
      aSrc = div255(pipe->aInput * alpha);
705
1.70k
      if (softMaskPtr) {
706
113
  if (pipe->alphaIsShape) {
707
0
    fSrc = div255(fSrc * *softMaskPtr);
708
0
  }
709
113
  aSrc = div255(aSrc * *softMaskPtr);
710
113
  ++softMaskPtr;
711
113
      }
712
713
      //----- non-isolated group correction
714
715
1.70k
      if (pipe->nonIsolatedGroup && alpha != 0) {
716
  // This path is only used when Splash::composite() is called to
717
  // composite a non-isolated group onto the backdrop.  In this
718
  // case, alpha is the source (group) alpha.
719
  //
720
  // In a nested non-isolated group, i.e., if the destination is
721
  // also a non-isolated group (state->inNonIsolatedGroup), we
722
  // need to compute the corrected alpha, because the
723
  // destination is is storing group alpha (same computation as
724
  // blitCorrectedAlpha).
725
1.30k
  if (alpha0Ptr) {
726
0
    t = *alpha0Ptr;
727
0
    t = (Guchar)(aDest + t - div255(aDest * t));
728
1.30k
  } else {
729
1.30k
    t = aDest;
730
1.30k
  }
731
1.30k
  t = (t * 255) / alpha - t;
732
1.30k
  switch (bitmap->mode) {
733
0
#if SPLASH_CMYK
734
0
  case splashModeCMYK8:
735
0
    cSrc[3] = clip255(cSrc[3] + ((cSrc[3] - cDest[3]) * t) / 255);
736
0
#endif
737
1.30k
  case splashModeRGB8:
738
1.30k
  case splashModeBGR8:
739
1.30k
    cSrc[2] = clip255(cSrc[2] + ((cSrc[2] - cDest[2]) * t) / 255);
740
1.30k
    cSrc[1] = clip255(cSrc[1] + ((cSrc[1] - cDest[1]) * t) / 255);
741
1.30k
  case splashModeMono1:
742
1.30k
  case splashModeMono8:
743
1.30k
    cSrc[0] = clip255(cSrc[0] + ((cSrc[0] - cDest[0]) * t) / 255);
744
1.30k
    break;
745
1.30k
  }
746
1.30k
      }
747
748
      //----- blend function
749
750
1.70k
      if (state->blendFunc) {
751
486
#if SPLASH_CMYK
752
486
  if (bitmap->mode == splashModeCMYK8) {
753
    // convert colors to additive
754
0
    cSrc2[0] = (Guchar)(0xff - cSrc[0]);
755
0
    cSrc2[1] = (Guchar)(0xff - cSrc[1]);
756
0
    cSrc2[2] = (Guchar)(0xff - cSrc[2]);
757
0
    cSrc2[3] = (Guchar)(0xff - cSrc[3]);
758
0
    cBlendIn2[0] = (Guchar)(0xff - cBlendIn[0]);
759
0
    cBlendIn2[1] = (Guchar)(0xff - cBlendIn[1]);
760
0
    cBlendIn2[2] = (Guchar)(0xff - cBlendIn[2]);
761
0
    cBlendIn2[3] = (Guchar)(0xff - cBlendIn[3]);
762
0
    (*state->blendFunc)(cSrc2, cBlendIn2, cBlend, bitmap->mode);
763
    // convert result back to subtractive
764
0
    cBlend[0] = (Guchar)(0xff - cBlend[0]);
765
0
    cBlend[1] = (Guchar)(0xff - cBlend[1]);
766
0
    cBlend[2] = (Guchar)(0xff - cBlend[2]);
767
0
    cBlend[3] = (Guchar)(0xff - cBlend[3]);
768
0
  } else
769
486
#endif
770
486
  (*state->blendFunc)(cSrc, cBlendIn, cBlend, bitmap->mode);
771
486
      }
772
773
      //----- result shape
774
775
1.70k
      fResult = (Guchar)(fSrc + fDest - div255(fSrc * fDest));
776
777
      //----- result alpha
778
      //      and setup for result color
779
780
      // alphaI = alpha_i
781
      // alphaIm1 = alpha_(i-1)
782
783
1.70k
      if (pipe->noTransparency) {
784
0
  alphaI = aResult = 255;
785
0
  m1 = m2 = m3 = m4 = 0; // make gcc happy
786
1.70k
      } else if (alpha0Ptr) {     // non-isolated
787
115
  alpha0 = *alpha0Ptr++;
788
115
  if (color0Ptr) {          // non-isolated, knockout
789
0
    aResult = (Guchar)(div255((255 - fSrc) * aDest) + aSrc);
790
0
    alphaI = (Guchar)(alpha0 + aResult - div255(alpha0 * aResult));
791
0
    alphaIm1 = (Guchar)(alpha0 + aDest - div255(alpha0 * aDest));
792
0
    m1 = div255((255 - fSrc) * alphaIm1);
793
0
    m2 = div255((fSrc - aSrc) * alpha0);
794
0
    m3 = div255(aSrc * (255 - alpha0));
795
0
    m4 = div255(aSrc * alpha0);
796
115
  } else {                  // non-isolated, non-knockout
797
115
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
798
115
    alphaI = (Guchar)(alpha0 + aResult - div255(alpha0 * aResult));
799
115
    alphaIm1 = (Guchar)(alpha0 + aDest - div255(alpha0 * aDest));
800
115
    m1 = alphaI - aSrc;
801
115
    m2 = 0;
802
115
    m3 = div255(aSrc * (255 - alphaIm1));
803
115
    m4 = div255(aSrc * alphaIm1);
804
115
  }
805
1.58k
      } else {
806
1.58k
  if (color0Ptr) {          // isolated, knockout
807
0
    aResult = (Guchar)(div255((255 - fSrc) * aDest) + aSrc);
808
0
    alphaI = aResult;
809
0
    alphaIm1 = aDest;
810
0
    m1 = div255((255 - fSrc) * alphaIm1);
811
0
    m2 = 0;
812
0
    m3 = aSrc;
813
0
    m4 = 0;
814
1.58k
  } else {                  // isolated, non-knockout
815
1.58k
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
816
1.58k
    alphaI = aResult;
817
1.58k
    alphaIm1 = aDest;
818
1.58k
    m1 = alphaI - aSrc;
819
1.58k
    m2 = 0;
820
1.58k
    m3 = div255(aSrc * (255 - alphaIm1));
821
1.58k
    m4 = div255(aSrc * alphaIm1);
822
1.58k
  }
823
1.58k
      }
824
825
      //----- result color
826
827
1.70k
      switch (pipe->resultColorCtrl) {
828
829
0
      case splashPipeResultColorNoAlphaBlendMono:
830
0
  cResult0 = div255((255 - aDest) * cSrc[0] + aDest * cBlend[0]);
831
0
  break;
832
0
      case splashPipeResultColorNoAlphaBlendRGB:
833
0
  cResult0 = div255((255 - aDest) * cSrc[0] + aDest * cBlend[0]);
834
0
  cResult1 = div255((255 - aDest) * cSrc[1] + aDest * cBlend[1]);
835
0
  cResult2 = div255((255 - aDest) * cSrc[2] + aDest * cBlend[2]);
836
0
  break;
837
0
#if SPLASH_CMYK
838
0
      case splashPipeResultColorNoAlphaBlendCMYK:
839
0
  cResult0 = div255((255 - aDest) * cSrc[0] + aDest * cBlend[0]);
840
0
  cResult1 = div255((255 - aDest) * cSrc[1] + aDest * cBlend[1]);
841
0
  cResult2 = div255((255 - aDest) * cSrc[2] + aDest * cBlend[2]);
842
0
  cResult3 = div255((255 - aDest) * cSrc[3] + aDest * cBlend[3]);
843
0
  break;
844
0
#endif
845
846
0
      case splashPipeResultColorAlphaNoBlendMono:
847
0
  if (alphaI == 0) {
848
0
    cResult0 = 0;
849
0
  } else {
850
0
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] + aSrc * cSrc[0])
851
0
            / alphaI);
852
0
  }
853
0
  break;
854
1.21k
      case splashPipeResultColorAlphaNoBlendRGB:
855
1.21k
  if (alphaI == 0) {
856
18
    cResult0 = 0;
857
18
    cResult1 = 0;
858
18
    cResult2 = 0;
859
1.19k
  } else {
860
1.19k
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] + aSrc * cSrc[0])
861
1.19k
            / alphaI);
862
1.19k
    cResult1 = (Guchar)((m1 * cDest[1] + m2 * c0[1] + aSrc * cSrc[1])
863
1.19k
            / alphaI);
864
1.19k
    cResult2 = (Guchar)((m1 * cDest[2] + m2 * c0[2] + aSrc * cSrc[2])
865
1.19k
            / alphaI);
866
1.19k
  }
867
1.21k
  break;
868
0
#if SPLASH_CMYK
869
0
      case splashPipeResultColorAlphaNoBlendCMYK:
870
0
  if (alphaI == 0) {
871
0
    cResult0 = 0;
872
0
    cResult1 = 0;
873
0
    cResult2 = 0;
874
0
    cResult3 = 0;
875
0
  } else {
876
0
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] + aSrc * cSrc[0])
877
0
            / alphaI);
878
0
    cResult1 = (Guchar)((m1 * cDest[1] + m2 * c0[1] + aSrc * cSrc[1])
879
0
            / alphaI);
880
0
    cResult2 = (Guchar)((m1 * cDest[2] + m2 * c0[2] + aSrc * cSrc[2])
881
0
            / alphaI);
882
0
    cResult3 = (Guchar)((m1 * cDest[3] + m2 * c0[2] + aSrc * cSrc[3])
883
0
            / alphaI);
884
0
  }
885
0
  break;
886
0
#endif
887
888
0
      case splashPipeResultColorAlphaBlendMono:
889
0
  if (alphaI == 0) {
890
0
    cResult0 = 0;
891
0
  } else {
892
0
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] +
893
0
             m3 * cSrc[0] + m4 * cBlend[0])
894
0
            / alphaI);
895
0
  }
896
0
  break;
897
486
      case splashPipeResultColorAlphaBlendRGB:
898
486
  if (alphaI == 0) {
899
24
    cResult0 = 0;
900
24
    cResult1 = 0;
901
24
    cResult2 = 0;
902
462
  } else {
903
462
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] +
904
462
             m3 * cSrc[0] + m4 * cBlend[0])
905
462
            / alphaI);
906
462
    cResult1 = (Guchar)((m1 * cDest[1] + m2 * c0[1] +
907
462
             m3 * cSrc[1] + m4 * cBlend[1])
908
462
            / alphaI);
909
462
    cResult2 = (Guchar)((m1 * cDest[2] + m2 * c0[2] +
910
462
             m3 * cSrc[2] + m4 * cBlend[2])
911
462
            / alphaI);
912
462
  }
913
486
  break;
914
0
#if SPLASH_CMYK
915
0
      case splashPipeResultColorAlphaBlendCMYK:
916
0
  if (alphaI == 0) {
917
0
    cResult0 = 0;
918
0
    cResult1 = 0;
919
0
    cResult2 = 0;
920
0
    cResult3 = 0;
921
0
  } else {
922
0
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] +
923
0
             m3 * cSrc[0] + m4 * cBlend[0])
924
0
            / alphaI);
925
0
    cResult1 = (Guchar)((m1 * cDest[1] + m2 * c0[1] +
926
0
             m3 * cSrc[1] + m4 * cBlend[1])
927
0
            / alphaI);
928
0
    cResult2 = (Guchar)((m1 * cDest[2] + m2 * c0[2] +
929
0
             m3 * cSrc[2] + m4 * cBlend[2])
930
0
            / alphaI);
931
0
    cResult3 = (Guchar)((m1 * cDest[3] + m2 * c0[3] +
932
0
             m3 * cSrc[3] + m4 * cBlend[3])
933
0
            / alphaI);
934
0
  }
935
0
  break;
936
1.70k
#endif
937
1.70k
      }
938
939
1.70k
    } // if (noTransparency && !blendFunc)
940
941
    //----- write destination pixel
942
943
1.70k
    switch (bitmap->mode) {
944
0
    case splashModeMono1:
945
0
      if (state->screen->test(x, y, cResult0)) {
946
0
  *destColorPtr |= destColorMask;
947
0
      } else {
948
0
  *destColorPtr &= (Guchar)~destColorMask;
949
0
      }
950
0
      destColorPtr += destColorMask & 1;
951
0
      destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
952
0
      break;
953
0
    case splashModeMono8:
954
0
      *destColorPtr++ = cResult0;
955
0
      break;
956
1.70k
    case splashModeRGB8:
957
1.70k
      destColorPtr[0] = cResult0;
958
1.70k
      destColorPtr[1] = cResult1;
959
1.70k
      destColorPtr[2] = cResult2;
960
1.70k
      destColorPtr += 3;
961
1.70k
      break;
962
0
    case splashModeBGR8:
963
0
      destColorPtr[0] = cResult2;
964
0
      destColorPtr[1] = cResult1;
965
0
      destColorPtr[2] = cResult0;
966
0
      destColorPtr += 3;
967
0
      break;
968
0
#if SPLASH_CMYK
969
0
    case splashModeCMYK8:
970
0
      destColorPtr[0] = cResult0;
971
0
      destColorPtr[1] = cResult1;
972
0
      destColorPtr[2] = cResult2;
973
0
      destColorPtr[3] = cResult3;
974
0
      destColorPtr += 4;
975
0
      break;
976
1.70k
#endif
977
1.70k
    }
978
1.70k
    if (destShapePtr) {
979
0
      *destShapePtr++ = fResult;
980
0
    }
981
1.70k
    if (destAlphaPtr) {
982
1.70k
      *destAlphaPtr++ = aResult;
983
1.70k
    }
984
985
1.70k
    cSrcPtr += cSrcStride;
986
1.70k
    shapePtr2 += shapeStride;
987
1.70k
    alphaPtr2 += alphaStride;
988
1.70k
  } // for (x ...)
989
990
1.70k
  updateModX(lastX);
991
1.70k
}
992
993
// special case:
994
// !pipe->pattern && pipe->noTransparency && !state->blendFunc &&
995
// bitmap->mode == splashModeMono1 && !bitmap->alpha) {
996
void Splash::pipeRunSimpleMono1(SplashPipe *pipe, int x0, int x1, int y,
997
        Guchar *shapePtr, Guchar *alphaPtr,
998
0
        SplashColorPtr cSrcPtr) {
999
0
  Guchar cResult0;
1000
0
  SplashColorPtr destColorPtr;
1001
0
  Guchar destColorMask;
1002
0
  SplashScreenCursor screenCursor;
1003
0
  int cSrcStride, x;
1004
1005
0
  if (cSrcPtr) {
1006
0
    cSrcStride = 1;
1007
0
  } else {
1008
0
    cSrcPtr = pipe->cSrcVal;
1009
0
    cSrcStride = 0;
1010
0
  }
1011
0
  if (x0 > x1) {
1012
0
    return;
1013
0
  }
1014
0
  updateModX(x0);
1015
0
  updateModX(x1);
1016
0
  updateModY(y);
1017
1018
0
  useBitmapRow(y);
1019
1020
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + (x0 >> 3)];
1021
0
  destColorMask = (Guchar)(0x80 >> (x0 & 7));
1022
1023
0
  screenCursor = state->screen->getTestCursor(y);
1024
1025
0
  for (x = x0; x <= x1; ++x) {
1026
1027
    //----- write destination pixel
1028
0
    cResult0 = state->grayTransfer[cSrcPtr[0]];
1029
0
    if (state->screen->testWithCursor(screenCursor, x, cResult0)) {
1030
0
      *destColorPtr |= destColorMask;
1031
0
    } else {
1032
0
      *destColorPtr &= (Guchar)~destColorMask;
1033
0
    }
1034
0
    destColorPtr += destColorMask & 1;
1035
0
    destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1036
1037
0
    cSrcPtr += cSrcStride;
1038
0
  }
1039
0
}
1040
1041
// special case:
1042
// !pipe->pattern && pipe->noTransparency && !state->blendFunc &&
1043
// bitmap->mode == splashModeMono8 && bitmap->alpha) {
1044
void Splash::pipeRunSimpleMono8(SplashPipe *pipe, int x0, int x1, int y,
1045
        Guchar *shapePtr, Guchar *alphaPtr,
1046
0
        SplashColorPtr cSrcPtr) {
1047
0
  SplashColorPtr destColorPtr;
1048
0
  Guchar *destAlphaPtr;
1049
0
  int cSrcStride, x;
1050
1051
0
  if (cSrcPtr) {
1052
0
    cSrcStride = 1;
1053
0
  } else {
1054
0
    cSrcPtr = pipe->cSrcVal;
1055
0
    cSrcStride = 0;
1056
0
  }
1057
0
  if (x0 > x1) {
1058
0
    return;
1059
0
  }
1060
0
  updateModX(x0);
1061
0
  updateModX(x1);
1062
0
  updateModY(y);
1063
1064
0
  useBitmapRow(y);
1065
1066
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
1067
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1068
1069
0
  for (x = x0; x <= x1; ++x) {
1070
1071
    //----- write destination pixel
1072
0
    *destColorPtr++ = state->grayTransfer[cSrcPtr[0]];
1073
0
    *destAlphaPtr++ = 255;
1074
1075
0
    cSrcPtr += cSrcStride;
1076
0
  }
1077
0
}
1078
1079
// special case:
1080
// !pipe->pattern && pipe->noTransparency && !state->blendFunc &&
1081
// bitmap->mode == splashModeRGB8 && bitmap->alpha) {
1082
void Splash::pipeRunSimpleRGB8(SplashPipe *pipe, int x0, int x1, int y,
1083
             Guchar *shapePtr, Guchar *alphaPtr,
1084
0
             SplashColorPtr cSrcPtr) {
1085
0
  SplashColorPtr destColorPtr;
1086
0
  Guchar *destAlphaPtr;
1087
0
  int cSrcStride, x;
1088
1089
0
  if (cSrcPtr) {
1090
0
    cSrcStride = 3;
1091
0
  } else {
1092
0
    cSrcPtr = pipe->cSrcVal;
1093
0
    cSrcStride = 0;
1094
0
  }
1095
0
  if (x0 > x1) {
1096
0
    return;
1097
0
  }
1098
0
  updateModX(x0);
1099
0
  updateModX(x1);
1100
0
  updateModY(y);
1101
1102
0
  useBitmapRow(y);
1103
1104
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1105
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1106
1107
0
  for (x = x0; x <= x1; ++x) {
1108
1109
    //----- write destination pixel
1110
0
    destColorPtr[0] = state->rgbTransferR[cSrcPtr[0]];
1111
0
    destColorPtr[1] = state->rgbTransferG[cSrcPtr[1]];
1112
0
    destColorPtr[2] = state->rgbTransferB[cSrcPtr[2]];
1113
0
    destColorPtr += 3;
1114
0
    *destAlphaPtr++ = 255;
1115
1116
0
    cSrcPtr += cSrcStride;
1117
0
  }
1118
0
}
1119
1120
// special case:
1121
// !pipe->pattern && pipe->noTransparency && !state->blendFunc &&
1122
// bitmap->mode == splashModeBGR8 && bitmap->alpha) {
1123
void Splash::pipeRunSimpleBGR8(SplashPipe *pipe, int x0, int x1, int y,
1124
             Guchar *shapePtr, Guchar *alphaPtr,
1125
0
             SplashColorPtr cSrcPtr) {
1126
0
  SplashColorPtr destColorPtr;
1127
0
  Guchar *destAlphaPtr;
1128
0
  int cSrcStride, x;
1129
1130
0
  if (cSrcPtr) {
1131
0
    cSrcStride = 3;
1132
0
  } else {
1133
0
    cSrcPtr = pipe->cSrcVal;
1134
0
    cSrcStride = 0;
1135
0
  }
1136
0
  if (x0 > x1) {
1137
0
    return;
1138
0
  }
1139
0
  updateModX(x0);
1140
0
  updateModX(x1);
1141
0
  updateModY(y);
1142
1143
0
  useBitmapRow(y);
1144
1145
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1146
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1147
1148
0
  for (x = x0; x <= x1; ++x) {
1149
1150
    //----- write destination pixel
1151
0
    destColorPtr[0] = state->rgbTransferB[cSrcPtr[2]];
1152
0
    destColorPtr[1] = state->rgbTransferG[cSrcPtr[1]];
1153
0
    destColorPtr[2] = state->rgbTransferR[cSrcPtr[0]];
1154
0
    destColorPtr += 3;
1155
0
    *destAlphaPtr++ = 255;
1156
1157
0
    cSrcPtr += cSrcStride;
1158
0
  }
1159
0
}
1160
1161
#if SPLASH_CMYK
1162
// special case:
1163
// !pipe->pattern && pipe->noTransparency && !state->blendFunc &&
1164
// bitmap->mode == splashModeCMYK8 && bitmap->alpha) {
1165
void Splash::pipeRunSimpleCMYK8(SplashPipe *pipe, int x0, int x1, int y,
1166
        Guchar *shapePtr, Guchar *alphaPtr,
1167
0
        SplashColorPtr cSrcPtr) {
1168
0
  SplashColorPtr destColorPtr;
1169
0
  Guchar *destAlphaPtr;
1170
0
  int cSrcStride, x;
1171
1172
0
  if (cSrcPtr) {
1173
0
    cSrcStride = 4;
1174
0
  } else {
1175
0
    cSrcPtr = pipe->cSrcVal;
1176
0
    cSrcStride = 0;
1177
0
  }
1178
0
  if (x0 > x1) {
1179
0
    return;
1180
0
  }
1181
0
  updateModX(x0);
1182
0
  updateModX(x1);
1183
0
  updateModY(y);
1184
1185
0
  useBitmapRow(y);
1186
1187
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 4 * x0];
1188
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1189
1190
0
  for (x = x0; x <= x1; ++x) {
1191
1192
    //----- write destination pixel
1193
0
    destColorPtr[0] = state->cmykTransferC[cSrcPtr[0]];
1194
0
    destColorPtr[1] = state->cmykTransferM[cSrcPtr[1]];
1195
0
    destColorPtr[2] = state->cmykTransferY[cSrcPtr[2]];
1196
0
    destColorPtr[3] = state->cmykTransferK[cSrcPtr[3]];
1197
0
    destColorPtr += 4;
1198
0
    *destAlphaPtr++ = 255;
1199
1200
0
    cSrcPtr += cSrcStride;
1201
0
  }
1202
0
}
1203
#endif
1204
1205
1206
// special case:
1207
// !pipe->pattern && pipe->alphaOnly && !state->blendFunc &&
1208
// bitmap->mode == splashModeMono1 && !bitmap->alpha
1209
void Splash::pipeRunShapeMono1(SplashPipe *pipe, int x0, int x1, int y,
1210
             Guchar *shapePtr, Guchar *alphaPtr,
1211
47.3k
             SplashColorPtr cSrcPtr) {
1212
47.3k
  Guchar alpha, aSrc, cSrc0, cDest0, cResult0;
1213
47.3k
  SplashColorPtr destColorPtr;
1214
47.3k
  Guchar destColorMask;
1215
47.3k
  SplashScreenCursor screenCursor;
1216
47.3k
  int cSrcStride, x, lastX;
1217
1218
47.3k
  if (cSrcPtr) {
1219
0
    cSrcStride = 1;
1220
47.3k
  } else {
1221
47.3k
    cSrcPtr = pipe->cSrcVal;
1222
47.3k
    cSrcStride = 0;
1223
47.3k
  }
1224
34.1M
  for (; x0 <= x1; ++x0) {
1225
34.1M
    if (*alphaPtr) {
1226
15.2k
      break;
1227
15.2k
    }
1228
34.0M
    cSrcPtr += cSrcStride;
1229
34.0M
    ++alphaPtr;
1230
34.0M
  }
1231
47.3k
  if (x0 > x1) {
1232
32.1k
    return;
1233
32.1k
  }
1234
15.2k
  updateModX(x0);
1235
15.2k
  updateModY(y);
1236
15.2k
  lastX = x0;
1237
1238
15.2k
  useBitmapRow(y);
1239
1240
15.2k
  destColorPtr = &bitmap->data[y * bitmap->rowSize + (x0 >> 3)];
1241
15.2k
  destColorMask = (Guchar)(0x80 >> (x0 & 7));
1242
1243
15.2k
  screenCursor = state->screen->getTestCursor(y);
1244
1245
10.6M
  for (x = x0; x <= x1; ++x) {
1246
1247
    //----- alpha
1248
10.6M
    alpha = *alphaPtr;
1249
10.6M
    if (!alpha) {
1250
3.12M
      destColorPtr += destColorMask & 1;
1251
3.12M
      destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1252
3.12M
      cSrcPtr += cSrcStride;
1253
3.12M
      ++alphaPtr;
1254
3.12M
      continue;
1255
3.12M
    }
1256
7.49M
    lastX = x;
1257
1258
    //----- source color
1259
7.49M
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1260
1261
    //----- source alpha
1262
7.49M
    aSrc = alpha;
1263
1264
    //----- special case for aSrc = 255
1265
7.49M
    if (aSrc == 255) {
1266
7.49M
      cResult0 = cSrc0;
1267
7.49M
    } else {
1268
1269
      //----- read destination pixel
1270
0
      cDest0 = (*destColorPtr & destColorMask) ? 0xff : 0x00;
1271
1272
      //----- result color
1273
      // note: aDest = alphaI = aResult = 0xff
1274
0
      cResult0 = (Guchar)div255((0xff - aSrc) * cDest0 + aSrc * cSrc0);
1275
0
    }
1276
1277
    //----- write destination pixel
1278
7.49M
    if (state->screen->testWithCursor(screenCursor, x, cResult0)) {
1279
7.49M
      *destColorPtr |= destColorMask;
1280
7.49M
    } else {
1281
0
      *destColorPtr &= (Guchar)~destColorMask;
1282
0
    }
1283
7.49M
    destColorPtr += destColorMask & 1;
1284
7.49M
    destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1285
1286
7.49M
    cSrcPtr += cSrcStride;
1287
7.49M
    ++alphaPtr;
1288
7.49M
  }
1289
1290
15.2k
  updateModX(lastX);
1291
15.2k
}
1292
1293
// special case:
1294
// !pipe->pattern && pipe->alphaOnly && !state->blendFunc &&
1295
// bitmap->mode == splashModeMono8 && bitmap->alpha
1296
void Splash::pipeRunShapeMono8(SplashPipe *pipe, int x0, int x1, int y,
1297
             Guchar *shapePtr, Guchar *alphaPtr,
1298
483
             SplashColorPtr cSrcPtr) {
1299
483
  Guchar alpha, aSrc, aDest, alphaI, aResult, cSrc0, cDest0, cResult0;
1300
483
  SplashColorPtr destColorPtr;
1301
483
  Guchar *destAlphaPtr;
1302
483
  int cSrcStride, x, lastX;
1303
1304
483
  if (cSrcPtr) {
1305
0
    cSrcStride = 1;
1306
483
  } else {
1307
483
    cSrcPtr = pipe->cSrcVal;
1308
483
    cSrcStride = 0;
1309
483
  }
1310
506
  for (; x0 <= x1; ++x0) {
1311
483
    if (*alphaPtr) {
1312
460
      break;
1313
460
    }
1314
23
    cSrcPtr += cSrcStride;
1315
23
    ++alphaPtr;
1316
23
  }
1317
483
  if (x0 > x1) {
1318
23
    return;
1319
23
  }
1320
460
  updateModX(x0);
1321
460
  updateModY(y);
1322
460
  lastX = x0;
1323
1324
460
  useBitmapRow(y);
1325
1326
460
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
1327
460
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1328
1329
920
  for (x = x0; x <= x1; ++x) {
1330
1331
    //----- alpha
1332
460
    alpha = *alphaPtr;
1333
460
    if (!alpha) {
1334
0
      ++destColorPtr;
1335
0
      ++destAlphaPtr;
1336
0
      cSrcPtr += cSrcStride;
1337
0
      ++alphaPtr;
1338
0
      continue;
1339
0
    }
1340
460
    lastX = x;
1341
1342
    //----- source color
1343
460
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1344
1345
    //----- source alpha
1346
460
    aSrc = alpha;
1347
1348
    //----- special case for aSrc = 255
1349
460
    if (aSrc == 255) {
1350
420
      aResult = 255;
1351
420
      cResult0 = cSrc0;
1352
420
    } else {
1353
1354
      //----- read destination alpha
1355
40
      aDest = *destAlphaPtr;
1356
1357
      //----- special case for aDest = 0
1358
40
      if (aDest == 0) {
1359
40
  aResult = aSrc;
1360
40
  cResult0 = cSrc0;
1361
40
      } else {
1362
1363
  //----- read destination pixel
1364
0
  cDest0 = *destColorPtr;
1365
1366
  //----- result alpha and non-isolated group element correction
1367
0
  aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
1368
0
  alphaI = aResult;
1369
1370
  //----- result color
1371
0
  cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
1372
0
      }
1373
40
    }
1374
1375
    //----- write destination pixel
1376
460
    *destColorPtr++ = cResult0;
1377
460
    *destAlphaPtr++ = aResult;
1378
1379
460
    cSrcPtr += cSrcStride;
1380
460
    ++alphaPtr;
1381
460
  }
1382
1383
460
  updateModX(lastX);
1384
460
}
1385
1386
// special case:
1387
// !pipe->pattern && pipe->alphaOnly && !state->blendFunc &&
1388
// bitmap->mode == splashModeRGB8 && bitmap->alpha
1389
void Splash::pipeRunShapeRGB8(SplashPipe *pipe, int x0, int x1, int y,
1390
            Guchar *shapePtr, Guchar *alphaPtr,
1391
8.62k
            SplashColorPtr cSrcPtr) {
1392
8.62k
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1393
8.62k
  Guchar cSrc0, cSrc1, cSrc2;
1394
8.62k
  Guchar cDest0, cDest1, cDest2;
1395
8.62k
  Guchar cResult0, cResult1, cResult2;
1396
8.62k
  SplashColorPtr destColorPtr;
1397
8.62k
  Guchar *destAlphaPtr;
1398
8.62k
  int cSrcStride, x, lastX;
1399
1400
8.62k
  if (cSrcPtr) {
1401
208
    cSrcStride = 3;
1402
8.41k
  } else {
1403
8.41k
    cSrcPtr = pipe->cSrcVal;
1404
8.41k
    cSrcStride = 0;
1405
8.41k
  }
1406
9.96k
  for (; x0 <= x1; ++x0) {
1407
8.62k
    if (*alphaPtr) {
1408
7.28k
      break;
1409
7.28k
    }
1410
1.34k
    cSrcPtr += cSrcStride;
1411
1.34k
    ++alphaPtr;
1412
1.34k
  }
1413
8.62k
  if (x0 > x1) {
1414
1.34k
    return;
1415
1.34k
  }
1416
7.28k
  updateModX(x0);
1417
7.28k
  updateModY(y);
1418
7.28k
  lastX = x0;
1419
1420
7.28k
  useBitmapRow(y);
1421
1422
7.28k
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1423
7.28k
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1424
1425
14.5k
  for (x = x0; x <= x1; ++x) {
1426
1427
    //----- alpha
1428
7.28k
    alpha = *alphaPtr;
1429
7.28k
    if (!alpha) {
1430
0
      destColorPtr += 3;
1431
0
      ++destAlphaPtr;
1432
0
      cSrcPtr += cSrcStride;
1433
0
      ++alphaPtr;
1434
0
      continue;
1435
0
    }
1436
7.28k
    lastX = x;
1437
1438
    //----- source color
1439
7.28k
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
1440
7.28k
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
1441
7.28k
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
1442
1443
    //----- source alpha
1444
7.28k
    aSrc = alpha;
1445
1446
    //----- special case for aSrc = 255
1447
7.28k
    if (aSrc == 255) {
1448
5.02k
      aResult = 255;
1449
5.02k
      cResult0 = cSrc0;
1450
5.02k
      cResult1 = cSrc1;
1451
5.02k
      cResult2 = cSrc2;
1452
5.02k
    } else {
1453
1454
      //----- read destination alpha
1455
2.25k
      aDest = *destAlphaPtr;
1456
1457
      //----- special case for aDest = 0
1458
2.25k
      if (aDest == 0) {
1459
450
  aResult = aSrc;
1460
450
  cResult0 = cSrc0;
1461
450
  cResult1 = cSrc1;
1462
450
  cResult2 = cSrc2;
1463
1.80k
      } else {
1464
1465
  //----- read destination pixel
1466
1.80k
  cDest0 = destColorPtr[0];
1467
1.80k
  cDest1 = destColorPtr[1];
1468
1.80k
  cDest2 = destColorPtr[2];
1469
1470
  //----- result alpha and non-isolated group element correction
1471
1.80k
  aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
1472
1.80k
  alphaI = aResult;
1473
1474
  //----- result color
1475
1.80k
  cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
1476
1.80k
  cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
1477
1.80k
  cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
1478
1.80k
      }
1479
2.25k
    }
1480
1481
    //----- write destination pixel
1482
7.28k
    destColorPtr[0] = cResult0;
1483
7.28k
    destColorPtr[1] = cResult1;
1484
7.28k
    destColorPtr[2] = cResult2;
1485
7.28k
    destColorPtr += 3;
1486
7.28k
    *destAlphaPtr++ = aResult;
1487
1488
7.28k
    cSrcPtr += cSrcStride;
1489
7.28k
    ++alphaPtr;
1490
7.28k
  }
1491
1492
7.28k
  updateModX(lastX);
1493
7.28k
}
1494
1495
// special case:
1496
// !pipe->pattern && pipe->alphaOnly && !state->blendFunc &&
1497
// bitmap->mode == splashModeBGR8 && bitmap->alpha
1498
void Splash::pipeRunShapeBGR8(SplashPipe *pipe, int x0, int x1, int y,
1499
            Guchar *shapePtr, Guchar *alphaPtr,
1500
0
            SplashColorPtr cSrcPtr) {
1501
0
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1502
0
  Guchar cSrc0, cSrc1, cSrc2;
1503
0
  Guchar cDest0, cDest1, cDest2;
1504
0
  Guchar cResult0, cResult1, cResult2;
1505
0
  SplashColorPtr destColorPtr;
1506
0
  Guchar *destAlphaPtr;
1507
0
  int cSrcStride, x, lastX;
1508
1509
0
  if (cSrcPtr) {
1510
0
    cSrcStride = 3;
1511
0
  } else {
1512
0
    cSrcPtr = pipe->cSrcVal;
1513
0
    cSrcStride = 0;
1514
0
  }
1515
0
  for (; x0 <= x1; ++x0) {
1516
0
    if (*alphaPtr) {
1517
0
      break;
1518
0
    }
1519
0
    cSrcPtr += cSrcStride;
1520
0
    ++alphaPtr;
1521
0
  }
1522
0
  if (x0 > x1) {
1523
0
    return;
1524
0
  }
1525
0
  updateModX(x0);
1526
0
  updateModY(y);
1527
0
  lastX = x0;
1528
1529
0
  useBitmapRow(y);
1530
1531
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1532
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1533
1534
0
  for (x = x0; x <= x1; ++x) {
1535
1536
    //----- alpha
1537
0
    alpha = *alphaPtr;
1538
0
    if (!alpha) {
1539
0
      destColorPtr += 3;
1540
0
      ++destAlphaPtr;
1541
0
      cSrcPtr += cSrcStride;
1542
0
      ++alphaPtr;
1543
0
      continue;
1544
0
    }
1545
0
    lastX = x;
1546
1547
    //----- source color
1548
0
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
1549
0
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
1550
0
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
1551
1552
    //----- source alpha
1553
0
    aSrc = alpha;
1554
1555
    //----- special case for aSrc = 255
1556
0
    if (aSrc == 255) {
1557
0
      aResult = 255;
1558
0
      cResult0 = cSrc0;
1559
0
      cResult1 = cSrc1;
1560
0
      cResult2 = cSrc2;
1561
0
    } else {
1562
1563
      //----- read destination alpha
1564
0
      aDest = *destAlphaPtr;
1565
1566
      //----- special case for aDest = 0
1567
0
      if (aDest == 0) {
1568
0
  aResult = aSrc;
1569
0
  cResult0 = cSrc0;
1570
0
  cResult1 = cSrc1;
1571
0
  cResult2 = cSrc2;
1572
0
      } else {
1573
1574
  //----- read destination pixel
1575
0
  cDest0 = destColorPtr[2];
1576
0
  cDest1 = destColorPtr[1];
1577
0
  cDest2 = destColorPtr[0];
1578
1579
  //----- result alpha and non-isolated group element correction
1580
0
  aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
1581
0
  alphaI = aResult;
1582
1583
  //----- result color
1584
0
  cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
1585
0
  cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
1586
0
  cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
1587
0
      }
1588
0
    }
1589
1590
    //----- write destination pixel
1591
0
    destColorPtr[0] = cResult2;
1592
0
    destColorPtr[1] = cResult1;
1593
0
    destColorPtr[2] = cResult0;
1594
0
    destColorPtr += 3;
1595
0
    *destAlphaPtr++ = aResult;
1596
1597
0
    cSrcPtr += cSrcStride;
1598
0
    ++alphaPtr;
1599
0
  }
1600
1601
0
  updateModX(lastX);
1602
0
}
1603
1604
#if SPLASH_CMYK
1605
// special case:
1606
// !pipe->pattern && pipe->alphaOnly && !state->blendFunc &&
1607
// bitmap->mode == splashModeCMYK8 && bitmap->alpha
1608
void Splash::pipeRunShapeCMYK8(SplashPipe *pipe, int x0, int x1, int y,
1609
             Guchar *shapePtr, Guchar *alphaPtr,
1610
0
             SplashColorPtr cSrcPtr) {
1611
0
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1612
0
  Guchar cSrc0, cSrc1, cSrc2, cSrc3;
1613
0
  Guchar cDest0, cDest1, cDest2, cDest3;
1614
0
  Guchar cResult0, cResult1, cResult2, cResult3;
1615
0
  SplashColorPtr destColorPtr;
1616
0
  Guchar *destAlphaPtr;
1617
0
  int cSrcStride, x, lastX;
1618
1619
0
  if (cSrcPtr) {
1620
0
    cSrcStride = 4;
1621
0
  } else {
1622
0
    cSrcPtr = pipe->cSrcVal;
1623
0
    cSrcStride = 0;
1624
0
  }
1625
0
  for (; x0 <= x1; ++x0) {
1626
0
    if (*alphaPtr) {
1627
0
      break;
1628
0
    }
1629
0
    cSrcPtr += cSrcStride;
1630
0
    ++alphaPtr;
1631
0
  }
1632
0
  if (x0 > x1) {
1633
0
    return;
1634
0
  }
1635
0
  updateModX(x0);
1636
0
  updateModY(y);
1637
0
  lastX = x0;
1638
1639
0
  useBitmapRow(y);
1640
1641
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 4 * x0];
1642
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1643
1644
0
  for (x = x0; x <= x1; ++x) {
1645
1646
    //----- alpha
1647
0
    alpha = *alphaPtr;
1648
0
    if (!alpha) {
1649
0
      destColorPtr += 4;
1650
0
      ++destAlphaPtr;
1651
0
      cSrcPtr += cSrcStride;
1652
0
      ++alphaPtr;
1653
0
      continue;
1654
0
    }
1655
0
    lastX = x;
1656
1657
    //----- read destination pixel
1658
0
    cDest0 = destColorPtr[0];
1659
0
    cDest1 = destColorPtr[1];
1660
0
    cDest2 = destColorPtr[2];
1661
0
    cDest3 = destColorPtr[3];
1662
0
    aDest = *destAlphaPtr;
1663
1664
    //----- overprint
1665
0
    cSrc0 = state->cmykTransferC[cSrcPtr[0]];
1666
0
    cSrc1 = state->cmykTransferM[cSrcPtr[1]];
1667
0
    cSrc2 = state->cmykTransferY[cSrcPtr[2]];
1668
0
    cSrc3 = state->cmykTransferK[cSrcPtr[3]];
1669
1670
    //----- source alpha
1671
0
    aSrc = alpha;
1672
1673
    //----- special case for aSrc = 255
1674
0
    if (aSrc == 255) {
1675
0
      aResult = 255;
1676
0
      cResult0 = cSrc0;
1677
0
      cResult1 = cSrc1;
1678
0
      cResult2 = cSrc2;
1679
0
      cResult3 = cSrc3;
1680
0
    } else {
1681
1682
      //----- special case for aDest = 0
1683
0
      if (aDest == 0) {
1684
0
  aResult = aSrc;
1685
0
  cResult0 = cSrc0;
1686
0
  cResult1 = cSrc1;
1687
0
  cResult2 = cSrc2;
1688
0
  cResult3 = cSrc3;
1689
0
      } else {
1690
1691
  //----- result alpha and non-isolated group element correction
1692
0
  aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
1693
0
  alphaI = aResult;
1694
1695
  //----- result color
1696
0
  cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
1697
0
  cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
1698
0
  cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
1699
0
  cResult3 = (Guchar)(((alphaI - aSrc) * cDest3 + aSrc * cSrc3) / alphaI);
1700
0
      }
1701
0
    }
1702
1703
    //----- write destination pixel
1704
0
    destColorPtr[0] = cResult0;
1705
0
    destColorPtr[1] = cResult1;
1706
0
    destColorPtr[2] = cResult2;
1707
0
    destColorPtr[3] = cResult3;
1708
0
    destColorPtr += 4;
1709
0
    *destAlphaPtr++ = aResult;
1710
1711
0
    cSrcPtr += cSrcStride;
1712
0
    ++alphaPtr;
1713
0
  }
1714
1715
0
  updateModX(lastX);
1716
0
}
1717
#endif
1718
1719
1720
// special case:
1721
// !pipe->pattern && pipe->alphaOnly && !state->blendFunc &&
1722
// bitmap->mode == splashModeMono8 && !bitmap->alpha
1723
void Splash::pipeRunShapeNoAlphaMono8(SplashPipe *pipe, int x0, int x1, int y,
1724
                                      Guchar *shapePtr, Guchar *alphaPtr,
1725
1.28k
                                      SplashColorPtr cSrcPtr) {
1726
1.28k
  Guchar alpha, aSrc, cSrc0, cDest0, cResult0;
1727
1.28k
  SplashColorPtr destColorPtr;
1728
1.28k
  int cSrcStride, x, lastX;
1729
1730
1.28k
  if (cSrcPtr) {
1731
172
    cSrcStride = 1;
1732
1.11k
  } else {
1733
1.11k
    cSrcPtr = pipe->cSrcVal;
1734
1.11k
    cSrcStride = 0;
1735
1.11k
  }
1736
21.4k
  for (; x0 <= x1; ++x0) {
1737
20.9k
    if (*alphaPtr) {
1738
808
      break;
1739
808
    }
1740
20.1k
    cSrcPtr += cSrcStride;
1741
20.1k
    ++alphaPtr;
1742
20.1k
  }
1743
1.28k
  if (x0 > x1) {
1744
474
    return;
1745
474
  }
1746
808
  updateModX(x0);
1747
808
  updateModY(y);
1748
808
  lastX = x0;
1749
1750
808
  useBitmapRow(y);
1751
1752
808
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
1753
1754
60.1k
  for (x = x0; x <= x1; ++x) {
1755
1756
    //----- alpha
1757
59.3k
    alpha = *alphaPtr;
1758
59.3k
    if (!alpha) {
1759
0
      ++destColorPtr;
1760
0
      cSrcPtr += cSrcStride;
1761
0
      ++alphaPtr;
1762
0
      continue;
1763
0
    }
1764
59.3k
    lastX = x;
1765
1766
    //----- source color
1767
59.3k
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1768
1769
    //----- source alpha
1770
59.3k
    aSrc = alpha;
1771
1772
    //----- special case for aSrc = 255
1773
59.3k
    if (aSrc == 255) {
1774
30.1k
      cResult0 = cSrc0;
1775
30.1k
    } else {
1776
1777
      //----- read destination pixel
1778
29.2k
      cDest0 = *destColorPtr;
1779
1780
      //----- result color
1781
29.2k
      cResult0 = div255((255 - aSrc) * cDest0 + aSrc * cSrc0);
1782
29.2k
    }
1783
1784
    //----- write destination pixel
1785
59.3k
    *destColorPtr++ = cResult0;
1786
1787
59.3k
    cSrcPtr += cSrcStride;
1788
59.3k
    ++alphaPtr;
1789
59.3k
  }
1790
1791
808
  updateModX(lastX);
1792
808
}
1793
1794
// special case:
1795
// !pipe->pattern && !pipe->noTransparency && !state->softMask &&
1796
// usesAlpha && !pipe->alpha0Ptr && !state->blendFunc &&
1797
// !pipe->nonIsolatedGroup &&
1798
// bitmap->mode == splashModeMono1 && !bitmap->alpha
1799
void Splash::pipeRunAAMono1(SplashPipe *pipe, int x0, int x1, int y,
1800
          Guchar *shapePtr, Guchar *alphaPtr,
1801
0
          SplashColorPtr cSrcPtr) {
1802
0
  Guchar alpha, aSrc, cSrc0, cDest0, cResult0;
1803
0
  SplashColorPtr destColorPtr;
1804
0
  Guchar destColorMask;
1805
0
  SplashScreenCursor screenCursor;
1806
0
  int cSrcStride, x, lastX;
1807
1808
0
  if (cSrcPtr) {
1809
0
    cSrcStride = 1;
1810
0
  } else {
1811
0
    cSrcPtr = pipe->cSrcVal;
1812
0
    cSrcStride = 0;
1813
0
  }
1814
0
  for (; x0 <= x1; ++x0) {
1815
0
    if (*alphaPtr) {
1816
0
      break;
1817
0
    }
1818
0
    cSrcPtr += cSrcStride;
1819
0
    ++alphaPtr;
1820
0
  }
1821
0
  if (x0 > x1) {
1822
0
    return;
1823
0
  }
1824
0
  updateModX(x0);
1825
0
  updateModY(y);
1826
0
  lastX = x0;
1827
1828
0
  useBitmapRow(y);
1829
1830
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + (x0 >> 3)];
1831
0
  destColorMask = (Guchar)(0x80 >> (x0 & 7));
1832
1833
0
  screenCursor = state->screen->getTestCursor(y);
1834
1835
0
  for (x = x0; x <= x1; ++x) {
1836
1837
    //----- alpha
1838
0
    alpha = *alphaPtr;
1839
0
    if (!alpha) {
1840
0
      destColorPtr += destColorMask & 1;
1841
0
      destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1842
0
      cSrcPtr += cSrcStride;
1843
0
      ++alphaPtr;
1844
0
      continue;
1845
0
    }
1846
0
    lastX = x;
1847
1848
    //----- read destination pixel
1849
0
    cDest0 = (*destColorPtr & destColorMask) ? 0xff : 0x00;
1850
1851
    //----- source color
1852
0
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1853
1854
    //----- source alpha
1855
0
    aSrc = div255(pipe->aInput * alpha);
1856
1857
    //----- result color
1858
    // note: aDest = alphaI = aResult = 0xff
1859
0
    cResult0 = (Guchar)div255((0xff - aSrc) * cDest0 + aSrc * cSrc0);
1860
1861
    //----- write destination pixel
1862
0
    if (state->screen->testWithCursor(screenCursor, x, cResult0)) {
1863
0
      *destColorPtr |= destColorMask;
1864
0
    } else {
1865
0
      *destColorPtr &= (Guchar)~destColorMask;
1866
0
    }
1867
0
    destColorPtr += destColorMask & 1;
1868
0
    destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1869
1870
0
    cSrcPtr += cSrcStride;
1871
0
    ++alphaPtr;
1872
0
  }
1873
1874
0
  updateModX(lastX);
1875
0
}
1876
1877
// special case:
1878
// !pipe->pattern && !pipe->noTransparency && !state->softMask &&
1879
// usesAlpha && !pipe->alpha0Ptr && !state->blendFunc &&
1880
// !pipe->nonIsolatedGroup &&
1881
// bitmap->mode == splashModeMono8 && bitmap->alpha
1882
void Splash::pipeRunAAMono8(SplashPipe *pipe, int x0, int x1, int y,
1883
          Guchar *shapePtr, Guchar *alphaPtr,
1884
0
          SplashColorPtr cSrcPtr) {
1885
0
  Guchar alpha, aSrc, aDest, alphaI, aResult, cSrc0, cDest0, cResult0;
1886
0
  SplashColorPtr destColorPtr;
1887
0
  Guchar *destAlphaPtr;
1888
0
  int cSrcStride, x, lastX;
1889
1890
0
  if (cSrcPtr) {
1891
0
    cSrcStride = 1;
1892
0
  } else {
1893
0
    cSrcPtr = pipe->cSrcVal;
1894
0
    cSrcStride = 0;
1895
0
  }
1896
0
  for (; x0 <= x1; ++x0) {
1897
0
    if (*alphaPtr) {
1898
0
      break;
1899
0
    }
1900
0
    cSrcPtr += cSrcStride;
1901
0
    ++alphaPtr;
1902
0
  }
1903
0
  if (x0 > x1) {
1904
0
    return;
1905
0
  }
1906
0
  updateModX(x0);
1907
0
  updateModY(y);
1908
0
  lastX = x0;
1909
1910
0
  useBitmapRow(y);
1911
1912
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
1913
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1914
1915
0
  for (x = x0; x <= x1; ++x) {
1916
1917
    //----- alpha
1918
0
    alpha = *alphaPtr;
1919
0
    if (!alpha) {
1920
0
      ++destColorPtr;
1921
0
      ++destAlphaPtr;
1922
0
      cSrcPtr += cSrcStride;
1923
0
      ++alphaPtr;
1924
0
      continue;
1925
0
    }
1926
0
    lastX = x;
1927
1928
    //----- read destination pixel
1929
0
    cDest0 = *destColorPtr;
1930
0
    aDest = *destAlphaPtr;
1931
1932
    //----- source color
1933
0
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1934
1935
    //----- source alpha
1936
0
    aSrc = div255(pipe->aInput * alpha);
1937
1938
    //----- result alpha and non-isolated group element correction
1939
0
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
1940
0
    alphaI = aResult;
1941
1942
    //----- result color
1943
0
    if (alphaI == 0) {
1944
0
      cResult0 = 0;
1945
0
    } else {
1946
0
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
1947
0
    }
1948
1949
    //----- write destination pixel
1950
0
    *destColorPtr++ = cResult0;
1951
0
    *destAlphaPtr++ = aResult;
1952
1953
0
    cSrcPtr += cSrcStride;
1954
0
    ++alphaPtr;
1955
0
  }
1956
1957
0
  updateModX(lastX);
1958
0
}
1959
1960
// special case:
1961
// !pipe->pattern && !pipe->noTransparency && !state->softMask &&
1962
// usesAlpha && !pipe->alpha0Ptr && !state->blendFunc &&
1963
// !pipe->nonIsolatedGroup &&
1964
// bitmap->mode == splashModeRGB8 && bitmap->alpha
1965
void Splash::pipeRunAARGB8(SplashPipe *pipe, int x0, int x1, int y,
1966
         Guchar *shapePtr, Guchar *alphaPtr,
1967
628
         SplashColorPtr cSrcPtr) {
1968
628
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1969
628
  Guchar cSrc0, cSrc1, cSrc2;
1970
628
  Guchar cDest0, cDest1, cDest2;
1971
628
  Guchar cResult0, cResult1, cResult2;
1972
628
  SplashColorPtr destColorPtr;
1973
628
  Guchar *destAlphaPtr;
1974
628
  int cSrcStride, x, lastX;
1975
1976
628
  if (cSrcPtr) {
1977
9
    cSrcStride = 3;
1978
619
  } else {
1979
619
    cSrcPtr = pipe->cSrcVal;
1980
619
    cSrcStride = 0;
1981
619
  }
1982
630
  for (; x0 <= x1; ++x0) {
1983
628
    if (*alphaPtr) {
1984
626
      break;
1985
626
    }
1986
2
    cSrcPtr += cSrcStride;
1987
2
    ++alphaPtr;
1988
2
  }
1989
628
  if (x0 > x1) {
1990
2
    return;
1991
2
  }
1992
626
  updateModX(x0);
1993
626
  updateModY(y);
1994
626
  lastX = x0;
1995
1996
626
  useBitmapRow(y);
1997
1998
626
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1999
626
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2000
2001
1.25k
  for (x = x0; x <= x1; ++x) {
2002
2003
    //----- alpha
2004
626
    alpha = *alphaPtr;
2005
626
    if (!alpha) {
2006
0
      destColorPtr += 3;
2007
0
      ++destAlphaPtr;
2008
0
      cSrcPtr += cSrcStride;
2009
0
      ++alphaPtr;
2010
0
      continue;
2011
0
    }
2012
626
    lastX = x;
2013
2014
    //----- read destination pixel
2015
626
    cDest0 = destColorPtr[0];
2016
626
    cDest1 = destColorPtr[1];
2017
626
    cDest2 = destColorPtr[2];
2018
626
    aDest = *destAlphaPtr;
2019
2020
    //----- source color
2021
626
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2022
626
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2023
626
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2024
2025
    //----- source alpha
2026
626
    aSrc = div255(pipe->aInput * alpha);
2027
2028
    //----- result alpha and non-isolated group element correction
2029
626
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2030
626
    alphaI = aResult;
2031
2032
    //----- result color
2033
626
    if (alphaI == 0) {
2034
149
      cResult0 = 0;
2035
149
      cResult1 = 0;
2036
149
      cResult2 = 0;
2037
477
    } else {
2038
477
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2039
477
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2040
477
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2041
477
    }
2042
2043
    //----- write destination pixel
2044
626
    destColorPtr[0] = cResult0;
2045
626
    destColorPtr[1] = cResult1;
2046
626
    destColorPtr[2] = cResult2;
2047
626
    destColorPtr += 3;
2048
626
    *destAlphaPtr++ = aResult;
2049
2050
626
    cSrcPtr += cSrcStride;
2051
626
    ++alphaPtr;
2052
626
  }
2053
2054
626
  updateModX(lastX);
2055
626
}
2056
2057
// special case:
2058
// !pipe->pattern && !pipe->noTransparency && !state->softMask &&
2059
// usesAlpha && !pipe->alpha0Ptr && !state->blendFunc &&
2060
// !pipe->nonIsolatedGroup &&
2061
// bitmap->mode == splashModeBGR8 && bitmap->alpha
2062
void Splash::pipeRunAABGR8(SplashPipe *pipe, int x0, int x1, int y,
2063
         Guchar *shapePtr, Guchar *alphaPtr,
2064
0
         SplashColorPtr cSrcPtr) {
2065
0
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2066
0
  Guchar cSrc0, cSrc1, cSrc2;
2067
0
  Guchar cDest0, cDest1, cDest2;
2068
0
  Guchar cResult0, cResult1, cResult2;
2069
0
  SplashColorPtr destColorPtr;
2070
0
  Guchar *destAlphaPtr;
2071
0
  int cSrcStride, x, lastX;
2072
2073
0
  if (cSrcPtr) {
2074
0
    cSrcStride = 3;
2075
0
  } else {
2076
0
    cSrcPtr = pipe->cSrcVal;
2077
0
    cSrcStride = 0;
2078
0
  }
2079
0
  for (; x0 <= x1; ++x0) {
2080
0
    if (*alphaPtr) {
2081
0
      break;
2082
0
    }
2083
0
    cSrcPtr += cSrcStride;
2084
0
    ++alphaPtr;
2085
0
  }
2086
0
  if (x0 > x1) {
2087
0
    return;
2088
0
  }
2089
0
  updateModX(x0);
2090
0
  updateModY(y);
2091
0
  lastX = x0;
2092
2093
0
  useBitmapRow(y);
2094
2095
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
2096
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2097
2098
0
  for (x = x0; x <= x1; ++x) {
2099
2100
    //----- alpha
2101
0
    alpha = *alphaPtr;
2102
0
    if (!alpha) {
2103
0
      destColorPtr += 3;
2104
0
      ++destAlphaPtr;
2105
0
      cSrcPtr += cSrcStride;
2106
0
      ++alphaPtr;
2107
0
      continue;
2108
0
    }
2109
0
    lastX = x;
2110
2111
    //----- read destination pixel
2112
0
    cDest0 = destColorPtr[2];
2113
0
    cDest1 = destColorPtr[1];
2114
0
    cDest2 = destColorPtr[0];
2115
0
    aDest = *destAlphaPtr;
2116
2117
    //----- source color
2118
0
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2119
0
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2120
0
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2121
2122
    //----- source alpha
2123
0
    aSrc = div255(pipe->aInput * alpha);
2124
2125
    //----- result alpha and non-isolated group element correction
2126
0
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2127
0
    alphaI = aResult;
2128
2129
    //----- result color
2130
0
    if (alphaI == 0) {
2131
0
      cResult0 = 0;
2132
0
      cResult1 = 0;
2133
0
      cResult2 = 0;
2134
0
    } else {
2135
0
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2136
0
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2137
0
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2138
0
    }
2139
2140
    //----- write destination pixel
2141
0
    destColorPtr[0] = cResult2;
2142
0
    destColorPtr[1] = cResult1;
2143
0
    destColorPtr[2] = cResult0;
2144
0
    destColorPtr += 3;
2145
0
    *destAlphaPtr++ = aResult;
2146
2147
0
    cSrcPtr += cSrcStride;
2148
0
    ++alphaPtr;
2149
0
  }
2150
2151
0
  updateModX(lastX);
2152
0
}
2153
2154
#if SPLASH_CMYK
2155
// special case:
2156
// !pipe->pattern && !pipe->noTransparency && !state->softMask &&
2157
// usesAlpha && !pipe->alpha0Ptr && !state->blendFunc &&
2158
// !pipe->nonIsolatedGroup &&
2159
// bitmap->mode == splashModeCMYK8 && bitmap->alpha
2160
void Splash::pipeRunAACMYK8(SplashPipe *pipe, int x0, int x1, int y,
2161
          Guchar *shapePtr, Guchar *alphaPtr,
2162
0
          SplashColorPtr cSrcPtr) {
2163
0
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2164
0
  Guchar cSrc0, cSrc1, cSrc2, cSrc3;
2165
0
  Guchar cDest0, cDest1, cDest2, cDest3;
2166
0
  Guchar cResult0, cResult1, cResult2, cResult3;
2167
0
  SplashColorPtr destColorPtr;
2168
0
  Guchar *destAlphaPtr;
2169
0
  int cSrcStride, x, lastX;
2170
2171
0
  if (cSrcPtr) {
2172
0
    cSrcStride = 4;
2173
0
  } else {
2174
0
    cSrcPtr = pipe->cSrcVal;
2175
0
    cSrcStride = 0;
2176
0
  }
2177
0
  for (; x0 <= x1; ++x0) {
2178
0
    if (*alphaPtr) {
2179
0
      break;
2180
0
    }
2181
0
    cSrcPtr += cSrcStride;
2182
0
    ++alphaPtr;
2183
0
  }
2184
0
  if (x0 > x1) {
2185
0
    return;
2186
0
  }
2187
0
  updateModX(x0);
2188
0
  updateModY(y);
2189
0
  lastX = x0;
2190
2191
0
  useBitmapRow(y);
2192
2193
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 4 * x0];
2194
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2195
2196
0
  for (x = x0; x <= x1; ++x) {
2197
2198
    //----- alpha
2199
0
    alpha = *alphaPtr;
2200
0
    if (!alpha) {
2201
0
      destColorPtr += 4;
2202
0
      ++destAlphaPtr;
2203
0
      cSrcPtr += cSrcStride;
2204
0
      ++alphaPtr;
2205
0
      continue;
2206
0
    }
2207
0
    lastX = x;
2208
2209
    //----- read destination pixel
2210
0
    cDest0 = destColorPtr[0];
2211
0
    cDest1 = destColorPtr[1];
2212
0
    cDest2 = destColorPtr[2];
2213
0
    cDest3 = destColorPtr[3];
2214
0
    aDest = *destAlphaPtr;
2215
2216
    //----- overprint
2217
0
    if (state->overprintMask & 1) {
2218
0
      cSrc0 = state->cmykTransferC[cSrcPtr[0]];
2219
0
    } else {
2220
0
      cSrc0 = div255(aDest * cDest0);
2221
0
    }
2222
0
    if (state->overprintMask & 2) {
2223
0
      cSrc1 = state->cmykTransferM[cSrcPtr[1]];
2224
0
    } else {
2225
0
      cSrc1 = div255(aDest * cDest1);
2226
0
    }
2227
0
    if (state->overprintMask & 4) {
2228
0
      cSrc2 = state->cmykTransferY[cSrcPtr[2]];
2229
0
    } else {
2230
0
      cSrc2 = div255(aDest * cDest2);
2231
0
    }
2232
0
    if (state->overprintMask & 8) {
2233
0
      cSrc3 = state->cmykTransferK[cSrcPtr[3]];
2234
0
    } else {
2235
0
      cSrc3 = div255(aDest * cDest3);
2236
0
    }
2237
2238
    //----- source alpha
2239
0
    aSrc = div255(pipe->aInput * alpha);
2240
2241
    //----- result alpha and non-isolated group element correction
2242
0
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2243
0
    alphaI = aResult;
2244
2245
    //----- result color
2246
0
    if (alphaI == 0) {
2247
0
      cResult0 = 0;
2248
0
      cResult1 = 0;
2249
0
      cResult2 = 0;
2250
0
      cResult3 = 0;
2251
0
    } else {
2252
0
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2253
0
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2254
0
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2255
0
      cResult3 = (Guchar)(((alphaI - aSrc) * cDest3 + aSrc * cSrc3) / alphaI);
2256
0
    }
2257
2258
    //----- write destination pixel
2259
0
    destColorPtr[0] = cResult0;
2260
0
    destColorPtr[1] = cResult1;
2261
0
    destColorPtr[2] = cResult2;
2262
0
    destColorPtr[3] = cResult3;
2263
0
    destColorPtr += 4;
2264
0
    *destAlphaPtr++ = aResult;
2265
2266
0
    cSrcPtr += cSrcStride;
2267
0
    ++alphaPtr;
2268
0
  }
2269
2270
0
  updateModX(lastX);
2271
0
}
2272
#endif
2273
2274
2275
// special case:
2276
// !pipe->pattern && aInput == 255 && state->softMask && usesAlpha &&
2277
// !state->inNonIsolatedGroup && !state->inKnockoutGroup &&
2278
// !nonIsolatedGroup && state->overprintMask == 0xffffffff &&
2279
// !state->blendFunc &&
2280
// bitmap->mode == splashModeMono8 && bitmap->alpha
2281
void Splash::pipeRunSoftMaskMono8(SplashPipe *pipe, int x0, int x1, int y,
2282
          Guchar *shapePtr, Guchar *alphaPtr,
2283
139
          SplashColorPtr cSrcPtr) {
2284
139
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2285
139
  Guchar cSrc0, cDest0, cResult0;
2286
139
  SplashColorPtr destColorPtr;
2287
139
  Guchar *destAlphaPtr;
2288
139
  SplashColorPtr softMaskPtr;
2289
139
  int cSrcStride, x, lastX;
2290
2291
139
  if (cSrcPtr) {
2292
0
    cSrcStride = 1;
2293
139
  } else {
2294
139
    cSrcPtr = pipe->cSrcVal;
2295
139
    cSrcStride = 0;
2296
139
  }
2297
190
  for (; x0 <= x1; ++x0) {
2298
139
    if (*alphaPtr) {
2299
88
      break;
2300
88
    }
2301
51
    cSrcPtr += cSrcStride;
2302
51
    ++alphaPtr;
2303
51
  }
2304
139
  if (x0 > x1) {
2305
51
    return;
2306
51
  }
2307
88
  updateModX(x0);
2308
88
  updateModY(y);
2309
88
  lastX = x0;
2310
2311
88
  useBitmapRow(y);
2312
2313
88
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
2314
88
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2315
88
  softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
2316
2317
176
  for (x = x0; x <= x1; ++x) {
2318
2319
    //----- alpha
2320
88
    alpha = *alphaPtr;
2321
88
    if (!alpha) {
2322
0
      ++destColorPtr;
2323
0
      ++destAlphaPtr;
2324
0
      ++softMaskPtr;
2325
0
      cSrcPtr += cSrcStride;
2326
0
      ++alphaPtr;
2327
0
      continue;
2328
0
    }
2329
88
    lastX = x;
2330
2331
    //----- read source color
2332
88
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
2333
2334
    //----- source alpha
2335
88
    aSrc = div255(*softMaskPtr++ * alpha);
2336
2337
    //----- special case for aSrc = 255
2338
88
    if (aSrc == 255) {
2339
0
      aResult = 255;
2340
0
      cResult0 = cSrc0;
2341
88
    } else {
2342
2343
      //----- read destination alpha
2344
88
      aDest = *destAlphaPtr;
2345
2346
      //----- special case for aDest = 0
2347
88
      if (aDest == 0) {
2348
77
        aResult = aSrc;
2349
77
        cResult0 = cSrc0;
2350
77
      } else {
2351
2352
        //----- read destination pixel
2353
11
        cDest0 = destColorPtr[0];
2354
2355
        //----- result alpha and non-isolated group element correction
2356
11
        aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2357
11
        alphaI = aResult;
2358
2359
        //----- result color
2360
11
        cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2361
11
      }
2362
88
    }
2363
2364
    //----- write destination pixel
2365
88
    destColorPtr[0] = cResult0;
2366
88
    ++destColorPtr;
2367
88
    *destAlphaPtr++ = aResult;
2368
2369
88
    cSrcPtr += cSrcStride;
2370
88
    ++alphaPtr;
2371
88
  }
2372
2373
88
  updateModX(lastX);
2374
88
}
2375
2376
// special case:
2377
// !pipe->pattern && aInput == 255 && state->softMask && usesAlpha &&
2378
// !state->inNonIsolatedGroup && !state->inKnockoutGroup &&
2379
// !nonIsolatedGroup && state->overprintMask == 0xffffffff &&
2380
// !state->blendFunc &&
2381
// bitmap->mode == splashModeRGB8 && bitmap->alpha
2382
void Splash::pipeRunSoftMaskRGB8(SplashPipe *pipe, int x0, int x1, int y,
2383
                                 Guchar *shapePtr, Guchar *alphaPtr,
2384
249
         SplashColorPtr cSrcPtr) {
2385
249
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2386
249
  Guchar cSrc0, cSrc1, cSrc2;
2387
249
  Guchar cDest0, cDest1, cDest2;
2388
249
  Guchar cResult0, cResult1, cResult2;
2389
249
  SplashColorPtr destColorPtr;
2390
249
  Guchar *destAlphaPtr;
2391
249
  SplashColorPtr softMaskPtr;
2392
249
  int cSrcStride, x, lastX;
2393
2394
249
  if (cSrcPtr) {
2395
80
    cSrcStride = 3;
2396
169
  } else {
2397
169
    cSrcPtr = pipe->cSrcVal;
2398
169
    cSrcStride = 0;
2399
169
  }
2400
314
  for (; x0 <= x1; ++x0) {
2401
249
    if (*alphaPtr) {
2402
184
      break;
2403
184
    }
2404
65
    cSrcPtr += cSrcStride;
2405
65
    ++alphaPtr;
2406
65
  }
2407
249
  if (x0 > x1) {
2408
65
    return;
2409
65
  }
2410
184
  updateModX(x0);
2411
184
  updateModY(y);
2412
184
  lastX = x0;
2413
2414
184
  useBitmapRow(y);
2415
2416
184
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 3];
2417
184
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2418
184
  softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
2419
2420
368
  for (x = x0; x <= x1; ++x) {
2421
2422
    //----- alpha
2423
184
    alpha = *alphaPtr;
2424
184
    if (!alpha) {
2425
0
      destColorPtr += 3;
2426
0
      ++destAlphaPtr;
2427
0
      ++softMaskPtr;
2428
0
      cSrcPtr += cSrcStride;
2429
0
      ++alphaPtr;
2430
0
      continue;
2431
0
    }
2432
184
    lastX = x;
2433
2434
    //----- read source color
2435
184
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2436
184
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2437
184
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2438
2439
    //----- source alpha
2440
184
    aSrc = div255(*softMaskPtr++ * alpha);
2441
2442
    //----- special case for aSrc = 255
2443
184
    if (aSrc == 255) {
2444
48
      aResult = 255;
2445
48
      cResult0 = cSrc0;
2446
48
      cResult1 = cSrc1;
2447
48
      cResult2 = cSrc2;
2448
136
    } else {
2449
2450
      //----- read destination alpha
2451
136
      aDest = *destAlphaPtr;
2452
2453
      //----- special case for aDest = 0
2454
136
      if (aDest == 0) {
2455
105
        aResult = aSrc;
2456
105
        cResult0 = cSrc0;
2457
105
        cResult1 = cSrc1;
2458
105
        cResult2 = cSrc2;
2459
105
      } else {
2460
2461
        //----- read destination pixel
2462
31
        cDest0 = destColorPtr[0];
2463
31
        cDest1 = destColorPtr[1];
2464
31
        cDest2 = destColorPtr[2];
2465
2466
        //----- result alpha and non-isolated group element correction
2467
31
        aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2468
31
        alphaI = aResult;
2469
2470
        //----- result color
2471
31
        cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2472
31
        cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2473
31
        cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2474
31
      }
2475
136
    }
2476
2477
    //----- write destination pixel
2478
184
    destColorPtr[0] = cResult0;
2479
184
    destColorPtr[1] = cResult1;
2480
184
    destColorPtr[2] = cResult2;
2481
184
    destColorPtr += 3;
2482
184
    *destAlphaPtr++ = aResult;
2483
2484
184
    cSrcPtr += cSrcStride;
2485
184
    ++alphaPtr;
2486
184
  }
2487
2488
184
  updateModX(lastX);
2489
184
}
2490
2491
// special case:
2492
// !pipe->pattern && aInput == 255 && state->softMask && usesAlpha &&
2493
// !state->inNonIsolatedGroup && !state->inKnockoutGroup &&
2494
// !nonIsolatedGroup && state->overprintMask == 0xffffffff &&
2495
// !state->blendFunc &&
2496
// bitmap->mode == splashModeBGR8 && bitmap->alpha
2497
void Splash::pipeRunSoftMaskBGR8(SplashPipe *pipe, int x0, int x1, int y,
2498
                                 Guchar *shapePtr, Guchar *alphaPtr,
2499
0
         SplashColorPtr cSrcPtr) {
2500
0
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2501
0
  Guchar cSrc0, cSrc1, cSrc2;
2502
0
  Guchar cDest0, cDest1, cDest2;
2503
0
  Guchar cResult0, cResult1, cResult2;
2504
0
  SplashColorPtr destColorPtr;
2505
0
  Guchar *destAlphaPtr;
2506
0
  SplashColorPtr softMaskPtr;
2507
0
  int cSrcStride, x, lastX;
2508
2509
0
  if (cSrcPtr) {
2510
0
    cSrcStride = 3;
2511
0
  } else {
2512
0
    cSrcPtr = pipe->cSrcVal;
2513
0
    cSrcStride = 0;
2514
0
  }
2515
0
  for (; x0 <= x1; ++x0) {
2516
0
    if (*alphaPtr) {
2517
0
      break;
2518
0
    }
2519
0
    cSrcPtr += cSrcStride;
2520
0
    ++alphaPtr;
2521
0
  }
2522
0
  if (x0 > x1) {
2523
0
    return;
2524
0
  }
2525
0
  updateModX(x0);
2526
0
  updateModY(y);
2527
0
  lastX = x0;
2528
2529
0
  useBitmapRow(y);
2530
2531
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 3];
2532
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2533
0
  softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
2534
2535
0
  for (x = x0; x <= x1; ++x) {
2536
2537
    //----- alpha
2538
0
    alpha = *alphaPtr;
2539
0
    if (!alpha) {
2540
0
      destColorPtr += 3;
2541
0
      ++destAlphaPtr;
2542
0
      ++softMaskPtr;
2543
0
      cSrcPtr += cSrcStride;
2544
0
      ++alphaPtr;
2545
0
      continue;
2546
0
    }
2547
0
    lastX = x;
2548
2549
    //----- read source color
2550
0
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2551
0
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2552
0
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2553
2554
    //----- source alpha
2555
0
    aSrc = div255(*softMaskPtr++ * alpha);
2556
2557
    //----- special case for aSrc = 255
2558
0
    if (aSrc == 255) {
2559
0
      aResult = 255;
2560
0
      cResult0 = cSrc0;
2561
0
      cResult1 = cSrc1;
2562
0
      cResult2 = cSrc2;
2563
0
    } else {
2564
2565
      //----- read destination alpha
2566
0
      aDest = *destAlphaPtr;
2567
2568
      //----- special case for aDest = 0
2569
0
      if (aDest == 0) {
2570
0
        aResult = aSrc;
2571
0
        cResult0 = cSrc0;
2572
0
        cResult1 = cSrc1;
2573
0
        cResult2 = cSrc2;
2574
0
      } else {
2575
2576
        //----- read destination pixel
2577
0
        cDest0 = destColorPtr[2];
2578
0
        cDest1 = destColorPtr[1];
2579
0
        cDest2 = destColorPtr[0];
2580
2581
        //----- result alpha and non-isolated group element correction
2582
0
        aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2583
0
        alphaI = aResult;
2584
2585
        //----- result color
2586
0
        cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2587
0
        cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2588
0
        cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2589
0
      }
2590
0
    }
2591
2592
    //----- write destination pixel
2593
0
    destColorPtr[2] = cResult0;
2594
0
    destColorPtr[1] = cResult1;
2595
0
    destColorPtr[0] = cResult2;
2596
0
    destColorPtr += 3;
2597
0
    *destAlphaPtr++ = aResult;
2598
2599
0
    cSrcPtr += cSrcStride;
2600
0
    ++alphaPtr;
2601
0
  }
2602
2603
0
  updateModX(lastX);
2604
0
}
2605
2606
#if SPLASH_CMYK
2607
// special case:
2608
// !pipe->pattern && aInput == 255 && state->softMask && usesAlpha &&
2609
// !state->inNonIsolatedGroup && !state->inKnockoutGroup &&
2610
// !nonIsolatedGroup && state->overprintMask == 0xffffffff &&
2611
// !state->blendFunc &&
2612
// bitmap->mode == splashModeCMYK8 && bitmap->alpha
2613
void Splash::pipeRunSoftMaskCMYK8(SplashPipe *pipe, int x0, int x1, int y,
2614
          Guchar *shapePtr, Guchar *alphaPtr,
2615
0
          SplashColorPtr cSrcPtr) {
2616
0
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2617
0
  Guchar cSrc0, cSrc1, cSrc2, cSrc3;
2618
0
  Guchar cDest0, cDest1, cDest2, cDest3;
2619
0
  Guchar cResult0, cResult1, cResult2, cResult3;
2620
0
  SplashColorPtr destColorPtr;
2621
0
  Guchar *destAlphaPtr;
2622
0
  SplashColorPtr softMaskPtr;
2623
0
  int cSrcStride, x, lastX;
2624
2625
0
  if (cSrcPtr) {
2626
0
    cSrcStride = 4;
2627
0
  } else {
2628
0
    cSrcPtr = pipe->cSrcVal;
2629
0
    cSrcStride = 0;
2630
0
  }
2631
0
  for (; x0 <= x1; ++x0) {
2632
0
    if (*alphaPtr) {
2633
0
      break;
2634
0
    }
2635
0
    cSrcPtr += cSrcStride;
2636
0
    ++alphaPtr;
2637
0
  }
2638
0
  if (x0 > x1) {
2639
0
    return;
2640
0
  }
2641
0
  updateModX(x0);
2642
0
  updateModY(y);
2643
0
  lastX = x0;
2644
2645
0
  useBitmapRow(y);
2646
2647
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 4];
2648
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2649
0
  softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
2650
2651
0
  for (x = x0; x <= x1; ++x) {
2652
2653
    //----- alpha
2654
0
    alpha = *alphaPtr;
2655
0
    if (!alpha) {
2656
0
      destColorPtr += 4;
2657
0
      ++destAlphaPtr;
2658
0
      ++softMaskPtr;
2659
0
      cSrcPtr += cSrcStride;
2660
0
      ++alphaPtr;
2661
0
      continue;
2662
0
    }
2663
0
    lastX = x;
2664
2665
    //----- read destination pixel
2666
0
    cDest0 = destColorPtr[0];
2667
0
    cDest1 = destColorPtr[1];
2668
0
    cDest2 = destColorPtr[2];
2669
0
    cDest3 = destColorPtr[3];
2670
2671
    //----- read destination alpha
2672
0
    aDest = *destAlphaPtr;
2673
2674
    //----- overprint
2675
0
    cSrc0 = state->cmykTransferC[cSrcPtr[0]];
2676
0
    cSrc1 = state->cmykTransferM[cSrcPtr[1]];
2677
0
    cSrc2 = state->cmykTransferY[cSrcPtr[2]];
2678
0
    cSrc3 = state->cmykTransferK[cSrcPtr[3]];
2679
2680
    //----- source alpha
2681
0
    aSrc = div255(*softMaskPtr++ * alpha);
2682
2683
    //----- special case for aSrc = 255
2684
0
    if (aSrc == 255) {
2685
0
      aResult = 255;
2686
0
      cResult0 = cSrc0;
2687
0
      cResult1 = cSrc1;
2688
0
      cResult2 = cSrc2;
2689
0
      cResult3 = cSrc3;
2690
0
    } else {
2691
2692
      //----- special case for aDest = 0
2693
0
      if (aDest == 0) {
2694
0
        aResult = aSrc;
2695
0
        cResult0 = cSrc0;
2696
0
        cResult1 = cSrc1;
2697
0
        cResult2 = cSrc2;
2698
0
        cResult3 = cSrc3;
2699
0
      } else {
2700
2701
        //----- result alpha and non-isolated group element correction
2702
0
        aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2703
0
        alphaI = aResult;
2704
2705
        //----- result color
2706
0
        cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2707
0
        cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2708
0
        cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2709
0
        cResult3 = (Guchar)(((alphaI - aSrc) * cDest3 + aSrc * cSrc3) / alphaI);
2710
0
      }
2711
0
    }
2712
2713
    //----- write destination pixel
2714
0
    destColorPtr[0] = cResult0;
2715
0
    destColorPtr[1] = cResult1;
2716
0
    destColorPtr[2] = cResult2;
2717
0
    destColorPtr[3] = cResult3;
2718
0
    destColorPtr += 4;
2719
0
    *destAlphaPtr++ = aResult;
2720
2721
0
    cSrcPtr += cSrcStride;
2722
0
    ++alphaPtr;
2723
0
  }
2724
2725
0
  updateModX(lastX);
2726
0
}
2727
#endif
2728
2729
2730
void Splash::pipeRunNonIsoMono8(SplashPipe *pipe, int x0, int x1, int y,
2731
        Guchar *shapePtr, Guchar *alphaPtr,
2732
0
        SplashColorPtr cSrcPtr) {
2733
0
  Guchar alpha, aSrc, aDest, alphaI, alpha0, aResult;
2734
0
  Guchar cSrc0, cDest0, cResult0;
2735
0
  SplashColorPtr destColorPtr;
2736
0
  Guchar *destAlphaPtr;
2737
0
  Guchar *alpha0Ptr;
2738
0
  int cSrcStride, x, lastX;
2739
2740
0
  if (cSrcPtr) {
2741
0
    cSrcStride = 1;
2742
0
  } else {
2743
0
    cSrcPtr = pipe->cSrcVal;
2744
0
    cSrcStride = 0;
2745
0
  }
2746
0
  for (; x0 <= x1; ++x0) {
2747
0
    if (*alphaPtr) {
2748
0
      break;
2749
0
    }
2750
0
    cSrcPtr += cSrcStride;
2751
0
    ++alphaPtr;
2752
0
  }
2753
0
  if (x0 > x1) {
2754
0
    return;
2755
0
  }
2756
0
  updateModX(x0);
2757
0
  updateModY(y);
2758
0
  lastX = x0;
2759
2760
0
  useBitmapRow(y);
2761
2762
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
2763
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2764
0
  if (alpha0Bitmap) {
2765
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
2766
0
  } else {
2767
0
    alpha0Ptr = &parent->bitmap->alpha[
2768
0
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
2769
0
        + (parentOffsetX + x0)];
2770
0
  }
2771
2772
0
  for (x = x0; x <= x1; ++x) {
2773
2774
    //----- alpha
2775
0
    alpha = *alphaPtr;
2776
0
    if (!alpha) {
2777
0
      destColorPtr += 1;
2778
0
      ++destAlphaPtr;
2779
0
      ++alpha0Ptr;
2780
0
      cSrcPtr += cSrcStride;
2781
0
      ++alphaPtr;
2782
0
      continue;
2783
0
    }
2784
0
    lastX = x;
2785
2786
    //----- read destination pixel
2787
0
    cDest0 = destColorPtr[0];
2788
0
    aDest = *destAlphaPtr;
2789
2790
    //----- source color
2791
0
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
2792
2793
    //----- source alpha
2794
0
    aSrc = div255(pipe->aInput * alpha);
2795
2796
    //----- result alpha and non-isolated group element correction
2797
0
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2798
0
    alpha0 = *alpha0Ptr++;
2799
0
    alphaI = (Guchar)(aResult + alpha0 - div255(aResult * alpha0));
2800
2801
    //----- result color
2802
0
    if (alphaI == 0) {
2803
0
      cResult0 = 0;
2804
0
    } else {
2805
0
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2806
0
    }
2807
2808
    //----- write destination pixel
2809
0
    *destColorPtr++ = cResult0;
2810
0
    *destAlphaPtr++ = aResult;
2811
2812
0
    cSrcPtr += cSrcStride;
2813
0
    ++alphaPtr;
2814
0
  } // for (x ...)
2815
2816
0
  updateModX(lastX);
2817
0
}
2818
2819
void Splash::pipeRunNonIsoRGB8(SplashPipe *pipe, int x0, int x1, int y,
2820
             Guchar *shapePtr, Guchar *alphaPtr,
2821
20.8k
             SplashColorPtr cSrcPtr) {
2822
20.8k
  Guchar alpha, aSrc, aDest, alphaI, alpha0, aResult;
2823
20.8k
  Guchar cSrc0, cSrc1, cSrc2;
2824
20.8k
  Guchar cDest0, cDest1, cDest2;
2825
20.8k
  Guchar cResult0, cResult1, cResult2;
2826
20.8k
  SplashColorPtr destColorPtr;
2827
20.8k
  Guchar *destAlphaPtr;
2828
20.8k
  Guchar *alpha0Ptr;
2829
20.8k
  int cSrcStride, x, lastX;
2830
2831
20.8k
  if (cSrcPtr) {
2832
0
    cSrcStride = 3;
2833
20.8k
  } else {
2834
20.8k
    cSrcPtr = pipe->cSrcVal;
2835
20.8k
    cSrcStride = 0;
2836
20.8k
  }
2837
21.8k
  for (; x0 <= x1; ++x0) {
2838
20.8k
    if (*alphaPtr) {
2839
19.8k
      break;
2840
19.8k
    }
2841
1.00k
    cSrcPtr += cSrcStride;
2842
1.00k
    ++alphaPtr;
2843
1.00k
  }
2844
20.8k
  if (x0 > x1) {
2845
1.00k
    return;
2846
1.00k
  }
2847
19.8k
  updateModX(x0);
2848
19.8k
  updateModY(y);
2849
19.8k
  lastX = x0;
2850
2851
19.8k
  useBitmapRow(y);
2852
2853
19.8k
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 3];
2854
19.8k
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2855
19.8k
  if (alpha0Bitmap) {
2856
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
2857
19.8k
  } else {
2858
19.8k
    alpha0Ptr = &parent->bitmap->alpha[
2859
19.8k
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
2860
19.8k
        + (parentOffsetX + x0)];
2861
19.8k
  }
2862
2863
39.6k
  for (x = x0; x <= x1; ++x) {
2864
2865
    //----- alpha
2866
19.8k
    alpha = *alphaPtr;
2867
19.8k
    if (!alpha) {
2868
0
      destColorPtr += 3;
2869
0
      ++destAlphaPtr;
2870
0
      ++alpha0Ptr;
2871
0
      cSrcPtr += cSrcStride;
2872
0
      ++alphaPtr;
2873
0
      continue;
2874
0
    }
2875
19.8k
    lastX = x;
2876
2877
    //----- read destination pixel
2878
19.8k
    cDest0 = destColorPtr[0];
2879
19.8k
    cDest1 = destColorPtr[1];
2880
19.8k
    cDest2 = destColorPtr[2];
2881
19.8k
    aDest = *destAlphaPtr;
2882
2883
    //----- source color
2884
19.8k
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2885
19.8k
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2886
19.8k
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2887
2888
    //----- source alpha
2889
19.8k
    aSrc = div255(pipe->aInput * alpha);
2890
2891
    //----- result alpha and non-isolated group element correction
2892
19.8k
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2893
19.8k
    alpha0 = *alpha0Ptr++;
2894
19.8k
    alphaI = (Guchar)(aResult + alpha0 - div255(aResult * alpha0));
2895
2896
    //----- result color
2897
19.8k
    if (alphaI == 0) {
2898
6
      cResult0 = 0;
2899
6
      cResult1 = 0;
2900
6
      cResult2 = 0;
2901
19.8k
    } else {
2902
19.8k
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2903
19.8k
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2904
19.8k
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2905
19.8k
    }
2906
2907
    //----- write destination pixel
2908
19.8k
    destColorPtr[0] = cResult0;
2909
19.8k
    destColorPtr[1] = cResult1;
2910
19.8k
    destColorPtr[2] = cResult2;
2911
19.8k
    destColorPtr += 3;
2912
19.8k
    *destAlphaPtr++ = aResult;
2913
2914
19.8k
    cSrcPtr += cSrcStride;
2915
19.8k
    ++alphaPtr;
2916
19.8k
  } // for (x ...)
2917
2918
19.8k
  updateModX(lastX);
2919
19.8k
}
2920
2921
void Splash::pipeRunNonIsoBGR8(SplashPipe *pipe, int x0, int x1, int y,
2922
             Guchar *shapePtr, Guchar *alphaPtr,
2923
0
             SplashColorPtr cSrcPtr) {
2924
0
  Guchar alpha, aSrc, aDest, alphaI, alpha0, aResult;
2925
0
  Guchar cSrc0, cSrc1, cSrc2;
2926
0
  Guchar cDest0, cDest1, cDest2;
2927
0
  Guchar cResult0, cResult1, cResult2;
2928
0
  SplashColorPtr destColorPtr;
2929
0
  Guchar *destAlphaPtr;
2930
0
  Guchar *alpha0Ptr;
2931
0
  int cSrcStride, x, lastX;
2932
2933
0
  if (cSrcPtr) {
2934
0
    cSrcStride = 3;
2935
0
  } else {
2936
0
    cSrcPtr = pipe->cSrcVal;
2937
0
    cSrcStride = 0;
2938
0
  }
2939
0
  for (; x0 <= x1; ++x0) {
2940
0
    if (*alphaPtr) {
2941
0
      break;
2942
0
    }
2943
0
    cSrcPtr += cSrcStride;
2944
0
    ++alphaPtr;
2945
0
  }
2946
0
  if (x0 > x1) {
2947
0
    return;
2948
0
  }
2949
0
  updateModX(x0);
2950
0
  updateModY(y);
2951
0
  lastX = x0;
2952
2953
0
  useBitmapRow(y);
2954
2955
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 3];
2956
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2957
0
  if (alpha0Bitmap) {
2958
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
2959
0
  } else {
2960
0
    alpha0Ptr = &parent->bitmap->alpha[
2961
0
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
2962
0
        + (parentOffsetX + x0)];
2963
0
  }
2964
2965
0
  for (x = x0; x <= x1; ++x) {
2966
2967
    //----- alpha
2968
0
    alpha = *alphaPtr;
2969
0
    if (!alpha) {
2970
0
      destColorPtr += 3;
2971
0
      ++destAlphaPtr;
2972
0
      ++alpha0Ptr;
2973
0
      cSrcPtr += cSrcStride;
2974
0
      ++alphaPtr;
2975
0
      continue;
2976
0
    }
2977
0
    lastX = x;
2978
2979
    //----- read destination pixel
2980
0
    cDest0 = destColorPtr[2];
2981
0
    cDest1 = destColorPtr[1];
2982
0
    cDest2 = destColorPtr[0];
2983
0
    aDest = *destAlphaPtr;
2984
2985
    //----- source color
2986
0
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2987
0
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2988
0
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2989
2990
    //----- source alpha
2991
0
    aSrc = div255(pipe->aInput * alpha);
2992
2993
    //----- result alpha and non-isolated group element correction
2994
0
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2995
0
    alpha0 = *alpha0Ptr++;
2996
0
    alphaI = (Guchar)(aResult + alpha0 - div255(aResult * alpha0));
2997
2998
    //----- result color
2999
0
    if (alphaI == 0) {
3000
0
      cResult0 = 0;
3001
0
      cResult1 = 0;
3002
0
      cResult2 = 0;
3003
0
    } else {
3004
0
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
3005
0
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
3006
0
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
3007
0
    }
3008
3009
    //----- write destination pixel
3010
0
    destColorPtr[0] = cResult2;
3011
0
    destColorPtr[1] = cResult1;
3012
0
    destColorPtr[2] = cResult0;
3013
0
    destColorPtr += 3;
3014
0
    *destAlphaPtr++ = aResult;
3015
3016
0
    cSrcPtr += cSrcStride;
3017
0
    ++alphaPtr;
3018
0
  } // for (x ...)
3019
3020
0
  updateModX(lastX);
3021
0
}
3022
3023
#if SPLASH_CMYK
3024
void Splash::pipeRunNonIsoCMYK8(SplashPipe *pipe, int x0, int x1, int y,
3025
        Guchar *shapePtr, Guchar *alphaPtr,
3026
0
        SplashColorPtr cSrcPtr) {
3027
0
  Guchar alpha, aSrc, aDest, alphaI, alpha0, aResult, aPrev;
3028
0
  Guchar cSrc0, cSrc1, cSrc2, cSrc3;
3029
0
  Guchar cDest0, cDest1, cDest2, cDest3;
3030
0
  Guchar cResult0, cResult1, cResult2, cResult3;
3031
0
  SplashColorPtr destColorPtr;
3032
0
  Guchar *destAlphaPtr;
3033
0
  Guchar *alpha0Ptr;
3034
0
  int cSrcStride, x, lastX;
3035
3036
0
  if (cSrcPtr) {
3037
0
    cSrcStride = 4;
3038
0
  } else {
3039
0
    cSrcPtr = pipe->cSrcVal;
3040
0
    cSrcStride = 0;
3041
0
  }
3042
0
  for (; x0 <= x1; ++x0) {
3043
0
    if (*alphaPtr) {
3044
0
      break;
3045
0
    }
3046
0
    cSrcPtr += cSrcStride;
3047
0
    ++alphaPtr;
3048
0
  }
3049
0
  if (x0 > x1) {
3050
0
    return;
3051
0
  }
3052
0
  updateModX(x0);
3053
0
  updateModY(y);
3054
0
  lastX = x0;
3055
3056
0
  useBitmapRow(y);
3057
3058
0
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 4];
3059
0
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
3060
0
  if (alpha0Bitmap) {
3061
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
3062
0
  } else {
3063
0
    alpha0Ptr = &parent->bitmap->alpha[
3064
0
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
3065
0
        + (parentOffsetX + x0)];
3066
0
  }
3067
3068
0
  for (x = x0; x <= x1; ++x) {
3069
3070
    //----- alpha
3071
0
    alpha = *alphaPtr;
3072
0
    if (!alpha) {
3073
0
      destColorPtr += 4;
3074
0
      ++destAlphaPtr;
3075
0
      ++alpha0Ptr;
3076
0
      cSrcPtr += cSrcStride;
3077
0
      ++alphaPtr;
3078
0
      continue;
3079
0
    }
3080
0
    lastX = x;
3081
3082
    //----- read destination pixel
3083
0
    cDest0 = destColorPtr[0];
3084
0
    cDest1 = destColorPtr[1];
3085
0
    cDest2 = destColorPtr[2];
3086
0
    cDest3 = destColorPtr[3];
3087
0
    aDest = *destAlphaPtr;
3088
3089
    //----- overprint
3090
0
    aPrev = (Guchar)(*alpha0Ptr + aDest - div255(*alpha0Ptr * aDest));
3091
0
    if (state->overprintMask & 0x01) {
3092
0
      cSrc0 = state->cmykTransferC[cSrcPtr[0]];
3093
0
    } else {
3094
0
      cSrc0 = div255(aPrev * cDest0);
3095
0
    }
3096
0
    if (state->overprintMask & 0x02) {
3097
0
      cSrc1 = state->cmykTransferM[cSrcPtr[1]];
3098
0
    } else {
3099
0
      cSrc1 = div255(aPrev * cDest1);
3100
0
    }
3101
0
    if (state->overprintMask & 0x04) {
3102
0
      cSrc2 = state->cmykTransferY[cSrcPtr[2]];
3103
0
    } else {
3104
0
      cSrc2 = div255(aPrev * cDest2);
3105
0
    }
3106
0
    if (state->overprintMask & 0x08) {
3107
0
      cSrc3 = state->cmykTransferK[cSrcPtr[3]];
3108
0
    } else {
3109
0
      cSrc3 = div255(aPrev * cDest3);
3110
0
    }
3111
3112
    //----- source alpha
3113
0
    aSrc = div255(pipe->aInput * alpha);
3114
3115
    //----- result alpha and non-isolated group element correction
3116
0
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
3117
0
    alpha0 = *alpha0Ptr++;
3118
0
    alphaI = (Guchar)(aResult + alpha0 - div255(aResult * alpha0));
3119
3120
    //----- result color
3121
0
    if (alphaI == 0) {
3122
0
      cResult0 = 0;
3123
0
      cResult1 = 0;
3124
0
      cResult2 = 0;
3125
0
      cResult3 = 0;
3126
0
    } else {
3127
0
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
3128
0
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
3129
0
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
3130
0
      cResult3 = (Guchar)(((alphaI - aSrc) * cDest3 + aSrc * cSrc3) / alphaI);
3131
0
    }
3132
3133
    //----- write destination pixel
3134
0
    destColorPtr[0] = cResult0;
3135
0
    destColorPtr[1] = cResult1;
3136
0
    destColorPtr[2] = cResult2;
3137
0
    destColorPtr[3] = cResult3;
3138
0
    destColorPtr += 4;
3139
0
    *destAlphaPtr++ = aResult;
3140
3141
0
    cSrcPtr += cSrcStride;
3142
0
    ++alphaPtr;
3143
0
  } // for (x ...)
3144
3145
0
  updateModX(lastX);
3146
0
}
3147
#endif
3148
3149
3150
55.9k
void Splash::useBitmapRow(int y) {
3151
55.9k
  if (!deferredInit) {
3152
31.1k
    return;
3153
31.1k
  }
3154
3155
24.8k
  int y0, y1;
3156
24.8k
  if (deferredInitYMin > deferredInitYMax) {
3157
5.92k
    y0 = y1 = y;
3158
5.92k
    deferredInitYMin = deferredInitYMax = y;
3159
18.8k
  } else if (y < deferredInitYMin) {
3160
0
    y0 = y;
3161
0
    y1 = deferredInitYMin - 1;
3162
0
    deferredInitYMin = y;
3163
18.8k
  } else if (y > deferredInitYMax) {
3164
0
    y0 = deferredInitYMax + 1;
3165
0
    y1 = y;
3166
0
    deferredInitYMax = y;
3167
18.8k
  } else {
3168
18.8k
    return;
3169
18.8k
  }
3170
3171
5.92k
  parent->useBitmapRow(parentOffsetY + y);
3172
5.92k
  if (state->inNonIsolatedGroup) {
3173
7.86k
    for (int yy = y0; yy <= y1; ++yy) {
3174
3.93k
      copyParentRowColor(yy);
3175
3.93k
    }
3176
3.93k
  } else {
3177
3.98k
    for (int yy = y0; yy <= y1; ++yy) {
3178
1.99k
      zeroRowColor(yy);
3179
1.99k
    }
3180
1.99k
  }
3181
5.92k
  if (bitmap->alpha) {
3182
11.8k
    for (int yy = y0; yy <= y1; ++yy) {
3183
5.92k
      zeroRowAlpha(yy);
3184
5.92k
    }
3185
5.92k
  }
3186
5.92k
  if (alpha0Bitmap) {
3187
0
    for (int yy = y0; yy <= y1; ++yy) {
3188
0
      computeAlpha0Row(yy);
3189
0
    }
3190
0
  }
3191
5.92k
}
3192
3193
3.93k
void Splash::copyParentRowColor(int y) {
3194
3.93k
  if (parent->bitmap->mode != bitmap->mode) {
3195
0
    return;
3196
0
  }
3197
3.93k
  if (bitmap->mode == splashModeMono1) {
3198
0
    SplashColorPtr p = &bitmap->data[y * bitmap->rowSize];
3199
0
    Guchar mask = (Guchar)0x80;
3200
0
    SplashColorPtr q =
3201
0
        &parent->bitmap->data[(parentOffsetY + y) * parent->bitmap->rowSize
3202
0
            + (parentOffsetX >> 3)];
3203
0
    Guchar srcMask = (Guchar)(0x80 >> (parentOffsetX & 7));
3204
0
    for (int x = 0; x < bitmap->width; ++x) {
3205
0
      if (*q & srcMask) {
3206
0
  *p |= mask;
3207
0
      } else {
3208
0
  *p &= (Guchar)~mask;
3209
0
      }
3210
0
      if (!(mask = (Guchar)(mask >> 1))) {
3211
0
  mask = 0x80;
3212
0
  ++p;
3213
0
      }
3214
0
      if (!(srcMask = (Guchar)(srcMask >> 1))) {
3215
0
  srcMask = 0x80;
3216
0
  ++q;
3217
0
      }
3218
0
    }
3219
3.93k
  } else {
3220
3.93k
    SplashColorPtr p = &bitmap->data[y * bitmap->rowSize];
3221
3.93k
    SplashColorPtr q =
3222
3.93k
        &parent->bitmap->data[(parentOffsetY + y) * parent->bitmap->rowSize
3223
3.93k
            + bitmapComps * parentOffsetX];
3224
3.93k
    memcpy(p, q, bitmapComps * bitmap->width);
3225
3.93k
  }
3226
3.93k
}
3227
3228
1.99k
void Splash::zeroRowColor(int y) {
3229
1.99k
  memset(&bitmap->data[y * bitmap->rowSize], 0,
3230
1.99k
   bitmap->rowSize < 0 ? -bitmap->rowSize : bitmap->rowSize);
3231
1.99k
}
3232
3233
5.92k
void Splash::zeroRowAlpha(int y) {
3234
5.92k
  memset(&bitmap->alpha[y * bitmap->alphaRowSize], 0, bitmap->width);
3235
5.92k
}
3236
3237
0
void Splash::computeAlpha0Row(int y) {
3238
0
  Guchar *p = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize];
3239
0
  Guchar *q0;
3240
0
  if (parent->alpha0Bitmap) {
3241
0
    q0 = &parent->alpha0Bitmap->alpha[
3242
0
       (parentOffsetY + y) * parent->alpha0Bitmap->alphaRowSize
3243
0
       + parentOffsetX];
3244
0
  } else {
3245
0
    q0 = &parent->parent->bitmap->alpha[
3246
0
       (parentOffsetY + parent->parentOffsetY + y)
3247
0
         * parent->parent->bitmap->alphaRowSize
3248
0
       + parentOffsetX + parent->parentOffsetX];
3249
0
  }
3250
0
  Guchar *q1 = &parent->bitmap->alpha[
3251
0
       (parentOffsetY + y) * parent->bitmap->alphaRowSize
3252
0
       + parentOffsetX];
3253
0
  int w = alpha0Bitmap->width;
3254
0
  for (int x = 0; x < w; ++x) {
3255
0
    Guchar a0 = *q0++;
3256
0
    Guchar a1 = *q1++;
3257
0
    *p++ = (Guchar)(a0 + a1 - div255(a0 * a1));
3258
0
  }
3259
0
}
3260
3261
//------------------------------------------------------------------------
3262
3263
// Transform a point from user space to device space.
3264
inline void Splash::transform(SplashCoord *matrix,
3265
            SplashCoord xi, SplashCoord yi,
3266
66.1M
            SplashCoord *xo, SplashCoord *yo) {
3267
  //                          [ m[0] m[1] 0 ]
3268
  // [xo yo 1] = [xi yi 1] *  [ m[2] m[3] 0 ]
3269
  //                          [ m[4] m[5] 1 ]
3270
66.1M
  *xo = xi * matrix[0] + yi * matrix[2] + matrix[4];
3271
66.1M
  *yo = xi * matrix[1] + yi * matrix[3] + matrix[5];
3272
66.1M
}
3273
3274
//------------------------------------------------------------------------
3275
// SplashImageCache
3276
//------------------------------------------------------------------------
3277
3278
207k
SplashImageCache::SplashImageCache() {
3279
207k
  tag = NULL;
3280
207k
  width = 0;
3281
207k
  height = 0;
3282
207k
  mode = splashModeRGB8;
3283
207k
  alpha = gFalse;
3284
207k
  interpolate = gFalse;
3285
207k
  colorData = NULL;
3286
207k
  alphaData = NULL;
3287
207k
  refCount = 1;
3288
207k
}
3289
3290
207k
SplashImageCache::~SplashImageCache() {
3291
207k
  if (tag) {
3292
84
    delete tag;
3293
84
  }
3294
207k
  gfree(colorData);
3295
207k
  gfree(alphaData);
3296
207k
}
3297
3298
GBool SplashImageCache::match(GString *aTag, int aWidth, int aHeight,
3299
            SplashColorMode aMode, GBool aAlpha,
3300
1.08k
            GBool aInterpolate) {
3301
1.08k
  return aTag && tag && !aTag->cmp(tag) &&
3302
179
   aWidth == width && aHeight == height &&
3303
179
   aMode == mode && aAlpha == alpha &&
3304
179
   aInterpolate == interpolate;
3305
1.08k
}
3306
3307
void SplashImageCache::reset(GString *aTag, int aWidth, int aHeight,
3308
           SplashColorMode aMode, GBool aAlpha,
3309
903
           GBool aInterpolate) {
3310
903
  if (tag) {
3311
169
    delete tag;
3312
169
  }
3313
903
  if (aTag) {
3314
253
    tag = aTag->copy();
3315
650
  } else {
3316
650
    tag = NULL;
3317
650
  }
3318
903
  width = aWidth;
3319
903
  height = aHeight;
3320
903
  mode = aMode;
3321
903
  alpha = aAlpha;
3322
903
  interpolate = aInterpolate; 
3323
903
  gfree(colorData);
3324
903
  colorData = NULL;
3325
903
  gfree(alphaData);
3326
903
  alphaData = NULL;
3327
903
}
3328
3329
9.57k
void SplashImageCache::incRefCount() {
3330
9.57k
  ++refCount;
3331
9.57k
}
3332
3333
216k
void SplashImageCache::decRefCount() {
3334
216k
  --refCount;
3335
216k
  if (refCount == 0) {
3336
207k
    delete this;
3337
207k
  }
3338
216k
}
3339
3340
//------------------------------------------------------------------------
3341
// ImageScaler
3342
//------------------------------------------------------------------------
3343
3344
// Abstract base class.
3345
class ImageScaler {
3346
public:
3347
3348
864
  ImageScaler() {}
3349
864
  virtual ~ImageScaler() {}
3350
3351
  // Compute the next line of the scaled image.  This can be called up
3352
  // to [scaledHeight] times.
3353
  virtual void nextLine() = 0;
3354
3355
  // Retrieve the color and alpha data generated by the most recent
3356
  // call to nextLine().
3357
  virtual Guchar *colorData() = 0;
3358
  virtual Guchar *alphaData() = 0;
3359
};
3360
3361
//------------------------------------------------------------------------
3362
// BasicImageScaler
3363
//------------------------------------------------------------------------
3364
3365
// The actual image scaler.
3366
class BasicImageScaler: public ImageScaler {
3367
public:
3368
3369
  BasicImageScaler(SplashImageSource aSrc, void *aSrcData,
3370
       int aSrcWidth, int aSrcHeight, int aNComps, GBool aHasAlpha,
3371
       int aScaledWidth, int aScaledHeight, GBool aInterpolate);
3372
  virtual ~BasicImageScaler();
3373
  virtual void nextLine();
3374
1.72k
  virtual Guchar *colorData() { return colorLine; }
3375
34
  virtual Guchar *alphaData() { return alphaLine; }
3376
3377
protected:
3378
3379
  void vertDownscaleHorizDownscale();
3380
  void vertDownscaleHorizUpscaleNoInterp();
3381
  void vertDownscaleHorizUpscaleInterp();
3382
  void vertUpscaleHorizDownscaleNoInterp();
3383
  void vertUpscaleHorizDownscaleInterp();
3384
  void vertUpscaleHorizUpscaleNoInterp();
3385
  void vertUpscaleHorizUpscaleInterp();
3386
3387
  // source image data function
3388
  SplashImageSource src;
3389
  void *srcData;
3390
3391
  // source image size
3392
  int srcWidth;
3393
  int srcHeight;
3394
3395
  // scaled image size
3396
  int scaledWidth;
3397
  int scaledHeight;
3398
3399
  // number of color and alpha components
3400
  int nComps;
3401
  GBool hasAlpha;
3402
3403
  // params/state for vertical scaling
3404
  int yp, yq;
3405
  int yt, yn;
3406
  int ySrcCur, yScaledCur;
3407
  SplashCoord yInvScale;
3408
3409
  // params for horizontal scaling
3410
  int xp, xq;
3411
  SplashCoord xInvScale;
3412
3413
  // scaling function
3414
  void (BasicImageScaler::*scalingFunc)();
3415
3416
  // temporary buffers for vertical scaling
3417
  Guchar *colorTmpBuf0;
3418
  Guchar *colorTmpBuf1;
3419
  Guchar *colorTmpBuf2;
3420
  Guchar *alphaTmpBuf0;
3421
  Guchar *alphaTmpBuf1;
3422
  Guchar *alphaTmpBuf2;
3423
  Guint *colorAccBuf;
3424
  Guint *alphaAccBuf;
3425
3426
  // output of horizontal scaling
3427
  Guchar *colorLine;
3428
  Guchar *alphaLine;
3429
};
3430
3431
BasicImageScaler::BasicImageScaler(SplashImageSource aSrc, void *aSrcData,
3432
           int aSrcWidth, int aSrcHeight,
3433
           int aNComps, GBool aHasAlpha,
3434
           int aScaledWidth, int aScaledHeight,
3435
864
           GBool aInterpolate) {
3436
864
  colorTmpBuf0 = NULL;
3437
864
  colorTmpBuf1 = NULL;
3438
864
  colorTmpBuf2 = NULL;
3439
864
  alphaTmpBuf0 = NULL;
3440
864
  alphaTmpBuf1 = NULL;
3441
864
  alphaTmpBuf2 = NULL;
3442
864
  colorAccBuf = NULL;
3443
864
  alphaAccBuf = NULL;
3444
864
  colorLine = NULL;
3445
864
  alphaLine = NULL;
3446
3447
864
  src = aSrc;
3448
864
  srcData = aSrcData;
3449
864
  srcWidth = aSrcWidth;
3450
864
  srcHeight = aSrcHeight;
3451
864
  scaledWidth = aScaledWidth;
3452
864
  scaledHeight = aScaledHeight;
3453
864
  nComps = aNComps;
3454
864
  hasAlpha = aHasAlpha;
3455
3456
  // select scaling function; allocate buffers
3457
864
  if (scaledHeight <= srcHeight) {
3458
    // vertical downscaling
3459
864
    yp = srcHeight / scaledHeight;
3460
864
    yq = srcHeight % scaledHeight;
3461
864
    yt = 0;
3462
864
    colorTmpBuf0 = (Guchar *)gmallocn(srcWidth, nComps);
3463
864
    colorAccBuf = (Guint *)gmallocn(srcWidth, nComps * (int)sizeof(Guint));
3464
864
    if (hasAlpha) {
3465
17
      alphaTmpBuf0 = (Guchar *)gmalloc(srcWidth);
3466
17
      alphaAccBuf = (Guint *)gmallocn(srcWidth, sizeof(Guint));
3467
17
    }
3468
864
    if (scaledWidth <= srcWidth) {
3469
864
      scalingFunc = &BasicImageScaler::vertDownscaleHorizDownscale;
3470
864
    } else {
3471
0
      if (aInterpolate) {
3472
0
  scalingFunc = &BasicImageScaler::vertDownscaleHorizUpscaleInterp;
3473
0
      } else {
3474
0
  scalingFunc = &BasicImageScaler::vertDownscaleHorizUpscaleNoInterp;
3475
0
      }
3476
0
    }
3477
864
  } else {
3478
    // vertical upscaling
3479
0
    yp = scaledHeight / srcHeight;
3480
0
    yq = scaledHeight % srcHeight;
3481
0
    yt = 0;
3482
0
    yn = 0;
3483
0
    if (aInterpolate) {
3484
0
      yInvScale = (SplashCoord)srcHeight / (SplashCoord)scaledHeight;
3485
0
      colorTmpBuf0 = (Guchar *)gmallocn(srcWidth, nComps);
3486
0
      colorTmpBuf1 = (Guchar *)gmallocn(srcWidth, nComps);
3487
0
      if (hasAlpha) {
3488
0
  alphaTmpBuf0 = (Guchar *)gmalloc(srcWidth);
3489
0
  alphaTmpBuf1 = (Guchar *)gmalloc(srcWidth);
3490
0
      }
3491
0
      ySrcCur = 0;
3492
0
      yScaledCur = 0;
3493
0
      if (scaledWidth <= srcWidth) {
3494
0
  scalingFunc = &BasicImageScaler::vertUpscaleHorizDownscaleInterp;
3495
0
      } else {
3496
0
  colorTmpBuf2 = (Guchar *)gmallocn(srcWidth, nComps);
3497
0
  if (hasAlpha) {
3498
0
    alphaTmpBuf2 = (Guchar *)gmalloc(srcWidth);
3499
0
  }
3500
0
  scalingFunc = &BasicImageScaler::vertUpscaleHorizUpscaleInterp;
3501
0
      }
3502
0
    } else {
3503
0
      colorTmpBuf0 = (Guchar *)gmallocn(srcWidth, nComps);
3504
0
      if (hasAlpha) {
3505
0
  alphaTmpBuf0 = (Guchar *)gmalloc(srcWidth);
3506
0
      }
3507
0
      if (scaledWidth <= srcWidth) {
3508
0
  scalingFunc = &BasicImageScaler::vertUpscaleHorizDownscaleNoInterp;
3509
0
      } else {
3510
0
  scalingFunc = &BasicImageScaler::vertUpscaleHorizUpscaleNoInterp;
3511
0
      }
3512
0
    }
3513
0
  }
3514
864
  if (scaledWidth <= srcWidth) {
3515
864
    xp = srcWidth / scaledWidth;
3516
864
    xq = srcWidth % scaledWidth;
3517
864
  } else {
3518
0
    xp = scaledWidth / srcWidth;
3519
0
    xq = scaledWidth % srcWidth;
3520
0
    if (aInterpolate) {
3521
0
      xInvScale = (SplashCoord)srcWidth / (SplashCoord)scaledWidth;
3522
0
    }
3523
0
  }
3524
864
  colorLine = (Guchar *)gmallocn(scaledWidth, nComps);
3525
864
  if (hasAlpha) {
3526
17
    alphaLine = (Guchar *)gmalloc(scaledWidth);
3527
17
  }
3528
864
}
3529
3530
864
BasicImageScaler::~BasicImageScaler() {
3531
864
  gfree(colorTmpBuf0);
3532
864
  gfree(colorTmpBuf1);
3533
864
  gfree(colorTmpBuf2);
3534
864
  gfree(alphaTmpBuf0);
3535
864
  gfree(alphaTmpBuf1);
3536
864
  gfree(alphaTmpBuf2);
3537
864
  gfree(colorAccBuf);
3538
864
  gfree(alphaAccBuf);
3539
864
  gfree(colorLine);
3540
864
  gfree(alphaLine);
3541
864
}
3542
3543
864
void BasicImageScaler::nextLine() {
3544
864
  (this->*scalingFunc)();
3545
864
}
3546
3547
864
void BasicImageScaler::vertDownscaleHorizDownscale() {
3548
  //--- vert downscale
3549
864
  int yStep = yp;
3550
864
  yt += yq;
3551
864
  if (yt >= scaledHeight) {
3552
0
    yt -= scaledHeight;
3553
0
    ++yStep;
3554
0
  }
3555
864
  memset(colorAccBuf, 0, (srcWidth * nComps) * sizeof(Guint));
3556
864
  if (hasAlpha) {
3557
17
    memset(alphaAccBuf, 0, srcWidth * sizeof(Guint));
3558
17
  }
3559
864
  int nRowComps = srcWidth * nComps;
3560
82.8k
  for (int i = 0; i < yStep; ++i) {
3561
82.0k
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3562
71.0M
    for (int j = 0; j < nRowComps; ++j) {
3563
70.9M
      colorAccBuf[j] += colorTmpBuf0[j];
3564
70.9M
    }
3565
82.0k
    if (hasAlpha) {
3566
4.99M
      for (int j = 0; j < srcWidth; ++j) {
3567
4.98M
  alphaAccBuf[j] += alphaTmpBuf0[j];
3568
4.98M
      }
3569
11.3k
    }
3570
82.0k
  }
3571
3572
  //--- horiz downscale
3573
864
  int colorAcc[splashMaxColorComps];
3574
864
  int xt = 0;
3575
864
  int unscaledColorIdx = 0;
3576
864
  int unscaledAlphaIdx = 0;
3577
864
  int scaledColorIdx = 0;
3578
3579
1.72k
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
3580
864
    int xStep = xp;
3581
864
    xt += xq;
3582
864
    if (xt >= scaledWidth) {
3583
0
      xt -= scaledWidth;
3584
0
      ++xStep;
3585
0
    }
3586
3587
2.90k
    for (int j = 0; j < nComps; ++j) {
3588
2.04k
      colorAcc[j] = 0;
3589
2.04k
    }
3590
82.9k
    for (int i = 0; i < xStep; ++i) {
3591
267k
      for (int j = 0; j < nComps; ++j) {
3592
185k
  colorAcc[j] += colorAccBuf[unscaledColorIdx + j];
3593
185k
      }
3594
82.1k
      unscaledColorIdx += nComps;
3595
82.1k
    }
3596
864
    int nPixels = yStep * xStep;
3597
2.90k
    for (int j = 0; j < nComps; ++j) {
3598
2.04k
      colorLine[scaledColorIdx + j] = (Guchar)(colorAcc[j] / nPixels);
3599
2.04k
    }
3600
864
    scaledColorIdx += nComps;
3601
3602
864
    if (hasAlpha) {
3603
17
      int alphaAcc = 0;
3604
8.10k
      for (int i = 0; i < xStep; ++i) {
3605
8.09k
  alphaAcc += alphaAccBuf[unscaledAlphaIdx];
3606
8.09k
  ++unscaledAlphaIdx;
3607
8.09k
      }
3608
17
      alphaLine[scaledIdx] = (Guchar)(alphaAcc / nPixels);
3609
17
    }
3610
864
  }
3611
864
}
3612
3613
0
void BasicImageScaler::vertDownscaleHorizUpscaleNoInterp() {
3614
  //--- vert downscale
3615
0
  int yStep = yp;
3616
0
  yt += yq;
3617
0
  if (yt >= scaledHeight) {
3618
0
    yt -= scaledHeight;
3619
0
    ++yStep;
3620
0
  }
3621
0
  memset(colorAccBuf, 0, (srcWidth * nComps) * sizeof(Guint));
3622
0
  if (hasAlpha) {
3623
0
    memset(alphaAccBuf, 0, srcWidth * sizeof(Guint));
3624
0
  }
3625
0
  int nRowComps = srcWidth * nComps;
3626
0
  for (int i = 0; i < yStep; ++i) {
3627
0
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3628
0
    for (int j = 0; j < nRowComps; ++j) {
3629
0
      colorAccBuf[j] += colorTmpBuf0[j];
3630
0
    }
3631
0
    if (hasAlpha) {
3632
0
      for (int j = 0; j < srcWidth; ++j) {
3633
0
  alphaAccBuf[j] += alphaTmpBuf0[j];
3634
0
      }
3635
0
    }
3636
0
  }
3637
3638
  //--- horiz upscale
3639
0
  Guchar colorBuf[splashMaxColorComps];
3640
0
  int xt = 0;
3641
0
  int scaledColorIdx = 0;
3642
0
  int srcColorIdx = 0;
3643
0
  int scaledAlphaIdx = 0;
3644
0
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
3645
0
    int xStep = xp;
3646
0
    xt += xq;
3647
0
    if (xt >= srcWidth) {
3648
0
      xt -= srcWidth;
3649
0
      ++xStep;
3650
0
    }
3651
0
    for (int j = 0; j < nComps; ++j) {
3652
0
      colorBuf[j] = (Guchar)(colorAccBuf[srcColorIdx + j] / yStep);
3653
0
    }
3654
0
    srcColorIdx += nComps;
3655
0
    for (int i = 0; i < xStep; ++i) {
3656
0
      for (int j = 0; j < nComps; ++j) {
3657
0
  colorLine[scaledColorIdx + j] = colorBuf[j];
3658
0
      }
3659
0
      scaledColorIdx += nComps;
3660
0
    }
3661
0
    if (hasAlpha) {
3662
0
      Guchar alphaBuf = (Guchar)(alphaAccBuf[srcIdx] / yStep);
3663
0
      for (int i = 0; i < xStep; ++i) {
3664
0
  alphaLine[scaledAlphaIdx] = alphaBuf;
3665
0
  ++scaledAlphaIdx;
3666
0
      }
3667
0
    }
3668
0
  }
3669
0
}
3670
3671
0
void BasicImageScaler::vertDownscaleHorizUpscaleInterp() {
3672
  //--- vert downscale
3673
0
  int yStep = yp;
3674
0
  yt += yq;
3675
0
  if (yt >= scaledHeight) {
3676
0
    yt -= scaledHeight;
3677
0
    ++yStep;
3678
0
  }
3679
0
  memset(colorAccBuf, 0, (srcWidth * nComps) * sizeof(Guint));
3680
0
  if (hasAlpha) {
3681
0
    memset(alphaAccBuf, 0, srcWidth * sizeof(Guint));
3682
0
  }
3683
0
  int nRowComps = srcWidth * nComps;
3684
0
  for (int i = 0; i < yStep; ++i) {
3685
0
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3686
0
    for (int j = 0; j < nRowComps; ++j) {
3687
0
      colorAccBuf[j] += colorTmpBuf0[j];
3688
0
    }
3689
0
    if (hasAlpha) {
3690
0
      for (int j = 0; j < srcWidth; ++j) {
3691
0
  alphaAccBuf[j] += alphaTmpBuf0[j];
3692
0
      }
3693
0
    }
3694
0
  }
3695
0
  for (int j = 0; j < srcWidth * nComps; ++j) {
3696
0
    colorAccBuf[j] /= yStep;
3697
0
  }
3698
0
  if (hasAlpha) {
3699
0
    for (int j = 0; j < srcWidth; ++j) {
3700
0
      alphaAccBuf[j] /= yStep;
3701
0
    }
3702
0
  }
3703
3704
  //--- horiz upscale
3705
0
  int scaledColorIdx = 0;
3706
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
3707
0
    SplashCoord xs = ((SplashCoord)scaledIdx + 0.5) * xInvScale;
3708
0
    int x0 = splashFloor(xs - 0.5);
3709
0
    int x1 = x0 + 1;
3710
0
    SplashCoord s0 = (SplashCoord)x1 + 0.5 - xs;
3711
0
    SplashCoord s1 = (SplashCoord)1 - s0;
3712
0
    if (x0 < 0) {
3713
0
      x0 = 0;
3714
0
    }
3715
0
    if (x1 >= srcWidth) {
3716
0
      x1 = srcWidth - 1;
3717
0
    }
3718
0
    for (int j = 0; j < nComps; ++j) {
3719
0
      colorLine[scaledColorIdx + j] =
3720
0
    (Guchar)(int)(s0 * colorAccBuf[x0 * nComps + j] +
3721
0
      s1 * colorAccBuf[x1 * nComps + j]);
3722
0
    }
3723
0
    scaledColorIdx += nComps;
3724
0
    if (hasAlpha) {
3725
0
      alphaLine[scaledIdx] = (Guchar)(int)(s0 * alphaAccBuf[x0] +
3726
0
             s1 * alphaAccBuf[x1]);
3727
0
    }
3728
0
  }
3729
0
}
3730
3731
0
void BasicImageScaler::vertUpscaleHorizDownscaleNoInterp() {
3732
  //--- vert upscale
3733
0
  if (yn == 0) {
3734
0
    yn = yp;
3735
0
    yt += yq;
3736
0
    if (yt >= srcHeight) {
3737
0
      yt -= srcHeight;
3738
0
      ++yn;
3739
0
    }
3740
0
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3741
0
  }
3742
0
  --yn;
3743
3744
  //--- horiz downscale
3745
0
  int colorAcc[splashMaxColorComps];
3746
0
  int xt = 0;
3747
0
  int unscaledColorIdx = 0;
3748
0
  int unscaledAlphaIdx = 0;
3749
0
  int scaledColorIdx = 0;
3750
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
3751
0
    int xStep = xp;
3752
0
    xt += xq;
3753
0
    if (xt >= scaledWidth) {
3754
0
      xt -= scaledWidth;
3755
0
      ++xStep;
3756
0
    }
3757
3758
0
    for (int j = 0; j < nComps; ++j) {
3759
0
      colorAcc[j] = 0;
3760
0
    }
3761
0
    for (int i = 0; i < xStep; ++i) {
3762
0
      for (int j = 0; j < nComps; ++j) {
3763
0
  colorAcc[j] += colorTmpBuf0[unscaledColorIdx + j];
3764
0
      }
3765
0
      unscaledColorIdx += nComps;
3766
0
    }
3767
0
    for (int j = 0; j < nComps; ++j) {
3768
0
      colorLine[scaledColorIdx + j] = (Guchar)(colorAcc[j] / xStep);
3769
0
    }
3770
0
    scaledColorIdx += nComps;
3771
3772
0
    if (hasAlpha) {
3773
0
      int alphaAcc = 0;
3774
0
      for (int i = 0; i < xStep; ++i) {
3775
0
  alphaAcc += alphaTmpBuf0[unscaledAlphaIdx];
3776
0
  ++unscaledAlphaIdx;
3777
0
      }
3778
0
      alphaLine[scaledIdx] = (Guchar)(alphaAcc / xStep);
3779
0
    }
3780
0
  }
3781
0
}
3782
3783
0
void BasicImageScaler::vertUpscaleHorizDownscaleInterp() {
3784
  //--- vert upscale
3785
0
  if (ySrcCur == 0) {
3786
0
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3787
0
    (*src)(srcData, colorTmpBuf1, alphaTmpBuf1);
3788
0
    ySrcCur = 1;
3789
0
  }
3790
0
  SplashCoord ys = ((SplashCoord)yScaledCur + 0.5) * yInvScale;
3791
0
  int y0 = splashFloor(ys - 0.5);
3792
0
  int y1 = y0 + 1;
3793
0
  SplashCoord vs0 = (SplashCoord)y1 + 0.5 - ys;
3794
0
  SplashCoord vs1 = (SplashCoord)1 - vs0;
3795
0
  if (y1 > ySrcCur && ySrcCur < srcHeight - 1) {
3796
0
    Guchar *t = colorTmpBuf0;
3797
0
    colorTmpBuf0 = colorTmpBuf1;
3798
0
    colorTmpBuf1 = t;
3799
0
    if (hasAlpha) {
3800
0
      t = alphaTmpBuf0;
3801
0
      alphaTmpBuf0 = alphaTmpBuf1;
3802
0
      alphaTmpBuf1 = t;
3803
0
    }
3804
0
    (*src)(srcData, colorTmpBuf1, alphaTmpBuf1);
3805
0
    ++ySrcCur;
3806
0
  }
3807
0
  Guchar *color0 = colorTmpBuf0;
3808
0
  Guchar *color1 = colorTmpBuf1;
3809
0
  Guchar *alpha0 = alphaTmpBuf0;
3810
0
  Guchar *alpha1 = alphaTmpBuf1;
3811
0
  if (y0 < 0) {
3812
0
    y0 = 0;
3813
0
    color1 = color0;
3814
0
    alpha1 = alpha0;
3815
0
  }
3816
0
  if (y1 >= srcHeight) {
3817
0
    y1 = srcHeight - 1;
3818
0
    color0 = color1;
3819
0
    alpha0 = alpha1;
3820
0
  }
3821
0
  ++yScaledCur;
3822
3823
  //--- horiz downscale
3824
0
  int colorAcc[splashMaxColorComps];
3825
0
  int xt = 0;
3826
0
  int unscaledColorIdx = 0;
3827
0
  int unscaledAlphaIdx = 0;
3828
0
  int scaledColorIdx = 0;
3829
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
3830
0
    int xStep = xp;
3831
0
    xt += xq;
3832
0
    if (xt >= scaledWidth) {
3833
0
      xt -= scaledWidth;
3834
0
      ++xStep;
3835
0
    }
3836
3837
0
    for (int j = 0; j < nComps; ++j) {
3838
0
      colorAcc[j] = 0;
3839
0
    }
3840
0
    for (int i = 0; i < xStep; ++i) {
3841
0
      for (int j = 0; j < nComps; ++j) {
3842
0
  colorAcc[j] += (int)(vs0 * (int)color0[unscaledColorIdx + j] +
3843
0
           vs1 * (int)color1[unscaledColorIdx + j]);
3844
0
      }
3845
0
      unscaledColorIdx += nComps;
3846
0
    }
3847
0
    for (int j = 0; j < nComps; ++j) {
3848
0
      colorLine[scaledColorIdx + j] = (Guchar)(colorAcc[j] / xStep);
3849
0
    }
3850
0
    scaledColorIdx += nComps;
3851
3852
0
    if (hasAlpha) {
3853
0
      int alphaAcc = 0;
3854
0
      for (int i = 0; i < xStep; ++i) {
3855
0
  alphaAcc += (int)(vs0 * (int)alpha0[unscaledAlphaIdx] +
3856
0
        vs1 * (int)alpha1[unscaledAlphaIdx]);
3857
0
  ++unscaledAlphaIdx;
3858
0
      }
3859
0
      alphaLine[scaledIdx] = (Guchar)(alphaAcc / xStep);
3860
0
    }
3861
0
  }
3862
0
}
3863
3864
0
void BasicImageScaler::vertUpscaleHorizUpscaleNoInterp() {
3865
  //--- vert upscale
3866
0
  if (yn == 0) {
3867
0
    yn = yp;
3868
0
    yt += yq;
3869
0
    if (yt >= srcHeight) {
3870
0
      yt -= srcHeight;
3871
0
      ++yn;
3872
0
    }
3873
0
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3874
0
  }
3875
0
  --yn;
3876
3877
  //--- horiz upscale
3878
0
  int xt = 0;
3879
0
  int scaledColorIdx = 0;
3880
0
  int srcColorIdx = 0;
3881
0
  int scaledAlphaIdx = 0;
3882
0
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
3883
0
    int xStep = xp;
3884
0
    xt += xq;
3885
0
    if (xt >= srcWidth) {
3886
0
      xt -= srcWidth;
3887
0
      ++xStep;
3888
0
    }
3889
0
    for (int i = 0; i < xStep; ++i) {
3890
0
      for (int j = 0; j < nComps; ++j) {
3891
0
  colorLine[scaledColorIdx + j] = colorTmpBuf0[srcColorIdx + j];
3892
0
      }
3893
0
      scaledColorIdx += nComps;
3894
0
    }
3895
0
    srcColorIdx += nComps;
3896
0
    if (hasAlpha) {
3897
0
      Guchar alphaBuf = alphaTmpBuf0[srcIdx];
3898
0
      for (int i = 0; i < xStep; ++i) {
3899
0
  alphaLine[scaledAlphaIdx] = alphaBuf;
3900
0
  ++scaledAlphaIdx;
3901
0
      }
3902
0
    }
3903
0
  }
3904
0
}
3905
3906
0
void BasicImageScaler::vertUpscaleHorizUpscaleInterp() {
3907
  //--- vert upscale
3908
0
  if (ySrcCur == 0) {
3909
0
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3910
0
    (*src)(srcData, colorTmpBuf1, alphaTmpBuf1);
3911
0
    ySrcCur = 1;
3912
0
  }
3913
0
  SplashCoord ys = ((SplashCoord)yScaledCur + 0.5) * yInvScale;
3914
0
  int y0 = splashFloor(ys - 0.5);
3915
0
  int y1 = y0 + 1;
3916
0
  SplashCoord vs0 = (SplashCoord)y1 + 0.5 - ys;
3917
0
  SplashCoord vs1 = (SplashCoord)1 - vs0;
3918
0
  if (y1 > ySrcCur && ySrcCur < srcHeight - 1) {
3919
0
    Guchar *t = colorTmpBuf0;
3920
0
    colorTmpBuf0 = colorTmpBuf1;
3921
0
    colorTmpBuf1 = t;
3922
0
    if (hasAlpha) {
3923
0
      t = alphaTmpBuf0;
3924
0
      alphaTmpBuf0 = alphaTmpBuf1;
3925
0
      alphaTmpBuf1 = t;
3926
0
    }
3927
0
    (*src)(srcData, colorTmpBuf1, alphaTmpBuf1);
3928
0
    ++ySrcCur;
3929
0
  }
3930
0
  Guchar *color0 = colorTmpBuf0;
3931
0
  Guchar *color1 = colorTmpBuf1;
3932
0
  Guchar *alpha0 = alphaTmpBuf0;
3933
0
  Guchar *alpha1 = alphaTmpBuf1;
3934
0
  if (y0 < 0) {
3935
0
    y0 = 0;
3936
0
    color1 = color0;
3937
0
    alpha1 = alpha0;
3938
0
  }
3939
0
  if (y1 >= srcHeight) {
3940
0
    y1 = srcHeight - 1;
3941
0
    color0 = color1;
3942
0
    alpha0 = alpha1;
3943
0
  }
3944
0
  ++yScaledCur;
3945
0
  for (int srcIdx = 0; srcIdx < srcWidth * nComps; ++srcIdx) {
3946
0
    colorTmpBuf2[srcIdx] = (Guchar)(int)(vs0 * (int)color0[srcIdx] +
3947
0
           vs1 * (int)color1[srcIdx]);
3948
0
  }
3949
0
  if (hasAlpha) {
3950
0
    for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
3951
0
      alphaTmpBuf2[srcIdx] = (Guchar)(int)(vs0 * (int)alpha0[srcIdx] +
3952
0
             vs1 * (int)alpha1[srcIdx]);
3953
0
    }
3954
0
  }
3955
3956
  //--- horiz upscale
3957
0
  int scaledColorIdx = 0;
3958
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
3959
0
    SplashCoord xs = ((SplashCoord)scaledIdx + 0.5) * xInvScale;
3960
0
    int x0 = splashFloor(xs - 0.5);
3961
0
    int x1 = x0 + 1;
3962
0
    SplashCoord hs0 = (SplashCoord)x1 + 0.5 - xs;
3963
0
    SplashCoord hs1 = (SplashCoord)1 - hs0;
3964
0
    if (x0 < 0) {
3965
0
      x0 = 0;
3966
0
    }
3967
0
    if (x1 >= srcWidth) {
3968
0
      x1 = srcWidth - 1;
3969
0
    }
3970
0
    for (int j = 0; j < nComps; ++j) {
3971
0
      colorLine[scaledColorIdx + j] =
3972
0
    (Guchar)(int)(hs0 * (int)colorTmpBuf2[x0 * nComps + j] +
3973
0
      hs1 * (int)colorTmpBuf2[x1 * nComps + j]);
3974
0
    }
3975
0
    scaledColorIdx += nComps;
3976
0
    if (hasAlpha) {
3977
0
      alphaLine[scaledIdx] = (Guchar)(int)(hs0 * (int)alphaTmpBuf2[x0] +
3978
0
             hs1 * (int)alphaTmpBuf2[x1]);
3979
0
    }
3980
0
  }
3981
0
}
3982
3983
//------------------------------------------------------------------------
3984
// SavingImageScaler
3985
//------------------------------------------------------------------------
3986
3987
// Wrapper around BasicImageScaler that saves the scaled image for use
3988
// by ReplayImageScaler.
3989
class SavingImageScaler: public BasicImageScaler {
3990
public:
3991
3992
  SavingImageScaler(SplashImageSource aSrc, void *aSrcData,
3993
        int aSrcWidth, int aSrcHeight, int aNComps, GBool aHasAlpha,
3994
        int aScaledWidth, int aScaledHeight, GBool aInterpolate,
3995
        Guchar *aColorCache, Guchar *aAlphaCache);
3996
  virtual void nextLine();
3997
3998
private:
3999
4000
  Guchar *colorPtr;
4001
  Guchar *alphaPtr;
4002
};
4003
4004
SavingImageScaler::SavingImageScaler(SplashImageSource aSrc, void *aSrcData,
4005
             int aSrcWidth, int aSrcHeight,
4006
             int aNComps, GBool aHasAlpha,
4007
             int aScaledWidth, int aScaledHeight,
4008
             GBool aInterpolate,
4009
             Guchar *aColorCache, Guchar *aAlphaCache):
4010
864
  BasicImageScaler(aSrc, aSrcData, aSrcWidth, aSrcHeight, aNComps, aHasAlpha,
4011
864
       aScaledWidth, aScaledHeight, aInterpolate)
4012
864
{
4013
864
  colorPtr = aColorCache;
4014
864
  alphaPtr = aAlphaCache;
4015
864
}
4016
4017
864
void SavingImageScaler::nextLine() {
4018
864
  BasicImageScaler::nextLine();
4019
864
  memcpy(colorPtr, colorData(), scaledWidth * nComps);
4020
864
  colorPtr += scaledWidth * nComps;
4021
864
  if (hasAlpha) {
4022
17
    memcpy(alphaPtr, alphaData(), scaledWidth);
4023
17
    alphaPtr += scaledWidth;
4024
17
  }
4025
864
}
4026
4027
//------------------------------------------------------------------------
4028
// ReplayImageScaler
4029
//------------------------------------------------------------------------
4030
4031
// "Replay" a scaled image saved by SavingImageScaler.
4032
class ReplayImageScaler: public ImageScaler {
4033
public:
4034
4035
  ReplayImageScaler(int aNComps, GBool aHasAlpha,
4036
        int aScaledWidth,
4037
        Guchar *aColorCache, Guchar *aAlphaCache);
4038
  virtual void nextLine();
4039
0
  virtual Guchar *colorData() { return colorLine; }
4040
0
  virtual Guchar *alphaData() { return alphaLine; }
4041
4042
private:
4043
4044
  int nComps;
4045
  GBool hasAlpha;
4046
  int scaledWidth;
4047
  Guchar *colorPtr;
4048
  Guchar *alphaPtr;
4049
  Guchar *colorLine;
4050
  Guchar *alphaLine;
4051
};
4052
4053
ReplayImageScaler::ReplayImageScaler(int aNComps, GBool aHasAlpha,
4054
             int aScaledWidth,
4055
0
             Guchar *aColorCache, Guchar *aAlphaCache) {
4056
0
  nComps = aNComps;
4057
0
  hasAlpha = aHasAlpha;
4058
0
  scaledWidth = aScaledWidth;
4059
0
  colorPtr = aColorCache;
4060
0
  alphaPtr = aAlphaCache;
4061
0
  colorLine = NULL;
4062
0
  alphaLine = NULL;
4063
0
}
4064
4065
0
void ReplayImageScaler::nextLine() {
4066
0
  colorLine = colorPtr;
4067
0
  alphaLine = alphaPtr;
4068
0
  colorPtr += scaledWidth * nComps;
4069
0
  if (hasAlpha) {
4070
0
    alphaPtr += scaledWidth;
4071
0
  }
4072
0
}
4073
4074
//------------------------------------------------------------------------
4075
// ImageMaskScaler
4076
//------------------------------------------------------------------------
4077
4078
class ImageMaskScaler {
4079
public:
4080
4081
  // Set up a MaskScaler to scale from [srcWidth]x[srcHeight] to
4082
  // [scaledWidth]x[scaledHeight].  The [interpolate] flag controls
4083
  // filtering on upsampling, and the [antialias] flag controls
4084
  // filtering on downsampling.
4085
  ImageMaskScaler(SplashImageMaskSource aSrc, void *aSrcData,
4086
      int aSrcWidth, int aSrcHeight,
4087
      int aScaledWidth, int aScaledHeight,
4088
      GBool aInterpolate, GBool aAntialias);
4089
4090
  ~ImageMaskScaler();
4091
4092
  // Compute the next line of the scaled image mask.  This can be
4093
  // called up to [scaledHeight] times.
4094
  void nextLine();
4095
4096
  // Retrieve the data generated by the most recent call to
4097
  // nextLine().
4098
10.1M
  Guchar *data() { return line; }
4099
4100
private:
4101
4102
  void vertDownscaleHorizDownscale();
4103
  void vertDownscaleHorizDownscaleThresh();
4104
  void vertDownscaleHorizUpscaleNoInterp();
4105
  void vertDownscaleHorizUpscaleInterp();
4106
  void vertDownscaleHorizUpscaleThresh();
4107
  void vertUpscaleHorizDownscaleNoInterp();
4108
  void vertUpscaleHorizDownscaleInterp();
4109
  void vertUpscaleHorizDownscaleThresh();
4110
  void vertUpscaleHorizUpscaleNoInterp();
4111
  void vertUpscaleHorizUpscaleInterp();
4112
4113
  // source image data function
4114
  SplashImageMaskSource src;
4115
  void *srcData;
4116
4117
  // source image size
4118
  int srcWidth;
4119
  int srcHeight;
4120
4121
  // scaled image size
4122
  int scaledWidth;
4123
  int scaledHeight;
4124
4125
  // params/state for vertical scaling
4126
  int yp, yq;
4127
  int yt, yn;
4128
  int ySrcCur, yScaledCur;
4129
  SplashCoord yInvScale;
4130
4131
  // params for horizontal scaling
4132
  int xp, xq;
4133
  SplashCoord xInvScale;
4134
4135
  // vertical and horizontal scaling functions
4136
  void (ImageMaskScaler::*scalingFunc)();
4137
4138
  // temporary buffers for vertical scaling
4139
  Guchar *tmpBuf0;
4140
  Guchar *tmpBuf1;
4141
  Guchar *tmpBuf2;
4142
  Guint *accBuf;
4143
4144
  // output of horizontal scaling
4145
  Guchar *line;
4146
};
4147
4148
ImageMaskScaler::ImageMaskScaler(SplashImageMaskSource aSrc, void *aSrcData,
4149
         int aSrcWidth, int aSrcHeight,
4150
         int aScaledWidth, int aScaledHeight,
4151
756
         GBool aInterpolate, GBool aAntialias) {
4152
756
  tmpBuf0 = NULL;
4153
756
  tmpBuf1 = NULL;
4154
756
  tmpBuf2 = NULL;
4155
756
  accBuf = NULL;
4156
756
  line = NULL;
4157
4158
756
  src = aSrc;
4159
756
  srcData = aSrcData;
4160
756
  srcWidth = aSrcWidth;
4161
756
  srcHeight = aSrcHeight;
4162
756
  scaledWidth = aScaledWidth;
4163
756
  scaledHeight = aScaledHeight;
4164
4165
  // select scaling function; allocate buffers
4166
756
  if (scaledHeight <= srcHeight) {
4167
    // vertical downscaling
4168
703
    yp = srcHeight / scaledHeight;
4169
703
    yq = srcHeight % scaledHeight;
4170
703
    yt = 0;
4171
703
    tmpBuf0 = (Guchar *)gmalloc(srcWidth);
4172
703
    accBuf = (Guint *)gmallocn(srcWidth, sizeof(Guint));
4173
703
    if (scaledWidth <= srcWidth) {
4174
639
      if (!aAntialias) {
4175
0
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizDownscaleThresh;
4176
639
      } else {
4177
639
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizDownscale;
4178
639
      }
4179
639
    } else {
4180
64
      if (!aAntialias) {
4181
0
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizUpscaleThresh;
4182
64
      } else if (aInterpolate) {
4183
0
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizUpscaleInterp;
4184
64
      } else {
4185
64
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizUpscaleNoInterp;
4186
64
      }
4187
64
    }
4188
703
  } else {
4189
    // vertical upscaling
4190
53
    yp = scaledHeight / srcHeight;
4191
53
    yq = scaledHeight % srcHeight;
4192
53
    yt = 0;
4193
53
    yn = 0;
4194
53
    if (!aAntialias) {
4195
0
      tmpBuf0 = (Guchar *)gmalloc(srcWidth);
4196
0
      if (scaledWidth <= srcWidth) {
4197
0
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizDownscaleThresh;
4198
0
      } else {
4199
0
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizUpscaleNoInterp;
4200
0
      }
4201
53
    } else if (aInterpolate) {
4202
0
      yInvScale = (SplashCoord)srcHeight / (SplashCoord)scaledHeight;
4203
0
      tmpBuf0 = (Guchar *)gmalloc(srcWidth);
4204
0
      tmpBuf1 = (Guchar *)gmalloc(srcWidth);
4205
0
      ySrcCur = 0;
4206
0
      yScaledCur = 0;
4207
0
      if (scaledWidth <= srcWidth) {
4208
0
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizDownscaleInterp;
4209
0
      } else {
4210
0
  tmpBuf2 = (Guchar *)gmalloc(srcWidth);
4211
0
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizUpscaleInterp;
4212
0
      }
4213
53
    } else {
4214
53
      tmpBuf0 = (Guchar *)gmalloc(srcWidth);
4215
53
      if (scaledWidth <= srcWidth) {
4216
20
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizDownscaleNoInterp;
4217
33
      } else {
4218
33
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizUpscaleNoInterp;
4219
33
      }
4220
53
    }
4221
53
  }
4222
756
  if (scaledWidth <= srcWidth) {
4223
659
    xp = srcWidth / scaledWidth;
4224
659
    xq = srcWidth % scaledWidth;
4225
659
  } else {
4226
97
    xp = scaledWidth / srcWidth;
4227
97
    xq = scaledWidth % srcWidth;
4228
97
    if (aInterpolate) {
4229
0
      xInvScale = (SplashCoord)srcWidth / (SplashCoord)scaledWidth;
4230
0
    }
4231
97
  }
4232
756
  line = (Guchar *)gmalloc(scaledWidth);
4233
756
}
4234
4235
755
ImageMaskScaler::~ImageMaskScaler() {
4236
755
  gfree(tmpBuf0);
4237
755
  gfree(tmpBuf1);
4238
755
  gfree(tmpBuf2);
4239
755
  gfree(accBuf);
4240
755
  gfree(line);
4241
755
}
4242
4243
10.1M
void ImageMaskScaler::nextLine() {
4244
10.1M
  (this->*scalingFunc)();
4245
10.1M
}
4246
4247
21.0k
void ImageMaskScaler::vertDownscaleHorizDownscale() {
4248
  //--- vert downscale
4249
21.0k
  int yStep = yp;
4250
21.0k
  yt += yq;
4251
21.0k
  if (yt >= scaledHeight) {
4252
722
    yt -= scaledHeight;
4253
722
    ++yStep;
4254
722
  }
4255
21.0k
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4256
57.5k
  for (int i = 0; i < yStep; ++i) {
4257
36.5k
    (*src)(srcData, tmpBuf0);
4258
3.15M
    for (int j = 0; j < srcWidth; ++j) {
4259
3.11M
      accBuf[j] += tmpBuf0[j];
4260
3.11M
    }
4261
36.5k
  }
4262
4263
  //--- horiz downscale
4264
21.0k
  int acc;
4265
21.0k
  int xt = 0;
4266
21.0k
  int unscaledIdx = 0;
4267
2.57M
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4268
2.55M
    int xStep = xp;
4269
2.55M
    xt += xq;
4270
2.55M
    if (xt >= scaledWidth) {
4271
0
      xt -= scaledWidth;
4272
0
      ++xStep;
4273
0
    }
4274
4275
2.55M
    acc = 0;
4276
5.19M
    for (int i = 0; i < xStep; ++i) {
4277
2.63M
      acc += accBuf[unscaledIdx];
4278
2.63M
      ++unscaledIdx;
4279
2.63M
    }
4280
2.55M
    line[scaledIdx] = (Guchar)((255 * acc) / (xStep * yStep));
4281
2.55M
  }
4282
21.0k
}
4283
4284
0
void ImageMaskScaler::vertDownscaleHorizDownscaleThresh() {
4285
  //--- vert downscale
4286
0
  int yStep = yp;
4287
0
  yt += yq;
4288
0
  if (yt >= scaledHeight) {
4289
0
    yt -= scaledHeight;
4290
0
    ++yStep;
4291
0
  }
4292
0
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4293
0
  for (int i = 0; i < yStep; ++i) {
4294
0
    (*src)(srcData, tmpBuf0);
4295
0
    for (int j = 0; j < srcWidth; ++j) {
4296
0
      accBuf[j] += tmpBuf0[j];
4297
0
    }
4298
0
  }
4299
4300
  //--- horiz downscale
4301
0
  int acc;
4302
0
  int xt = 0;
4303
0
  int unscaledIdx = 0;
4304
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4305
0
    int xStep = xp;
4306
0
    xt += xq;
4307
0
    if (xt >= scaledWidth) {
4308
0
      xt -= scaledWidth;
4309
0
      ++xStep;
4310
0
    }
4311
4312
0
    acc = 0;
4313
0
    for (int i = 0; i < xStep; ++i) {
4314
0
      acc += accBuf[unscaledIdx];
4315
0
      ++unscaledIdx;
4316
0
    }
4317
0
    line[scaledIdx] = acc > ((xStep * yStep) >> 1) ? (Guchar)255 : (Guchar)0;
4318
0
  }
4319
0
}
4320
4321
27.3k
void ImageMaskScaler::vertDownscaleHorizUpscaleNoInterp() {
4322
  //--- vert downscale
4323
27.3k
  int yStep = yp;
4324
27.3k
  yt += yq;
4325
27.3k
  if (yt >= scaledHeight) {
4326
0
    yt -= scaledHeight;
4327
0
    ++yStep;
4328
0
  }
4329
27.3k
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4330
54.8k
  for (int i = 0; i < yStep; ++i) {
4331
27.5k
    (*src)(srcData, tmpBuf0);
4332
1.52M
    for (int j = 0; j < srcWidth; ++j) {
4333
1.49M
      accBuf[j] += tmpBuf0[j];
4334
1.49M
    }
4335
27.5k
  }
4336
4337
  //--- horiz upscale
4338
27.3k
  int xt = 0;
4339
27.3k
  int scaledIdx = 0;
4340
1.51M
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
4341
1.49M
    int xStep = xp;
4342
1.49M
    xt += xq;
4343
1.49M
    if (xt >= srcWidth) {
4344
132k
      xt -= srcWidth;
4345
132k
      ++xStep;
4346
132k
    }
4347
1.49M
    Guchar buf = (Guchar)((255 * accBuf[srcIdx]) / yStep);
4348
103M
    for (int i = 0; i < xStep; ++i) {
4349
102M
      line[scaledIdx] = buf;
4350
102M
      ++scaledIdx;
4351
102M
    }
4352
1.49M
  }
4353
27.3k
}
4354
4355
0
void ImageMaskScaler::vertDownscaleHorizUpscaleInterp() {
4356
  //--- vert downscale
4357
0
  int yStep = yp;
4358
0
  yt += yq;
4359
0
  if (yt >= scaledHeight) {
4360
0
    yt -= scaledHeight;
4361
0
    ++yStep;
4362
0
  }
4363
0
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4364
0
  for (int i = 0; i < yStep; ++i) {
4365
0
    (*src)(srcData, tmpBuf0);
4366
0
    for (int j = 0; j < srcWidth; ++j) {
4367
0
      accBuf[j] += tmpBuf0[j];
4368
0
    }
4369
0
  }
4370
0
  for (int j = 0; j < srcWidth; ++j) {
4371
0
    accBuf[j] = (255 * accBuf[j]) / yStep;
4372
0
  }
4373
4374
  //--- horiz upscale
4375
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4376
0
    SplashCoord xs = ((SplashCoord)scaledIdx + 0.5) * xInvScale;
4377
0
    int x0 = splashFloor(xs - 0.5);
4378
0
    int x1 = x0 + 1;
4379
0
    SplashCoord s0 = (SplashCoord)x1 + 0.5 - xs;
4380
0
    SplashCoord s1 = (SplashCoord)1 - s0;
4381
0
    if (x0 < 0) {
4382
0
      x0 = 0;
4383
0
    }
4384
0
    if (x1 >= srcWidth) {
4385
0
      x1 = srcWidth - 1;
4386
0
    }
4387
0
    line[scaledIdx] = (Guchar)(int)(s0 * accBuf[x0] + s1 * accBuf[x1]);
4388
0
  }
4389
0
}
4390
4391
0
void ImageMaskScaler::vertDownscaleHorizUpscaleThresh() {
4392
  //--- vert downscale
4393
0
  int yStep = yp;
4394
0
  yt += yq;
4395
0
  if (yt >= scaledHeight) {
4396
0
    yt -= scaledHeight;
4397
0
    ++yStep;
4398
0
  }
4399
0
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4400
0
  for (int i = 0; i < yStep; ++i) {
4401
0
    (*src)(srcData, tmpBuf0);
4402
0
    for (int j = 0; j < srcWidth; ++j) {
4403
0
      accBuf[j] += tmpBuf0[j];
4404
0
    }
4405
0
  }
4406
4407
  //--- horiz upscale
4408
0
  int xt = 0;
4409
0
  int scaledIdx = 0;
4410
0
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
4411
0
    int xStep = xp;
4412
0
    xt += xq;
4413
0
    if (xt >= srcWidth) {
4414
0
      xt -= srcWidth;
4415
0
      ++xStep;
4416
0
    }
4417
0
    Guchar buf = accBuf[srcIdx] > (Guint)(yStep >> 1) ? (Guchar)255 : (Guchar)0;
4418
0
    for (int i = 0; i < xStep; ++i) {
4419
0
      line[scaledIdx] = buf;
4420
0
      ++scaledIdx;
4421
0
    }
4422
0
  }
4423
0
}
4424
4425
10.1M
void ImageMaskScaler::vertUpscaleHorizDownscaleNoInterp() {
4426
  //--- vert upscale
4427
10.1M
  if (yn == 0) {
4428
2.99k
    yn = yp;
4429
2.99k
    yt += yq;
4430
2.99k
    if (yt >= srcHeight) {
4431
828
      yt -= srcHeight;
4432
828
      ++yn;
4433
828
    }
4434
2.99k
    (*src)(srcData, tmpBuf0);
4435
2.99k
  }
4436
10.1M
  --yn;
4437
4438
  //--- horiz downscale
4439
10.1M
  int acc;
4440
10.1M
  int xt = 0;
4441
10.1M
  int unscaledIdx = 0;
4442
21.0M
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4443
10.9M
    int xStep = xp;
4444
10.9M
    xt += xq;
4445
10.9M
    if (xt >= scaledWidth) {
4446
0
      xt -= scaledWidth;
4447
0
      ++xStep;
4448
0
    }
4449
4450
10.9M
    acc = 0;
4451
82.5M
    for (int i = 0; i < xStep; ++i) {
4452
71.5M
      acc += tmpBuf0[unscaledIdx];
4453
71.5M
      ++unscaledIdx;
4454
71.5M
    }
4455
10.9M
    line[scaledIdx] = (Guchar)((255 * acc) / xStep);
4456
10.9M
  }
4457
10.1M
}
4458
4459
0
void ImageMaskScaler::vertUpscaleHorizDownscaleInterp() {
4460
  //--- vert upscale
4461
0
  if (ySrcCur == 0) {
4462
0
    (*src)(srcData, tmpBuf0);
4463
0
    (*src)(srcData, tmpBuf1);
4464
0
    ySrcCur = 1;
4465
0
  }
4466
0
  SplashCoord ys = ((SplashCoord)yScaledCur + 0.5) * yInvScale;
4467
0
  int y0 = splashFloor(ys - 0.5);
4468
0
  int y1 = y0 + 1;
4469
0
  SplashCoord vs0 = (SplashCoord)y1 + 0.5 - ys;
4470
0
  SplashCoord vs1 = (SplashCoord)1 - vs0;
4471
0
  if (y1 > ySrcCur && ySrcCur < srcHeight - 1) {
4472
0
    Guchar *t = tmpBuf0;
4473
0
    tmpBuf0 = tmpBuf1;
4474
0
    tmpBuf1 = t;
4475
0
    (*src)(srcData, tmpBuf1);
4476
0
    ++ySrcCur;
4477
0
  }
4478
0
  Guchar *mask0 = tmpBuf0;
4479
0
  Guchar *mask1 = tmpBuf1;
4480
0
  if (y0 < 0) {
4481
0
    y0 = 0;
4482
0
    mask1 = mask0;
4483
0
  }
4484
0
  if (y1 >= srcHeight) {
4485
0
    y1 = srcHeight - 1;
4486
0
    mask0 = mask1;
4487
0
  }
4488
0
  ++yScaledCur;
4489
4490
  //--- horiz downscale
4491
0
  int acc;
4492
0
  int xt = 0;
4493
0
  int unscaledIdx = 0;
4494
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4495
0
    int xStep = xp;
4496
0
    xt += xq;
4497
0
    if (xt >= scaledWidth) {
4498
0
      xt -= scaledWidth;
4499
0
      ++xStep;
4500
0
    }
4501
4502
0
    acc = 0;
4503
0
    for (int i = 0; i < xStep; ++i) {
4504
0
      acc += (int)(vs0 * (int)mask0[unscaledIdx] +
4505
0
       vs1 * (int)mask1[unscaledIdx]);
4506
0
      ++unscaledIdx;
4507
0
    }
4508
0
    line[scaledIdx] = (Guchar)((255 * acc) / xStep);
4509
0
  }
4510
0
}
4511
4512
0
void ImageMaskScaler::vertUpscaleHorizDownscaleThresh() {
4513
  //--- vert upscale
4514
0
  if (yn == 0) {
4515
0
    yn = yp;
4516
0
    yt += yq;
4517
0
    if (yt >= srcHeight) {
4518
0
      yt -= srcHeight;
4519
0
      ++yn;
4520
0
    }
4521
0
    (*src)(srcData, tmpBuf0);
4522
0
  }
4523
0
  --yn;
4524
4525
  //--- horiz downscale
4526
0
  int acc;
4527
0
  int xt = 0;
4528
0
  int unscaledIdx = 0;
4529
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4530
0
    int xStep = xp;
4531
0
    xt += xq;
4532
0
    if (xt >= scaledWidth) {
4533
0
      xt -= scaledWidth;
4534
0
      ++xStep;
4535
0
    }
4536
4537
0
    acc = 0;
4538
0
    for (int i = 0; i < xStep; ++i) {
4539
0
      acc += tmpBuf0[unscaledIdx];
4540
0
      ++unscaledIdx;
4541
0
    }
4542
0
    line[scaledIdx] = acc > (xStep >> 1) ? (Guchar)255 : (Guchar)0;
4543
0
  }
4544
0
}
4545
4546
8.17k
void ImageMaskScaler::vertUpscaleHorizUpscaleNoInterp() {
4547
  //--- vert upscale
4548
8.17k
  if (yn == 0) {
4549
6.04k
    yn = yp;
4550
6.04k
    yt += yq;
4551
6.04k
    if (yt >= srcHeight) {
4552
665
      yt -= srcHeight;
4553
665
      ++yn;
4554
665
    }
4555
6.04k
    (*src)(srcData, tmpBuf0);
4556
6.04k
  }
4557
8.17k
  --yn;
4558
4559
  //--- horiz upscale
4560
8.17k
  int xt = 0;
4561
8.17k
  int scaledIdx = 0;
4562
79.0k
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
4563
70.8k
    int xStep = xp;
4564
70.8k
    xt += xq;
4565
70.8k
    if (xt >= srcWidth) {
4566
33.1k
      xt -= srcWidth;
4567
33.1k
      ++xStep;
4568
33.1k
    }
4569
70.8k
    Guchar buf = (Guchar)(255 * tmpBuf0[srcIdx]);
4570
13.8M
    for (int i = 0; i < xStep; ++i) {
4571
13.7M
      line[scaledIdx] = buf;
4572
13.7M
      ++scaledIdx;
4573
13.7M
    }
4574
70.8k
  }
4575
8.17k
}
4576
4577
0
void ImageMaskScaler::vertUpscaleHorizUpscaleInterp() {
4578
  //--- vert upscale
4579
0
  if (ySrcCur == 0) {
4580
0
    (*src)(srcData, tmpBuf0);
4581
0
    (*src)(srcData, tmpBuf1);
4582
0
    ySrcCur = 1;
4583
0
  }
4584
0
  SplashCoord ys = ((SplashCoord)yScaledCur + 0.5) * yInvScale;
4585
0
  int y0 = splashFloor(ys - 0.5);
4586
0
  int y1 = y0 + 1;
4587
0
  SplashCoord vs0 = (SplashCoord)255 * ((SplashCoord)y1 + 0.5 - ys);
4588
0
  SplashCoord vs1 = (SplashCoord)255 - vs0;
4589
0
  if (y1 > ySrcCur && ySrcCur < srcHeight - 1) {
4590
0
    Guchar *t = tmpBuf0;
4591
0
    tmpBuf0 = tmpBuf1;
4592
0
    tmpBuf1 = t;
4593
0
    (*src)(srcData, tmpBuf1);
4594
0
    ++ySrcCur;
4595
0
  }
4596
0
  Guchar *mask0 = tmpBuf0;
4597
0
  Guchar *mask1 = tmpBuf1;
4598
0
  if (y0 < 0) {
4599
0
    y0 = 0;
4600
0
    mask1 = mask0;
4601
0
  }
4602
0
  if (y1 >= srcHeight) {
4603
0
    y1 = srcHeight - 1;
4604
0
    mask0 = mask1;
4605
0
  }
4606
0
  ++yScaledCur;
4607
0
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
4608
0
    tmpBuf2[srcIdx] = (Guchar)(int)(vs0 * (int)mask0[srcIdx] +
4609
0
            vs1 * (int)mask1[srcIdx]);
4610
0
  }
4611
4612
  //--- horiz upscale
4613
0
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4614
0
    SplashCoord xs = ((SplashCoord)scaledIdx + 0.5) * xInvScale;
4615
0
    int x0 = splashFloor(xs - 0.5);
4616
0
    int x1 = x0 + 1;
4617
0
    SplashCoord hs0 = (SplashCoord)x1 + 0.5 - xs;
4618
0
    SplashCoord hs1 = (SplashCoord)1 - hs0;
4619
0
    if (x0 < 0) {
4620
0
      x0 = 0;
4621
0
    }
4622
0
    if (x1 >= srcWidth) {
4623
0
      x1 = srcWidth - 1;
4624
0
    }
4625
0
    line[scaledIdx] = (Guchar)(int)(hs0 * (int)tmpBuf2[x0] +
4626
0
            hs1 * (int)tmpBuf2[x1]);
4627
0
  }
4628
0
}
4629
4630
//------------------------------------------------------------------------
4631
// Splash
4632
//------------------------------------------------------------------------
4633
4634
Splash::Splash(SplashBitmap *bitmapA, GBool vectorAntialiasA,
4635
         SplashImageCache *imageCacheA,
4636
208k
         SplashScreenParams *screenParams) {
4637
208k
  bitmap = bitmapA;
4638
208k
  bitmapComps = splashColorModeNComps[bitmap->mode];
4639
208k
  vectorAntialias = vectorAntialiasA;
4640
208k
  state = new SplashState(bitmap->width, bitmap->height, vectorAntialias,
4641
208k
        screenParams);
4642
208k
  scanBuf = (Guchar *)gmalloc(bitmap->width);
4643
208k
  if (bitmap->mode == splashModeMono1) {
4644
159
    scanBuf2 = (Guchar *)gmalloc(bitmap->width);
4645
208k
  } else {
4646
208k
    scanBuf2 = NULL;
4647
208k
  }
4648
208k
  parent = NULL;
4649
208k
  deferredInit = gFalse;
4650
208k
  alpha0Bitmap = NULL;
4651
208k
  shapeBitmap = NULL;
4652
208k
  overprintMaskBitmap = NULL;
4653
208k
  minLineWidth = 0;
4654
208k
  clearModRegion();
4655
208k
  debugMode = gFalse;
4656
4657
208k
  if (imageCacheA) {
4658
1.53k
    imageCache = imageCacheA;
4659
1.53k
    imageCache->incRefCount();
4660
207k
  } else {
4661
207k
    imageCache = new SplashImageCache();
4662
207k
  }
4663
208k
}
4664
4665
Splash::Splash(SplashBitmap *bitmapA, GBool vectorAntialiasA,
4666
8.03k
         SplashImageCache *imageCacheA, SplashScreen *screenA) {
4667
8.03k
  bitmap = bitmapA;
4668
8.03k
  bitmapComps = splashColorModeNComps[bitmap->mode];
4669
8.03k
  vectorAntialias = vectorAntialiasA;
4670
8.03k
  state = new SplashState(bitmap->width, bitmap->height, vectorAntialias,
4671
8.03k
        screenA);
4672
8.03k
  scanBuf = (Guchar *)gmalloc(bitmap->width);
4673
8.03k
  if (bitmap->mode == splashModeMono1) {
4674
0
    scanBuf2 = (Guchar *)gmalloc(bitmap->width);
4675
8.03k
  } else {
4676
8.03k
    scanBuf2 = NULL;
4677
8.03k
  }
4678
8.03k
  parent = NULL;
4679
8.03k
  deferredInit = gFalse;
4680
8.03k
  alpha0Bitmap = NULL;
4681
8.03k
  shapeBitmap = NULL;
4682
8.03k
  overprintMaskBitmap = NULL;
4683
8.03k
  minLineWidth = 0;
4684
8.03k
  clearModRegion();
4685
8.03k
  debugMode = gFalse;
4686
4687
8.03k
  if (imageCacheA) {
4688
8.03k
    imageCache = imageCacheA;
4689
8.03k
    imageCache->incRefCount();
4690
8.03k
  } else {
4691
0
    imageCache = new SplashImageCache();
4692
0
  }
4693
8.03k
}
4694
4695
216k
Splash::~Splash() {
4696
216k
  imageCache->decRefCount();
4697
4698
217k
  while (state->next) {
4699
97
    restoreState();
4700
97
  }
4701
216k
  delete state;
4702
216k
  gfree(scanBuf);
4703
216k
  gfree(scanBuf2);
4704
216k
}
4705
4706
//------------------------------------------------------------------------
4707
// state read
4708
//------------------------------------------------------------------------
4709
4710
26.2k
SplashCoord *Splash::getMatrix() {
4711
26.2k
  return state->matrix;
4712
26.2k
}
4713
4714
6.60k
SplashPattern *Splash::getStrokePattern() {
4715
6.60k
  return state->strokePattern;
4716
6.60k
}
4717
4718
6.60k
SplashPattern *Splash::getFillPattern() {
4719
6.60k
  return state->fillPattern;
4720
6.60k
}
4721
4722
8.03k
SplashScreen *Splash::getScreen() {
4723
8.03k
  return state->screen;
4724
8.03k
}
4725
4726
0
SplashBlendFunc Splash::getBlendFunc() {
4727
0
  return state->blendFunc;
4728
0
}
4729
4730
0
SplashCoord Splash::getStrokeAlpha() {
4731
0
  return state->strokeAlpha;
4732
0
}
4733
4734
0
SplashCoord Splash::getFillAlpha() {
4735
0
  return state->fillAlpha;
4736
0
}
4737
4738
6.56k
SplashCoord Splash::getLineWidth() {
4739
6.56k
  return state->lineWidth;
4740
6.56k
}
4741
4742
6.56k
int Splash::getLineCap() {
4743
6.56k
  return state->lineCap;
4744
6.56k
}
4745
4746
6.56k
int Splash::getLineJoin() {
4747
6.56k
  return state->lineJoin;
4748
6.56k
}
4749
4750
0
SplashCoord Splash::getMiterLimit() {
4751
0
  return state->miterLimit;
4752
0
}
4753
4754
0
SplashCoord Splash::getFlatness() {
4755
0
  return state->flatness;
4756
0
}
4757
4758
6.56k
SplashCoord *Splash::getLineDash() {
4759
6.56k
  return state->lineDash;
4760
6.56k
}
4761
4762
6.56k
int Splash::getLineDashLength() {
4763
6.56k
  return state->lineDashLength;
4764
6.56k
}
4765
4766
6.56k
SplashCoord Splash::getLineDashPhase() {
4767
6.56k
  return state->lineDashPhase;
4768
6.56k
}
4769
4770
2.13M
SplashStrokeAdjustMode Splash::getStrokeAdjust() {
4771
2.13M
  return state->strokeAdjust;
4772
2.13M
}
4773
4774
33.1k
SplashClip *Splash::getClip() {
4775
33.1k
  return state->clip;
4776
33.1k
}
4777
4778
6.10k
SplashBitmap *Splash::getSoftMask() {
4779
6.10k
  return state->softMask;
4780
6.10k
}
4781
4782
10.3k
GBool Splash::getInNonIsolatedGroup() {
4783
10.3k
  return state->inNonIsolatedGroup;
4784
10.3k
}
4785
4786
6.07k
GBool Splash::getInKnockoutGroup() {
4787
6.07k
  return state->inKnockoutGroup;
4788
6.07k
}
4789
4790
//------------------------------------------------------------------------
4791
// state write
4792
//------------------------------------------------------------------------
4793
4794
2.62M
void Splash::setMatrix(SplashCoord *matrix) {
4795
2.62M
  memcpy(state->matrix, matrix, 6 * sizeof(SplashCoord));
4796
2.62M
}
4797
4798
491k
void Splash::setStrokePattern(SplashPattern *strokePattern) {
4799
491k
  state->setStrokePattern(strokePattern);
4800
491k
}
4801
4802
1.95M
void Splash::setFillPattern(SplashPattern *fillPattern) {
4803
1.95M
  state->setFillPattern(fillPattern);
4804
1.95M
}
4805
4806
0
void Splash::setScreen(SplashScreen *screen) {
4807
0
  state->setScreen(screen);
4808
0
}
4809
4810
4.27k
void Splash::setBlendFunc(SplashBlendFunc func) {
4811
4.27k
  state->blendFunc = func;
4812
4.27k
}
4813
4814
5.38k
void Splash::setStrokeAlpha(SplashCoord alpha) {
4815
5.38k
  state->strokeAlpha = alpha;
4816
5.38k
}
4817
4818
5.57k
void Splash::setFillAlpha(SplashCoord alpha) {
4819
5.57k
  state->fillAlpha = alpha;
4820
5.57k
}
4821
4822
351k
void Splash::setLineWidth(SplashCoord lineWidth) {
4823
351k
  state->lineWidth = lineWidth;
4824
351k
}
4825
4826
448k
void Splash::setLineCap(int lineCap) {
4827
448k
  if (lineCap >= 0 && lineCap <= 2) {
4828
448k
    state->lineCap = lineCap;
4829
448k
  } else {
4830
0
    state->lineCap = 0;
4831
0
  }
4832
448k
}
4833
4834
410k
void Splash::setLineJoin(int lineJoin) {
4835
410k
  if (lineJoin >= 0 && lineJoin <= 2) {
4836
410k
    state->lineJoin = lineJoin;
4837
410k
  } else {
4838
0
    state->lineJoin = 0;
4839
0
  }
4840
410k
}
4841
4842
417k
void Splash::setMiterLimit(SplashCoord miterLimit) {
4843
417k
  state->miterLimit = miterLimit;
4844
417k
}
4845
4846
193k
void Splash::setFlatness(SplashCoord flatness) {
4847
193k
  if (flatness < 1) {
4848
0
    state->flatness = 1;
4849
193k
  } else {
4850
193k
    state->flatness = flatness;
4851
193k
  }
4852
193k
}
4853
4854
void Splash::setLineDash(SplashCoord *lineDash, int lineDashLength,
4855
403k
       SplashCoord lineDashPhase) {
4856
403k
  state->setLineDash(lineDash, lineDashLength, lineDashPhase);
4857
403k
}
4858
4859
4.49M
void Splash::setStrokeAdjust(SplashStrokeAdjustMode strokeAdjust) {
4860
4.49M
  state->strokeAdjust = strokeAdjust;
4861
4.49M
}
4862
4863
3.23k
void Splash::setAlphaIsShape(GBool alphaIsShape) {
4864
3.23k
  state->alphaIsShape = alphaIsShape;
4865
3.23k
}
4866
4867
void Splash::clipResetToRect(SplashCoord x0, SplashCoord y0,
4868
0
           SplashCoord x1, SplashCoord y1) {
4869
0
  state->clipResetToRect(x0, y0, x1, y1);
4870
0
}
4871
4872
SplashError Splash::clipToRect(SplashCoord x0, SplashCoord y0,
4873
0
             SplashCoord x1, SplashCoord y1) {
4874
0
  return state->clipToRect(x0, y0, x1, y1);
4875
0
}
4876
4877
176k
SplashError Splash::clipToPath(SplashPath *path, GBool eo) {
4878
176k
  return state->clipToPath(path, eo);
4879
176k
}
4880
4881
12.4k
void Splash::setSoftMask(SplashBitmap *softMask, GBool deleteBitmap) {
4882
12.4k
  state->setSoftMask(softMask, deleteBitmap);
4883
12.4k
}
4884
4885
void Splash::setInTransparencyGroup(Splash *parentA,
4886
            int parentOffsetXA, int parentOffsetYA,
4887
            SplashAlphaBitmap *alpha0BitmapA,
4888
            SplashAlphaBitmap *shapeBitmapA,
4889
6.41k
            GBool nonIsolated, GBool knockout) {
4890
6.41k
  parent = parentA;
4891
6.41k
  parentOffsetX = parentOffsetXA;
4892
6.41k
  parentOffsetY = parentOffsetYA;
4893
6.41k
  deferredInit = gTrue;
4894
6.41k
  deferredInitYMin = 1;
4895
6.41k
  deferredInitYMax = 0;
4896
6.41k
  alpha0Bitmap = alpha0BitmapA;
4897
6.41k
  shapeBitmap = shapeBitmapA;
4898
6.41k
  state->inNonIsolatedGroup = nonIsolated;
4899
6.41k
  state->inKnockoutGroup = knockout;
4900
6.41k
}
4901
4902
1.90k
void Splash::forceDeferredInit(int y, int h) {
4903
1.90k
  useBitmapRow(y);
4904
1.90k
  useBitmapRow(y + h - 1);
4905
1.90k
}
4906
4907
// Check that alpha is 0 in the specified rectangle.
4908
// NB: This doesn't work correctly in a non-isolated group, because
4909
//     the rasterizer temporarily stores alpha_g instead of alpha.
4910
5.85k
GBool Splash::checkTransparentRect(int x, int y, int w, int h) {
4911
5.85k
  if (state->inNonIsolatedGroup) {
4912
0
    return gFalse;
4913
0
  }
4914
4915
4916
5.85k
  if (!bitmap->alpha) {
4917
0
    return gFalse;
4918
0
  }
4919
5.85k
  int yy0, yy1;
4920
5.85k
  if (deferredInit) {
4921
    // all transparency groups initialize alpha to zero, so anything
4922
    // outside of groupDestInit[YMin,YMax] will have alpha=0
4923
26
    yy0 = (y > deferredInitYMin) ? y : deferredInitYMin;
4924
26
    yy1 = (y + h - 1 < deferredInitYMax) ? y + h - 1 : deferredInitYMax;
4925
5.82k
  } else {
4926
5.82k
    yy0 = y;
4927
5.82k
    yy1 = y + h - 1;
4928
5.82k
  }
4929
5.85k
  Guchar *alphaP = &bitmap->alpha[yy0 * bitmap->alphaRowSize + x];
4930
7.69k
  for (int yy = yy0; yy <= yy1; ++yy) {
4931
7.66k
    for (int xx = 0; xx < w; ++xx) {
4932
5.82k
      if (alphaP[xx] != 0) {
4933
3.99k
  return gFalse;
4934
3.99k
      }
4935
5.82k
    }
4936
1.83k
    alphaP += bitmap->getAlphaRowSize();
4937
1.83k
  }
4938
1.86k
  return gTrue;
4939
5.85k
}
4940
4941
void Splash::setTransfer(Guchar *red, Guchar *green, Guchar *blue,
4942
598
       Guchar *gray) {
4943
598
  state->setTransfer(red, green, blue, gray);
4944
598
}
4945
4946
856k
void Splash::setOverprintMask(Guint overprintMask) {
4947
856k
  state->overprintMask = overprintMask;
4948
856k
}
4949
4950
4951
216k
void Splash::setEnablePathSimplification(GBool en) {
4952
216k
  state->enablePathSimplification = en;
4953
216k
}
4954
4955
//------------------------------------------------------------------------
4956
// state save/restore
4957
//------------------------------------------------------------------------
4958
4959
2.81M
void Splash::saveState() {
4960
2.81M
  SplashState *newState;
4961
4962
2.81M
  newState = state->copy();
4963
2.81M
  newState->next = state;
4964
2.81M
  state = newState;
4965
2.81M
}
4966
4967
2.94M
SplashError Splash::restoreState() {
4968
2.94M
  SplashState *oldState;
4969
4970
2.94M
  if (!state->next) {
4971
128k
    return splashErrNoSave;
4972
128k
  }
4973
2.81M
  oldState = state;
4974
2.81M
  state = state->next;
4975
2.81M
  delete oldState;
4976
2.81M
  return splashOk;
4977
2.94M
}
4978
4979
//------------------------------------------------------------------------
4980
// drawing operations
4981
//------------------------------------------------------------------------
4982
4983
208k
void Splash::clear(SplashColorPtr color, Guchar alpha) {
4984
208k
  SplashColorPtr row, p;
4985
208k
  Guchar mono;
4986
208k
  int x, y;
4987
4988
208k
  switch (bitmap->mode) {
4989
159
  case splashModeMono1:
4990
159
    mono = (color[0] & 0x80) ? 0xff : 0x00;
4991
159
    if (bitmap->rowSize < 0) {
4992
0
      memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
4993
0
       mono, -bitmap->rowSize * bitmap->height);
4994
159
    } else {
4995
159
      memset(bitmap->data, mono, bitmap->rowSize * bitmap->height);
4996
159
    }
4997
159
    break;
4998
151
  case splashModeMono8:
4999
151
    if (bitmap->rowSize < 0) {
5000
0
      memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
5001
0
       color[0], -bitmap->rowSize * bitmap->height);
5002
151
    } else {
5003
151
      memset(bitmap->data, color[0], bitmap->rowSize * bitmap->height);
5004
151
    }
5005
151
    break;
5006
207k
  case splashModeRGB8:
5007
207k
    if (color[0] == color[1] && color[1] == color[2]) {
5008
207k
      if (bitmap->rowSize < 0) {
5009
0
  memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
5010
0
         color[0], -bitmap->rowSize * bitmap->height);
5011
207k
      } else {
5012
207k
  memset(bitmap->data, color[0], bitmap->rowSize * bitmap->height);
5013
207k
      }
5014
207k
    } else {
5015
0
      row = bitmap->data;
5016
0
      for (y = 0; y < bitmap->height; ++y) {
5017
0
  p = row;
5018
0
  for (x = 0; x < bitmap->width; ++x) {
5019
0
    *p++ = color[0];
5020
0
    *p++ = color[1];
5021
0
    *p++ = color[2];
5022
0
  }
5023
0
  row += bitmap->rowSize;
5024
0
      }
5025
0
    }
5026
207k
    break;
5027
0
  case splashModeBGR8:
5028
0
    if (color[0] == color[1] && color[1] == color[2]) {
5029
0
      if (bitmap->rowSize < 0) {
5030
0
  memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
5031
0
         color[0], -bitmap->rowSize * bitmap->height);
5032
0
      } else {
5033
0
  memset(bitmap->data, color[0], bitmap->rowSize * bitmap->height);
5034
0
      }
5035
0
    } else {
5036
0
      row = bitmap->data;
5037
0
      for (y = 0; y < bitmap->height; ++y) {
5038
0
  p = row;
5039
0
  for (x = 0; x < bitmap->width; ++x) {
5040
0
    *p++ = color[2];
5041
0
    *p++ = color[1];
5042
0
    *p++ = color[0];
5043
0
  }
5044
0
  row += bitmap->rowSize;
5045
0
      }
5046
0
    }
5047
0
    break;
5048
0
#if SPLASH_CMYK
5049
0
  case splashModeCMYK8:
5050
0
    if (color[0] == color[1] && color[1] == color[2] && color[2] == color[3]) {
5051
0
      if (bitmap->rowSize < 0) {
5052
0
  memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
5053
0
         color[0], -bitmap->rowSize * bitmap->height);
5054
0
      } else {
5055
0
  memset(bitmap->data, color[0], bitmap->rowSize * bitmap->height);
5056
0
      }
5057
0
    } else {
5058
0
      row = bitmap->data;
5059
0
      for (y = 0; y < bitmap->height; ++y) {
5060
0
  p = row;
5061
0
  for (x = 0; x < bitmap->width; ++x) {
5062
0
    *p++ = color[0];
5063
0
    *p++ = color[1];
5064
0
    *p++ = color[2];
5065
0
    *p++ = color[3];
5066
0
  }
5067
0
  row += bitmap->rowSize;
5068
0
      }
5069
0
    }
5070
0
    break;
5071
208k
#endif
5072
208k
  }
5073
5074
208k
  if (bitmap->alpha) {
5075
207k
    memset(bitmap->alpha, alpha, bitmap->alphaRowSize * bitmap->height);
5076
207k
  }
5077
5078
208k
  updateModX(0);
5079
208k
  updateModY(0);
5080
208k
  updateModX(bitmap->width - 1);
5081
208k
  updateModY(bitmap->height - 1);
5082
208k
}
5083
5084
53.4k
SplashError Splash::stroke(SplashPath *path) {
5085
53.4k
  SplashPath *path2, *dPath;
5086
53.4k
  SplashCoord w, w2, lineDashMax, lineDashTotal;
5087
53.4k
  int lineCap, lineJoin, i;
5088
5089
53.4k
  if (debugMode) {
5090
0
    printf("stroke [dash:%d] [width:%.2f]:\n",
5091
0
     state->lineDashLength, (double)state->lineWidth);
5092
0
    dumpPath(path);
5093
0
  }
5094
53.4k
  opClipRes = splashClipAllOutside;
5095
53.4k
  if (path->length == 0) {
5096
1.23k
    return splashErrEmptyPath;
5097
1.23k
  }
5098
52.1k
  if (pathAllOutside(path, gTrue)) {
5099
29.0k
    return splashOk;
5100
29.0k
  }
5101
23.1k
  path2 = flattenPath(path, state->matrix, state->flatness);
5102
5103
  // Compute an approximation of the transformed line width.
5104
  // Given a CTM of [m0 m1],
5105
  //                [m2 m3]
5106
  // if |m0|*|m3| >= |m1|*|m2| then use min{|m0|,|m3|}, else
5107
  // use min{|m1|,|m2|}.
5108
  // This handles the common cases -- [s 0   ] and [0    s] --
5109
  //                                  [0 +/-s]     [+/-s 0]
5110
  // well, and still does something reasonable for the uncommon
5111
  // case transforms.
5112
23.1k
  double t0 = splashAbs(state->matrix[0]);
5113
23.1k
  double t1 = splashAbs(state->matrix[1]);
5114
23.1k
  double t2 = splashAbs(state->matrix[2]);
5115
23.1k
  double t3 = splashAbs(state->matrix[3]);
5116
23.1k
  double t01 = t0 * t0 + t1 * t1;
5117
23.1k
  double t23 = t2 * t2 + t3 * t3;
5118
23.1k
  w = sqrt((t01 > t23) ? t01 : t23);
5119
23.1k
  w2 = w * state->lineWidth;
5120
5121
  // construct the dashed path
5122
23.1k
  if (state->lineDashLength > 0) {
5123
5124
    // check the maximum transformed dash element length (using the
5125
    // same approximation as for line width) -- if it's less than 0.1
5126
    // pixel, don't apply the dash pattern; this avoids a huge
5127
    // performance/memory hit with PDF files that use absurd dash
5128
    // patterns like [0.0007 0.0003]
5129
    //
5130
    // we also skip the dash pattern for very long paths with short
5131
    // (but not necessarily sub-pixel) dash patterns -- Adobe appears
5132
    // to do something similar
5133
354
    lineDashTotal = 0;
5134
354
    lineDashMax = 0;
5135
4.14k
    for (i = 0; i < state->lineDashLength; ++i) {
5136
3.79k
      lineDashTotal += state->lineDash[i];
5137
3.79k
      if (state->lineDash[i] > lineDashMax) {
5138
573
  lineDashMax = state->lineDash[i];
5139
573
      }
5140
3.79k
    }
5141
    // Acrobat simply draws nothing if the dash array is [0]
5142
354
    if (lineDashTotal == 0) {
5143
5
      delete path2;
5144
5
      return splashOk;
5145
5
    }
5146
349
    if (w * lineDashMax > 0.1 &&
5147
339
  w * lineDashMax > 0.0001 * path->length) {
5148
339
      dPath = makeDashedPath(path2);
5149
339
      delete path2;
5150
339
      path2 = dPath;
5151
339
      if (path2->length == 0) {
5152
6
  delete path2;
5153
6
  return splashErrEmptyPath;
5154
6
      }
5155
339
    }
5156
349
  }
5157
5158
  // round caps on narrow lines look bad, and can't be
5159
  // stroke-adjusted, so use projecting caps instead (but we can't do
5160
  // this if there are zero-length dashes or segments, because those
5161
  // turn into round dots)
5162
23.1k
  lineCap = state->lineCap;
5163
23.1k
  lineJoin = state->lineJoin;
5164
23.1k
  if (state->strokeAdjust == splashStrokeAdjustCAD &&
5165
0
      w2 < 3.5) {
5166
0
    if (lineCap == splashLineCapRound &&
5167
0
  !state->lineDashContainsZeroLengthDashes() &&
5168
0
  !path->containsZeroLengthSubpaths()) {
5169
0
      lineCap = splashLineCapProjecting;
5170
0
    }
5171
0
    if (lineJoin == splashLineJoinRound) {
5172
0
      lineJoin = splashLineJoinBevel;
5173
0
    }
5174
0
  }
5175
5176
  // if there is a min line width set, and the transformed line width
5177
  // is smaller, use the min line width
5178
23.1k
  if (w > 0 && w2 < minLineWidth) {
5179
293
    strokeWide(path2, minLineWidth / w, splashLineCapButt, splashLineJoinBevel);
5180
22.8k
  } else if (bitmap->mode == splashModeMono1 || !vectorAntialias) {
5181
    // in monochrome mode or if antialiasing is disabled, use 0-width
5182
    // lines for any transformed line width <= 1 -- lines less than 1
5183
    // pixel wide look too fat without antialiasing
5184
0
    if (w2 < 1.001) {
5185
0
      strokeNarrow(path2);
5186
0
    } else {
5187
0
      strokeWide(path2, state->lineWidth, lineCap, lineJoin);
5188
0
    }
5189
22.8k
  } else {
5190
    // in gray and color modes, only use 0-width lines if the line
5191
    // width is explicitly set to 0
5192
22.8k
    if (state->lineWidth == 0) {
5193
2.07k
      strokeNarrow(path2);
5194
20.7k
    } else {
5195
20.7k
      strokeWide(path2, state->lineWidth, lineCap, lineJoin);
5196
20.7k
    }
5197
22.8k
  }
5198
5199
23.1k
  delete path2;
5200
23.1k
  return splashOk;
5201
23.1k
}
5202
5203
2.07k
void Splash::strokeNarrow(SplashPath *path) {
5204
2.07k
  SplashPipe pipe;
5205
2.07k
  SplashXPath *xPath;
5206
2.07k
  SplashXPathSeg *seg;
5207
2.07k
  int x0, x1, y0, y1, xa, xb, y;
5208
2.07k
  SplashCoord dxdy;
5209
2.07k
  SplashClipResult clipRes;
5210
2.07k
  int nClipRes[3];
5211
2.07k
  int i;
5212
5213
2.07k
  nClipRes[0] = nClipRes[1] = nClipRes[2] = 0;
5214
5215
2.07k
  xPath = new SplashXPath(path, state->matrix, state->flatness, gFalse,
5216
2.07k
        state->enablePathSimplification,
5217
2.07k
        state->strokeAdjust, state->clip);
5218
5219
2.07k
  pipeInit(&pipe, state->strokePattern,
5220
2.07k
     (Guchar)splashRound(state->strokeAlpha * 255),
5221
2.07k
     gFalse, gTrue, gFalse);
5222
5223
2.11M
  for (i = 0, seg = xPath->segs; i < xPath->length; ++i, ++seg) {
5224
2.11M
    if (seg->y0 <= seg->y1) {
5225
2.03M
      y0 = splashFloor(seg->y0);
5226
2.03M
      y1 = splashFloor(seg->y1);
5227
2.03M
      x0 = splashFloor(seg->x0);
5228
2.03M
      x1 = splashFloor(seg->x1);
5229
2.03M
    } else {
5230
72.9k
      y0 = splashFloor(seg->y1);
5231
72.9k
      y1 = splashFloor(seg->y0);
5232
72.9k
      x0 = splashFloor(seg->x1);
5233
72.9k
      x1 = splashFloor(seg->x0);
5234
72.9k
    }
5235
5236
    // With CAD-mode stroke adjustment, and a simple rectangular clip
5237
    // region, horizontal and vertical stroke-adjusted edges that fall
5238
    // slightly outside the clip region are adjusted back inside the
5239
    // clip region.  This avoids problems with narrow lines in
5240
    // slightly mismatched clip rectangles, which appear to be
5241
    // generated somewhat commonly by buggy CAD software.  This is
5242
    // similar to the code in SplashXPath::strokeAdjust() -- narrow
5243
    // strokes aren't stroke-adjusted, so this tweak has to be done
5244
    // here.
5245
2.11M
    if (y0 == y1 &&
5246
2.10M
  seg->y0 == seg->y1 &&
5247
2.03M
  state->clip->getIsSimple() &&
5248
1.18M
  state->strokeAdjust == splashStrokeAdjustCAD) {
5249
0
      SplashCoord cy0 = state->clip->getYMin();
5250
0
      SplashCoord cy1 = state->clip->getYMax();
5251
0
      int cyi0 = state->clip->getYMinI(state->strokeAdjust);
5252
0
      int cyi1 = state->clip->getYMaxI(state->strokeAdjust);
5253
0
      if (y0 == cyi0 - 1 && cy0 - seg->y0 < 0.5) {
5254
0
  y0 = y1 = y0 + 1;
5255
0
      } else if (y0 == cyi1 + 1 && seg->y0 - cy1 < 0.5) {
5256
0
  y0 = y1 = y0 - 1;
5257
0
      }
5258
2.11M
    } else if (x0 == x1 &&
5259
2.10M
  seg->x0 == seg->x1 &&
5260
2.10M
  state->clip->getIsSimple() &&
5261
1.24M
  state->strokeAdjust == splashStrokeAdjustCAD) {
5262
0
      SplashCoord cx0 = state->clip->getXMin();
5263
0
      SplashCoord cx1 = state->clip->getXMax();
5264
0
      int cxi0 = state->clip->getXMinI(state->strokeAdjust);
5265
0
      int cxi1 = state->clip->getXMaxI(state->strokeAdjust);
5266
0
      if (x0 == cxi0 - 1 && cx0 - seg->x0 < 0.5) {
5267
0
  x0 = x1 = x0 + 1;
5268
0
      } else if (x0 == cxi1 + 1 && seg->x0 - cx1 < 0.5) {
5269
0
  x0 = x1 = x0 - 1;
5270
0
      }
5271
0
    }
5272
5273
2.11M
    if ((clipRes = state->clip->testRect(x0 <= x1 ? x0 : x1, y0,
5274
2.11M
           x0 <= x1 ? x1 : x0, y1,
5275
2.11M
           state->strokeAdjust))
5276
2.11M
  != splashClipAllOutside) {
5277
5.52k
      if (y0 == y1) {
5278
1.39k
  if (x0 <= x1) {
5279
1.29k
    drawStrokeSpan(&pipe, x0, x1, y0, clipRes == splashClipAllInside);
5280
1.29k
  } else {
5281
97
    drawStrokeSpan(&pipe, x1, x0, y0, clipRes == splashClipAllInside);
5282
97
  }
5283
4.12k
      } else if (x0 == x1) {
5284
1.89k
  y = state->clip->getYMinI(state->strokeAdjust);
5285
1.89k
  if (y0 < y) {
5286
1.63k
    y0 = y;
5287
1.63k
  }
5288
1.89k
  y = state->clip->getYMaxI(state->strokeAdjust);
5289
1.89k
  if (y1 > y) {
5290
1.89k
    y1 = y;
5291
1.89k
  }
5292
3.78k
  for (y = y0; y <= y1; ++y) {
5293
1.89k
    drawStrokeSpan(&pipe, x0, x0, y, clipRes == splashClipAllInside);
5294
1.89k
  }
5295
2.23k
      } else {
5296
2.23k
  dxdy = seg->dxdy;
5297
2.23k
  y = state->clip->getYMinI(state->strokeAdjust);
5298
2.23k
  if (y0 < y) {
5299
1.86k
    y0 = y;
5300
1.86k
    x0 = splashFloor(seg->x0 + ((SplashCoord)y0 - seg->y0) * dxdy);
5301
1.86k
  }
5302
2.23k
  y = state->clip->getYMaxI(state->strokeAdjust);
5303
2.23k
  if (y1 > y) {
5304
2.03k
    y1 = y;
5305
2.03k
    x1 = splashFloor(seg->x0 + ((SplashCoord)y1 + 1 - seg->y0) * dxdy);
5306
2.03k
  }
5307
2.23k
  if (x0 <= x1) {
5308
1.03k
    xa = x0;
5309
2.06k
    for (y = y0; y <= y1; ++y) {
5310
1.03k
      if (y < y1) {
5311
0
        xb = splashFloor(seg->x0 +
5312
0
             ((SplashCoord)y + 1 - seg->y0) * dxdy);
5313
1.03k
      } else {
5314
1.03k
        xb = x1 + 1;
5315
1.03k
      }
5316
1.03k
      if (xa == xb) {
5317
0
        drawStrokeSpan(&pipe, xa, xa, y, clipRes == splashClipAllInside);
5318
1.03k
      } else {
5319
1.03k
        drawStrokeSpan(&pipe, xa, xb - 1, y,
5320
1.03k
           clipRes == splashClipAllInside);
5321
1.03k
      }
5322
1.03k
      xa = xb;
5323
1.03k
    }
5324
1.20k
  } else {
5325
1.20k
    xa = x0;
5326
2.41k
    for (y = y0; y <= y1; ++y) {
5327
1.20k
      if (y < y1) {
5328
0
        xb = splashFloor(seg->x0 +
5329
0
             ((SplashCoord)y + 1 - seg->y0) * dxdy);
5330
1.20k
      } else {
5331
1.20k
        xb = x1 - 1;
5332
1.20k
      }
5333
1.20k
      if (xa == xb) {
5334
0
        drawStrokeSpan(&pipe, xa, xa, y, clipRes == splashClipAllInside);
5335
1.20k
      } else {
5336
1.20k
        drawStrokeSpan(&pipe, xb + 1, xa, y,
5337
1.20k
           clipRes == splashClipAllInside);
5338
1.20k
      }
5339
1.20k
      xa = xb;
5340
1.20k
    }
5341
1.20k
  }
5342
2.23k
      }
5343
5.52k
    }
5344
2.11M
    ++nClipRes[clipRes];
5345
2.11M
  }
5346
5347
2.07k
  if (nClipRes[splashClipPartial] ||
5348
1.36k
      (nClipRes[splashClipAllInside] && nClipRes[splashClipAllOutside])) {
5349
1.36k
    opClipRes = splashClipPartial;
5350
1.36k
  } else if (nClipRes[splashClipAllInside]) {
5351
315
    opClipRes = splashClipAllInside;
5352
392
  } else {
5353
392
    opClipRes = splashClipAllOutside;
5354
392
  }
5355
5356
2.07k
  delete xPath;
5357
2.07k
}
5358
5359
void Splash::drawStrokeSpan(SplashPipe *pipe, int x0, int x1, int y,
5360
5.52k
          GBool noClip) {
5361
5.52k
  int x;
5362
5363
5.52k
  x = state->clip->getXMinI(state->strokeAdjust);
5364
5.52k
  if (x > x0) {
5365
1.24k
    x0 = x;
5366
1.24k
  }
5367
5.52k
  x = state->clip->getXMaxI(state->strokeAdjust);
5368
5.52k
  if (x < x1) {
5369
1.24k
    x1 = x;
5370
1.24k
  }
5371
5.52k
  if (x0 > x1) {
5372
194
    return;
5373
194
  }
5374
10.6k
  for (x = x0; x <= x1; ++x) {
5375
5.32k
    scanBuf[x] = 0xff;
5376
5.32k
  }
5377
5.32k
  if (!noClip) {
5378
4.35k
    if (!state->clip->clipSpanBinary(scanBuf, y, x0, x1, state->strokeAdjust)) {
5379
246
      return;
5380
246
    }
5381
4.35k
  }
5382
5.08k
  (this->*pipe->run)(pipe, x0, x1, y, scanBuf + x0, scanBuf + x0, NULL);
5383
5.08k
}
5384
5385
void Splash::strokeWide(SplashPath *path, SplashCoord w,
5386
21.0k
      int lineCap, int lineJoin) {
5387
21.0k
  SplashPath *path2;
5388
5389
21.0k
  path2 = makeStrokePath(path, w, lineCap, lineJoin, gFalse);
5390
21.0k
  fillWithPattern(path2, gFalse, state->strokePattern, state->strokeAlpha);
5391
21.0k
  delete path2;
5392
21.0k
}
5393
5394
SplashPath *Splash::flattenPath(SplashPath *path, SplashCoord *matrix,
5395
23.2k
        SplashCoord flatness) {
5396
23.2k
  SplashPath *fPath;
5397
23.2k
  SplashCoord flatness2;
5398
23.2k
  Guchar flag;
5399
23.2k
  int i;
5400
5401
23.2k
  fPath = new SplashPath();
5402
#if USE_FIXEDPOINT
5403
  flatness2 = flatness;
5404
#else
5405
23.2k
  flatness2 = flatness * flatness;
5406
23.2k
#endif
5407
23.2k
  i = 0;
5408
163k
  while (i < path->length) {
5409
140k
    flag = path->flags[i];
5410
140k
    if (flag & splashPathFirst) {
5411
37.2k
      fPath->moveTo(path->pts[i].x, path->pts[i].y);
5412
37.2k
      ++i;
5413
103k
    } else {
5414
103k
      if (flag & splashPathCurve) {
5415
10.9k
  flattenCurve(path->pts[i-1].x, path->pts[i-1].y,
5416
10.9k
         path->pts[i  ].x, path->pts[i  ].y,
5417
10.9k
         path->pts[i+1].x, path->pts[i+1].y,
5418
10.9k
         path->pts[i+2].x, path->pts[i+2].y,
5419
10.9k
         matrix, flatness2, fPath);
5420
10.9k
  i += 3;
5421
92.4k
      } else {
5422
92.4k
  fPath->lineTo(path->pts[i].x, path->pts[i].y);
5423
92.4k
  ++i;
5424
92.4k
      }
5425
103k
      if (path->flags[i-1] & splashPathClosed) {
5426
23.2k
  fPath->close();
5427
23.2k
      }
5428
103k
    }
5429
140k
  }
5430
23.2k
  return fPath;
5431
23.2k
}
5432
5433
void Splash::flattenCurve(SplashCoord x0, SplashCoord y0,
5434
        SplashCoord x1, SplashCoord y1,
5435
        SplashCoord x2, SplashCoord y2,
5436
        SplashCoord x3, SplashCoord y3,
5437
        SplashCoord *matrix, SplashCoord flatness2,
5438
10.9k
        SplashPath *fPath) {
5439
10.9k
  SplashCoord cx[splashMaxCurveSplits + 1][3];
5440
10.9k
  SplashCoord cy[splashMaxCurveSplits + 1][3];
5441
10.9k
  int cNext[splashMaxCurveSplits + 1];
5442
10.9k
  SplashCoord xl0, xl1, xl2, xr0, xr1, xr2, xr3, xx1, xx2, xh;
5443
10.9k
  SplashCoord yl0, yl1, yl2, yr0, yr1, yr2, yr3, yy1, yy2, yh;
5444
10.9k
  SplashCoord dx, dy, mx, my, tx, ty, d1, d2;
5445
10.9k
  int p1, p2, p3;
5446
5447
  // initial segment
5448
10.9k
  p1 = 0;
5449
10.9k
  p2 = splashMaxCurveSplits;
5450
10.9k
  cx[p1][0] = x0;  cy[p1][0] = y0;
5451
10.9k
  cx[p1][1] = x1;  cy[p1][1] = y1;
5452
10.9k
  cx[p1][2] = x2;  cy[p1][2] = y2;
5453
10.9k
  cx[p2][0] = x3;  cy[p2][0] = y3;
5454
10.9k
  cNext[p1] = p2;
5455
5456
19.8M
  while (p1 < splashMaxCurveSplits) {
5457
5458
    // get the next segment
5459
19.8M
    xl0 = cx[p1][0];  yl0 = cy[p1][0];
5460
19.8M
    xx1 = cx[p1][1];  yy1 = cy[p1][1];
5461
19.8M
    xx2 = cx[p1][2];  yy2 = cy[p1][2];
5462
19.8M
    p2 = cNext[p1];
5463
19.8M
    xr3 = cx[p2][0];  yr3 = cy[p2][0];
5464
5465
    // compute the distances (in device space) from the control points
5466
    // to the midpoint of the straight line (this is a bit of a hack,
5467
    // but it's much faster than computing the actual distances to the
5468
    // line)
5469
19.8M
    transform(matrix, (xl0 + xr3) * 0.5, (yl0 + yr3) * 0.5, &mx, &my);
5470
19.8M
    transform(matrix, xx1, yy1, &tx, &ty);
5471
#if USE_FIXEDPOINT
5472
    d1 = splashDist(tx, ty, mx, my);
5473
#else
5474
19.8M
    dx = tx - mx;
5475
19.8M
    dy = ty - my;
5476
19.8M
    d1 = dx*dx + dy*dy;
5477
19.8M
#endif
5478
19.8M
    transform(matrix, xx2, yy2, &tx, &ty);
5479
#if USE_FIXEDPOINT
5480
    d2 = splashDist(tx, ty, mx, my);
5481
#else
5482
19.8M
    dx = tx - mx;
5483
19.8M
    dy = ty - my;
5484
19.8M
    d2 = dx*dx + dy*dy;
5485
19.8M
#endif
5486
5487
    // if the curve is flat enough, or no more subdivisions are
5488
    // allowed, add the straight line segment
5489
19.8M
    if (p2 - p1 == 1 || (d1 <= flatness2 && d2 <= flatness2)) {
5490
9.91M
      fPath->lineTo(xr3, yr3);
5491
9.91M
      p1 = p2;
5492
5493
    // otherwise, subdivide the curve
5494
9.91M
    } else {
5495
9.90M
      xl1 = splashAvg(xl0, xx1);
5496
9.90M
      yl1 = splashAvg(yl0, yy1);
5497
9.90M
      xh = splashAvg(xx1, xx2);
5498
9.90M
      yh = splashAvg(yy1, yy2);
5499
9.90M
      xl2 = splashAvg(xl1, xh);
5500
9.90M
      yl2 = splashAvg(yl1, yh);
5501
9.90M
      xr2 = splashAvg(xx2, xr3);
5502
9.90M
      yr2 = splashAvg(yy2, yr3);
5503
9.90M
      xr1 = splashAvg(xh, xr2);
5504
9.90M
      yr1 = splashAvg(yh, yr2);
5505
9.90M
      xr0 = splashAvg(xl2, xr1);
5506
9.90M
      yr0 = splashAvg(yl2, yr1);
5507
      // add the new subdivision points
5508
9.90M
      p3 = (p1 + p2) / 2;
5509
9.90M
      cx[p1][1] = xl1;  cy[p1][1] = yl1;
5510
9.90M
      cx[p1][2] = xl2;  cy[p1][2] = yl2;
5511
9.90M
      cNext[p1] = p3;
5512
9.90M
      cx[p3][0] = xr0;  cy[p3][0] = yr0;
5513
9.90M
      cx[p3][1] = xr1;  cy[p3][1] = yr1;
5514
9.90M
      cx[p3][2] = xr2;  cy[p3][2] = yr2;
5515
9.90M
      cNext[p3] = p2;
5516
9.90M
    }
5517
19.8M
  }
5518
10.9k
}
5519
5520
339
SplashPath *Splash::makeDashedPath(SplashPath *path) {
5521
339
  SplashPath *dPath;
5522
339
  SplashCoord lineDashTotal;
5523
339
  SplashCoord lineDashStartPhase, lineDashDist, segLen;
5524
339
  SplashCoord x0, y0, x1, y1, xa, ya;
5525
339
  GBool lineDashStartOn, lineDashEndOn, lineDashOn, newPath;
5526
339
  int lineDashStartIdx, lineDashIdx, subpathStart, nDashes;
5527
339
  int i, j, k;
5528
5529
339
  lineDashTotal = 0;
5530
4.07k
  for (i = 0; i < state->lineDashLength; ++i) {
5531
3.73k
    lineDashTotal += state->lineDash[i];
5532
3.73k
  }
5533
  // Acrobat simply draws nothing if the dash array is [0]
5534
339
  if (lineDashTotal == 0) {
5535
0
    return new SplashPath();
5536
0
  }
5537
339
  lineDashStartPhase = state->lineDashPhase;
5538
339
  if (lineDashStartPhase > 0) {
5539
47
    i = splashFloor(lineDashStartPhase / lineDashTotal);
5540
47
    lineDashStartPhase -= lineDashTotal * i;
5541
292
  } else {
5542
292
    i = splashCeil(-lineDashStartPhase / lineDashTotal);
5543
292
    lineDashStartPhase += lineDashTotal * i;
5544
292
  }
5545
339
  lineDashStartOn = !((state->lineDashLength & 1) && (i & 1));
5546
339
  lineDashStartIdx = 0;
5547
339
  if (lineDashStartPhase > 0) {
5548
147
    while (lineDashStartPhase >= state->lineDash[lineDashStartIdx]) {
5549
100
      lineDashStartOn = !lineDashStartOn;
5550
100
      lineDashStartPhase -= state->lineDash[lineDashStartIdx];
5551
100
      if (++lineDashStartIdx == state->lineDashLength) {
5552
0
  lineDashStartIdx = 0;
5553
0
      }
5554
100
    }
5555
47
  }
5556
5557
339
  dPath = new SplashPath();
5558
5559
  // process each subpath
5560
339
  i = 0;
5561
1.01k
  while (i < path->length) {
5562
5563
    // find the end of the subpath
5564
673
    for (j = i;
5565
36.5k
   j < path->length - 1 && !(path->flags[j] & splashPathLast);
5566
35.9k
   ++j) ;
5567
5568
    // initialize the dash parameters
5569
673
    lineDashOn = lineDashStartOn;
5570
673
    lineDashEndOn = lineDashStartOn;
5571
673
    lineDashIdx = lineDashStartIdx;
5572
673
    lineDashDist = state->lineDash[lineDashIdx] - lineDashStartPhase;
5573
673
    subpathStart = dPath->length;
5574
673
    nDashes = 0;
5575
5576
    // process each segment of the subpath
5577
673
    newPath = gTrue;
5578
36.5k
    for (k = i; k < j; ++k) {
5579
5580
      // grab the segment
5581
35.9k
      x0 = path->pts[k].x;
5582
35.9k
      y0 = path->pts[k].y;
5583
35.9k
      x1 = path->pts[k+1].x;
5584
35.9k
      y1 = path->pts[k+1].y;
5585
35.9k
      segLen = splashDist(x0, y0, x1, y1);
5586
5587
      // Special case for zero-length subpath: copy the zero-length
5588
      // segment into the dashed path so that the round line cap
5589
      // special case is handled.
5590
35.9k
      if (j == i+1 && segLen == 0) {
5591
34
  dPath->moveTo(x0, y0);
5592
34
  dPath->lineTo(x0, y0);
5593
34
      }
5594
5595
      // process the segment
5596
7.79M
      while (segLen > 0) {
5597
5598
  // Special case for zero-length dash segments: draw a very
5599
  // short -- but not zero-length -- segment.  This ensures that
5600
  // we get the correct behavior with butt and projecting line
5601
  // caps.  The PS/PDF specs imply that zero-length segments are
5602
  // not drawn unless the line cap is round, but Acrobat and
5603
  // Ghostscript both draw very short segments (for butt caps)
5604
  // and squares (for projecting caps).
5605
7.76M
  if (lineDashDist == 0) {
5606
995k
    if (lineDashOn) {
5607
408k
      if (newPath) {
5608
408k
        dPath->moveTo(x0, y0);
5609
408k
        newPath = gFalse;
5610
408k
        ++nDashes;
5611
408k
      }
5612
408k
      xa = x0 + ((SplashCoord)0.001 / segLen) * (x1 - x0);
5613
408k
      ya = y0 + ((SplashCoord)0.001 / segLen) * (y1 - y0);
5614
408k
      dPath->lineTo(xa, ya);
5615
408k
    }
5616
5617
6.76M
  } else if (lineDashDist >= segLen) {
5618
35.8k
    if (lineDashOn) {
5619
22.5k
      if (newPath) {
5620
4.78k
        dPath->moveTo(x0, y0);
5621
4.78k
        newPath = gFalse;
5622
4.78k
        ++nDashes;
5623
4.78k
      }
5624
22.5k
      dPath->lineTo(x1, y1);
5625
22.5k
    }
5626
35.8k
    lineDashDist -= segLen;
5627
35.8k
    segLen = 0;
5628
5629
6.73M
  } else {
5630
6.73M
    xa = x0 + (lineDashDist / segLen) * (x1 - x0);
5631
6.73M
    ya = y0 + (lineDashDist / segLen) * (y1 - y0);
5632
6.73M
    if (lineDashOn) {
5633
3.45M
      if (newPath) {
5634
3.45M
        dPath->moveTo(x0, y0);
5635
3.45M
        newPath = gFalse;
5636
3.45M
        ++nDashes;
5637
3.45M
      }
5638
3.45M
      dPath->lineTo(xa, ya);
5639
3.45M
    }
5640
6.73M
    x0 = xa;
5641
6.73M
    y0 = ya;
5642
6.73M
    segLen -= lineDashDist;
5643
6.73M
    lineDashDist = 0;
5644
6.73M
  }
5645
5646
7.76M
  lineDashEndOn = lineDashOn;
5647
5648
  // get the next entry in the dash array
5649
7.76M
  if (lineDashDist <= 0) {
5650
7.72M
    lineDashOn = !lineDashOn;
5651
7.72M
    if (++lineDashIdx == state->lineDashLength) {
5652
1.86M
      lineDashIdx = 0;
5653
1.86M
    }
5654
7.72M
    lineDashDist = state->lineDash[lineDashIdx];
5655
7.72M
    newPath = gTrue;
5656
7.72M
  }
5657
7.76M
      }
5658
35.9k
    }
5659
5660
    // in a closed subpath, where the dash pattern is "on" at both the
5661
    // start and end of the subpath, we need to merge the start and
5662
    // end to get a proper line join
5663
673
    if ((path->flags[j] & splashPathClosed) &&
5664
177
  lineDashStartOn &&
5665
176
  lineDashEndOn) {
5666
154
      if (nDashes == 1) {
5667
87
  dPath->close();
5668
87
      } else if (nDashes > 1) {
5669
34
  k = subpathStart;
5670
54
  do {
5671
54
    ++k;
5672
54
    dPath->lineTo(dPath->pts[k].x, dPath->pts[k].y);
5673
54
  } while (!(dPath->flags[k] & splashPathLast));
5674
34
  ++k;
5675
34
  memmove(&dPath->pts[subpathStart], &dPath->pts[k],
5676
34
    (dPath->length - k) * sizeof(SplashPathPoint));
5677
34
  memmove(&dPath->flags[subpathStart], &dPath->flags[k],
5678
34
    (dPath->length - k) * sizeof(Guchar));
5679
34
  dPath->length -= k - subpathStart;
5680
34
  dPath->curSubpath -= k - subpathStart;
5681
34
      }
5682
154
    }
5683
5684
673
    i = j + 1;
5685
673
  }
5686
5687
339
  return dPath;
5688
339
}
5689
5690
1.44M
SplashError Splash::fill(SplashPath *path, GBool eo) {
5691
1.44M
  if (debugMode) {
5692
0
    printf("fill [eo:%d]:\n", eo);
5693
0
    dumpPath(path);
5694
0
  }
5695
1.44M
  if (path->length == 0) {
5696
2.91k
    return splashErrEmptyPath;
5697
2.91k
  }
5698
1.44M
  if (pathAllOutside(path, gFalse)) {
5699
1.40M
    opClipRes = splashClipAllOutside;
5700
1.40M
    return splashOk;
5701
1.40M
  }
5702
35.5k
  return fillWithPattern(path, eo, state->fillPattern, state->fillAlpha);
5703
1.44M
}
5704
5705
SplashError Splash::fillWithPattern(SplashPath *path, GBool eo,
5706
            SplashPattern *pattern,
5707
56.5k
            SplashCoord alpha) {
5708
56.5k
  SplashPipe pipe;
5709
56.5k
  SplashPath *path2;
5710
56.5k
  SplashXPath *xPath;
5711
56.5k
  SplashXPathScanner *scanner;
5712
56.5k
  int xMin, yMin, xMax, xMin2, xMax2, yMax, y, t;
5713
56.5k
  SplashClipResult clipRes;
5714
5715
56.5k
  path2 = tweakFillPath(path);
5716
5717
56.5k
  xPath = new SplashXPath(path2, state->matrix, state->flatness, gTrue,
5718
56.5k
        state->enablePathSimplification,
5719
56.5k
        state->strokeAdjust, state->clip);
5720
56.5k
  if (path2 != path) {
5721
1.71k
    delete path2;
5722
1.71k
  }
5723
56.5k
  xMin = xPath->getXMin();
5724
56.5k
  yMin = xPath->getYMin();
5725
56.5k
  xMax = xPath->getXMax();
5726
56.5k
  yMax = xPath->getYMax();
5727
56.5k
  if (xMin > xMax || yMin > yMax) {
5728
21
    delete xPath;
5729
21
    return splashOk;
5730
21
  }
5731
56.5k
  scanner = new SplashXPathScanner(xPath, eo, yMin, yMax);
5732
5733
  // check clipping
5734
56.5k
  if ((clipRes = state->clip->testRect(xMin, yMin, xMax, yMax,
5735
56.5k
               state->strokeAdjust))
5736
56.5k
      != splashClipAllOutside) {
5737
5738
30.3k
    if ((t = state->clip->getXMinI(state->strokeAdjust)) > xMin) {
5739
8.03k
      xMin = t;
5740
8.03k
    }
5741
30.3k
    if ((t = state->clip->getXMaxI(state->strokeAdjust)) < xMax) {
5742
12.8k
      xMax = t;
5743
12.8k
    }
5744
30.3k
    if ((t = state->clip->getYMinI(state->strokeAdjust)) > yMin) {
5745
23.3k
      yMin = t;
5746
23.3k
    }
5747
30.3k
    if ((t = state->clip->getYMaxI(state->strokeAdjust)) < yMax) {
5748
23.7k
      yMax = t;
5749
23.7k
    }
5750
30.3k
    if (xMin > xMax || yMin > yMax) {
5751
0
      delete scanner;
5752
0
      delete xPath;
5753
0
      return splashOk;
5754
0
    }
5755
5756
30.3k
    pipeInit(&pipe, pattern, (Guchar)splashRound(alpha * 255),
5757
30.3k
       gFalse, gTrue, gFalse);
5758
5759
    // draw the spans
5760
30.3k
    if (vectorAntialias) {
5761
60.4k
      for (y = yMin; y <= yMax; ++y) {
5762
30.2k
  scanner->getSpan(scanBuf, y, xMin, xMax, &xMin2, &xMax2);
5763
30.2k
  if (xMin2 <= xMax2) {
5764
25.2k
    if (clipRes != splashClipAllInside) {
5765
23.0k
      state->clip->clipSpan(scanBuf, y, xMin2, xMax2,
5766
23.0k
          state->strokeAdjust);
5767
23.0k
    }
5768
25.2k
    (this->*pipe.run)(&pipe, xMin2, xMax2, y,
5769
25.2k
          NULL, scanBuf + xMin2, NULL);
5770
25.2k
  }
5771
30.2k
      }
5772
30.2k
    } else {
5773
292
      for (y = yMin; y <= yMax; ++y) {
5774
146
  scanner->getSpanBinary(scanBuf, y, xMin, xMax, &xMin2, &xMax2);
5775
146
  if (xMin2 <= xMax2) {
5776
146
    if (clipRes != splashClipAllInside) {
5777
146
      state->clip->clipSpanBinary(scanBuf, y, xMin2, xMax2,
5778
146
          state->strokeAdjust);
5779
146
    }
5780
146
    (this->*pipe.run)(&pipe, xMin2, xMax2, y,
5781
146
          NULL, scanBuf + xMin2, NULL);
5782
146
  }
5783
146
      }
5784
146
    }
5785
30.3k
  }
5786
56.5k
  opClipRes = clipRes;
5787
5788
56.5k
  delete scanner;
5789
56.5k
  delete xPath;
5790
56.5k
  return splashOk;
5791
56.5k
}
5792
5793
// Applies various tweaks to a fill path:
5794
// (1) add stroke adjust hints to a filled rectangle
5795
// (2) applies a minimum width to a zero-width filled rectangle (so
5796
//     stroke adjustment works correctly
5797
// (3) convert a degenerate fill ('moveto lineto fill' and 'moveto
5798
//     lineto closepath fill') to a minimum-width filled rectangle
5799
//
5800
// These tweaks only apply to paths with a single subpath.
5801
//
5802
// Returns either the unchanged input path or a new path (in which
5803
// case the returned path must be deleted by the caller).
5804
56.5k
SplashPath *Splash::tweakFillPath(SplashPath *path) {
5805
56.5k
  SplashPath *path2;
5806
56.5k
  SplashCoord xx0, yy0, xx1, yy1, dx, dy, d, wx, wy, w;
5807
56.5k
  int n;
5808
5809
56.5k
  if (state->strokeAdjust == splashStrokeAdjustOff || path->hints) {
5810
22.6k
    return path;
5811
22.6k
  }
5812
5813
33.9k
  n = path->getLength();
5814
33.9k
  if (!((n == 2) ||
5815
33.2k
  (n == 3 &&
5816
921
   path->flags[1] == 0) ||
5817
32.3k
  (n == 4 &&
5818
1.64k
   path->flags[1] == 0 &&
5819
1.11k
   path->flags[2] == 0) ||
5820
31.2k
  (n == 5 &&
5821
22.7k
   path->flags[1] == 0 &&
5822
22.6k
   path->flags[2] == 0 &&
5823
22.4k
   path->flags[3] == 0))) {
5824
8.77k
    return path;
5825
8.77k
  }
5826
5827
25.1k
  path2 = path;
5828
5829
  // degenerate fill (2 or 3 points) or rectangle of (nearly) zero
5830
  // width --> replace with a min-width rectangle and hint
5831
25.1k
  if (n == 2 ||
5832
24.5k
      (n == 3 && (path->flags[0] & splashPathClosed)) ||
5833
24.1k
      (n == 3 && (splashAbs(path->pts[0].x - path->pts[2].x) < 0.001 &&
5834
252
      splashAbs(path->pts[0].y - path->pts[2].y) < 0.001)) ||
5835
23.9k
      ((n == 4 ||
5836
22.8k
  (n == 5 && (path->flags[0] & splashPathClosed))) &&
5837
23.5k
       ((splashAbs(path->pts[0].x - path->pts[1].x) < 0.001 &&
5838
624
   splashAbs(path->pts[0].y - path->pts[1].y) < 0.001 &&
5839
376
   splashAbs(path->pts[2].x - path->pts[3].x) < 0.001 &&
5840
325
   splashAbs(path->pts[2].y - path->pts[3].y) < 0.001) ||
5841
23.2k
  (splashAbs(path->pts[0].x - path->pts[3].x) < 0.001 &&
5842
22.5k
   splashAbs(path->pts[0].y - path->pts[3].y) < 0.001 &&
5843
336
   splashAbs(path->pts[1].x - path->pts[2].x) < 0.001 &&
5844
1.71k
   splashAbs(path->pts[1].y - path->pts[2].y) < 0.001)))) {
5845
1.71k
    wx = state->matrix[0] + state->matrix[2];
5846
1.71k
    wy = state->matrix[1] + state->matrix[3];
5847
1.71k
    w = splashSqrt(wx*wx + wy*wy);
5848
1.71k
    if (w < 0.001) {
5849
744
      w = 0;
5850
975
    } else {
5851
      // min width is 0.1 -- this constant is minWidth * sqrt(2)
5852
975
      w = (SplashCoord)0.1414 / w;
5853
975
    }
5854
1.71k
    xx0 = path->pts[0].x;
5855
1.71k
    yy0 = path->pts[0].y;
5856
1.71k
    if (n <= 3) {
5857
1.18k
      xx1 = path->pts[1].x;
5858
1.18k
      yy1 = path->pts[1].y;
5859
1.18k
    } else {
5860
531
      xx1 = path->pts[2].x;
5861
531
      yy1 = path->pts[2].y;
5862
531
    }
5863
1.71k
    dx = xx1 - xx0;
5864
1.71k
    dy = yy1 - yy0;
5865
1.71k
    d = splashSqrt(dx * dx + dy * dy);
5866
1.71k
    if (d < 0.001) {
5867
531
      d = 0;
5868
1.18k
    } else {
5869
1.18k
      d = w / d;
5870
1.18k
    }
5871
1.71k
    dx *= d;
5872
1.71k
    dy *= d;
5873
1.71k
    path2 = new SplashPath();
5874
1.71k
    path2->moveTo(xx0 + dy, yy0 - dx);
5875
1.71k
    path2->lineTo(xx1 + dy, yy1 - dx);
5876
1.71k
    path2->lineTo(xx1 - dy, yy1 + dx);
5877
1.71k
    path2->lineTo(xx0 - dy, yy0 + dx);
5878
1.71k
    path2->close(gTrue);
5879
1.71k
    path2->addStrokeAdjustHint(0, 2, 0, 4);
5880
1.71k
    path2->addStrokeAdjustHint(1, 3, 0, 4);
5881
5882
  // unclosed rectangle --> close and hint
5883
23.4k
  } else if (n == 4 && !(path->flags[0] & splashPathClosed)) {
5884
629
    path2->close(gTrue);
5885
629
    path2->addStrokeAdjustHint(0, 2, 0, 4);
5886
629
    path2->addStrokeAdjustHint(1, 3, 0, 4);
5887
5888
  // closed rectangle --> hint
5889
22.8k
  } else if (n == 5 && (path->flags[0] & splashPathClosed)) {
5890
22.2k
    path2->addStrokeAdjustHint(0, 2, 0, 4);
5891
22.2k
    path2->addStrokeAdjustHint(1, 3, 0, 4);
5892
22.2k
  }
5893
5894
25.1k
  return path2;
5895
33.9k
}
5896
5897
// Returns true if [path] is entirely outside the current clipping
5898
// path.  The path coordinates have not been stroke adjusted, so we
5899
// compare against the floating point clip rect.  If [stroke] is true,
5900
// allow for the stroke width and miter limit.
5901
1.49M
GBool Splash::pathAllOutside(SplashPath *path, GBool stroke) {
5902
1.49M
  SplashCoord xMin1, yMin1, xMax1, yMax1;
5903
1.49M
  SplashCoord xMin2, yMin2, xMax2, yMax2;
5904
1.49M
  SplashCoord x, y;
5905
1.49M
  int i;
5906
5907
  //--- compute the path's bbox in user space
5908
1.49M
  xMin1 = xMax1 = path->pts[0].x;
5909
1.49M
  yMin1 = yMax1 = path->pts[0].y;
5910
18.4M
  for (i = 1; i < path->length; ++i) {
5911
17.0M
    if (path->pts[i].x < xMin1) {
5912
4.52M
      xMin1 = path->pts[i].x;
5913
12.4M
    } else if (path->pts[i].x > xMax1) {
5914
3.81M
      xMax1 = path->pts[i].x;
5915
3.81M
    }
5916
17.0M
    if (path->pts[i].y < yMin1) {
5917
2.35M
      yMin1 = path->pts[i].y;
5918
14.6M
    } else if (path->pts[i].y > yMax1) {
5919
2.65M
      yMax1 = path->pts[i].y;
5920
2.65M
    }
5921
17.0M
  }
5922
5923
  //--- allow for stroke width and miter limit
5924
1.49M
  if (stroke && state->lineWidth > 0) {
5925
48.0k
    SplashCoord w = state->lineWidth * 0.5;
5926
48.0k
    if (state->lineJoin == splashLineJoinMiter) {
5927
45.1k
      w *= state->miterLimit;
5928
45.1k
    }
5929
48.0k
    xMin1 -= w;
5930
48.0k
    yMin1 -= w;
5931
48.0k
    xMax1 += w;
5932
48.0k
    yMax1 += w;
5933
48.0k
  }
5934
5935
  //--- convert path bbox to device space
5936
1.49M
  transform(state->matrix, xMin1, yMin1, &x, &y);
5937
1.49M
  xMin2 = xMax2 = x;
5938
1.49M
  yMin2 = yMax2 = y;
5939
1.49M
  transform(state->matrix, xMin1, yMax1, &x, &y);
5940
1.49M
  if (x < xMin2) {
5941
6.64k
    xMin2 = x;
5942
1.49M
  } else if (x > xMax2) {
5943
5.49k
    xMax2 = x;
5944
5.49k
  }
5945
1.49M
  if (y < yMin2) {
5946
41.9k
    yMin2 = y;
5947
1.45M
  } else if (y > yMax2) {
5948
632k
    yMax2 = y;
5949
632k
  }
5950
1.49M
  transform(state->matrix, xMax1, yMin1, &x, &y);
5951
1.49M
  if (x < xMin2) {
5952
45.7k
    xMin2 = x;
5953
1.45M
  } else if (x > xMax2) {
5954
636k
    xMax2 = x;
5955
636k
  }
5956
1.49M
  if (y < yMin2) {
5957
3.19k
    yMin2 = y;
5958
1.49M
  } else if (y > yMax2) {
5959
2.66k
    yMax2 = y;
5960
2.66k
  }
5961
1.49M
  transform(state->matrix, xMax1, yMax1, &x, &y);
5962
1.49M
  if (x < xMin2) {
5963
4.66k
    xMin2 = x;
5964
1.49M
  } else if (x > xMax2) {
5965
2.72k
    xMax2 = x;
5966
2.72k
  }
5967
1.49M
  if (y < yMin2) {
5968
5.18k
    yMin2 = y;
5969
1.49M
  } else if (y > yMax2) {
5970
4.42k
    yMax2 = y;
5971
4.42k
  }
5972
5973
  //--- handle zero-width strokes
5974
1.49M
  if (stroke && state->lineWidth == 0) {
5975
3.55k
    xMin1 -= 1;
5976
3.55k
    yMin1 -= 1;
5977
3.55k
    xMax1 += 1;
5978
3.55k
    yMax1 += 1;
5979
3.55k
  }
5980
5981
  //--- check against the clip rect
5982
1.49M
  return xMin2 > state->clip->getXMax() ||
5983
781k
         xMax2 < state->clip->getXMin() ||
5984
745k
         yMin2 > state->clip->getYMax() ||
5985
65.9k
         yMax2 < state->clip->getYMin();
5986
1.49M
}
5987
5988
SplashError Splash::fillChar(SplashCoord x, SplashCoord y,
5989
627k
           Guint c, SplashFont *font) {
5990
627k
  SplashGlyphBitmap glyph;
5991
627k
  SplashCoord xt, yt;
5992
627k
  int x0, y0, xFrac, yFrac;
5993
627k
  SplashError err;
5994
5995
627k
  if (debugMode) {
5996
0
    printf("fillChar: x=%.2f y=%.2f c=%3u=0x%02x='%c'\n",
5997
0
     (double)x, (double)y, c, c, c);
5998
0
  }
5999
627k
  transform(state->matrix, x, y, &xt, &yt);
6000
627k
  x0 = splashFloor(xt);
6001
627k
  xFrac = splashFloor((xt - x0) * splashFontFraction);
6002
627k
  y0 = splashFloor(yt);
6003
627k
  yFrac = splashFloor((yt - y0) * splashFontFraction);
6004
627k
  if (!font->getGlyph(c, xFrac, yFrac, &glyph)) {
6005
604k
    return splashErrNoGlyph;
6006
604k
  }
6007
23.5k
  err = fillGlyph2(x0, y0, &glyph);
6008
23.5k
  if (glyph.freeData) {
6009
14.7k
    gfree(glyph.data);
6010
14.7k
  }
6011
23.5k
  return err;
6012
627k
}
6013
6014
SplashError Splash::fillGlyph(SplashCoord x, SplashCoord y,
6015
751
            SplashGlyphBitmap *glyph) {
6016
751
  SplashCoord xt, yt;
6017
751
  int x0, y0;
6018
6019
751
  transform(state->matrix, x, y, &xt, &yt);
6020
751
  x0 = splashFloor(xt);
6021
751
  y0 = splashFloor(yt);
6022
751
  return fillGlyph2(x0, y0, glyph);
6023
751
}
6024
6025
24.2k
SplashError Splash::fillGlyph2(int x0, int y0, SplashGlyphBitmap *glyph) {
6026
24.2k
  SplashPipe pipe;
6027
24.2k
  SplashClipResult clipRes;
6028
24.2k
  Guchar alpha;
6029
24.2k
  Guchar *p;
6030
24.2k
  int xMin, yMin, xMax, yMax;
6031
24.2k
  int x, y, xg, yg, xx, t;
6032
6033
24.2k
  xg = x0 - glyph->x;
6034
24.2k
  yg = y0 - glyph->y;
6035
24.2k
  xMin = xg;
6036
24.2k
  xMax = xg + glyph->w - 1;
6037
24.2k
  yMin = yg;
6038
24.2k
  yMax = yg + glyph->h - 1;
6039
24.2k
  if ((clipRes = state->clip->testRect(xMin, yMin, xMax, yMax,
6040
24.2k
               state->strokeAdjust))
6041
24.2k
      != splashClipAllOutside) {
6042
8
    pipeInit(&pipe, state->fillPattern,
6043
8
       (Guchar)splashRound(state->fillAlpha * 255),
6044
8
       gFalse, gTrue, gFalse);
6045
8
    if (clipRes == splashClipAllInside) {
6046
0
      if (glyph->aa) {
6047
0
  p = glyph->data;
6048
0
  for (y = yMin; y <= yMax; ++y) {
6049
0
    (this->*pipe.run)(&pipe, xMin, xMax, y,
6050
0
          NULL,
6051
0
          glyph->data + (y - yMin) * glyph->w,
6052
0
          NULL);
6053
0
  }
6054
0
      } else {
6055
0
  p = glyph->data;
6056
0
  for (y = yMin; y <= yMax; ++y) {
6057
0
    for (x = xMin; x <= xMax; x += 8) {
6058
0
      alpha = *p++;
6059
0
      for (xx = 0; xx < 8 && x + xx <= xMax; ++xx) {
6060
0
        scanBuf[x + xx] = (alpha & 0x80) ? 0xff : 0x00;
6061
0
        alpha = (Guchar)(alpha << 1);
6062
0
      }
6063
0
    }
6064
0
    (this->*pipe.run)(&pipe, xMin, xMax, y,
6065
0
          NULL, scanBuf + xMin, NULL);
6066
0
  }
6067
0
      }
6068
8
    } else {
6069
8
      if ((t = state->clip->getXMinI(state->strokeAdjust)) > xMin) {
6070
8
  xMin = t;
6071
8
      }
6072
8
      if ((t = state->clip->getXMaxI(state->strokeAdjust)) < xMax) {
6073
8
  xMax = t;
6074
8
      }
6075
8
      if ((t = state->clip->getYMinI(state->strokeAdjust)) > yMin) {
6076
8
  yMin = t;
6077
8
      }
6078
8
      if ((t = state->clip->getYMaxI(state->strokeAdjust)) < yMax) {
6079
8
  yMax = t;
6080
8
      }
6081
8
      if (xMin <= xMax && yMin <= yMax) {
6082
8
  if (glyph->aa) {
6083
16
    for (y = yMin; y <= yMax; ++y) {
6084
8
      p = glyph->data + (y - yg) * glyph->w + (xMin - xg);
6085
8
      memcpy(scanBuf + xMin, p, xMax - xMin + 1);
6086
8
      state->clip->clipSpan(scanBuf, y, xMin, xMax,
6087
8
          state->strokeAdjust);
6088
8
      (this->*pipe.run)(&pipe, xMin, xMax, y,
6089
8
            NULL, scanBuf + xMin, NULL);
6090
8
    }
6091
8
  } else {
6092
0
    for (y = yMin; y <= yMax; ++y) {
6093
0
      p = glyph->data + (y - yg) * ((glyph->w + 7) >> 3)
6094
0
        + ((xMin - xg) >> 3);
6095
0
      alpha = *p++;
6096
0
      xx = (xMin - xg) & 7;
6097
0
      alpha = (Guchar)(alpha << xx);
6098
0
      for (x = xMin; xx < 8 && x <= xMax; ++x, ++xx) {
6099
0
        scanBuf[x] = (alpha & 0x80) ? 255 : 0;
6100
0
        alpha = (Guchar)(alpha << 1);
6101
0
      }
6102
0
      for (; x <= xMax; x += 8) {
6103
0
        alpha = *p++;
6104
0
        for (xx = 0; xx < 8 && x + xx <= xMax; ++xx) {
6105
0
    scanBuf[x + xx] = (alpha & 0x80) ? 255 : 0;
6106
0
    alpha = (Guchar)(alpha << 1);
6107
0
        }
6108
0
      }
6109
0
      state->clip->clipSpanBinary(scanBuf, y, xMin, xMax,
6110
0
          state->strokeAdjust);
6111
0
      (this->*pipe.run)(&pipe, xMin, xMax, y,
6112
0
            NULL, scanBuf + xMin, NULL);
6113
0
    }
6114
0
  }
6115
8
      }
6116
8
    }
6117
8
  }
6118
24.2k
  opClipRes = clipRes;
6119
6120
24.2k
  return splashOk;
6121
24.2k
}
6122
6123
void Splash::getImageBounds(SplashCoord xyMin, SplashCoord xyMax,
6124
109k
          int *xyMinI, int *xyMaxI) {
6125
109k
  if (state->strokeAdjust == splashStrokeAdjustOff) {
6126
    // make sure the coords fit in 32-bit ints
6127
#if USE_FIXEDPOINT
6128
    if (xyMin < -32767) {
6129
      xyMin = -32767;
6130
    } else if (xyMin > 32767) {
6131
      xyMin = 32767;
6132
    }
6133
    if (xyMax < -32767) {
6134
      xyMax = -32767;
6135
    } else if (xyMax > 32767) {
6136
      xyMax = 32767;
6137
    }
6138
#else
6139
35.4k
    if (xyMin < -1e9) {
6140
15.4k
      xyMin = -1e9;
6141
19.9k
    } else if (xyMin > 1e9) {
6142
7.74k
      xyMin = 1e9;
6143
7.74k
    }
6144
35.4k
    if (xyMax < -1e9) {
6145
13.9k
      xyMax = -1e9;
6146
21.4k
    } else if (xyMax > 1e9) {
6147
14.9k
      xyMax = 1e9;
6148
14.9k
    }
6149
35.4k
#endif
6150
35.4k
    *xyMinI = splashFloor(xyMin);
6151
35.4k
    *xyMaxI = splashFloor(xyMax);
6152
35.4k
    if (*xyMaxI <= *xyMinI) {
6153
26.7k
      *xyMaxI = *xyMinI + 1;
6154
26.7k
    }
6155
74.3k
  } else {
6156
74.3k
    splashStrokeAdjust(xyMin, xyMax, xyMinI, xyMaxI, state->strokeAdjust);
6157
74.3k
  }
6158
109k
}
6159
6160
struct SplashDrawImageMaskRowData {
6161
  SplashPipe pipe;
6162
};
6163
6164
// The glyphMode flag is not currently used, but may be useful if the
6165
// stroke adjustment behavior is changed.
6166
SplashError Splash::fillImageMask(GString *imageTag,
6167
          SplashImageMaskSource src, void *srcData,
6168
          int w, int h, SplashCoord *mat,
6169
          GBool glyphMode, GBool interpolate,
6170
46.3k
          GBool antialias) {
6171
46.3k
  if (debugMode) {
6172
0
    printf("fillImageMask: w=%d h=%d mat=[%.2f %.2f %.2f %.2f %.2f %.2f]\n",
6173
0
     w, h, (double)mat[0], (double)mat[1], (double)mat[2],
6174
0
     (double)mat[3], (double)mat[4], (double)mat[5]);
6175
0
  }
6176
6177
  //--- check for singular matrix
6178
46.3k
  if (!splashCheckDet(mat[0], mat[1], mat[2], mat[3], 0.000001)) {
6179
875
    return splashErrSingularMatrix;
6180
875
  }
6181
6182
  //--- compute image bbox, check clipping
6183
45.4k
  GBool flipsOnly = splashAbs(mat[1]) <= 0.0001 && splashAbs(mat[2]) <= 0.0001;
6184
45.4k
  GBool rot90Only = splashAbs(mat[0]) <= 0.0001 && splashAbs(mat[3]) <= 0.0001;
6185
45.4k
  GBool horizFlip = gFalse;
6186
45.4k
  GBool vertFlip = gFalse;
6187
45.4k
  int xMin, yMin, xMax, yMax;
6188
45.4k
  if (flipsOnly) {
6189
22.2k
    horizFlip = mat[0] < 0;
6190
22.2k
    vertFlip = mat[3] < 0;
6191
22.2k
    if (horizFlip) {
6192
17.4k
      getImageBounds(mat[0] + mat[4], mat[4], &xMin, &xMax);
6193
17.4k
    } else {
6194
4.77k
      getImageBounds(mat[4], mat[0] + mat[4], &xMin, &xMax);
6195
4.77k
    }
6196
22.2k
    if (vertFlip) {
6197
6.65k
      getImageBounds(mat[3] + mat[5], mat[5], &yMin, &yMax);
6198
15.5k
    } else {
6199
15.5k
      getImageBounds(mat[5], mat[3] + mat[5], &yMin, &yMax);
6200
15.5k
    }
6201
23.2k
  } else if (rot90Only) {
6202
9.89k
    horizFlip = mat[2] < 0;
6203
9.89k
    vertFlip = mat[1] < 0;
6204
9.89k
    if (horizFlip) {
6205
1.02k
      getImageBounds(mat[2] + mat[4], mat[4], &xMin, &xMax);
6206
8.86k
    } else {
6207
8.86k
      getImageBounds(mat[4], mat[2] + mat[4], &xMin, &xMax);
6208
8.86k
    }
6209
9.89k
    if (vertFlip) {
6210
2.34k
      getImageBounds(mat[1] + mat[5], mat[5], &yMin, &yMax);
6211
7.54k
    } else {
6212
7.54k
      getImageBounds(mat[5], mat[1] + mat[5], &yMin, &yMax);
6213
7.54k
    }
6214
13.3k
  } else {
6215
13.3k
    int xx = splashRound(mat[4]);   // (0,0)
6216
13.3k
    int yy = splashRound(mat[5]);
6217
13.3k
    xMin = xMax = xx;
6218
13.3k
    yMin = yMax = yy;
6219
13.3k
    xx = splashRound(mat[0] + mat[4]);    // (1,0)
6220
13.3k
    yy = splashRound(mat[1] + mat[5]);
6221
13.3k
    if (xx < xMin) {
6222
578
      xMin = xx;
6223
12.7k
    } else if (xx > xMax) {
6224
847
      xMax = xx;
6225
847
    }
6226
13.3k
    if (yy < yMin) {
6227
500
      yMin = yy;
6228
12.8k
    } else if (yy > yMax) {
6229
390
      yMax = yy;
6230
390
    }
6231
13.3k
    xx = splashRound(mat[2] + mat[4]);    // (0,1)
6232
13.3k
    yy = splashRound(mat[3] + mat[5]);
6233
13.3k
    if (xx < xMin) {
6234
656
      xMin = xx;
6235
12.6k
    } else if (xx > xMax) {
6236
241
      xMax = xx;
6237
241
    }
6238
13.3k
    if (yy < yMin) {
6239
243
      yMin = yy;
6240
13.0k
    } else if (yy > yMax) {
6241
59
      yMax = yy;
6242
59
    }
6243
13.3k
    xx = splashRound(mat[0] + mat[2] + mat[4]); // (1,1)
6244
13.3k
    yy = splashRound(mat[1] + mat[3] + mat[5]);
6245
13.3k
    if (xx < xMin) {
6246
1.27k
      xMin = xx;
6247
12.0k
    } else if (xx > xMax) {
6248
408
      xMax = xx;
6249
408
    }
6250
13.3k
    if (yy < yMin) {
6251
1.76k
      yMin = yy;
6252
11.5k
    } else if (yy > yMax) {
6253
132
      yMax = yy;
6254
132
    }
6255
13.3k
    if (xMax <= xMin) {
6256
10.1k
      xMax = xMin + 1;
6257
10.1k
    }
6258
13.3k
    if (yMax <= yMin) {
6259
10.5k
      yMax = yMin + 1;
6260
10.5k
    }
6261
13.3k
  }
6262
45.4k
  SplashClipResult clipRes =
6263
45.4k
      state->clip->testRect(xMin, yMin, xMax - 1, yMax - 1,
6264
45.4k
          state->strokeAdjust);
6265
45.4k
  if (clipRes == splashClipAllOutside) {
6266
44.6k
    return splashOk;
6267
44.6k
  }
6268
  // If the scaled mask is much wider and/or taller than the clip
6269
  // region, we use the "arbitrary" scaling path, to avoid a
6270
  // potentially very slow loop in the flips-only path (which scans
6271
  // the full width and height of the scaled mask, regardless of the
6272
  // clip region).
6273
779
  int clipW = state->clip->getXMaxI(state->strokeAdjust)
6274
779
        - state->clip->getXMinI(state->strokeAdjust);
6275
779
  int clipH = state->clip->getYMaxI(state->strokeAdjust)
6276
779
        - state->clip->getYMinI(state->strokeAdjust);
6277
779
  GBool veryLarge = ((xMax - xMin) / 8 > clipW && xMax - xMin > 1000) ||
6278
248
                    ((yMax - yMin) / 8 > clipH && yMax - yMin > 1000);
6279
6280
  //--- set up the SplashDrawImageMaskRowData object and the pipes
6281
779
  SplashDrawImageMaskRowData dd;
6282
779
  pipeInit(&dd.pipe, state->fillPattern,
6283
779
     (Guchar)splashRound(state->fillAlpha * 255),
6284
779
     gFalse, gTrue, gFalse);
6285
6286
  //--- choose the drawRow function
6287
779
  SplashDrawImageMaskRowFunc drawRowFunc;
6288
779
  if (clipRes == splashClipAllInside) {
6289
159
    drawRowFunc = &Splash::drawImageMaskRowNoClip;
6290
620
  } else {
6291
620
    if (antialias) {
6292
620
      drawRowFunc = &Splash::drawImageMaskRowClipAA;
6293
620
    } else {
6294
0
      drawRowFunc = &Splash::drawImageMaskRowClipNoAA;
6295
0
    }
6296
620
  }
6297
6298
  //--- horizontal/vertical flips only
6299
779
  if (flipsOnly && !veryLarge) {
6300
159
    int scaledWidth = xMax - xMin;
6301
159
    int scaledHeight = yMax - yMin;
6302
159
    ImageMaskScaler scaler(src, srcData, w, h,
6303
159
         scaledWidth, scaledHeight, interpolate, antialias);
6304
159
    Guchar *tmpLine = NULL;
6305
159
    if (horizFlip) {
6306
0
      tmpLine = (Guchar *)gmalloc(scaledWidth);
6307
0
    }
6308
159
    if (vertFlip) {
6309
0
      if (horizFlip) {    // bottom-up, mirrored
6310
0
  for (int y = 0; y < scaledHeight; ++y) {
6311
0
    scaler.nextLine();
6312
0
    mirrorImageMaskRow(scaler.data(), tmpLine, scaledWidth);
6313
0
    (this->*drawRowFunc)(&dd, tmpLine,
6314
0
             xMin, yMax - 1 - y, scaledWidth);
6315
0
  }
6316
0
      } else {            // bottom-up
6317
0
  for (int y = 0; y < scaledHeight; ++y) {
6318
0
    scaler.nextLine();
6319
0
    (this->*drawRowFunc)(&dd, scaler.data(),
6320
0
             xMin, yMax - 1 - y, scaledWidth);
6321
0
  }
6322
0
      }
6323
159
    } else {
6324
159
      if (horizFlip) {    // top-down, mirrored
6325
0
  for (int y = 0; y < scaledHeight; ++y) {
6326
0
    scaler.nextLine();
6327
0
    mirrorImageMaskRow(scaler.data(), tmpLine, scaledWidth);
6328
0
    (this->*drawRowFunc)(&dd, tmpLine,
6329
0
             xMin, yMin + y, scaledWidth);
6330
0
  }
6331
159
      } else {            // top-down
6332
47.5k
  for (int y = 0; y < scaledHeight; ++y) {
6333
47.3k
    scaler.nextLine();
6334
47.3k
    (this->*drawRowFunc)(&dd, scaler.data(),
6335
47.3k
             xMin, yMin + y, scaledWidth);
6336
47.3k
  }
6337
159
      }
6338
159
    }
6339
159
    gfree(tmpLine);
6340
6341
  //--- 90/270 rotation
6342
620
  } else if (rot90Only && !veryLarge) {
6343
6344
    // scale the mask
6345
0
    int scaledWidth = yMax - yMin;
6346
0
    int scaledHeight = xMax - xMin;
6347
0
    ImageMaskScaler scaler(src, srcData, w, h,
6348
0
         scaledWidth, scaledHeight, interpolate, antialias);
6349
0
    Guchar *scaledMask = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
6350
0
    Guchar *ptr = scaledMask;
6351
0
    for (int y = 0; y < scaledHeight; ++y) {
6352
0
      scaler.nextLine();
6353
0
      memcpy(ptr, scaler.data(), scaledWidth);
6354
0
      ptr += scaledWidth;
6355
0
    }
6356
6357
    // draw it
6358
0
    Guchar *tmpLine = (Guchar *)gmalloc(scaledHeight);
6359
0
    for (int y = 0; y < scaledWidth; ++y) {
6360
0
      if (vertFlip) {
6361
0
  ptr = scaledMask + (scaledWidth - 1 - y);
6362
0
      } else {
6363
0
  ptr = scaledMask + y;
6364
0
      }
6365
0
      if (horizFlip) {
6366
0
  ptr += (scaledHeight - 1) * (SplashBitmapRowSize)scaledWidth;
6367
0
  for (int x = 0; x < scaledHeight; ++x) {
6368
0
    tmpLine[x] = *ptr;
6369
0
    ptr -= scaledWidth;
6370
0
  }
6371
0
      } else {
6372
0
  for (int x = 0; x < scaledHeight; ++x) {
6373
0
    tmpLine[x] = *ptr;
6374
0
    ptr += scaledWidth;
6375
0
  }
6376
0
      }
6377
0
      (this->*drawRowFunc)(&dd, tmpLine, xMin, yMin + y, scaledHeight);
6378
0
    }
6379
6380
0
    gfree(tmpLine);
6381
0
    gfree(scaledMask);
6382
6383
  //--- arbitrary transform
6384
620
  } else {
6385
    // estimate of size of scaled image
6386
620
    int scaledWidth = splashRound(splashSqrt(mat[0] * mat[0]
6387
620
               + mat[1] * mat[1]));
6388
620
    int scaledHeight = splashRound(splashSqrt(mat[2] * mat[2]
6389
620
                + mat[3] * mat[3]));
6390
620
    if (scaledWidth < 1) {
6391
603
      scaledWidth = 1;
6392
603
    }
6393
620
    if (scaledHeight < 1) {
6394
607
      scaledHeight = 1;
6395
607
    }
6396
620
    GBool downscaling = gTrue;
6397
620
    if (scaledWidth >= w && scaledHeight >= h) {
6398
23
      downscaling = gFalse;
6399
23
      scaledWidth = w;
6400
23
      scaledHeight = h;
6401
23
    }
6402
6403
    // compute mapping from device space to scaled image space
6404
620
    SplashCoord mat1[6];
6405
620
    mat1[0] = mat[0] / scaledWidth;
6406
620
    mat1[1] = mat[1] / scaledWidth;
6407
620
    mat1[2] = mat[2] / scaledHeight;
6408
620
    mat1[3] = mat[3] / scaledHeight;
6409
620
    mat1[4] = mat[4];
6410
620
    mat1[5] = mat[5];
6411
620
    SplashCoord det = mat1[0] * mat1[3] - mat1[1] * mat1[2];
6412
620
    if (splashAbs(det) < 1e-6) {
6413
      // this should be caught by the singular matrix check in drawImage
6414
0
      return splashErrSingularMatrix;
6415
0
    }
6416
620
    SplashCoord invMat[6];
6417
620
    invMat[0] = mat1[3] / det;
6418
620
    invMat[1] = -mat1[1] / det;
6419
620
    invMat[2] = -mat1[2] / det;
6420
620
    invMat[3] = mat1[0] / det;
6421
    // the extra "+ 0.5 * (...)" terms are here because the
6422
    // drawImageArbitrary(No)Interp functions multiply by pixel
6423
    // centers, (x + 0.5, y + 0.5)
6424
620
    invMat[4] = (mat1[2] * mat1[5] - mat1[3] * mat1[4]) / det
6425
620
                + (invMat[0] + invMat[2]) * 0.5;
6426
620
    invMat[5] = (mat1[1] * mat1[4] - mat1[0] * mat1[5]) / det
6427
620
                + (invMat[1] + invMat[3]) * 0.5;
6428
6429
    // if downscaling: store the downscaled image mask
6430
    // if upscaling: store the unscaled image mask
6431
620
    Guchar *scaledMask = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
6432
620
    if (downscaling) {
6433
597
      ImageMaskScaler scaler(src, srcData, w, h,
6434
597
           scaledWidth, scaledHeight, interpolate, antialias);
6435
597
      Guchar *ptr = scaledMask;
6436
10.1M
      for (int y = 0; y < scaledHeight; ++y) {
6437
10.1M
  scaler.nextLine();
6438
10.1M
  memcpy(ptr, scaler.data(), scaledWidth);
6439
10.1M
  ptr += scaledWidth;
6440
10.1M
      }
6441
597
    } else {
6442
23
      Guchar *ptr = scaledMask;
6443
55
      for (int y = 0; y < scaledHeight; ++y) {
6444
32
  (*src)(srcData, ptr);
6445
64
  for (int x = 0; x < scaledWidth; ++x) {
6446
32
    *ptr = (Guchar)(*ptr * 255);
6447
32
    ++ptr;
6448
32
  }
6449
32
      }
6450
23
    }
6451
6452
    // draw it
6453
620
    if (interpolate && antialias) {
6454
76
      drawImageMaskArbitraryInterp(scaledMask,
6455
76
           &dd, drawRowFunc, invMat,
6456
76
           scaledWidth, scaledHeight,
6457
76
           xMin, yMin, xMax, yMax);
6458
544
    } else {
6459
544
      drawImageMaskArbitraryNoInterp(scaledMask,
6460
544
             &dd, drawRowFunc, invMat,
6461
544
             scaledWidth, scaledHeight,
6462
544
             xMin, yMin, xMax, yMax);
6463
544
    }
6464
6465
    // free the downscaled/unscaled image
6466
620
    gfree(scaledMask);
6467
620
  }
6468
6469
779
  return splashOk;
6470
779
}
6471
6472
void Splash::drawImageMaskArbitraryNoInterp(
6473
           Guchar *scaledMask,
6474
           SplashDrawImageMaskRowData *dd,
6475
           SplashDrawImageMaskRowFunc drawRowFunc,
6476
           SplashCoord *invMat,
6477
           int scaledWidth, int scaledHeight,
6478
544
           int xMin, int yMin, int xMax, int yMax) {
6479
544
  int tt = state->clip->getXMinI(state->strokeAdjust);
6480
544
  if (tt > xMin) {
6481
72
    xMin = tt;
6482
72
  }
6483
544
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
6484
544
  if (tt < xMax) {
6485
520
    xMax = tt;
6486
520
  }
6487
544
  tt = state->clip->getYMinI(state->strokeAdjust);
6488
544
  if (tt > yMin) {
6489
105
    yMin = tt;
6490
105
  }
6491
544
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
6492
544
  if (tt < yMax) {
6493
538
    yMax = tt;
6494
538
  }
6495
544
  if (xMax <= xMin || yMax <= yMin) {
6496
0
    return;
6497
0
  }
6498
6499
544
  Guchar *buf = (Guchar *)gmalloc(xMax - xMin);
6500
6501
2.16k
  for (int y = yMin; y < yMax; ++y) {
6502
1.62k
    int rowMin = xMax;
6503
1.62k
    int rowMax = 0;
6504
81.6k
    for (int x = xMin; x < xMax; ++x) {
6505
      // note: invMat includes a "+0.5" factor so that this is really
6506
      // a multiply by (x+0.5, y+0.5)
6507
80.0k
      int xx = splashFloor((SplashCoord)x * invMat[0]
6508
80.0k
         + (SplashCoord)y * invMat[2] + invMat[4]);
6509
80.0k
      int yy = splashFloor((SplashCoord)x * invMat[1]
6510
80.0k
         + (SplashCoord)y * invMat[3] + invMat[5]);
6511
80.0k
      if (xx >= 0 && xx < scaledWidth &&
6512
79.8k
    yy >= 0 && yy < scaledHeight) {
6513
79.7k
  Guchar *p = scaledMask + (yy * (SplashBitmapRowSize)scaledWidth + xx);
6514
79.7k
  Guchar *q = buf + (x - xMin);
6515
79.7k
  *q = *p;
6516
79.7k
  if (x < rowMin) {
6517
1.52k
    rowMin = x;
6518
1.52k
  }
6519
79.7k
  rowMax = x + 1;
6520
79.7k
      }
6521
80.0k
    }
6522
1.62k
    if (rowMin < rowMax) {
6523
1.52k
      (this->*drawRowFunc)(dd, buf + (rowMin - xMin),
6524
1.52k
         rowMin, y, rowMax - rowMin);
6525
1.52k
    }
6526
1.62k
  }
6527
6528
544
  gfree(buf);
6529
544
}
6530
6531
void Splash::drawImageMaskArbitraryInterp(
6532
           Guchar *scaledMask,
6533
           SplashDrawImageMaskRowData *dd,
6534
           SplashDrawImageMaskRowFunc drawRowFunc,
6535
           SplashCoord *invMat,
6536
           int scaledWidth, int scaledHeight,
6537
76
           int xMin, int yMin, int xMax, int yMax) {
6538
76
  int tt = state->clip->getXMinI(state->strokeAdjust);
6539
76
  if (tt > xMin) {
6540
41
    xMin = tt;
6541
41
  }
6542
76
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
6543
76
  if (tt < xMax) {
6544
76
    xMax = tt;
6545
76
  }
6546
76
  tt = state->clip->getYMinI(state->strokeAdjust);
6547
76
  if (tt > yMin) {
6548
32
    yMin = tt;
6549
32
  }
6550
76
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
6551
76
  if (tt < yMax) {
6552
76
    yMax = tt;
6553
76
  }
6554
76
  if (xMax <= xMin || yMax <= yMin) {
6555
0
    return;
6556
0
  }
6557
6558
76
  Guchar *buf = (Guchar *)gmalloc(xMax - xMin);
6559
6560
152
  for (int y = yMin; y < yMax; ++y) {
6561
76
    int rowMin = xMax;
6562
76
    int rowMax = 0;
6563
152
    for (int x = xMin; x < xMax; ++x) {
6564
      // note: invMat includes a "+0.5" factor so that this is really
6565
      // a multiply by (x+0.5, y+0.5)
6566
76
      SplashCoord xs = (SplashCoord)x * invMat[0]
6567
76
                 + (SplashCoord)y * invMat[2] + invMat[4];
6568
76
      SplashCoord ys = (SplashCoord)x * invMat[1]
6569
76
                 + (SplashCoord)y * invMat[3] + invMat[5];
6570
76
      int x0 = splashFloor(xs - 0.5);
6571
76
      int x1 = x0 + 1;
6572
76
      int y0 = splashFloor(ys - 0.5);
6573
76
      int y1 = y0 + 1;
6574
76
      if (x1 >= 0 && x0 < scaledWidth && y1 >= 0 && y0 < scaledHeight) {
6575
76
  SplashCoord sx0 = (SplashCoord)x1 + 0.5 - xs;
6576
76
  SplashCoord sx1 = (SplashCoord)1 - sx0;
6577
76
  SplashCoord sy0 = (SplashCoord)y1 + 0.5 - ys;
6578
76
  SplashCoord sy1 = (SplashCoord)1 - sy0;
6579
76
  if (x0 < 0) {
6580
0
    x0 = 0;
6581
0
  }
6582
76
  if (x1 >= scaledWidth) {
6583
76
    x1 = scaledWidth - 1;
6584
76
  }
6585
76
  if (y0 < 0) {
6586
3
    y0 = 0;
6587
3
  }
6588
76
  if (y1 >= scaledHeight) {
6589
73
    y1 = scaledHeight - 1;
6590
73
  }
6591
76
  Guchar *p00 = scaledMask + (y0 * (SplashBitmapRowSize)scaledWidth + x0);
6592
76
  Guchar *p10 = scaledMask + (y0 * (SplashBitmapRowSize)scaledWidth + x1);
6593
76
  Guchar *p01 = scaledMask + (y1 * (SplashBitmapRowSize)scaledWidth + x0);
6594
76
  Guchar *p11 = scaledMask + (y1 * (SplashBitmapRowSize)scaledWidth + x1);
6595
76
  Guchar *q = buf + (x - xMin);
6596
76
  *q = (Guchar)(int)(sx0 * (sy0 * (int)*p00 + sy1 * (int)*p01) +
6597
76
         sx1 * (sy0 * (int)*p10 + sy1 * (int)*p11));
6598
76
  if (x < rowMin) {
6599
76
    rowMin = x;
6600
76
  }
6601
76
  rowMax = x + 1;
6602
76
      }
6603
76
    }
6604
76
    if (rowMin < rowMax) {
6605
76
      (this->*drawRowFunc)(dd, buf + (rowMin - xMin),
6606
76
         rowMin, y, rowMax - rowMin);
6607
76
    }
6608
76
  }
6609
6610
76
  gfree(buf);
6611
76
}
6612
6613
0
void Splash::mirrorImageMaskRow(Guchar *maskIn, Guchar *maskOut, int width) {
6614
0
  Guchar *p, *q;
6615
6616
0
  p = maskIn;
6617
0
  q = maskOut + (width - 1);
6618
0
  for (int i = 0; i < width; ++i) {
6619
0
    *q = *p;
6620
0
    ++p;
6621
0
    --q;
6622
0
  }
6623
0
}
6624
6625
void Splash::drawImageMaskRowNoClip(SplashDrawImageMaskRowData *data,
6626
            Guchar *maskData,
6627
47.3k
            int x, int y, int width) {
6628
47.3k
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
6629
47.3k
        NULL, maskData, NULL);
6630
47.3k
}
6631
6632
void Splash::drawImageMaskRowClipNoAA(SplashDrawImageMaskRowData *data,
6633
              Guchar *maskData,
6634
0
              int x, int y, int width) {
6635
0
  if (y < 0 || y >= bitmap->height) {
6636
0
    return;
6637
0
  }
6638
0
  if (x < 0) {
6639
0
    maskData -= x;
6640
0
    width += x;
6641
0
    x = 0;
6642
0
  }
6643
0
  if (x + width > bitmap->width) {
6644
0
    width = bitmap->width - x;
6645
0
  }
6646
0
  if (width <= 0) {
6647
0
    return;
6648
0
  }
6649
0
  memcpy(scanBuf + x, maskData, width);
6650
0
  state->clip->clipSpanBinary(scanBuf, y, x, x + width - 1,
6651
0
            state->strokeAdjust);
6652
0
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
6653
0
        NULL, scanBuf + x, NULL);
6654
0
}
6655
6656
void Splash::drawImageMaskRowClipAA(SplashDrawImageMaskRowData *data,
6657
            Guchar *maskData,
6658
1.60k
            int x, int y, int width) {
6659
1.60k
  if (y < 0 || y >= bitmap->height) {
6660
0
    return;
6661
0
  }
6662
1.60k
  if (x < 0) {
6663
0
    maskData -= x;
6664
0
    width += x;
6665
0
    x = 0;
6666
0
  }
6667
1.60k
  if (x + width > bitmap->width) {
6668
0
    width = bitmap->width - x;
6669
0
  }
6670
1.60k
  if (width <= 0) {
6671
0
    return;
6672
0
  }
6673
1.60k
  memcpy(scanBuf + x, maskData, width);
6674
1.60k
  state->clip->clipSpan(scanBuf, y, x, x + width - 1, state->strokeAdjust);
6675
1.60k
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
6676
1.60k
        NULL, scanBuf + x, NULL);
6677
1.60k
}
6678
6679
struct SplashDrawImageRowData {
6680
  int nComps;
6681
  GBool srcAlpha;
6682
  SplashPipe pipe;
6683
};
6684
6685
SplashError Splash::drawImage(GString *imageTag,
6686
            SplashImageSource src, void *srcData,
6687
            SplashColorMode srcMode, GBool srcAlpha,
6688
            int w, int h, SplashCoord *mat,
6689
30.7k
            GBool interpolate) {
6690
30.7k
  if (debugMode) {
6691
0
    printf("drawImage: srcMode=%d srcAlpha=%d w=%d h=%d mat=[%.2f %.2f %.2f %.2f %.2f %.2f]\n",
6692
0
     srcMode, srcAlpha, w, h, (double)mat[0], (double)mat[1], (double)mat[2],
6693
0
     (double)mat[3], (double)mat[4], (double)mat[5]);
6694
0
  }
6695
6696
  //--- check color modes
6697
30.7k
  GBool ok = gFalse;
6698
30.7k
  int nComps = 0;
6699
30.7k
  switch (bitmap->mode) {
6700
0
  case splashModeMono1:
6701
1.35k
  case splashModeMono8:
6702
1.35k
    ok = srcMode == splashModeMono8;
6703
1.35k
    nComps = 1;
6704
1.35k
    break;
6705
29.2k
  case splashModeRGB8:
6706
29.2k
  case splashModeBGR8:
6707
29.2k
    ok = srcMode == splashModeRGB8;
6708
29.2k
    nComps = 3;
6709
29.2k
    break;
6710
0
#if SPLASH_CMYK
6711
105
  case splashModeCMYK8:
6712
105
    ok = srcMode == splashModeCMYK8;
6713
105
    nComps = 4;
6714
105
    break;
6715
0
#endif
6716
0
  default:
6717
0
    ok = gFalse;
6718
0
    break;
6719
30.7k
  }
6720
30.7k
  if (!ok) {
6721
0
    return splashErrModeMismatch;
6722
0
  }
6723
6724
  //--- check for singular matrix
6725
30.7k
  if (!splashCheckDet(mat[0], mat[1], mat[2], mat[3], 0.000001)) {
6726
2.17k
    return splashErrSingularMatrix;
6727
2.17k
  }
6728
6729
  //--- compute image bbox, check clipping
6730
28.5k
  GBool flipsOnly = splashAbs(mat[1]) <= 0.0001 && splashAbs(mat[2]) <= 0.0001;
6731
28.5k
  GBool rot90Only = splashAbs(mat[0]) <= 0.0001 && splashAbs(mat[3]) <= 0.0001;
6732
28.5k
  GBool horizFlip = gFalse;
6733
28.5k
  GBool vertFlip = gFalse;
6734
28.5k
  int xMin, yMin, xMax, yMax;
6735
28.5k
  if (flipsOnly) {
6736
19.4k
    horizFlip = mat[0] < 0;
6737
19.4k
    vertFlip = mat[3] < 0;
6738
19.4k
    if (horizFlip) {
6739
6.88k
      getImageBounds(mat[0] + mat[4], mat[4], &xMin, &xMax);
6740
12.5k
    } else {
6741
12.5k
      getImageBounds(mat[4], mat[0] + mat[4], &xMin, &xMax);
6742
12.5k
    }
6743
19.4k
    if (vertFlip) {
6744
7.68k
      getImageBounds(mat[3] + mat[5], mat[5], &yMin, &yMax);
6745
11.7k
    } else {
6746
11.7k
      getImageBounds(mat[5], mat[3] + mat[5], &yMin, &yMax);
6747
11.7k
    }
6748
19.4k
  } else if (rot90Only) {
6749
3.33k
    horizFlip = mat[2] < 0;
6750
3.33k
    vertFlip = mat[1] < 0;
6751
3.33k
    if (horizFlip) {
6752
776
      getImageBounds(mat[2] + mat[4], mat[4], &xMin, &xMax);
6753
2.55k
    } else {
6754
2.55k
      getImageBounds(mat[4], mat[2] + mat[4], &xMin, &xMax);
6755
2.55k
    }
6756
3.33k
    if (vertFlip) {
6757
1.70k
      getImageBounds(mat[1] + mat[5], mat[5], &yMin, &yMax);
6758
1.70k
    } else {
6759
1.62k
      getImageBounds(mat[5], mat[1] + mat[5], &yMin, &yMax);
6760
1.62k
    }
6761
5.77k
  } else {
6762
5.77k
    int xx = splashRound(mat[4]);   // (0,0)
6763
5.77k
    int yy = splashRound(mat[5]);
6764
5.77k
    xMin = xMax = xx;
6765
5.77k
    yMin = yMax = yy;
6766
5.77k
    xx = splashRound(mat[0] + mat[4]);    // (1,0)
6767
5.77k
    yy = splashRound(mat[1] + mat[5]);
6768
5.77k
    if (xx < xMin) {
6769
258
      xMin = xx;
6770
5.51k
    } else if (xx > xMax) {
6771
1.19k
      xMax = xx;
6772
1.19k
    }
6773
5.77k
    if (yy < yMin) {
6774
323
      yMin = yy;
6775
5.45k
    } else if (yy > yMax) {
6776
611
      yMax = yy;
6777
611
    }
6778
5.77k
    xx = splashRound(mat[2] + mat[4]);    // (0,1)
6779
5.77k
    yy = splashRound(mat[3] + mat[5]);
6780
5.77k
    if (xx < xMin) {
6781
1.41k
      xMin = xx;
6782
4.35k
    } else if (xx > xMax) {
6783
225
      xMax = xx;
6784
225
    }
6785
5.77k
    if (yy < yMin) {
6786
789
      yMin = yy;
6787
4.98k
    } else if (yy > yMax) {
6788
321
      yMax = yy;
6789
321
    }
6790
5.77k
    xx = splashRound(mat[0] + mat[2] + mat[4]); // (1,1)
6791
5.77k
    yy = splashRound(mat[1] + mat[3] + mat[5]);
6792
5.77k
    if (xx < xMin) {
6793
720
      xMin = xx;
6794
5.05k
    } else if (xx > xMax) {
6795
245
      xMax = xx;
6796
245
    }
6797
5.77k
    if (yy < yMin) {
6798
991
      yMin = yy;
6799
4.78k
    } else if (yy > yMax) {
6800
49
      yMax = yy;
6801
49
    }
6802
5.77k
    if (xMax <= xMin) {
6803
3.02k
      xMax = xMin + 1;
6804
3.02k
    }
6805
5.77k
    if (yMax <= yMin) {
6806
3.50k
      yMax = yMin + 1;
6807
3.50k
    }
6808
5.77k
  }
6809
28.5k
  SplashClipResult clipRes =
6810
28.5k
      state->clip->testRect(xMin, yMin, xMax - 1, yMax - 1,
6811
28.5k
          state->strokeAdjust);
6812
28.5k
  if (clipRes == splashClipAllOutside) {
6813
27.4k
    return splashOk;
6814
27.4k
  }
6815
  // If the scaled image is much wider and/or taller than the clip
6816
  // region, we use the arbitrary transform path, to avoid a
6817
  // potentially very slow loop in the flips-only path (which scans
6818
  // the full width and height of the scaled image, regardless of the
6819
  // clip region).
6820
1.08k
  int clipW = state->clip->getXMaxI(state->strokeAdjust)
6821
1.08k
        - state->clip->getXMinI(state->strokeAdjust);
6822
1.08k
  int clipH = state->clip->getYMaxI(state->strokeAdjust)
6823
1.08k
        - state->clip->getYMinI(state->strokeAdjust);
6824
1.08k
  GBool veryLarge = ((xMax - xMin) / 8 > clipW && xMax - xMin > 1000) ||
6825
228
                    ((yMax - yMin) / 8 > clipH && yMax - yMin > 1000);
6826
6827
  //--- set up the SplashDrawImageRowData object and the pipes
6828
1.08k
  SplashDrawImageRowData dd;
6829
1.08k
  dd.nComps = nComps;
6830
1.08k
  dd.srcAlpha = srcAlpha;
6831
1.08k
  pipeInit(&dd.pipe, NULL,
6832
1.08k
     (Guchar)splashRound(state->fillAlpha * 255),
6833
1.08k
     gFalse, clipRes != splashClipAllInside || srcAlpha,
6834
1.08k
     gFalse, gFalse, state->alphaIsShape);
6835
6836
  //--- choose the drawRow function
6837
1.08k
  SplashDrawImageRowFunc drawRowFunc;
6838
1.08k
  if (clipRes == splashClipAllInside) {
6839
0
    if (srcAlpha) {
6840
0
      drawRowFunc = &Splash::drawImageRowNoClipAlpha;
6841
0
    } else {
6842
0
      drawRowFunc = &Splash::drawImageRowNoClipNoAlpha;
6843
0
    }
6844
1.08k
  } else {
6845
1.08k
    if (srcAlpha) {
6846
29
      if (vectorAntialias) {
6847
29
  drawRowFunc = &Splash::drawImageRowClipAlphaAA;
6848
29
      } else {
6849
0
  drawRowFunc = &Splash::drawImageRowClipAlphaNoAA;
6850
0
      }
6851
1.05k
    } else {
6852
1.05k
      if (vectorAntialias) {
6853
1.05k
  drawRowFunc = &Splash::drawImageRowClipNoAlphaAA;
6854
1.05k
      } else {
6855
0
  drawRowFunc = &Splash::drawImageRowClipNoAlphaNoAA;
6856
0
      }
6857
1.05k
    }
6858
1.08k
  }
6859
6860
  //--- horizontal/vertical flips only
6861
1.08k
  if (flipsOnly && !veryLarge) {
6862
0
    int scaledWidth = xMax - xMin;
6863
0
    int scaledHeight = yMax - yMin;
6864
0
    ImageScaler *scaler = getImageScaler(imageTag, src, srcData,
6865
0
           w, h, nComps,
6866
0
           scaledWidth, scaledHeight,
6867
0
           srcMode, srcAlpha, interpolate);
6868
0
    Guchar *tmpLine = NULL;
6869
0
    Guchar *tmpAlphaLine = NULL;
6870
0
    if (horizFlip) {
6871
0
      tmpLine = (Guchar *)gmallocn(scaledWidth, nComps);
6872
0
      if (srcAlpha) {
6873
0
  tmpAlphaLine = (Guchar *)gmalloc(scaledWidth);
6874
0
      }
6875
0
    }
6876
0
    if (vertFlip) {
6877
0
      if (horizFlip) {    // bottom-up, mirrored
6878
0
  for (int y = 0; y < scaledHeight; ++y) {
6879
0
    scaler->nextLine();
6880
0
    mirrorImageRow(scaler->colorData(), scaler->alphaData(),
6881
0
       tmpLine, tmpAlphaLine,
6882
0
       scaledWidth, nComps, srcAlpha);
6883
0
    (this->*drawRowFunc)(&dd, tmpLine, tmpAlphaLine,
6884
0
             xMin, yMax - 1 - y, scaledWidth);
6885
0
  }
6886
0
      } else {            // bottom-up
6887
0
  for (int y = 0; y < scaledHeight; ++y) {
6888
0
    scaler->nextLine();
6889
0
    (this->*drawRowFunc)(&dd, scaler->colorData(), scaler->alphaData(),
6890
0
             xMin, yMax - 1 - y, scaledWidth);
6891
0
  }
6892
0
      }
6893
0
    } else {
6894
0
      if (horizFlip) {    // top-down, mirrored
6895
0
  for (int y = 0; y < scaledHeight; ++y) {
6896
0
    scaler->nextLine();
6897
0
    mirrorImageRow(scaler->colorData(), scaler->alphaData(),
6898
0
       tmpLine, tmpAlphaLine,
6899
0
       scaledWidth, nComps, srcAlpha);
6900
0
    (this->*drawRowFunc)(&dd, tmpLine, tmpAlphaLine,
6901
0
             xMin, yMin + y, scaledWidth);
6902
0
  }
6903
0
      } else {            // top-down
6904
0
  for (int y = 0; y < scaledHeight; ++y) {
6905
0
    scaler->nextLine();
6906
0
    (this->*drawRowFunc)(&dd, scaler->colorData(), scaler->alphaData(),
6907
0
             xMin, yMin + y, scaledWidth);
6908
0
  }
6909
0
      }
6910
0
    }
6911
0
    gfree(tmpLine);
6912
0
    gfree(tmpAlphaLine);
6913
0
    delete scaler;
6914
6915
  //--- 90/270 rotation
6916
1.08k
  } else if (rot90Only && !veryLarge) {
6917
6918
    // scale the image
6919
0
    int scaledWidth = yMax - yMin;
6920
0
    int scaledHeight = xMax - xMin;
6921
0
    Guchar *scaledColor, *scaledAlpha;
6922
0
    GBool freeScaledImage;
6923
0
    getScaledImage(imageTag, src, srcData, w, h, nComps,
6924
0
       scaledWidth, scaledHeight, srcMode, srcAlpha, interpolate,
6925
0
       &scaledColor, &scaledAlpha, &freeScaledImage);
6926
6927
    // draw it
6928
0
    Guchar *tmpLine = (Guchar *)gmallocn(scaledHeight, nComps);
6929
0
    Guchar *tmpAlphaLine = NULL;
6930
0
    if (srcAlpha) {
6931
0
      tmpAlphaLine = (Guchar *)gmalloc(scaledHeight);
6932
0
    }
6933
0
    for (int y = 0; y < scaledWidth; ++y) {
6934
0
      Guchar *ptr = NULL;
6935
0
      Guchar *alphaPtr = NULL;
6936
0
      if (vertFlip) {
6937
0
  ptr = scaledColor + ((SplashBitmapRowSize)scaledWidth - 1 - y) * nComps;
6938
0
  if (srcAlpha) {
6939
0
    alphaPtr = scaledAlpha + (scaledWidth - 1 - y);
6940
0
  }
6941
0
      } else {
6942
0
  ptr = scaledColor + y * nComps;
6943
0
  if (srcAlpha) {
6944
0
    alphaPtr = scaledAlpha + y;
6945
0
  }
6946
0
      }
6947
0
      if (horizFlip) {
6948
0
  ptr += (scaledHeight - 1) * (SplashBitmapRowSize)scaledWidth * nComps;
6949
0
  Guchar *q = tmpLine;
6950
0
  for (int x = 0; x < scaledHeight; ++x) {
6951
0
    for (int i = 0; i < nComps; ++i) {
6952
0
      *q++ = ptr[i];
6953
0
    }
6954
0
    ptr -= scaledWidth * nComps;
6955
0
  }
6956
0
  if (srcAlpha) {
6957
0
    alphaPtr += (scaledHeight - 1) * (SplashBitmapRowSize)scaledWidth;
6958
0
    q = tmpAlphaLine;
6959
0
    for (int x = 0; x < scaledHeight; ++x) {
6960
0
      *q++ = *alphaPtr;
6961
0
      alphaPtr -= scaledWidth;
6962
0
    }
6963
0
  }
6964
0
      } else {
6965
0
  Guchar *q = tmpLine;
6966
0
  for (int x = 0; x < scaledHeight; ++x) {
6967
0
    for (int i = 0; i < nComps; ++i) {
6968
0
      *q++ = ptr[i];
6969
0
    }
6970
0
    ptr += scaledWidth * nComps;
6971
0
  }
6972
0
  if (srcAlpha) {
6973
0
    q = tmpAlphaLine;
6974
0
    for (int x = 0; x < scaledHeight; ++x) {
6975
0
      *q++ = *alphaPtr;
6976
0
      alphaPtr += scaledWidth;
6977
0
    }
6978
0
  }
6979
0
      }
6980
0
      (this->*drawRowFunc)(&dd, tmpLine, tmpAlphaLine,
6981
0
         xMin, yMin + y, scaledHeight);
6982
0
    }
6983
6984
0
    gfree(tmpLine);
6985
0
    gfree(tmpAlphaLine);
6986
0
    if (freeScaledImage) {
6987
0
      gfree(scaledColor);
6988
0
      gfree(scaledAlpha);
6989
0
    }
6990
6991
  //--- arbitrary transform
6992
1.08k
  } else {
6993
    // estimate of size of scaled image
6994
1.08k
    int scaledWidth = splashRound(splashSqrt(mat[0] * mat[0]
6995
1.08k
               + mat[1] * mat[1]));
6996
1.08k
    int scaledHeight = splashRound(splashSqrt(mat[2] * mat[2]
6997
1.08k
                + mat[3] * mat[3]));
6998
1.08k
    if (scaledWidth < 1) {
6999
1.08k
      scaledWidth = 1;
7000
1.08k
    }
7001
1.08k
    if (scaledHeight < 1) {
7002
1.08k
      scaledHeight = 1;
7003
1.08k
    }
7004
1.08k
    if (scaledWidth >= w && scaledHeight >= h) {
7005
39
      scaledWidth = w;
7006
39
      scaledHeight = h;
7007
39
    }
7008
7009
    // compute mapping from device space to scaled image space
7010
1.08k
    SplashCoord mat1[6];
7011
1.08k
    mat1[0] = mat[0] / scaledWidth;
7012
1.08k
    mat1[1] = mat[1] / scaledWidth;
7013
1.08k
    mat1[2] = mat[2] / scaledHeight;
7014
1.08k
    mat1[3] = mat[3] / scaledHeight;
7015
1.08k
    mat1[4] = mat[4];
7016
1.08k
    mat1[5] = mat[5];
7017
1.08k
    SplashCoord det = mat1[0] * mat1[3] - mat1[1] * mat1[2];
7018
1.08k
    if (splashAbs(det) < 1e-6) {
7019
      // this should be caught by the singular matrix check in drawImage
7020
0
      return splashErrSingularMatrix;
7021
0
    }
7022
1.08k
    SplashCoord invMat[6];
7023
1.08k
    invMat[0] = mat1[3] / det;
7024
1.08k
    invMat[1] = -mat1[1] / det;
7025
1.08k
    invMat[2] = -mat1[2] / det;
7026
1.08k
    invMat[3] = mat1[0] / det;
7027
    // the extra "+ 0.5 * (...)" terms are here because the
7028
    // drawImageArbitrary(No)Interp functions multiply by pixel
7029
    // centers, (x + 0.5, y + 0.5)
7030
1.08k
    invMat[4] = (mat1[2] * mat1[5] - mat1[3] * mat1[4]) / det
7031
1.08k
                + (invMat[0] + invMat[2]) * 0.5;
7032
1.08k
    invMat[5] = (mat1[1] * mat1[4] - mat1[0] * mat1[5]) / det
7033
1.08k
                + (invMat[1] + invMat[3]) * 0.5;
7034
7035
1.08k
    Guchar *scaledColor, *scaledAlpha;
7036
1.08k
    GBool freeScaledImage;
7037
1.08k
    getScaledImage(imageTag, src, srcData, w, h, nComps,
7038
1.08k
       scaledWidth, scaledHeight, srcMode, srcAlpha, interpolate,
7039
1.08k
       &scaledColor, &scaledAlpha, &freeScaledImage);
7040
7041
    // draw it
7042
1.08k
    if (interpolate) {
7043
102
      drawImageArbitraryInterp(scaledColor, scaledAlpha,
7044
102
             &dd, drawRowFunc, invMat,
7045
102
             scaledWidth, scaledHeight,
7046
102
             xMin, yMin, xMax, yMax,
7047
102
             nComps, srcAlpha);
7048
980
    } else {
7049
980
      drawImageArbitraryNoInterp(scaledColor, scaledAlpha,
7050
980
         &dd, drawRowFunc, invMat,
7051
980
         scaledWidth, scaledHeight,
7052
980
         xMin, yMin, xMax, yMax,
7053
980
         nComps, srcAlpha);
7054
980
    }
7055
7056
    // free the downscaled/unscaled image
7057
1.08k
    if (freeScaledImage) {
7058
0
      gfree(scaledColor);
7059
0
      gfree(scaledAlpha);
7060
0
    }
7061
1.08k
  }
7062
7063
1.08k
  return splashOk;
7064
1.08k
}
7065
7066
ImageScaler *Splash::getImageScaler(GString *imageTag,
7067
            SplashImageSource src, void *srcData,
7068
            int w, int h, int nComps,
7069
            int scaledWidth, int scaledHeight,
7070
            SplashColorMode srcMode,
7071
0
            GBool srcAlpha, GBool interpolate) {
7072
  // Notes:
7073
  //
7074
  // * If the scaled image is more than 8 Mpixels, we don't cache it.
7075
  //
7076
  // * Caching is done on the third consecutive use (second
7077
  //   consecutive reuse) of an image; this avoids overhead on the
7078
  //   common case of single-use images.
7079
7080
0
  if (scaledWidth < 8000000 / scaledHeight &&
7081
0
      imageCache->match(imageTag, scaledWidth, scaledHeight,
7082
0
      srcMode, srcAlpha, interpolate)) {
7083
0
    if (imageCache->colorData) {
7084
0
      return new ReplayImageScaler(nComps, srcAlpha, scaledWidth,
7085
0
           imageCache->colorData,
7086
0
           imageCache->alphaData);
7087
0
    } else {
7088
0
      int lineSize;
7089
0
      if (scaledWidth < INT_MAX / nComps) {
7090
0
  lineSize = scaledWidth * nComps;
7091
0
      } else {
7092
0
  lineSize = -1;
7093
0
      }
7094
0
      imageCache->colorData = (Guchar *)gmallocn64(scaledHeight, lineSize);
7095
0
      if (srcAlpha) {
7096
0
  imageCache->alphaData = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
7097
0
      }
7098
0
      return new SavingImageScaler(src, srcData,
7099
0
           w, h, nComps, srcAlpha,
7100
0
           scaledWidth, scaledHeight,
7101
0
           interpolate,
7102
0
           imageCache->colorData,
7103
0
           imageCache->alphaData);
7104
0
    }
7105
0
  } else {
7106
0
    imageCache->reset(imageTag, scaledWidth, scaledHeight,
7107
0
          srcMode, srcAlpha, interpolate);
7108
0
    return new BasicImageScaler(src, srcData,
7109
0
        w, h, nComps, srcAlpha,
7110
0
        scaledWidth, scaledHeight,
7111
0
        interpolate);
7112
0
  }
7113
0
}
7114
7115
void Splash::getScaledImage(GString *imageTag,
7116
          SplashImageSource src, void *srcData,
7117
          int w, int h, int nComps,
7118
          int scaledWidth, int scaledHeight,
7119
          SplashColorMode srcMode,
7120
          GBool srcAlpha, GBool interpolate,
7121
          Guchar **scaledColor, Guchar **scaledAlpha,
7122
1.08k
          GBool *freeScaledImage) {
7123
  // Notes:
7124
  //
7125
  // * If the scaled image is more than 8 Mpixels, we don't cache it.
7126
  //
7127
  // * This buffers the whole image anyway, so there's no reason to
7128
  //   skip caching on the first reuse.
7129
7130
1.08k
  if (scaledWidth >= 8000000 / scaledHeight) {
7131
0
    int lineSize;
7132
0
    if (scaledWidth < INT_MAX / nComps) {
7133
0
      lineSize = scaledWidth * nComps;
7134
0
    } else {
7135
0
      lineSize = -1;
7136
0
    }
7137
0
    *scaledColor = (Guchar *)gmallocn64(scaledHeight, lineSize);
7138
0
    if (srcAlpha) {
7139
0
      *scaledAlpha = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
7140
0
    } else {
7141
0
      *scaledAlpha = NULL;
7142
0
    }
7143
0
    *freeScaledImage = gTrue;
7144
0
    if (scaledWidth == w && scaledHeight == h) {
7145
0
      Guchar *colorPtr = *scaledColor;
7146
0
      Guchar *alphaPtr = *scaledAlpha;
7147
0
      for (int y = 0; y < scaledHeight; ++y) {
7148
0
  (*src)(srcData, colorPtr, alphaPtr);
7149
0
  colorPtr += scaledWidth * nComps;
7150
0
  if (srcAlpha) {
7151
0
    alphaPtr += scaledWidth;
7152
0
  }
7153
0
      }
7154
0
    } else {
7155
0
      BasicImageScaler scaler(src, srcData, w, h, nComps, srcAlpha,
7156
0
            scaledWidth, scaledHeight, interpolate);
7157
0
      Guchar *colorPtr = *scaledColor;
7158
0
      Guchar *alphaPtr = *scaledAlpha;
7159
0
      for (int y = 0; y < scaledHeight; ++y) {
7160
0
  scaler.nextLine();
7161
0
  memcpy(colorPtr, scaler.colorData(), scaledWidth * nComps);
7162
0
  colorPtr += scaledWidth * nComps;
7163
0
  if (srcAlpha) {
7164
0
    memcpy(alphaPtr, scaler.alphaData(), scaledWidth);
7165
0
    alphaPtr += scaledWidth;
7166
0
  }
7167
0
      }
7168
0
    }
7169
1.08k
  } else {
7170
1.08k
    if (!imageCache->match(imageTag, scaledWidth, scaledHeight,
7171
1.08k
        srcMode, srcAlpha, interpolate) ||
7172
903
  !imageCache->colorData) {
7173
903
      imageCache->reset(imageTag, scaledWidth, scaledHeight,
7174
903
      srcMode, srcAlpha, interpolate);
7175
903
      int lineSize;
7176
903
      if (scaledWidth < INT_MAX / nComps) {
7177
903
  lineSize = scaledWidth * nComps;
7178
903
      } else {
7179
0
  lineSize = -1;
7180
0
      }
7181
903
      imageCache->colorData = (Guchar *)gmallocn64(scaledHeight, lineSize);
7182
903
      if (srcAlpha) {
7183
17
  imageCache->alphaData = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
7184
17
      }
7185
903
      if (scaledWidth == w && scaledHeight == h) {
7186
39
  Guchar *colorPtr = imageCache->colorData;
7187
39
  Guchar *alphaPtr = imageCache->alphaData;
7188
78
  for (int y = 0; y < scaledHeight; ++y) {
7189
39
    (*src)(srcData, colorPtr, alphaPtr);
7190
39
    colorPtr += scaledWidth * nComps;
7191
39
    if (srcAlpha) {
7192
0
      alphaPtr += scaledWidth;
7193
0
    }
7194
39
  }
7195
864
      } else {
7196
864
  SavingImageScaler scaler(src, srcData, w, h, nComps, srcAlpha,
7197
864
         scaledWidth, scaledHeight, interpolate,
7198
864
         imageCache->colorData, imageCache->alphaData);
7199
864
  Guchar *colorPtr = imageCache->colorData;
7200
864
  Guchar *alphaPtr = imageCache->alphaData;
7201
1.72k
  for (int y = 0; y < scaledHeight; ++y) {
7202
864
    scaler.nextLine();
7203
864
    memcpy(colorPtr, scaler.colorData(), scaledWidth * nComps);
7204
864
    colorPtr += scaledWidth * nComps;
7205
864
    if (srcAlpha) {
7206
17
      memcpy(alphaPtr, scaler.alphaData(), scaledWidth);
7207
17
      alphaPtr += scaledWidth;
7208
17
    }
7209
864
  }
7210
864
      }
7211
903
    }
7212
1.08k
    *scaledColor = imageCache->colorData;
7213
1.08k
    *scaledAlpha = imageCache->alphaData;
7214
1.08k
    *freeScaledImage = gFalse;
7215
1.08k
  }
7216
1.08k
}
7217
7218
void Splash::drawImageArbitraryNoInterp(Guchar *scaledColor,
7219
          Guchar *scaledAlpha,
7220
          SplashDrawImageRowData *dd,
7221
          SplashDrawImageRowFunc drawRowFunc,
7222
          SplashCoord *invMat,
7223
          int scaledWidth, int scaledHeight,
7224
          int xMin, int yMin, int xMax, int yMax,
7225
980
          int nComps, GBool srcAlpha) {
7226
980
  int tt = state->clip->getXMinI(state->strokeAdjust);
7227
980
  if (tt > xMin) {
7228
182
    xMin = tt;
7229
182
  }
7230
980
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
7231
980
  if (tt < xMax) {
7232
980
    xMax = tt;
7233
980
  }
7234
980
  tt = state->clip->getYMinI(state->strokeAdjust);
7235
980
  if (tt > yMin) {
7236
440
    yMin = tt;
7237
440
  }
7238
980
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
7239
980
  if (tt < yMax) {
7240
980
    yMax = tt;
7241
980
  }
7242
980
  if (xMax <= xMin || yMax <= yMin) {
7243
0
    return;
7244
0
  }
7245
7246
980
  Guchar *colorBuf = (Guchar *)gmallocn(xMax - xMin, nComps);
7247
980
  Guchar *alphaBuf = NULL;
7248
980
  if (srcAlpha) {
7249
29
    alphaBuf = (Guchar *)gmalloc(xMax - xMin);
7250
29
  }
7251
7252
1.96k
  for (int y = yMin; y < yMax; ++y) {
7253
980
    int rowMin = xMax;
7254
980
    int rowMax = 0;
7255
1.96k
    for (int x = xMin; x < xMax; ++x) {
7256
      // note: invMat includes a "+0.5" factor so that this is really
7257
      // a multiply by (x+0.5, y+0.5)
7258
980
      int xx = splashFloor((SplashCoord)x * invMat[0]
7259
980
         + (SplashCoord)y * invMat[2] + invMat[4]);
7260
980
      int yy = splashFloor((SplashCoord)x * invMat[1]
7261
980
         + (SplashCoord)y * invMat[3] + invMat[5]);
7262
980
      if (xx >= 0 && xx < scaledWidth &&
7263
621
    yy >= 0 && yy < scaledHeight) {
7264
446
  Guchar *p = scaledColor +
7265
446
                (yy * (SplashBitmapRowSize)scaledWidth + xx) * nComps;
7266
446
  Guchar *q = colorBuf + (x - xMin) * nComps;
7267
1.44k
  for (int i = 0; i < nComps; ++i) {
7268
994
    *q++ = *p++;
7269
994
  }
7270
446
  if (srcAlpha) {
7271
29
    alphaBuf[x - xMin] =
7272
29
        scaledAlpha[yy * (SplashBitmapRowSize)scaledWidth + xx];
7273
29
  }
7274
446
  if (x < rowMin) {
7275
446
    rowMin = x;
7276
446
  }
7277
446
  rowMax = x + 1;
7278
446
      }
7279
980
    }
7280
980
    if (rowMin < rowMax) {
7281
446
      (this->*drawRowFunc)(dd,
7282
446
         colorBuf + (rowMin - xMin) * nComps,
7283
446
         alphaBuf + (rowMin - xMin),
7284
446
         rowMin, y, rowMax - rowMin);
7285
446
    }
7286
980
  }
7287
7288
980
  gfree(colorBuf);
7289
980
  gfree(alphaBuf);
7290
980
}
7291
7292
void Splash::drawImageArbitraryInterp(Guchar *scaledColor, Guchar *scaledAlpha,
7293
              SplashDrawImageRowData *dd,
7294
              SplashDrawImageRowFunc drawRowFunc,
7295
              SplashCoord *invMat,
7296
              int scaledWidth, int scaledHeight,
7297
              int xMin, int yMin, int xMax, int yMax,
7298
102
              int nComps, GBool srcAlpha) {
7299
102
  int tt = state->clip->getXMinI(state->strokeAdjust);
7300
102
  if (tt > xMin) {
7301
46
    xMin = tt;
7302
46
  }
7303
102
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
7304
102
  if (tt < xMax) {
7305
102
    xMax = tt;
7306
102
  }
7307
102
  tt = state->clip->getYMinI(state->strokeAdjust);
7308
102
  if (tt > yMin) {
7309
56
    yMin = tt;
7310
56
  }
7311
102
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
7312
102
  if (tt < yMax) {
7313
102
    yMax = tt;
7314
102
  }
7315
102
  if (xMax <= xMin || yMax <= yMin) {
7316
0
    return;
7317
0
  }
7318
7319
102
  Guchar *colorBuf = (Guchar *)gmallocn(xMax - xMin, nComps);
7320
102
  Guchar *alphaBuf = NULL;
7321
102
  if (srcAlpha) {
7322
0
    alphaBuf = (Guchar *)gmalloc(xMax - xMin);
7323
0
  }
7324
7325
204
  for (int y = yMin; y < yMax; ++y) {
7326
102
    int rowMin = xMax;
7327
102
    int rowMax = 0;
7328
204
    for (int x = xMin; x < xMax; ++x) {
7329
      // note: invMat includes a "+0.5" factor so that this is really
7330
      // a multiply by (x+0.5, y+0.5)
7331
102
      SplashCoord xs = (SplashCoord)x * invMat[0]
7332
102
                 + (SplashCoord)y * invMat[2] + invMat[4];
7333
102
      SplashCoord ys = (SplashCoord)x * invMat[1]
7334
102
                 + (SplashCoord)y * invMat[3] + invMat[5];
7335
102
      int x0 = splashFloor(xs - 0.5);
7336
102
      int x1 = x0 + 1;
7337
102
      int y0 = splashFloor(ys - 0.5);
7338
102
      int y1 = y0 + 1;
7339
102
      if (x1 >= 0 && x0 < scaledWidth && y1 >= 0 && y0 < scaledHeight) {
7340
102
  SplashCoord sx0 = (SplashCoord)x1 + 0.5 - xs;
7341
102
  SplashCoord sx1 = (SplashCoord)1 - sx0;
7342
102
  SplashCoord sy0 = (SplashCoord)y1 + 0.5 - ys;
7343
102
  SplashCoord sy1 = (SplashCoord)1 - sy0;
7344
102
  if (x0 < 0) {
7345
0
    x0 = 0;
7346
0
  }
7347
102
  if (x1 >= scaledWidth) {
7348
102
    x1 = scaledWidth - 1;
7349
102
  }
7350
102
  if (y0 < 0) {
7351
0
    y0 = 0;
7352
0
  }
7353
102
  if (y1 >= scaledHeight) {
7354
102
    y1 = scaledHeight - 1;
7355
102
  }
7356
102
  Guchar *p00 = scaledColor +
7357
102
                  (y0 * (SplashBitmapRowSize)scaledWidth + x0) * nComps;
7358
102
  Guchar *p10 = scaledColor +
7359
102
                  (y0 * (SplashBitmapRowSize)scaledWidth + x1) * nComps;
7360
102
  Guchar *p01 = scaledColor +
7361
102
                  (y1 * (SplashBitmapRowSize)scaledWidth + x0) * nComps;
7362
102
  Guchar *p11 = scaledColor +
7363
102
                  (y1 * (SplashBitmapRowSize)scaledWidth + x1) * nComps;
7364
102
  Guchar *q = colorBuf + (x - xMin) * nComps;
7365
408
  for (int i = 0; i < nComps; ++i) {
7366
306
    *q++ = (Guchar)(int)(sx0 * (sy0 * (int)*p00++ + sy1 * (int)*p01++) +
7367
306
             sx1 * (sy0 * (int)*p10++ + sy1 * (int)*p11++));
7368
306
  }
7369
102
  if (srcAlpha) {
7370
0
    p00 = scaledAlpha + (y0 * (SplashBitmapRowSize)scaledWidth + x0);
7371
0
    p10 = scaledAlpha + (y0 * (SplashBitmapRowSize)scaledWidth + x1);
7372
0
    p01 = scaledAlpha + (y1 * (SplashBitmapRowSize)scaledWidth + x0);
7373
0
    p11 = scaledAlpha + (y1 * (SplashBitmapRowSize)scaledWidth + x1);
7374
0
    q = alphaBuf + (x - xMin);
7375
0
    *q = (Guchar)(int)(sx0 * (sy0 * (int)*p00 + sy1 * (int)*p01) +
7376
0
           sx1 * (sy0 * (int)*p10 + sy1 * (int)*p11));
7377
0
  }
7378
102
  if (x < rowMin) {
7379
102
    rowMin = x;
7380
102
  }
7381
102
  rowMax = x + 1;
7382
102
      }
7383
102
    }
7384
102
    if (rowMin < rowMax) {
7385
102
      (this->*drawRowFunc)(dd,
7386
102
         colorBuf + (rowMin - xMin) * nComps,
7387
102
         alphaBuf + (rowMin - xMin),
7388
102
         rowMin, y, rowMax - rowMin);
7389
102
    }
7390
102
  }
7391
7392
102
  gfree(colorBuf);
7393
102
  gfree(alphaBuf);
7394
102
}
7395
7396
void Splash::mirrorImageRow(Guchar *colorIn, Guchar *alphaIn,
7397
          Guchar *colorOut, Guchar *alphaOut,
7398
0
          int width, int nComps, GBool srcAlpha) {
7399
0
  Guchar *p, *q;
7400
7401
0
  p = colorIn;
7402
0
  q = colorOut + (width - 1) * nComps;
7403
0
  for (int i = 0; i < width; ++i) {
7404
0
    for (int j = 0; j < nComps; ++j) {
7405
0
      q[j] = p[j];
7406
0
    }
7407
0
    p += nComps;
7408
0
    q -= nComps;
7409
0
  }
7410
7411
0
  if (srcAlpha) {
7412
0
    p = alphaIn;
7413
0
    q = alphaOut + (width - 1);
7414
0
    for (int i = 0; i < width; ++i) {
7415
0
      *q = *p;
7416
0
      ++p;
7417
0
      --q;
7418
0
    }
7419
0
  }
7420
0
}
7421
7422
void Splash::drawImageRowNoClipNoAlpha(SplashDrawImageRowData *data,
7423
               Guchar *colorData, Guchar *alphaData,
7424
0
               int x, int y, int width) {
7425
0
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7426
0
        NULL, NULL, colorData);
7427
0
}
7428
7429
void Splash::drawImageRowNoClipAlpha(SplashDrawImageRowData *data,
7430
             Guchar *colorData, Guchar *alphaData,
7431
0
             int x, int y, int width) {
7432
0
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7433
0
        NULL, alphaData, colorData);
7434
0
}
7435
7436
void Splash::drawImageRowClipNoAlphaNoAA(SplashDrawImageRowData *data,
7437
           Guchar *colorData,
7438
           Guchar *alphaData,
7439
0
           int x, int y, int width) {
7440
0
  if (y < 0 || y >= bitmap->height) {
7441
0
    return;
7442
0
  }
7443
0
  if (x < 0) {
7444
0
    colorData -= x * data->nComps;
7445
0
    width += x;
7446
0
    x = 0;
7447
0
  }
7448
0
  if (x + width > bitmap->width) {
7449
0
    width = bitmap->width - x;
7450
0
  }
7451
0
  if (width <= 0) {
7452
0
    return;
7453
0
  }
7454
0
  memset(scanBuf + x, 0xff, width);
7455
0
  state->clip->clipSpanBinary(scanBuf, y, x, x + width - 1,
7456
0
            state->strokeAdjust);
7457
0
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7458
0
        NULL, scanBuf + x, colorData);
7459
0
}
7460
7461
void Splash::drawImageRowClipNoAlphaAA(SplashDrawImageRowData *data,
7462
               Guchar *colorData,
7463
               Guchar *alphaData,
7464
519
               int x, int y, int width) {
7465
519
  if (y < 0 || y >= bitmap->height) {
7466
0
    return;
7467
0
  }
7468
519
  if (x < 0) {
7469
0
    colorData -= x * data->nComps;
7470
0
    width += x;
7471
0
    x = 0;
7472
0
  }
7473
519
  if (x + width > bitmap->width) {
7474
0
    width = bitmap->width - x;
7475
0
  }
7476
519
  if (width <= 0) {
7477
0
    return;
7478
0
  }
7479
519
  memset(scanBuf + x, 0xff, width);
7480
519
  state->clip->clipSpan(scanBuf, y, x, x + width - 1, state->strokeAdjust);
7481
519
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7482
519
        NULL, scanBuf + x, colorData);
7483
519
}
7484
7485
void Splash::drawImageRowClipAlphaNoAA(SplashDrawImageRowData *data,
7486
               Guchar *colorData,
7487
               Guchar *alphaData,
7488
0
               int x, int y, int width) {
7489
0
  if (y < 0 || y >= bitmap->height) {
7490
0
    return;
7491
0
  }
7492
0
  if (x < 0) {
7493
0
    colorData -= x * data->nComps;
7494
0
    alphaData -= x;
7495
0
    width += x;
7496
0
    x = 0;
7497
0
  }
7498
0
  if (x + width > bitmap->width) {
7499
0
    width = bitmap->width - x;
7500
0
  }
7501
0
  if (width <= 0) {
7502
0
    return;
7503
0
  }
7504
0
  memcpy(scanBuf + x, alphaData, width);
7505
0
  state->clip->clipSpanBinary(scanBuf, y, x, x + width - 1,
7506
0
            state->strokeAdjust);
7507
0
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7508
0
        NULL, scanBuf + x, colorData);
7509
0
}
7510
7511
void Splash::drawImageRowClipAlphaAA(SplashDrawImageRowData *data,
7512
             Guchar *colorData,
7513
             Guchar *alphaData,
7514
29
             int x, int y, int width) {
7515
29
  if (y < 0 || y >= bitmap->height) {
7516
0
    return;
7517
0
  }
7518
29
  if (x < 0) {
7519
0
    colorData -= x * data->nComps;
7520
0
    alphaData -= x;
7521
0
    width += x;
7522
0
    x = 0;
7523
0
  }
7524
29
  if (x + width > bitmap->width) {
7525
0
    width = bitmap->width - x;
7526
0
  }
7527
29
  if (width <= 0) {
7528
0
    return;
7529
0
  }
7530
29
  memcpy(scanBuf + x, alphaData, width);
7531
29
  state->clip->clipSpan(scanBuf, y, x, x + width - 1, state->strokeAdjust);
7532
29
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7533
29
        NULL, scanBuf + x, colorData);
7534
29
}
7535
7536
SplashError Splash::composite(SplashBitmap *src, SplashAlphaBitmap *srcShape,
7537
            int xSrc, int ySrc, int xDest, int yDest,
7538
4.19k
            int w, int h, GBool noClip, GBool nonIsolated) {
7539
4.19k
  SplashPipe pipe;
7540
4.19k
  Guchar *mono1Ptr, *lineBuf, *linePtr;
7541
4.19k
  Guchar mono1Mask, b;
7542
4.19k
  int x0, x1, x, y0, y1, y, t;
7543
7544
4.19k
  if (!(src->mode == bitmap->mode ||
7545
0
  (src->mode == splashModeMono8 && bitmap->mode == splashModeMono1) ||
7546
0
  (src->mode == splashModeRGB8 && bitmap->mode == splashModeBGR8))) {
7547
0
    return splashErrModeMismatch;
7548
0
  }
7549
7550
4.19k
  GBool usesShape = srcShape && shapeBitmap && src->alpha;
7551
4.19k
  GBool usesAlpha = !noClip || src->alpha;
7552
4.19k
  pipeInit(&pipe, NULL,
7553
4.19k
     (Guchar)splashRound(state->fillAlpha * 255),
7554
4.19k
     usesShape, usesAlpha, nonIsolated);
7555
4.19k
  if (src->mode == splashModeMono1) {
7556
    // in mono1 mode, pipeRun expects the source to be in mono8
7557
    // format, so we need to extract the source color values into
7558
    // scanBuf, expanding them from mono1 to mono8
7559
0
    if (noClip) {
7560
0
      if (usesShape) {
7561
0
  for (y = 0; y < h; ++y) {
7562
0
    mono1Ptr = src->data + (ySrc + y) * src->rowSize + (xSrc >> 3);
7563
0
    mono1Mask = (Guchar)(0x80 >> (xSrc & 7));
7564
0
    for (x = 0; x < w; ++x) {
7565
0
      scanBuf[x] = (*mono1Ptr & mono1Mask) ? 0xff : 0x00;
7566
0
      mono1Ptr += mono1Mask & 1;
7567
0
      mono1Mask = (Guchar)((mono1Mask << 7) | (mono1Mask >> 1));
7568
0
    }
7569
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7570
0
          srcShape->alpha +
7571
0
            (ySrc + y) * srcShape->alphaRowSize + xSrc,
7572
0
          src->alpha +
7573
0
            (ySrc + y) * src->alphaRowSize + xSrc,
7574
0
          scanBuf);
7575
0
  }
7576
0
      } else  if (usesAlpha) {
7577
0
  for (y = 0; y < h; ++y) {
7578
0
    mono1Ptr = src->data + (ySrc + y) * src->rowSize + (xSrc >> 3);
7579
0
    mono1Mask = (Guchar)(0x80 >> (xSrc & 7));
7580
0
    for (x = 0; x < w; ++x) {
7581
0
      scanBuf[x] = (*mono1Ptr & mono1Mask) ? 0xff : 0x00;
7582
0
      mono1Ptr += mono1Mask & 1;
7583
0
      mono1Mask = (Guchar)((mono1Mask << 7) | (mono1Mask >> 1));
7584
0
    }
7585
    // this uses shape instead of alpha, which isn't technically
7586
    // correct, but works out the same
7587
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7588
0
          NULL,
7589
0
          src->alpha +
7590
0
            (ySrc + y) * src->alphaRowSize + xSrc,
7591
0
          scanBuf);
7592
0
  }
7593
0
      } else {
7594
0
  for (y = 0; y < h; ++y) {
7595
0
    mono1Ptr = src->data + (ySrc + y) * src->rowSize + (xSrc >> 3);
7596
0
    mono1Mask = (Guchar)(0x80 >> (xSrc & 7));
7597
0
    for (x = 0; x < w; ++x) {
7598
0
      scanBuf[x] = (*mono1Ptr & mono1Mask) ? 0xff : 0x00;
7599
0
      mono1Ptr += mono1Mask & 1;
7600
0
      mono1Mask = (Guchar)((mono1Mask << 7) | (mono1Mask >> 1));
7601
0
    }
7602
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7603
0
          NULL, NULL, scanBuf);
7604
0
  }
7605
0
      }
7606
0
    } else {
7607
0
      x0 = xDest;
7608
0
      if ((t = state->clip->getXMinI(state->strokeAdjust)) > x0) {
7609
0
  x0 = t;
7610
0
      }
7611
0
      x1 = xDest + w;
7612
0
      if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < x1) {
7613
0
  x1 = t;
7614
0
      }
7615
0
      y0 = yDest;
7616
0
      if ((t = state->clip->getYMinI(state->strokeAdjust)) > y0) {
7617
0
  y0 = t;
7618
0
      }
7619
0
      y1 = yDest + h;
7620
0
      if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < y1) {
7621
0
  y1 = t;
7622
0
      }
7623
0
      if (x0 < x1 && y0 < y1) {
7624
0
  if (usesShape) {
7625
0
    for (y = y0; y < y1; ++y) {
7626
0
      mono1Ptr = src->data
7627
0
                 + (ySrc + y - yDest) * src->rowSize
7628
0
                 + ((xSrc + x0 - xDest) >> 3);
7629
0
      mono1Mask = (Guchar)(0x80 >> ((xSrc + x0 - xDest) & 7));
7630
0
      for (x = x0; x < x1; ++x) {
7631
0
        scanBuf[x] = (*mono1Ptr & mono1Mask) ? 0xff : 0x00;
7632
0
        mono1Ptr += mono1Mask & 1;
7633
0
        mono1Mask = (Guchar)((mono1Mask << 7) | (mono1Mask >> 1));
7634
0
      }
7635
0
      memcpy(scanBuf2 + x0,
7636
0
       srcShape->alpha +
7637
0
         (ySrc + y - yDest) * srcShape->alphaRowSize + 
7638
0
         (xSrc + x0 - xDest),
7639
0
       x1 - x0);
7640
0
      if (!state->clip->clipSpanBinary(scanBuf2, y, x0, x1 - 1,
7641
0
               state->strokeAdjust)) {
7642
0
        continue;
7643
0
      }
7644
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7645
0
            scanBuf2 + x0,
7646
0
            srcShape->alpha +
7647
0
              (ySrc + y - yDest) * srcShape->alphaRowSize + 
7648
0
              (xSrc + x0 - xDest),
7649
0
            scanBuf + x0);
7650
0
    }
7651
0
  } else if (usesAlpha) {
7652
0
    for (y = y0; y < y1; ++y) {
7653
0
      mono1Ptr = src->data
7654
0
                 + (ySrc + y - yDest) * src->rowSize
7655
0
                 + ((xSrc + x0 - xDest) >> 3);
7656
0
      mono1Mask = (Guchar)(0x80 >> ((xSrc + x0 - xDest) & 7));
7657
0
      for (x = x0; x < x1; ++x) {
7658
0
        scanBuf[x] = (*mono1Ptr & mono1Mask) ? 0xff : 0x00;
7659
0
        mono1Ptr += mono1Mask & 1;
7660
0
        mono1Mask = (Guchar)((mono1Mask << 7) | (mono1Mask >> 1));
7661
0
      }
7662
0
      memcpy(scanBuf2 + x0,
7663
0
       src->alpha + (ySrc + y - yDest) * src->alphaRowSize + 
7664
0
         (xSrc + x0 - xDest),
7665
0
       x1 - x0);
7666
0
      if (!state->clip->clipSpanBinary(scanBuf2, y, x0, x1 - 1,
7667
0
               state->strokeAdjust)) {
7668
0
        continue;
7669
0
      }
7670
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7671
0
            NULL, scanBuf2 + x0, scanBuf + x0);
7672
0
    }
7673
0
  } else {
7674
0
    for (y = y0; y < y1; ++y) {
7675
0
      mono1Ptr = src->data
7676
0
                 + (ySrc + y - yDest) * src->rowSize
7677
0
                 + ((xSrc + x0 - xDest) >> 3);
7678
0
      mono1Mask = (Guchar)(0x80 >> ((xSrc + x0 - xDest) & 7));
7679
0
      for (x = x0; x < x1; ++x) {
7680
0
        scanBuf[x] = (*mono1Ptr & mono1Mask) ? 0xff : 0x00;
7681
0
        mono1Ptr += mono1Mask & 1;
7682
0
        mono1Mask = (Guchar)((mono1Mask << 7) | (mono1Mask >> 1));
7683
0
      }
7684
0
      memset(scanBuf2 + x0, 0xff, x1 - x0);
7685
0
      if (!state->clip->clipSpanBinary(scanBuf2, y, x0, x1 - 1,
7686
0
               state->strokeAdjust)) {
7687
0
        continue;
7688
0
      }
7689
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7690
0
            NULL, scanBuf2 + x0, scanBuf + x0);
7691
0
    }
7692
0
  }
7693
0
      }
7694
0
    }
7695
7696
4.19k
  } else if (src->mode == splashModeBGR8) {
7697
    // in BGR8 mode, pipeRun expects the source to be in RGB8 format,
7698
    // so we need to swap bytes
7699
0
    lineBuf = (Guchar *)gmallocn(w, 3);
7700
0
    if (noClip) {
7701
0
      if (usesShape) {
7702
0
  for (y = 0; y < h; ++y) {
7703
0
    memcpy(lineBuf,
7704
0
     src->data + (ySrc + y) * src->rowSize + xSrc * 3,
7705
0
     w * 3);
7706
0
    for (x = 0, linePtr = lineBuf; x < w; ++x, linePtr += 3) {
7707
0
      b = linePtr[0];
7708
0
      linePtr[0] = linePtr[2];
7709
0
      linePtr[2] = b;
7710
0
    }
7711
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7712
0
          srcShape->alpha +
7713
0
            (ySrc + y) * srcShape->alphaRowSize + xSrc,
7714
0
          src->alpha +
7715
0
            (ySrc + y) * src->alphaRowSize + xSrc,
7716
0
          lineBuf);
7717
0
  }
7718
0
      } else if (usesAlpha) {
7719
0
  for (y = 0; y < h; ++y) {
7720
0
    memcpy(lineBuf,
7721
0
     src->data + (ySrc + y) * src->rowSize + xSrc * 3,
7722
0
     w * 3);
7723
0
    for (x = 0, linePtr = lineBuf; x < w; ++x, linePtr += 3) {
7724
0
      b = linePtr[0];
7725
0
      linePtr[0] = linePtr[2];
7726
0
      linePtr[2] = b;
7727
0
    }
7728
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7729
0
          NULL,
7730
0
          src->alpha +
7731
0
            (ySrc + y) * src->alphaRowSize + xSrc,
7732
0
          lineBuf);
7733
0
  }
7734
0
      } else {
7735
0
  for (y = 0; y < h; ++y) {
7736
0
    memcpy(lineBuf,
7737
0
     src->data + (ySrc + y) * src->rowSize + xSrc * 3,
7738
0
     w * 3);
7739
0
    for (x = 0, linePtr = lineBuf; x < w; ++x, linePtr += 3) {
7740
0
      b = linePtr[0];
7741
0
      linePtr[0] = linePtr[2];
7742
0
      linePtr[2] = b;
7743
0
    }
7744
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7745
0
          NULL, NULL, lineBuf);
7746
0
  }
7747
0
      }
7748
0
    } else {
7749
0
      x0 = xDest;
7750
0
      if ((t = state->clip->getXMinI(state->strokeAdjust)) > x0) {
7751
0
  x0 = t;
7752
0
      }
7753
0
      x1 = xDest + w;
7754
0
      if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < x1) {
7755
0
  x1 = t;
7756
0
      }
7757
0
      y0 = yDest;
7758
0
      if ((t = state->clip->getYMinI(state->strokeAdjust)) > y0) {
7759
0
  y0 = t;
7760
0
      }
7761
0
      y1 = yDest + h;
7762
0
      if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < y1) {
7763
0
  y1 = t;
7764
0
      }
7765
0
      if (x0 < x1 && y0 < y1) {
7766
0
  if (usesShape) {
7767
0
    for (y = y0; y < y1; ++y) {
7768
0
      memcpy(scanBuf + x0,
7769
0
       srcShape->alpha +
7770
0
         (ySrc + y - yDest) * srcShape->alphaRowSize + 
7771
0
         (xSrc + x0 - xDest),
7772
0
       x1 - x0);
7773
0
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7774
0
      memcpy(lineBuf,
7775
0
       src->data +
7776
0
         (ySrc + y - yDest) * src->rowSize +
7777
0
         (xSrc + x0 - xDest) * 3,
7778
0
       (x1 - x0) * 3);
7779
0
      for (x = 0, linePtr = lineBuf; x < x1 - x0; ++x, linePtr += 3) {
7780
0
        b = linePtr[0];
7781
0
        linePtr[0] = linePtr[2];
7782
0
        linePtr[2] = b;
7783
0
      }
7784
      // this uses shape instead of alpha, which isn't technically
7785
      // correct, but works out the same
7786
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7787
0
            scanBuf + x0,
7788
0
            src->alpha +
7789
0
              (ySrc + y - yDest) * src->alphaRowSize + 
7790
0
              (xSrc + x0 - xDest),
7791
0
            lineBuf);
7792
0
    }
7793
0
  } else if (usesAlpha) {
7794
0
    for (y = y0; y < y1; ++y) {
7795
0
      memcpy(scanBuf + x0,
7796
0
       src->alpha + (ySrc + y - yDest) * src->alphaRowSize + 
7797
0
         (xSrc + x0 - xDest),
7798
0
       x1 - x0);
7799
0
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7800
0
      memcpy(lineBuf,
7801
0
       src->data +
7802
0
         (ySrc + y - yDest) * src->rowSize +
7803
0
         (xSrc + x0 - xDest) * 3,
7804
0
       (x1 - x0) * 3);
7805
0
      for (x = 0, linePtr = lineBuf; x < x1 - x0; ++x, linePtr += 3) {
7806
0
        b = linePtr[0];
7807
0
        linePtr[0] = linePtr[2];
7808
0
        linePtr[2] = b;
7809
0
      }
7810
      // this uses shape instead of alpha, which isn't technically
7811
      // correct, but works out the same
7812
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7813
0
            NULL, scanBuf + x0, lineBuf);
7814
0
    }
7815
0
  } else {
7816
0
    for (y = y0; y < y1; ++y) {
7817
0
      memset(scanBuf + x0, 0xff, x1 - x0);
7818
0
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7819
0
      memcpy(lineBuf,
7820
0
       src->data +
7821
0
         (ySrc + y - yDest) * src->rowSize +
7822
0
         (xSrc + x0 - xDest) * 3,
7823
0
       (x1 - x0) * 3);
7824
0
      for (x = 0, linePtr = lineBuf; x < x1 - x0; ++x, linePtr += 3) {
7825
0
        b = linePtr[0];
7826
0
        linePtr[0] = linePtr[2];
7827
0
        linePtr[2] = b;
7828
0
      }
7829
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, yDest + y,
7830
0
            NULL,
7831
0
            scanBuf + x0,
7832
0
            src->data +
7833
0
              (ySrc + y - yDest) * src->rowSize +
7834
0
              (xSrc + x0 - xDest) * bitmapComps);
7835
0
    }
7836
0
  }
7837
0
      }
7838
0
    }
7839
0
    gfree(lineBuf);
7840
7841
4.19k
  } else { // src->mode not mono1 or BGR8
7842
4.19k
    if (noClip) {
7843
0
      if (usesShape) {
7844
0
  for (y = 0; y < h; ++y) {
7845
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7846
0
          srcShape->alpha +
7847
0
            (ySrc + y) * srcShape->alphaRowSize + xSrc,
7848
0
          src->alpha +
7849
0
            (ySrc + y) * src->alphaRowSize + xSrc,
7850
0
          src->data + (ySrc + y) * src->rowSize +
7851
0
          xSrc * bitmapComps);
7852
0
  }
7853
0
      } else if (usesAlpha) {
7854
0
  for (y = 0; y < h; ++y) {
7855
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7856
0
          NULL,
7857
0
          src->alpha +
7858
0
            (ySrc + y) * src->alphaRowSize + xSrc,
7859
0
          src->data + (ySrc + y) * src->rowSize +
7860
0
          xSrc * bitmapComps);
7861
0
  }
7862
0
      } else {
7863
0
  for (y = 0; y < h; ++y) {
7864
0
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7865
0
          NULL, NULL,
7866
0
          src->data + (ySrc + y) * src->rowSize +
7867
0
            xSrc * bitmapComps);
7868
0
  }
7869
0
      }
7870
4.19k
    } else {
7871
4.19k
      x0 = xDest;
7872
4.19k
      if ((t = state->clip->getXMinI(state->strokeAdjust)) > x0) {
7873
1.49k
  x0 = t;
7874
1.49k
      }
7875
4.19k
      x1 = xDest + w;
7876
4.19k
      if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < x1) {
7877
1.77k
  x1 = t;
7878
1.77k
      }
7879
4.19k
      y0 = yDest;
7880
4.19k
      if ((t = state->clip->getYMinI(state->strokeAdjust)) > y0) {
7881
2.15k
  y0 = t;
7882
2.15k
      }
7883
4.19k
      y1 = yDest + h;
7884
4.19k
      if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < y1) {
7885
1.63k
  y1 = t;
7886
1.63k
      }
7887
4.19k
      if (x0 < x1 && y0 < y1) {
7888
1.50k
  if (usesShape) {
7889
0
    for (y = y0; y < y1; ++y) {
7890
0
      memcpy(scanBuf + x0,
7891
0
       srcShape->alpha +
7892
0
         (ySrc + y - yDest) * srcShape->alphaRowSize + 
7893
0
         (xSrc + x0 - xDest),
7894
0
       x1 - x0);
7895
0
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7896
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7897
0
            scanBuf + x0,
7898
0
            src->alpha +
7899
0
              (ySrc + y - yDest) * src->alphaRowSize + 
7900
0
              (xSrc + x0 - xDest),
7901
0
            src->data +
7902
0
              (ySrc + y - yDest) * src->rowSize +
7903
0
              (xSrc + x0 - xDest) * bitmapComps);
7904
0
    }
7905
1.50k
  } else if (usesAlpha) {
7906
3.00k
    for (y = y0; y < y1; ++y) {
7907
1.50k
      memcpy(scanBuf + x0,
7908
1.50k
       src->alpha + (ySrc + y - yDest) * src->alphaRowSize + 
7909
1.50k
         (xSrc + x0 - xDest),
7910
1.50k
       x1 - x0);
7911
1.50k
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7912
1.50k
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7913
1.50k
            NULL,
7914
1.50k
            scanBuf + x0,
7915
1.50k
            src->data +
7916
1.50k
              (ySrc + y - yDest) * src->rowSize +
7917
1.50k
              (xSrc + x0 - xDest) * bitmapComps);
7918
1.50k
    }
7919
1.50k
  } else {
7920
0
    for (y = y0; y < y1; ++y) {
7921
0
      memset(scanBuf + x0, 0xff, x1 - x0);
7922
0
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7923
0
      (this->*pipe.run)(&pipe, x0, x1 - 1, yDest + y,
7924
0
            NULL,
7925
0
            scanBuf + x0,
7926
0
            src->data +
7927
0
              (ySrc + y - yDest) * src->rowSize +
7928
0
              (xSrc + x0 - xDest) * bitmapComps);
7929
0
    }
7930
0
  }
7931
1.50k
      }
7932
4.19k
    }
7933
4.19k
  }
7934
7935
4.19k
  return splashOk;
7936
4.19k
}
7937
7938
SplashError Splash::compositeWithOverprint(SplashBitmap *src,
7939
             SplashAlphaBitmap *srcShape,
7940
             Guint *srcOverprintMaskBitmap,
7941
             int xSrc, int ySrc,
7942
             int xDest, int yDest, int w, int h,
7943
0
             GBool noClip, GBool nonIsolated) {
7944
0
  SplashPipe pipe;
7945
0
  int x0, x1, y0, y1, y, t;
7946
7947
0
  if (!(src->mode == bitmap->mode ||
7948
0
  (src->mode == splashModeMono8 && bitmap->mode == splashModeMono1) ||
7949
0
  (src->mode == splashModeRGB8 && bitmap->mode == splashModeBGR8))) {
7950
0
    return splashErrModeMismatch;
7951
0
  }
7952
7953
0
  GBool usesShape = srcShape && shapeBitmap && src->alpha;
7954
0
  GBool usesAlpha = !noClip || src->alpha;
7955
0
  pipeInit(&pipe, NULL,
7956
0
     (Guchar)splashRound(state->fillAlpha * 255),
7957
0
     usesShape, usesAlpha, nonIsolated, gTrue);
7958
7959
0
  if (noClip) {
7960
0
    if (usesShape) {
7961
0
      for (y = 0; y < h; ++y) {
7962
0
  pipe.srcOverprintMaskPtr = srcOverprintMaskBitmap + y * w + xSrc;
7963
0
  (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7964
0
        srcShape->alpha +
7965
0
          (ySrc + y) * srcShape->alphaRowSize + xSrc,
7966
0
        src->alpha +
7967
0
          (ySrc + y) * src->alphaRowSize + xSrc,
7968
0
        src->data + (ySrc + y) * src->rowSize +
7969
0
          xSrc * bitmapComps);
7970
0
      }
7971
0
    } else if (usesAlpha) {
7972
0
      for (y = 0; y < h; ++y) {
7973
0
  pipe.srcOverprintMaskPtr = srcOverprintMaskBitmap + y * w + xSrc;
7974
0
  (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7975
0
        NULL,
7976
0
        src->alpha +
7977
0
          (ySrc + y) * src->alphaRowSize + xSrc,
7978
0
        src->data + (ySrc + y) * src->rowSize +
7979
0
          xSrc * bitmapComps);
7980
0
      }
7981
0
    } else {
7982
0
      for (y = 0; y < h; ++y) {
7983
0
  pipe.srcOverprintMaskPtr = srcOverprintMaskBitmap + y * w + xSrc;
7984
0
  (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7985
0
        NULL, NULL,
7986
0
        src->data + (ySrc + y) * src->rowSize +
7987
0
          xSrc * bitmapComps);
7988
0
      }
7989
0
    }
7990
0
  } else {
7991
0
    x0 = xDest;
7992
0
    if ((t = state->clip->getXMinI(state->strokeAdjust)) > x0) {
7993
0
      x0 = t;
7994
0
    }
7995
0
    x1 = xDest + w;
7996
0
    if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < x1) {
7997
0
      x1 = t;
7998
0
    }
7999
0
    y0 = yDest;
8000
0
    if ((t = state->clip->getYMinI(state->strokeAdjust)) > y0) {
8001
0
      y0 = t;
8002
0
    }
8003
0
    y1 = yDest + h;
8004
0
    if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < y1) {
8005
0
      y1 = t;
8006
0
    }
8007
0
    if (x0 < x1 && y0 < y1) {
8008
0
      if (usesShape) {
8009
0
  for (y = y0; y < y1; ++y) {
8010
0
    memcpy(scanBuf + x0,
8011
0
     srcShape->alpha +
8012
0
       (ySrc + y - yDest) * srcShape->alphaRowSize + 
8013
0
       (xSrc + x0 - xDest),
8014
0
     x1 - x0);
8015
0
    state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
8016
0
    pipe.srcOverprintMaskPtr = srcOverprintMaskBitmap
8017
0
                               + (ySrc + y - yDest) * w
8018
0
                               + (xSrc + x0 - xDest);
8019
0
    (this->*pipe.run)(&pipe, x0, x1 - 1, y,
8020
0
          scanBuf + x0,
8021
0
          src->alpha +
8022
0
            (ySrc + y - yDest) * src->alphaRowSize + 
8023
0
            (xSrc + x0 - xDest),
8024
0
          src->data +
8025
0
            (ySrc + y - yDest) * src->rowSize +
8026
0
            (xSrc + x0 - xDest) * bitmapComps);
8027
0
  }
8028
0
      } else if (usesAlpha) {
8029
0
  for (y = y0; y < y1; ++y) {
8030
0
    memcpy(scanBuf + x0,
8031
0
     src->alpha + (ySrc + y - yDest) * src->alphaRowSize + 
8032
0
       (xSrc + x0 - xDest),
8033
0
     x1 - x0);
8034
0
    state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
8035
0
    pipe.srcOverprintMaskPtr = srcOverprintMaskBitmap
8036
0
                               + (ySrc + y - yDest) * w
8037
0
                               + (xSrc + x0 - xDest);
8038
0
    (this->*pipe.run)(&pipe, x0, x1 - 1, y,
8039
0
          NULL,
8040
0
          scanBuf + x0,
8041
0
          src->data +
8042
0
            (ySrc + y - yDest) * src->rowSize +
8043
0
            (xSrc + x0 - xDest) * bitmapComps);
8044
0
  }
8045
0
      } else {
8046
0
  for (y = y0; y < y1; ++y) {
8047
0
    memset(scanBuf + x0, 0xff, x1 - x0);
8048
0
    state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
8049
0
    pipe.srcOverprintMaskPtr = srcOverprintMaskBitmap
8050
0
                               + (ySrc + y - yDest) * w
8051
0
                               + (xSrc + x0 - xDest);
8052
0
    (this->*pipe.run)(&pipe, x0, x1 - 1, y,
8053
0
          NULL,
8054
0
          scanBuf + x0,
8055
0
          src->data +
8056
0
            (ySrc + y - yDest) * src->rowSize +
8057
0
            (xSrc + x0 - xDest) * bitmapComps);
8058
0
  }
8059
0
      }
8060
0
    }
8061
0
  }
8062
8063
0
  return splashOk;
8064
0
}
8065
8066
1.15k
void Splash::compositeBackground(SplashColorPtr color) {
8067
1.15k
  SplashColorPtr p;
8068
1.15k
  Guchar *q;
8069
1.15k
  Guchar alpha, alpha1, c, color0, color1, color2, mask;
8070
1.15k
#if SPLASH_CMYK
8071
1.15k
  Guchar color3;
8072
1.15k
#endif
8073
1.15k
  int x, y;
8074
8075
1.15k
  switch (bitmap->mode) {
8076
0
  case splashModeMono1:
8077
0
    color0 = color[0];
8078
0
    for (y = 0; y < bitmap->height; ++y) {
8079
0
      p = &bitmap->data[y * bitmap->rowSize];
8080
0
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8081
0
      mask = 0x80;
8082
0
      for (x = 0; x < bitmap->width; ++x) {
8083
0
  alpha = *q++;
8084
0
  if (alpha == 0) {
8085
0
    if (color0 & 0x80) {
8086
0
      *p |= mask;
8087
0
    } else {
8088
0
      *p &= (Guchar)~mask;
8089
0
    }
8090
0
  } else if (alpha != 255) {
8091
0
    alpha1 = (Guchar)(255 - alpha);
8092
0
    c = (*p & mask) ? 0xff : 0x00;
8093
0
    c = div255(alpha1 * color0 + alpha * c);
8094
0
    if (c & 0x80) {
8095
0
      *p |= mask;
8096
0
    } else {
8097
0
      *p &= (Guchar)~mask;
8098
0
    }
8099
0
  }
8100
0
  if (!(mask = (Guchar)(mask >> 1))) {
8101
0
    mask = 0x80;
8102
0
    ++p;
8103
0
  }
8104
0
      }
8105
0
    }
8106
0
    break;
8107
996
  case splashModeMono8:
8108
996
    color0 = color[0];
8109
1.99k
    for (y = 0; y < bitmap->height; ++y) {
8110
996
      p = &bitmap->data[y * bitmap->rowSize];
8111
996
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8112
1.99k
      for (x = 0; x < bitmap->width; ++x) {
8113
996
  alpha = *q++;
8114
996
  if (alpha == 0) {
8115
559
    p[0] = color0;
8116
559
  } else if (alpha != 255) {
8117
40
    alpha1 = (Guchar)(255 - alpha);
8118
40
    p[0] = div255(alpha1 * color0 + alpha * p[0]);
8119
40
  }
8120
996
  ++p;
8121
996
      }
8122
996
    }
8123
996
    break;
8124
29
  case splashModeRGB8:
8125
29
  case splashModeBGR8:
8126
29
    color0 = color[0];
8127
29
    color1 = color[1];
8128
29
    color2 = color[2];
8129
58
    for (y = 0; y < bitmap->height; ++y) {
8130
29
      p = &bitmap->data[y * bitmap->rowSize];
8131
29
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8132
58
      for (x = 0; x < bitmap->width; ++x) {
8133
29
  alpha = *q++;
8134
29
  if (alpha == 0) {
8135
29
    p[0] = color0;
8136
29
    p[1] = color1;
8137
29
    p[2] = color2;
8138
29
  } else if (alpha != 255) {
8139
0
    alpha1 = (Guchar)(255 - alpha);
8140
0
    p[0] = div255(alpha1 * color0 + alpha * p[0]);
8141
0
    p[1] = div255(alpha1 * color1 + alpha * p[1]);
8142
0
    p[2] = div255(alpha1 * color2 + alpha * p[2]);
8143
0
  }
8144
29
  p += 3;
8145
29
      }
8146
29
    }
8147
29
    break;
8148
0
#if SPLASH_CMYK
8149
126
  case splashModeCMYK8:
8150
126
    color0 = color[0];
8151
126
    color1 = color[1];
8152
126
    color2 = color[2];
8153
126
    color3 = color[3];
8154
252
    for (y = 0; y < bitmap->height; ++y) {
8155
126
      p = &bitmap->data[y * bitmap->rowSize];
8156
126
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8157
252
      for (x = 0; x < bitmap->width; ++x) {
8158
126
  alpha = *q++;
8159
126
  if (alpha == 0) {
8160
126
    p[0] = color0;
8161
126
    p[1] = color1;
8162
126
    p[2] = color2;
8163
126
    p[3] = color3;
8164
126
  } else if (alpha != 255) {
8165
0
    alpha1 = (Guchar)(255 - alpha);
8166
0
    p[0] = div255(alpha1 * color0 + alpha * p[0]);
8167
0
    p[1] = div255(alpha1 * color1 + alpha * p[1]);
8168
0
    p[2] = div255(alpha1 * color2 + alpha * p[2]);
8169
0
    p[3] = div255(alpha1 * color3 + alpha * p[3]);
8170
0
  }
8171
126
  p += 4;
8172
126
      }
8173
126
    }
8174
126
    break;
8175
1.15k
#endif
8176
1.15k
  }
8177
1.15k
  memset(bitmap->alpha, 255, bitmap->alphaRowSize * bitmap->height);
8178
1.15k
}
8179
8180
SplashPath *Splash::makeStrokePath(SplashPath *path, SplashCoord w,
8181
           int lineCap, int lineJoin,
8182
21.1k
           GBool flatten) {
8183
21.1k
  SplashPath *pathIn, *dashPath, *pathOut;
8184
21.1k
  SplashCoord d, dx, dy, wdx, wdy, dxNext, dyNext, wdxNext, wdyNext;
8185
21.1k
  SplashCoord crossprod, dotprod, miter, m;
8186
21.1k
  SplashCoord angle, angleNext, dAngle, xc, yc;
8187
21.1k
  SplashCoord dxJoin, dyJoin, dJoin, kappa;
8188
21.1k
  SplashCoord cx1, cy1, cx2, cy2, cx3, cy3, cx4, cy4;
8189
21.1k
  GBool first, last, closed;
8190
21.1k
  int subpathStart0, subpathStart1, seg, i0, i1, j0, j1, k0, k1;
8191
21.1k
  int left0, left1, left2, right0, right1, right2, join0, join1, join2;
8192
21.1k
  int leftFirst, rightFirst, firstPt;
8193
8194
21.1k
  pathOut = new SplashPath();
8195
8196
21.1k
  if (path->length == 0) {
8197
0
    return pathOut;
8198
0
  }
8199
8200
21.1k
  if (flatten) {
8201
94
    pathIn = flattenPath(path, state->matrix, state->flatness);
8202
94
    if (state->lineDashLength > 0) {
8203
0
      dashPath = makeDashedPath(pathIn);
8204
0
      delete pathIn;
8205
0
      pathIn = dashPath;
8206
0
      if (pathIn->length == 0) {
8207
0
  delete pathIn;
8208
0
  return pathOut;
8209
0
      }
8210
0
    }
8211
21.0k
  } else {
8212
21.0k
    pathIn = path;
8213
21.0k
  }
8214
8215
21.1k
  subpathStart0 = subpathStart1 = 0; // make gcc happy
8216
21.1k
  seg = 0; // make gcc happy
8217
21.1k
  closed = gFalse; // make gcc happy
8218
21.1k
  left0 = left1 = right0 = right1 = join0 = join1 = 0; // make gcc happy
8219
21.1k
  leftFirst = rightFirst = firstPt = 0; // make gcc happy
8220
8221
21.1k
  i0 = 0;
8222
21.1k
  for (i1 = i0;
8223
28.1k
       !(pathIn->flags[i1] & splashPathLast) &&
8224
23.8k
   i1 + 1 < pathIn->length &&
8225
23.8k
   pathIn->pts[i1+1].x == pathIn->pts[i1].x &&
8226
9.40k
   pathIn->pts[i1+1].y == pathIn->pts[i1].y;
8227
21.1k
       ++i1) ;
8228
8229
14.8M
  while (i1 < pathIn->length) {
8230
14.7M
    if ((first = pathIn->flags[i0] & splashPathFirst)) {
8231
3.81M
      subpathStart0 = i0;
8232
3.81M
      subpathStart1 = i1;
8233
3.81M
      seg = 0;
8234
3.81M
      closed = pathIn->flags[i0] & splashPathClosed;
8235
3.81M
    }
8236
14.7M
    j0 = i1 + 1;
8237
14.7M
    if (j0 < pathIn->length) {
8238
14.7M
      for (j1 = j0;
8239
15.5M
     !(pathIn->flags[j1] & splashPathLast) &&
8240
11.7M
       j1 + 1 < pathIn->length &&
8241
11.7M
       pathIn->pts[j1+1].x == pathIn->pts[j1].x &&
8242
960k
       pathIn->pts[j1+1].y == pathIn->pts[j1].y;
8243
14.7M
     ++j1) ;
8244
14.7M
    } else {
8245
21.1k
      j1 = j0;
8246
21.1k
    }
8247
14.7M
    if (pathIn->flags[i1] & splashPathLast) {
8248
3.81M
      if (first && lineCap == splashLineCapRound) {
8249
  // special case: zero-length subpath with round line caps -->
8250
  // draw a circle
8251
1.24k
  pathOut->moveTo(pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8252
1.24k
      pathIn->pts[i0].y);
8253
1.24k
  pathOut->curveTo(pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8254
1.24k
       pathIn->pts[i0].y + bezierCircle2 * w,
8255
1.24k
       pathIn->pts[i0].x + bezierCircle2 * w,
8256
1.24k
       pathIn->pts[i0].y + (SplashCoord)0.5 * w,
8257
1.24k
       pathIn->pts[i0].x,
8258
1.24k
       pathIn->pts[i0].y + (SplashCoord)0.5 * w);
8259
1.24k
  pathOut->curveTo(pathIn->pts[i0].x - bezierCircle2 * w,
8260
1.24k
       pathIn->pts[i0].y + (SplashCoord)0.5 * w,
8261
1.24k
       pathIn->pts[i0].x - (SplashCoord)0.5 * w,
8262
1.24k
       pathIn->pts[i0].y + bezierCircle2 * w,
8263
1.24k
       pathIn->pts[i0].x - (SplashCoord)0.5 * w,
8264
1.24k
       pathIn->pts[i0].y);
8265
1.24k
  pathOut->curveTo(pathIn->pts[i0].x - (SplashCoord)0.5 * w,
8266
1.24k
       pathIn->pts[i0].y - bezierCircle2 * w,
8267
1.24k
       pathIn->pts[i0].x - bezierCircle2 * w,
8268
1.24k
       pathIn->pts[i0].y - (SplashCoord)0.5 * w,
8269
1.24k
       pathIn->pts[i0].x,
8270
1.24k
       pathIn->pts[i0].y - (SplashCoord)0.5 * w);
8271
1.24k
  pathOut->curveTo(pathIn->pts[i0].x + bezierCircle2 * w,
8272
1.24k
       pathIn->pts[i0].y - (SplashCoord)0.5 * w,
8273
1.24k
       pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8274
1.24k
       pathIn->pts[i0].y - bezierCircle2 * w,
8275
1.24k
       pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8276
1.24k
       pathIn->pts[i0].y);
8277
1.24k
  pathOut->close();
8278
1.24k
      }
8279
3.81M
      i0 = j0;
8280
3.81M
      i1 = j1;
8281
3.81M
      continue;
8282
3.81M
    }
8283
10.9M
    last = pathIn->flags[j1] & splashPathLast;
8284
10.9M
    if (last) {
8285
3.04M
      k0 = subpathStart1 + 1;
8286
7.91M
    } else {
8287
7.91M
      k0 = j1 + 1;
8288
7.91M
    }
8289
10.9M
    for (k1 = k0;
8290
10.9M
   !(pathIn->flags[k1] & splashPathLast) &&
8291
7.92M
     k1 + 1 < pathIn->length &&
8292
7.92M
     pathIn->pts[k1+1].x == pathIn->pts[k1].x &&
8293
139k
     pathIn->pts[k1+1].y == pathIn->pts[k1].y;
8294
10.9M
   ++k1) ;
8295
8296
    // compute the deltas for segment (i1, j0)
8297
#if USE_FIXEDPOINT
8298
    // the 1/d value can be small, which introduces significant
8299
    // inaccuracies in fixed point mode
8300
    d = splashDist(pathIn->pts[i1].x, pathIn->pts[i1].y,
8301
       pathIn->pts[j0].x, pathIn->pts[j0].y);
8302
    dx = (pathIn->pts[j0].x - pathIn->pts[i1].x) / d;
8303
    dy = (pathIn->pts[j0].y - pathIn->pts[i1].y) / d;
8304
#else
8305
10.9M
    d = (SplashCoord)1 / splashDist(pathIn->pts[i1].x, pathIn->pts[i1].y,
8306
10.9M
            pathIn->pts[j0].x, pathIn->pts[j0].y);
8307
10.9M
    dx = d * (pathIn->pts[j0].x - pathIn->pts[i1].x);
8308
10.9M
    dy = d * (pathIn->pts[j0].y - pathIn->pts[i1].y);
8309
10.9M
#endif
8310
10.9M
    wdx = (SplashCoord)0.5 * w * dx;
8311
10.9M
    wdy = (SplashCoord)0.5 * w * dy;
8312
8313
    // draw the start cap
8314
10.9M
    if (i0 == subpathStart0) {
8315
3.04M
      firstPt = pathOut->length;
8316
3.04M
    }
8317
10.9M
    if (first && !closed) {
8318
3.03M
      switch (lineCap) {
8319
2.69M
      case splashLineCapButt:
8320
2.69M
  pathOut->moveTo(pathIn->pts[i0].x - wdy, pathIn->pts[i0].y + wdx);
8321
2.69M
  pathOut->lineTo(pathIn->pts[i0].x + wdy, pathIn->pts[i0].y - wdx);
8322
2.69M
  break;
8323
11.1k
      case splashLineCapRound:
8324
11.1k
  pathOut->moveTo(pathIn->pts[i0].x - wdy, pathIn->pts[i0].y + wdx);
8325
11.1k
  pathOut->curveTo(pathIn->pts[i0].x - wdy - bezierCircle * wdx,
8326
11.1k
       pathIn->pts[i0].y + wdx - bezierCircle * wdy,
8327
11.1k
       pathIn->pts[i0].x - wdx - bezierCircle * wdy,
8328
11.1k
       pathIn->pts[i0].y - wdy + bezierCircle * wdx,
8329
11.1k
       pathIn->pts[i0].x - wdx,
8330
11.1k
       pathIn->pts[i0].y - wdy);
8331
11.1k
  pathOut->curveTo(pathIn->pts[i0].x - wdx + bezierCircle * wdy,
8332
11.1k
       pathIn->pts[i0].y - wdy - bezierCircle * wdx,
8333
11.1k
       pathIn->pts[i0].x + wdy - bezierCircle * wdx,
8334
11.1k
       pathIn->pts[i0].y - wdx - bezierCircle * wdy,
8335
11.1k
       pathIn->pts[i0].x + wdy,
8336
11.1k
       pathIn->pts[i0].y - wdx);
8337
11.1k
  break;
8338
328k
      case splashLineCapProjecting:
8339
328k
  pathOut->moveTo(pathIn->pts[i0].x - wdx - wdy,
8340
328k
      pathIn->pts[i0].y + wdx - wdy);
8341
328k
  pathOut->lineTo(pathIn->pts[i0].x - wdx + wdy,
8342
328k
      pathIn->pts[i0].y - wdx - wdy);
8343
328k
  break;
8344
3.03M
      }
8345
7.92M
    } else {
8346
7.92M
      pathOut->moveTo(pathIn->pts[i0].x - wdy, pathIn->pts[i0].y + wdx);
8347
7.92M
      pathOut->lineTo(pathIn->pts[i0].x + wdy, pathIn->pts[i0].y - wdx);
8348
7.92M
    }
8349
8350
    // draw the left side of the segment rectangle and the end cap
8351
10.9M
    left2 = pathOut->length - 1;
8352
10.9M
    if (last && !closed) {
8353
3.03M
      switch (lineCap) {
8354
2.69M
      case splashLineCapButt:
8355
2.69M
  pathOut->lineTo(pathIn->pts[j0].x + wdy, pathIn->pts[j0].y - wdx);
8356
2.69M
  pathOut->lineTo(pathIn->pts[j0].x - wdy, pathIn->pts[j0].y + wdx);
8357
2.69M
  break;
8358
11.1k
      case splashLineCapRound:
8359
11.1k
  pathOut->lineTo(pathIn->pts[j0].x + wdy, pathIn->pts[j0].y - wdx);
8360
11.1k
  pathOut->curveTo(pathIn->pts[j0].x + wdy + bezierCircle * wdx,
8361
11.1k
       pathIn->pts[j0].y - wdx + bezierCircle * wdy,
8362
11.1k
       pathIn->pts[j0].x + wdx + bezierCircle * wdy,
8363
11.1k
       pathIn->pts[j0].y + wdy - bezierCircle * wdx,
8364
11.1k
       pathIn->pts[j0].x + wdx,
8365
11.1k
       pathIn->pts[j0].y + wdy);
8366
11.1k
  pathOut->curveTo(pathIn->pts[j0].x + wdx - bezierCircle * wdy,
8367
11.1k
       pathIn->pts[j0].y + wdy + bezierCircle * wdx,
8368
11.1k
       pathIn->pts[j0].x - wdy + bezierCircle * wdx,
8369
11.1k
       pathIn->pts[j0].y + wdx + bezierCircle * wdy,
8370
11.1k
       pathIn->pts[j0].x - wdy,
8371
11.1k
       pathIn->pts[j0].y + wdx);
8372
11.1k
  break;
8373
328k
      case splashLineCapProjecting:
8374
328k
  pathOut->lineTo(pathIn->pts[j0].x + wdy + wdx,
8375
328k
      pathIn->pts[j0].y - wdx + wdy);
8376
328k
  pathOut->lineTo(pathIn->pts[j0].x - wdy + wdx,
8377
328k
      pathIn->pts[j0].y + wdx + wdy);
8378
328k
  break;
8379
3.03M
      }
8380
7.92M
    } else {
8381
7.92M
      pathOut->lineTo(pathIn->pts[j0].x + wdy, pathIn->pts[j0].y - wdx);
8382
7.92M
      pathOut->lineTo(pathIn->pts[j0].x - wdy, pathIn->pts[j0].y + wdx);
8383
7.92M
    }
8384
8385
    // draw the right side of the segment rectangle
8386
    // (NB: if stroke adjustment is enabled, the closepath operation MUST
8387
    // add a segment because this segment is used for a hint)
8388
10.9M
    right2 = pathOut->length - 1;
8389
10.9M
    pathOut->close(state->strokeAdjust != splashStrokeAdjustOff);
8390
8391
    // draw the join
8392
10.9M
    join2 = pathOut->length;
8393
10.9M
    if (!last || closed) {
8394
8395
      // compute the deltas for segment (j1, k0)
8396
#if USE_FIXEDPOINT
8397
      // the 1/d value can be small, which introduces significant
8398
      // inaccuracies in fixed point mode
8399
      d = splashDist(pathIn->pts[j1].x, pathIn->pts[j1].y,
8400
         pathIn->pts[k0].x, pathIn->pts[k0].y);
8401
      dxNext = (pathIn->pts[k0].x - pathIn->pts[j1].x) / d;
8402
      dyNext = (pathIn->pts[k0].y - pathIn->pts[j1].y) / d;
8403
#else
8404
7.92M
      d = (SplashCoord)1 / splashDist(pathIn->pts[j1].x, pathIn->pts[j1].y,
8405
7.92M
              pathIn->pts[k0].x, pathIn->pts[k0].y);
8406
7.92M
      dxNext = d * (pathIn->pts[k0].x - pathIn->pts[j1].x);
8407
7.92M
      dyNext = d * (pathIn->pts[k0].y - pathIn->pts[j1].y);
8408
7.92M
#endif
8409
7.92M
      wdxNext = (SplashCoord)0.5 * w * dxNext;
8410
7.92M
      wdyNext = (SplashCoord)0.5 * w * dyNext;
8411
8412
      // compute the join parameters
8413
7.92M
      crossprod = dx * dyNext - dy * dxNext;
8414
7.92M
      dotprod = -(dx * dxNext + dy * dyNext);
8415
7.92M
      if (dotprod > 0.9999) {
8416
  // avoid a divide-by-zero -- set miter to something arbitrary
8417
  // such that sqrt(miter) will exceed miterLimit (and m is never
8418
  // used in that situation)
8419
  // (note: the comparison value (0.9999) has to be less than
8420
  // 1-epsilon, where epsilon is the smallest value
8421
  // representable in the fixed point format)
8422
9.92k
  miter = (state->miterLimit + 1) * (state->miterLimit + 1);
8423
9.92k
  m = 0;
8424
7.91M
      } else {
8425
7.91M
  miter = (SplashCoord)2 / ((SplashCoord)1 - dotprod);
8426
7.91M
  if (miter < 1) {
8427
    // this can happen because of floating point inaccuracies
8428
1.28k
    miter = 1;
8429
1.28k
  }
8430
7.91M
  m = splashSqrt(miter - 1);
8431
7.91M
      }
8432
8433
      // round join
8434
7.92M
      if (lineJoin == splashLineJoinRound) {
8435
  // join angle < 180
8436
1.79M
  if (crossprod < 0) {
8437
622k
    angle = atan2((double)dx, (double)-dy);
8438
622k
    angleNext = atan2((double)dxNext, (double)-dyNext);
8439
622k
    if (angle < angleNext) {
8440
1.49k
      angle += 2 * M_PI;
8441
1.49k
    }
8442
622k
    dAngle = (angle  - angleNext) / M_PI;
8443
622k
    if (dAngle < 0.501) {
8444
      // span angle is <= 90 degrees -> draw a single arc
8445
621k
      kappa = dAngle * bezierCircle * w;
8446
621k
      cx1 = pathIn->pts[j0].x - wdy + kappa * dx;
8447
621k
      cy1 = pathIn->pts[j0].y + wdx + kappa * dy;
8448
621k
      cx2 = pathIn->pts[j0].x - wdyNext - kappa * dxNext;
8449
621k
      cy2 = pathIn->pts[j0].y + wdxNext - kappa * dyNext;
8450
621k
      pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8451
621k
      pathOut->lineTo(pathIn->pts[j0].x - wdyNext,
8452
621k
          pathIn->pts[j0].y + wdxNext);
8453
621k
      pathOut->curveTo(cx2, cy2, cx1, cy1,
8454
621k
           pathIn->pts[j0].x - wdy,
8455
621k
           pathIn->pts[j0].y + wdx);
8456
621k
    } else {
8457
      // span angle is > 90 degrees -> split into two arcs
8458
1.10k
      dJoin = splashDist(-wdy, wdx, -wdyNext, wdxNext);
8459
1.10k
      if (dJoin > 0) {
8460
1.10k
        dxJoin = (-wdyNext + wdy) / dJoin;
8461
1.10k
        dyJoin = (wdxNext - wdx) / dJoin;
8462
1.10k
        xc = pathIn->pts[j0].x
8463
1.10k
       + (SplashCoord)0.5 * w
8464
1.10k
         * cos((double)((SplashCoord)0.5 * (angle + angleNext)));
8465
1.10k
        yc = pathIn->pts[j0].y
8466
1.10k
       + (SplashCoord)0.5 * w
8467
1.10k
         * sin((double)((SplashCoord)0.5 * (angle + angleNext)));
8468
1.10k
        kappa = dAngle * bezierCircle2 * w;
8469
1.10k
        cx1 = pathIn->pts[j0].x - wdy + kappa * dx;
8470
1.10k
        cy1 = pathIn->pts[j0].y + wdx + kappa * dy;
8471
1.10k
        cx2 = xc - kappa * dxJoin;
8472
1.10k
        cy2 = yc - kappa * dyJoin;
8473
1.10k
        cx3 = xc + kappa * dxJoin;
8474
1.10k
        cy3 = yc + kappa * dyJoin;
8475
1.10k
        cx4 = pathIn->pts[j0].x - wdyNext - kappa * dxNext;
8476
1.10k
        cy4 = pathIn->pts[j0].y + wdxNext - kappa * dyNext;
8477
1.10k
        pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8478
1.10k
        pathOut->lineTo(pathIn->pts[j0].x - wdyNext,
8479
1.10k
            pathIn->pts[j0].y + wdxNext);
8480
1.10k
        pathOut->curveTo(cx4, cy4, cx3, cy3, xc, yc);
8481
1.10k
        pathOut->curveTo(cx2, cy2, cx1, cy1,
8482
1.10k
             pathIn->pts[j0].x - wdy,
8483
1.10k
             pathIn->pts[j0].y + wdx);
8484
1.10k
      }
8485
1.10k
    }
8486
8487
  // join angle >= 180
8488
1.16M
  } else {
8489
1.16M
    angle = atan2((double)-dx, (double)dy);
8490
1.16M
    angleNext = atan2((double)-dxNext, (double)dyNext);
8491
1.16M
    if (angleNext < angle) {
8492
5.48k
      angleNext += 2 * M_PI;
8493
5.48k
    }
8494
1.16M
    dAngle = (angleNext - angle) / M_PI;
8495
1.16M
    if (dAngle < 0.501) {
8496
      // span angle is <= 90 degrees -> draw a single arc
8497
1.16M
      kappa = dAngle * bezierCircle * w;
8498
1.16M
        cx1 = pathIn->pts[j0].x + wdy + kappa * dx;
8499
1.16M
        cy1 = pathIn->pts[j0].y - wdx + kappa * dy;
8500
1.16M
        cx2 = pathIn->pts[j0].x + wdyNext - kappa * dxNext;
8501
1.16M
        cy2 = pathIn->pts[j0].y - wdxNext - kappa * dyNext;
8502
1.16M
        pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8503
1.16M
        pathOut->lineTo(pathIn->pts[j0].x + wdy,
8504
1.16M
            pathIn->pts[j0].y - wdx);
8505
1.16M
        pathOut->curveTo(cx1, cy1, cx2, cy2,
8506
1.16M
             pathIn->pts[j0].x + wdyNext,
8507
1.16M
             pathIn->pts[j0].y - wdxNext);
8508
1.16M
    } else {
8509
      // span angle is > 90 degrees -> split into two arcs
8510
5.80k
      dJoin = splashDist(wdy, -wdx, wdyNext, -wdxNext);
8511
5.80k
      if (dJoin > 0) {
8512
5.68k
        dxJoin = (wdyNext - wdy) / dJoin;
8513
5.68k
        dyJoin = (-wdxNext + wdx) / dJoin;
8514
5.68k
        xc = pathIn->pts[j0].x
8515
5.68k
       + (SplashCoord)0.5 * w
8516
5.68k
         * cos((double)((SplashCoord)0.5 * (angle + angleNext)));
8517
5.68k
        yc = pathIn->pts[j0].y
8518
5.68k
       + (SplashCoord)0.5 * w
8519
5.68k
         * sin((double)((SplashCoord)0.5 * (angle + angleNext)));
8520
5.68k
        kappa = dAngle * bezierCircle2 * w;
8521
5.68k
        cx1 = pathIn->pts[j0].x + wdy + kappa * dx;
8522
5.68k
        cy1 = pathIn->pts[j0].y - wdx + kappa * dy;
8523
5.68k
        cx2 = xc - kappa * dxJoin;
8524
5.68k
        cy2 = yc - kappa * dyJoin;
8525
5.68k
        cx3 = xc + kappa * dxJoin;
8526
5.68k
        cy3 = yc + kappa * dyJoin;
8527
5.68k
        cx4 = pathIn->pts[j0].x + wdyNext - kappa * dxNext;
8528
5.68k
        cy4 = pathIn->pts[j0].y - wdxNext - kappa * dyNext;
8529
5.68k
        pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8530
5.68k
        pathOut->lineTo(pathIn->pts[j0].x + wdy,
8531
5.68k
            pathIn->pts[j0].y - wdx);
8532
5.68k
        pathOut->curveTo(cx1, cy1, cx2, cy2, xc, yc);
8533
5.68k
        pathOut->curveTo(cx3, cy3, cx4, cy4,
8534
5.68k
             pathIn->pts[j0].x + wdyNext,
8535
5.68k
             pathIn->pts[j0].y - wdxNext);
8536
5.68k
      }
8537
5.80k
    }
8538
1.16M
  }
8539
8540
6.13M
      } else {
8541
6.13M
  pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8542
8543
  // join angle < 180
8544
6.13M
  if (crossprod < 0) {
8545
2.26M
    pathOut->lineTo(pathIn->pts[j0].x - wdyNext,
8546
2.26M
        pathIn->pts[j0].y + wdxNext);
8547
    // miter join inside limit
8548
2.26M
    if (lineJoin == splashLineJoinMiter &&
8549
2.02M
        splashSqrt(miter) <= state->miterLimit) {
8550
1.84M
      pathOut->lineTo(pathIn->pts[j0].x - wdy + wdx * m,
8551
1.84M
          pathIn->pts[j0].y + wdx + wdy * m);
8552
1.84M
      pathOut->lineTo(pathIn->pts[j0].x - wdy,
8553
1.84M
          pathIn->pts[j0].y + wdx);
8554
    // bevel join or miter join outside limit
8555
1.84M
    } else {
8556
424k
      pathOut->lineTo(pathIn->pts[j0].x - wdy,
8557
424k
          pathIn->pts[j0].y + wdx);
8558
424k
    }
8559
8560
  // join angle >= 180
8561
3.87M
  } else {
8562
3.87M
    pathOut->lineTo(pathIn->pts[j0].x + wdy,
8563
3.87M
        pathIn->pts[j0].y - wdx);
8564
    // miter join inside limit
8565
3.87M
    if (lineJoin == splashLineJoinMiter &&
8566
3.47M
        splashSqrt(miter) <= state->miterLimit) {
8567
3.18M
      pathOut->lineTo(pathIn->pts[j0].x + wdy + wdx * m,
8568
3.18M
          pathIn->pts[j0].y - wdx + wdy * m);
8569
3.18M
      pathOut->lineTo(pathIn->pts[j0].x + wdyNext,
8570
3.18M
          pathIn->pts[j0].y - wdxNext);
8571
    // bevel join or miter join outside limit
8572
3.18M
    } else {
8573
682k
      pathOut->lineTo(pathIn->pts[j0].x + wdyNext,
8574
682k
          pathIn->pts[j0].y - wdxNext);
8575
682k
    }
8576
3.87M
  }
8577
6.13M
      }
8578
8579
7.92M
      pathOut->close();
8580
7.92M
    }
8581
8582
    // add stroke adjustment hints
8583
10.9M
    if (state->strokeAdjust != splashStrokeAdjustOff) {
8584
8585
      // subpath with one segment
8586
10.8M
      if (seg == 0 && last) {
8587
3.02M
  switch (lineCap) {
8588
2.68M
  case splashLineCapButt:
8589
2.68M
    pathOut->addStrokeAdjustHint(firstPt, left2 + 1,
8590
2.68M
               firstPt, pathOut->length - 1);
8591
2.68M
    break;
8592
327k
  case splashLineCapProjecting:
8593
327k
    pathOut->addStrokeAdjustHint(firstPt, left2 + 1,
8594
327k
               firstPt, pathOut->length - 1, gTrue);
8595
327k
    break;
8596
10.9k
  case splashLineCapRound:
8597
10.9k
    break;
8598
3.02M
  }
8599
3.02M
  pathOut->addStrokeAdjustHint(left2, right2,
8600
3.02M
             firstPt, pathOut->length - 1);
8601
7.85M
      } else {
8602
8603
  // start of subpath
8604
7.85M
  if (seg == 1) {
8605
8606
    // start cap
8607
18.1k
    if (!closed) {
8608
7.93k
      switch (lineCap) {
8609
6.86k
      case splashLineCapButt:
8610
6.86k
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8611
6.86k
             firstPt, firstPt + 1);
8612
6.86k
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8613
6.86k
             right1 + 1, right1 + 1);
8614
6.86k
        break;
8615
906
      case splashLineCapProjecting:
8616
906
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8617
906
             firstPt, firstPt + 1, gTrue);
8618
906
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8619
906
             right1 + 1, right1 + 1, gTrue);
8620
906
        break;
8621
165
      case splashLineCapRound:
8622
165
        break;
8623
7.93k
      }
8624
7.93k
    }
8625
8626
    // first segment
8627
18.1k
    pathOut->addStrokeAdjustHint(left1, right1, firstPt, left2);
8628
18.1k
    pathOut->addStrokeAdjustHint(left1, right1, right2 + 1, right2 + 1);
8629
18.1k
  }
8630
8631
  // middle of subpath
8632
7.85M
  if (seg > 1) {
8633
7.82M
    pathOut->addStrokeAdjustHint(left1, right1, left0 + 1, right0);
8634
7.82M
    pathOut->addStrokeAdjustHint(left1, right1, join0, left2);
8635
7.82M
    pathOut->addStrokeAdjustHint(left1, right1, right2 + 1, right2 + 1);
8636
7.82M
  }
8637
8638
  // end of subpath
8639
7.85M
  if (last) {
8640
8641
18.1k
    if (closed) {
8642
      // first segment
8643
10.2k
      pathOut->addStrokeAdjustHint(leftFirst, rightFirst,
8644
10.2k
           left2 + 1, right2);
8645
10.2k
      pathOut->addStrokeAdjustHint(leftFirst, rightFirst,
8646
10.2k
           join2, pathOut->length - 1);
8647
8648
      // last segment
8649
10.2k
      pathOut->addStrokeAdjustHint(left2, right2,
8650
10.2k
           left1 + 1, right1);
8651
10.2k
      pathOut->addStrokeAdjustHint(left2, right2,
8652
10.2k
           join1, pathOut->length - 1);
8653
10.2k
      pathOut->addStrokeAdjustHint(left2, right2,
8654
10.2k
           leftFirst - 1, leftFirst);
8655
10.2k
      pathOut->addStrokeAdjustHint(left2, right2,
8656
10.2k
           rightFirst + 1, rightFirst + 1);
8657
8658
10.2k
    } else {
8659
8660
      // last segment
8661
7.93k
      pathOut->addStrokeAdjustHint(left2, right2,
8662
7.93k
           left1 + 1, right1);
8663
7.93k
      pathOut->addStrokeAdjustHint(left2, right2,
8664
7.93k
           join1, pathOut->length - 1);
8665
8666
      // end cap
8667
7.93k
      switch (lineCap) {
8668
6.86k
      case splashLineCapButt:
8669
6.86k
        pathOut->addStrokeAdjustHint(left2 - 1, left2 + 1,
8670
6.86k
             left2 + 1, left2 + 2);
8671
6.86k
        break;
8672
906
      case splashLineCapProjecting:
8673
906
        pathOut->addStrokeAdjustHint(left2 - 1, left2 + 1,
8674
906
             left2 + 1, left2 + 2, gTrue);
8675
906
        break;
8676
165
      case splashLineCapRound:
8677
165
        break;
8678
7.93k
      }
8679
7.93k
    }
8680
18.1k
  }
8681
7.85M
      }
8682
8683
10.8M
      left0 = left1;
8684
10.8M
      left1 = left2;
8685
10.8M
      right0 = right1;
8686
10.8M
      right1 = right2;
8687
10.8M
      join0 = join1;
8688
10.8M
      join1 = join2;
8689
10.8M
      if (seg == 0) {
8690
3.04M
  leftFirst = left2;
8691
3.04M
  rightFirst = right2;
8692
3.04M
      }
8693
10.8M
    }
8694
8695
10.9M
    i0 = j0;
8696
10.9M
    i1 = j1;
8697
10.9M
    ++seg;
8698
10.9M
  }
8699
8700
21.1k
  if (pathIn != path) {
8701
94
    delete pathIn;
8702
94
  }
8703
8704
21.1k
  return pathOut;
8705
21.1k
}
8706
8707
SplashClipResult Splash::limitRectToClipRect(int *xMin, int *yMin,
8708
1.02k
               int *xMax, int *yMax) {
8709
1.02k
  int t;
8710
8711
1.02k
  if ((t = state->clip->getXMinI(state->strokeAdjust)) > *xMin) {
8712
982
    *xMin = t;
8713
982
  }
8714
1.02k
  if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < *xMax) {
8715
273
    *xMax = t;
8716
273
  }
8717
1.02k
  if ((t = state->clip->getYMinI(state->strokeAdjust)) > *yMin) {
8718
969
    *yMin = t;
8719
969
  }
8720
1.02k
  if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < *yMax) {
8721
262
    *yMax = t;
8722
262
  }
8723
1.02k
  if (*xMin >= *xMax || *yMin >= *yMax) {
8724
1.01k
    return splashClipAllOutside;
8725
1.01k
  }
8726
2
  return state->clip->testRect(*xMin, *yMin, *xMax - 1, *yMax - 1,
8727
2
             state->strokeAdjust);
8728
1.02k
}
8729
8730
0
void Splash::dumpPath(SplashPath *path) {
8731
0
  int i;
8732
8733
0
  for (i = 0; i < path->length; ++i) {
8734
0
    printf("  %3d: x=%8.2f y=%8.2f%s%s%s%s\n",
8735
0
     i, (double)path->pts[i].x, (double)path->pts[i].y,
8736
0
     (path->flags[i] & splashPathFirst) ? " first" : "",
8737
0
     (path->flags[i] & splashPathLast) ? " last" : "",
8738
0
     (path->flags[i] & splashPathClosed) ? " closed" : "",
8739
0
     (path->flags[i] & splashPathCurve) ? " curve" : "");
8740
0
  }
8741
0
  if (path->hintsLength == 0) {
8742
0
    printf("  no hints\n");
8743
0
  } else {
8744
0
    for (i = 0; i < path->hintsLength; ++i) {
8745
0
      printf("  hint %3d: ctrl0=%d ctrl1=%d pts=%d..%d\n",
8746
0
       i, path->hints[i].ctrl0, path->hints[i].ctrl1,
8747
0
       path->hints[i].firstPt, path->hints[i].lastPt);
8748
0
    }
8749
0
  }
8750
0
}
8751
8752
0
void Splash::dumpXPath(SplashXPath *path) {
8753
0
  int i;
8754
8755
0
  for (i = 0; i < path->length; ++i) {
8756
0
    printf("  %4d: x0=%8.2f y0=%8.2f x1=%8.2f y1=%8.2f count=%d\n",
8757
0
     i, (double)path->segs[i].x0, (double)path->segs[i].y0,
8758
0
     (double)path->segs[i].x1, (double)path->segs[i].y1,
8759
0
     path->segs[i].count);
8760
0
  }
8761
0
}
8762