Coverage Report

Created: 2026-03-31 07:04

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.49M
#define bezierCircle ((SplashCoord)0.55228475)
41
7.83k
#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
80.9k
static inline Guchar div255(int x) {
45
80.9k
  return (Guchar)((x + (x >> 8) + 0x80) >> 8);
46
80.9k
}
47
48
// Clip x to lie in [0, 255].
49
3.58k
static inline Guchar clip255(int x) {
50
3.58k
  return x < 0 ? 0 : x > 255 ? 255 : (Guchar)x;
51
3.58k
}
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
408k
void Splash::clearModRegion() {
142
408k
  modXMin = bitmap->width;
143
408k
  modYMin = bitmap->height;
144
408k
  modXMax = -1;
145
408k
  modYMax = -1;
146
408k
}
147
148
870k
inline void Splash::updateModX(int x) {
149
870k
  if (x < modXMin) {
150
401k
    modXMin = x;
151
401k
  }
152
870k
  if (x > modXMax) {
153
401k
    modXMax = x;
154
401k
  }
155
870k
}
156
157
833k
inline void Splash::updateModY(int y) {
158
833k
  if (y < modYMin) {
159
401k
    modYMin = y;
160
401k
  }
161
833k
  if (y > modYMax) {
162
401k
    modYMax = y;
163
401k
  }
164
833k
}
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
32.8k
           GBool alphaIsShape) {
174
32.8k
  SplashColorMode mode;
175
176
32.8k
  mode = bitmap->mode;
177
178
32.8k
  pipe->pattern = NULL;
179
180
  // source color
181
32.8k
  if (pattern && pattern->isStatic()) {
182
28.7k
    pattern->getColor(0, 0, pipe->cSrcVal);
183
28.7k
    pipe->pattern = NULL;
184
28.7k
  } else {
185
4.08k
    pipe->pattern = pattern;
186
4.08k
  }
187
188
  // source alpha
189
32.8k
  pipe->aInput = aInput;
190
191
  // source overprint mask
192
32.8k
  pipe->srcOverprintMaskPtr = NULL;
193
194
  // alpha is shape
195
32.8k
  pipe->alphaIsShape = alphaIsShape;
196
197
  // special cases
198
32.8k
  pipe->noTransparency = aInput == 255 &&
199
32.4k
                         !state->softMask &&
200
31.9k
                         !usesShape &&
201
31.9k
                         !usesAlpha &&
202
0
                         !state->inNonIsolatedGroup &&
203
0
       !state->inKnockoutGroup &&
204
0
                         !nonIsolatedGroup &&
205
0
                         state->overprintMask == 0xffffffff;
206
32.8k
  pipe->alphaOnly = aInput == 255 &&
207
32.4k
                    !state->softMask &&
208
31.9k
                    !usesShape &&
209
31.9k
                    usesAlpha &&
210
31.9k
                    !state->inNonIsolatedGroup &&
211
18.7k
        !state->inKnockoutGroup &&
212
18.7k
                    !nonIsolatedGroup &&
213
16.1k
                    state->overprintMask == 0xffffffff;
214
215
  // result color
216
32.8k
  if (pipe->noTransparency) {
217
    // the !state->blendFunc case is handled separately in pipeRun
218
0
    pipe->resultColorCtrl = pipeResultColorNoAlphaBlend[mode];
219
32.8k
  } else if (!state->blendFunc) {
220
32.0k
    pipe->resultColorCtrl = pipeResultColorAlphaNoBlend[mode];
221
32.0k
  } else {
222
781
    pipe->resultColorCtrl = pipeResultColorAlphaBlend[mode];
223
781
  }
224
225
  // non-isolated group correction
226
32.8k
  pipe->nonIsolatedGroup = nonIsolatedGroup;
227
228
  // select the 'run' function
229
32.8k
  pipe->run = &Splash::pipeRun;
230
32.8k
  if (shapeBitmap || overprintMaskBitmap || usesSrcOverprint) {
231
    // use Splash::pipeRun
232
32.8k
  } 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
32.8k
  } else if (!pipe->pattern && pipe->alphaOnly && !state->blendFunc) {
247
15.8k
    if (mode == splashModeMono1 && !bitmap->alpha) {
248
128
      pipe->run = &Splash::pipeRunShapeMono1;
249
15.6k
    } else if (mode == splashModeMono8 && bitmap->alpha) {
250
432
      pipe->run = &Splash::pipeRunShapeMono8;
251
15.2k
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
252
14.8k
      pipe->run = &Splash::pipeRunShapeRGB8;
253
14.8k
    } else if (mode == splashModeBGR8 && bitmap->alpha) {
254
0
      pipe->run = &Splash::pipeRunShapeBGR8;
255
0
#if SPLASH_CMYK
256
407
    } else if (mode == splashModeCMYK8 && bitmap->alpha) {
257
13
      pipe->run = &Splash::pipeRunShapeCMYK8;
258
13
#endif
259
394
    } else if (mode == splashModeMono8 && !bitmap->alpha) {
260
      // this is used when drawing soft-masked images
261
394
      pipe->run = &Splash::pipeRunShapeNoAlphaMono8;
262
394
    }
263
16.9k
  } else if (!pipe->pattern && !pipe->noTransparency && !state->softMask &&
264
16.4k
       usesAlpha &&
265
16.4k
       !(state->inNonIsolatedGroup && (alpha0Bitmap ||
266
13.1k
               parent->bitmap->alpha)) &&
267
3.28k
       !state->inKnockoutGroup &&
268
3.28k
       !state->blendFunc && !pipe->nonIsolatedGroup) {
269
237
    if (mode == splashModeMono1 && !bitmap->alpha) {
270
0
      pipe->run = &Splash::pipeRunAAMono1;
271
237
    } else if (mode == splashModeMono8 && bitmap->alpha) {
272
0
      pipe->run = &Splash::pipeRunAAMono8;
273
237
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
274
237
      pipe->run = &Splash::pipeRunAARGB8;
275
237
    } 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
16.7k
  } else if (!pipe->pattern &&
283
16.7k
             aInput == 255 &&
284
16.6k
             state->softMask &&
285
519
             usesAlpha &&
286
519
             !state->inNonIsolatedGroup &&
287
517
             !state->inKnockoutGroup &&
288
517
             !nonIsolatedGroup &&
289
502
             state->overprintMask == 0xffffffff &&
290
502
             !state->blendFunc) {
291
272
    if (mode == splashModeMono8 && bitmap->alpha) {
292
67
      pipe->run = &Splash::pipeRunSoftMaskMono8;
293
205
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
294
205
      pipe->run = &Splash::pipeRunSoftMaskRGB8;
295
205
    } 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
16.4k
  } else if (!pipe->pattern && !pipe->noTransparency && !state->softMask &&
303
16.2k
       usesAlpha &&
304
16.2k
       state->inNonIsolatedGroup && (alpha0Bitmap ||
305
13.1k
             parent->bitmap->alpha) &&
306
13.1k
       !state->inKnockoutGroup &&
307
13.1k
       !state->blendFunc && !pipe->nonIsolatedGroup) {
308
13.0k
    if (mode == splashModeMono8 && bitmap->alpha) {
309
0
      pipe->run = &Splash::pipeRunNonIsoMono8;
310
13.0k
    } else if (mode == splashModeRGB8 && bitmap->alpha) {
311
13.0k
      pipe->run = &Splash::pipeRunNonIsoRGB8;
312
13.0k
    } 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
13.0k
  }
320
32.8k
}
321
322
// general case
323
void Splash::pipeRun(SplashPipe *pipe, int x0, int x1, int y,
324
         Guchar *shapePtr, Guchar *alphaPtr,
325
1.94k
         SplashColorPtr cSrcPtr) {
326
1.94k
  Guchar *shapePtr2, *alphaPtr2;
327
1.94k
  Guchar shape, alpha, fSrc, aSrc, fDest, aDest;
328
1.94k
  Guchar alphaI, alphaIm1, alpha0, fResult, aResult;
329
1.94k
  Guchar m1, m2, m3, m4;
330
1.94k
  SplashColor cSrc, cDest, c0, cBlend;
331
1.94k
  Guchar shapeVal, alphaVal, cResult0, cResult1, cResult2, cResult3;
332
1.94k
  int cSrcStride, shapeStride, alphaStride, x, lastX, t;
333
1.94k
  SplashColorPtr cBlendIn, destColorPtr;
334
1.94k
  Guchar destColorMask;
335
1.94k
  Guchar *destShapePtr, *destAlphaPtr;
336
1.94k
  SplashColorPtr color0Ptr;
337
1.94k
  Guchar color0Mask;
338
1.94k
  Guchar *alpha0Ptr;
339
1.94k
  SplashColorPtr softMaskPtr;
340
1.94k
  Guint overprintMask;
341
1.94k
  Guint *overprintMaskPtr;
342
1.94k
#if SPLASH_CMYK
343
1.94k
  Guchar aPrev;
344
1.94k
  SplashColor cSrc2, cBlendIn2;
345
1.94k
#endif
346
1.94k
  int i;
347
348
1.94k
  if (cSrcPtr && !pipe->pattern) {
349
1.55k
    cSrcStride = bitmapComps;
350
1.55k
  } else {
351
386
    cSrcPtr = pipe->cSrcVal;
352
386
    cSrcStride = 0;
353
386
  }
354
1.94k
  if (shapePtr) {
355
0
    shapePtr2 = shapePtr;
356
0
    shapeStride = 1;
357
1.94k
  } else {
358
1.94k
    shapeVal = 0xff;
359
1.94k
    shapePtr2 = &shapeVal;
360
1.94k
    shapeStride = 0;
361
1.94k
  }
362
1.94k
  if (alphaPtr) {
363
1.94k
    alphaPtr2 = alphaPtr;
364
1.94k
    alphaStride = 1;
365
1.94k
  } else {
366
0
    alphaVal = 0xff;
367
0
    alphaPtr2 = &alphaVal;
368
0
    alphaStride = 0;
369
0
  }
370
371
1.94k
  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.94k
  } else if (alphaPtr) {
384
2.01k
    for (; x0 <= x1; ++x0) {
385
1.94k
      if (*alphaPtr2) {
386
1.86k
  break;
387
1.86k
      }
388
76
      cSrcPtr += cSrcStride;
389
76
      ++alphaPtr2;
390
76
      if (pipe->srcOverprintMaskPtr) {
391
0
  ++pipe->srcOverprintMaskPtr;
392
0
      }
393
76
    }
394
1.94k
  }
395
1.94k
  if (x0 > x1) {
396
76
    return;
397
76
  }
398
1.86k
  updateModX(x0);
399
1.86k
  updateModY(y);
400
1.86k
  lastX = x0;
401
402
1.86k
  useBitmapRow(y);
403
404
1.86k
  if (bitmap->mode == splashModeMono1) {
405
0
    destColorPtr = &bitmap->data[y * bitmap->rowSize + (x0 >> 3)];
406
0
    destColorMask = (Guchar)(0x80 >> (x0 & 7));
407
1.86k
  } else {
408
1.86k
    destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * bitmapComps];
409
1.86k
    destColorMask = 0; // make gcc happy
410
1.86k
  }
411
1.86k
  if (shapeBitmap) {
412
0
    destShapePtr = &shapeBitmap->alpha[y * shapeBitmap->alphaRowSize + x0];
413
1.86k
  } else {
414
1.86k
    destShapePtr = NULL;
415
1.86k
  }
416
1.86k
  if (bitmap->alpha) {
417
1.86k
    destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
418
1.86k
  } else {
419
0
    destAlphaPtr = NULL;
420
0
  }
421
1.86k
  if (state->softMask) {
422
237
    softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
423
1.62k
  } else {
424
1.62k
    softMaskPtr = NULL;
425
1.62k
  }
426
1.86k
  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.86k
  } else {
439
1.86k
    color0Ptr = NULL;
440
1.86k
    color0Mask = 0; // make gcc happy
441
1.86k
  }
442
1.86k
  if (alpha0Bitmap) {
443
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
444
1.86k
  } else if (state->inNonIsolatedGroup && parent->bitmap->alpha) {
445
93
    alpha0Ptr = &parent->bitmap->alpha[
446
93
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
447
93
        + (parentOffsetX + x0)];
448
1.77k
  } else {
449
1.77k
    alpha0Ptr = NULL;
450
1.77k
  }
451
1.86k
  if (overprintMaskBitmap) {
452
0
    overprintMaskPtr = overprintMaskBitmap + y * bitmap->width + x0;
453
1.86k
  } else {
454
1.86k
    overprintMaskPtr = NULL;
455
1.86k
  }
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
7.46k
  for (i = 0; i < bitmapComps; ++i) {
461
5.59k
    c0[i] = 0;
462
5.59k
  }
463
464
3.73k
  for (x = x0; x <= x1; ++x) {
465
466
    //----- shape and alpha
467
468
1.86k
    shape = *shapePtr2;
469
1.86k
    alpha = *alphaPtr2;
470
1.86k
    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.86k
    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.86k
    if (pipe->pattern) {
517
0
      pipe->pattern->getColor(x, y, pipe->cSrcVal);
518
0
    }
519
520
1.86k
    cResult0 = cResult1 = cResult2 = cResult3 = 0; // make gcc happy
521
522
1.86k
    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.86k
    } else { // if (noTransparency && !blendFunc)
556
557
      //----- read destination color
558
559
1.86k
      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.86k
      case splashModeRGB8:
567
1.86k
  cDest[0] = destColorPtr[0];
568
1.86k
  cDest[1] = destColorPtr[1];
569
1.86k
  cDest[2] = destColorPtr[2];
570
1.86k
  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.86k
#endif
584
1.86k
      }
585
586
      //----- read backdrop color
587
      //      (only used for non-iso knockout groups)
588
589
1.86k
      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.86k
      } else {
623
1.86k
  cBlendIn = cDest;
624
1.86k
      }
625
626
      //----- read destination shape and alpha
627
628
1.86k
      if (destShapePtr) {
629
0
  fDest = *destShapePtr;
630
1.86k
      } else {
631
1.86k
  fDest = 0xff;
632
1.86k
      }
633
634
1.86k
      if (destAlphaPtr) {
635
1.86k
  aDest = *destAlphaPtr;
636
1.86k
      } else {
637
0
  aDest = 0xff;
638
0
      }
639
640
      //----- read source color; handle overprint
641
642
1.86k
      if (pipe->srcOverprintMaskPtr) {
643
0
  overprintMask = *pipe->srcOverprintMaskPtr++;
644
1.86k
      } else {
645
1.86k
  overprintMask = state->overprintMask;
646
1.86k
      }
647
1.86k
      if (overprintMaskPtr) {
648
0
  *overprintMaskPtr++ |= overprintMask;
649
0
      }
650
651
1.86k
      switch (bitmap->mode) {
652
0
      case splashModeMono1:
653
0
      case splashModeMono8:
654
0
  cSrc[0] = state->grayTransfer[cSrcPtr[0]];
655
0
  break;
656
1.86k
      case splashModeRGB8:
657
1.86k
      case splashModeBGR8:
658
1.86k
  cSrc[0] = state->rgbTransferR[cSrcPtr[0]];
659
1.86k
  cSrc[1] = state->rgbTransferG[cSrcPtr[1]];
660
1.86k
  cSrc[2] = state->rgbTransferB[cSrcPtr[2]];
661
1.86k
  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.86k
#endif
699
1.86k
      }
700
701
      //----- source shape and alpha
702
703
1.86k
      fSrc = shape;
704
1.86k
      aSrc = div255(pipe->aInput * alpha);
705
1.86k
      if (softMaskPtr) {
706
237
  if (pipe->alphaIsShape) {
707
0
    fSrc = div255(fSrc * *softMaskPtr);
708
0
  }
709
237
  aSrc = div255(aSrc * *softMaskPtr);
710
237
  ++softMaskPtr;
711
237
      }
712
713
      //----- non-isolated group correction
714
715
1.86k
      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.19k
  if (alpha0Ptr) {
726
0
    t = *alpha0Ptr;
727
0
    t = (Guchar)(aDest + t - div255(aDest * t));
728
1.19k
  } else {
729
1.19k
    t = aDest;
730
1.19k
  }
731
1.19k
  t = (t * 255) / alpha - t;
732
1.19k
  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.19k
  case splashModeRGB8:
738
1.19k
  case splashModeBGR8:
739
1.19k
    cSrc[2] = clip255(cSrc[2] + ((cSrc[2] - cDest[2]) * t) / 255);
740
1.19k
    cSrc[1] = clip255(cSrc[1] + ((cSrc[1] - cDest[1]) * t) / 255);
741
1.19k
  case splashModeMono1:
742
1.19k
  case splashModeMono8:
743
1.19k
    cSrc[0] = clip255(cSrc[0] + ((cSrc[0] - cDest[0]) * t) / 255);
744
1.19k
    break;
745
1.19k
  }
746
1.19k
      }
747
748
      //----- blend function
749
750
1.86k
      if (state->blendFunc) {
751
713
#if SPLASH_CMYK
752
713
  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
713
#endif
770
713
  (*state->blendFunc)(cSrc, cBlendIn, cBlend, bitmap->mode);
771
713
      }
772
773
      //----- result shape
774
775
1.86k
      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.86k
      if (pipe->noTransparency) {
784
0
  alphaI = aResult = 255;
785
0
  m1 = m2 = m3 = m4 = 0; // make gcc happy
786
1.86k
      } else if (alpha0Ptr) {     // non-isolated
787
93
  alpha0 = *alpha0Ptr++;
788
93
  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
93
  } else {                  // non-isolated, non-knockout
797
93
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
798
93
    alphaI = (Guchar)(alpha0 + aResult - div255(alpha0 * aResult));
799
93
    alphaIm1 = (Guchar)(alpha0 + aDest - div255(alpha0 * aDest));
800
93
    m1 = alphaI - aSrc;
801
93
    m2 = 0;
802
93
    m3 = div255(aSrc * (255 - alphaIm1));
803
93
    m4 = div255(aSrc * alphaIm1);
804
93
  }
805
1.77k
      } else {
806
1.77k
  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.77k
  } else {                  // isolated, non-knockout
815
1.77k
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
816
1.77k
    alphaI = aResult;
817
1.77k
    alphaIm1 = aDest;
818
1.77k
    m1 = alphaI - aSrc;
819
1.77k
    m2 = 0;
820
1.77k
    m3 = div255(aSrc * (255 - alphaIm1));
821
1.77k
    m4 = div255(aSrc * alphaIm1);
822
1.77k
  }
823
1.77k
      }
824
825
      //----- result color
826
827
1.86k
      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.15k
      case splashPipeResultColorAlphaNoBlendRGB:
855
1.15k
  if (alphaI == 0) {
856
13
    cResult0 = 0;
857
13
    cResult1 = 0;
858
13
    cResult2 = 0;
859
1.14k
  } else {
860
1.14k
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] + aSrc * cSrc[0])
861
1.14k
            / alphaI);
862
1.14k
    cResult1 = (Guchar)((m1 * cDest[1] + m2 * c0[1] + aSrc * cSrc[1])
863
1.14k
            / alphaI);
864
1.14k
    cResult2 = (Guchar)((m1 * cDest[2] + m2 * c0[2] + aSrc * cSrc[2])
865
1.14k
            / alphaI);
866
1.14k
  }
867
1.15k
  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
713
      case splashPipeResultColorAlphaBlendRGB:
898
713
  if (alphaI == 0) {
899
19
    cResult0 = 0;
900
19
    cResult1 = 0;
901
19
    cResult2 = 0;
902
694
  } else {
903
694
    cResult0 = (Guchar)((m1 * cDest[0] + m2 * c0[0] +
904
694
             m3 * cSrc[0] + m4 * cBlend[0])
905
694
            / alphaI);
906
694
    cResult1 = (Guchar)((m1 * cDest[1] + m2 * c0[1] +
907
694
             m3 * cSrc[1] + m4 * cBlend[1])
908
694
            / alphaI);
909
694
    cResult2 = (Guchar)((m1 * cDest[2] + m2 * c0[2] +
910
694
             m3 * cSrc[2] + m4 * cBlend[2])
911
694
            / alphaI);
912
694
  }
913
713
  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.86k
#endif
937
1.86k
      }
938
939
1.86k
    } // if (noTransparency && !blendFunc)
940
941
    //----- write destination pixel
942
943
1.86k
    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.86k
    case splashModeRGB8:
957
1.86k
      destColorPtr[0] = cResult0;
958
1.86k
      destColorPtr[1] = cResult1;
959
1.86k
      destColorPtr[2] = cResult2;
960
1.86k
      destColorPtr += 3;
961
1.86k
      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.86k
#endif
977
1.86k
    }
978
1.86k
    if (destShapePtr) {
979
0
      *destShapePtr++ = fResult;
980
0
    }
981
1.86k
    if (destAlphaPtr) {
982
1.86k
      *destAlphaPtr++ = aResult;
983
1.86k
    }
984
985
1.86k
    cSrcPtr += cSrcStride;
986
1.86k
    shapePtr2 += shapeStride;
987
1.86k
    alphaPtr2 += alphaStride;
988
1.86k
  } // for (x ...)
989
990
1.86k
  updateModX(lastX);
991
1.86k
}
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
37.5k
             SplashColorPtr cSrcPtr) {
1212
37.5k
  Guchar alpha, aSrc, cSrc0, cDest0, cResult0;
1213
37.5k
  SplashColorPtr destColorPtr;
1214
37.5k
  Guchar destColorMask;
1215
37.5k
  SplashScreenCursor screenCursor;
1216
37.5k
  int cSrcStride, x, lastX;
1217
1218
37.5k
  if (cSrcPtr) {
1219
0
    cSrcStride = 1;
1220
37.5k
  } else {
1221
37.5k
    cSrcPtr = pipe->cSrcVal;
1222
37.5k
    cSrcStride = 0;
1223
37.5k
  }
1224
31.5M
  for (; x0 <= x1; ++x0) {
1225
31.5M
    if (*alphaPtr) {
1226
11.6k
      break;
1227
11.6k
    }
1228
31.5M
    cSrcPtr += cSrcStride;
1229
31.5M
    ++alphaPtr;
1230
31.5M
  }
1231
37.5k
  if (x0 > x1) {
1232
25.8k
    return;
1233
25.8k
  }
1234
11.6k
  updateModX(x0);
1235
11.6k
  updateModY(y);
1236
11.6k
  lastX = x0;
1237
1238
11.6k
  useBitmapRow(y);
1239
1240
11.6k
  destColorPtr = &bitmap->data[y * bitmap->rowSize + (x0 >> 3)];
1241
11.6k
  destColorMask = (Guchar)(0x80 >> (x0 & 7));
1242
1243
11.6k
  screenCursor = state->screen->getTestCursor(y);
1244
1245
9.94M
  for (x = x0; x <= x1; ++x) {
1246
1247
    //----- alpha
1248
9.93M
    alpha = *alphaPtr;
1249
9.93M
    if (!alpha) {
1250
2.91M
      destColorPtr += destColorMask & 1;
1251
2.91M
      destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1252
2.91M
      cSrcPtr += cSrcStride;
1253
2.91M
      ++alphaPtr;
1254
2.91M
      continue;
1255
2.91M
    }
1256
7.01M
    lastX = x;
1257
1258
    //----- source color
1259
7.01M
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1260
1261
    //----- source alpha
1262
7.01M
    aSrc = alpha;
1263
1264
    //----- special case for aSrc = 255
1265
7.01M
    if (aSrc == 255) {
1266
7.01M
      cResult0 = cSrc0;
1267
7.01M
    } 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.01M
    if (state->screen->testWithCursor(screenCursor, x, cResult0)) {
1279
7.01M
      *destColorPtr |= destColorMask;
1280
7.01M
    } else {
1281
0
      *destColorPtr &= (Guchar)~destColorMask;
1282
0
    }
1283
7.01M
    destColorPtr += destColorMask & 1;
1284
7.01M
    destColorMask = (Guchar)((destColorMask << 7) | (destColorMask >> 1));
1285
1286
7.01M
    cSrcPtr += cSrcStride;
1287
7.01M
    ++alphaPtr;
1288
7.01M
  }
1289
1290
11.6k
  updateModX(lastX);
1291
11.6k
}
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
432
             SplashColorPtr cSrcPtr) {
1299
432
  Guchar alpha, aSrc, aDest, alphaI, aResult, cSrc0, cDest0, cResult0;
1300
432
  SplashColorPtr destColorPtr;
1301
432
  Guchar *destAlphaPtr;
1302
432
  int cSrcStride, x, lastX;
1303
1304
432
  if (cSrcPtr) {
1305
0
    cSrcStride = 1;
1306
432
  } else {
1307
432
    cSrcPtr = pipe->cSrcVal;
1308
432
    cSrcStride = 0;
1309
432
  }
1310
439
  for (; x0 <= x1; ++x0) {
1311
432
    if (*alphaPtr) {
1312
425
      break;
1313
425
    }
1314
7
    cSrcPtr += cSrcStride;
1315
7
    ++alphaPtr;
1316
7
  }
1317
432
  if (x0 > x1) {
1318
7
    return;
1319
7
  }
1320
425
  updateModX(x0);
1321
425
  updateModY(y);
1322
425
  lastX = x0;
1323
1324
425
  useBitmapRow(y);
1325
1326
425
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
1327
425
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1328
1329
850
  for (x = x0; x <= x1; ++x) {
1330
1331
    //----- alpha
1332
425
    alpha = *alphaPtr;
1333
425
    if (!alpha) {
1334
0
      ++destColorPtr;
1335
0
      ++destAlphaPtr;
1336
0
      cSrcPtr += cSrcStride;
1337
0
      ++alphaPtr;
1338
0
      continue;
1339
0
    }
1340
425
    lastX = x;
1341
1342
    //----- source color
1343
425
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1344
1345
    //----- source alpha
1346
425
    aSrc = alpha;
1347
1348
    //----- special case for aSrc = 255
1349
425
    if (aSrc == 255) {
1350
168
      aResult = 255;
1351
168
      cResult0 = cSrc0;
1352
257
    } else {
1353
1354
      //----- read destination alpha
1355
257
      aDest = *destAlphaPtr;
1356
1357
      //----- special case for aDest = 0
1358
257
      if (aDest == 0) {
1359
257
  aResult = aSrc;
1360
257
  cResult0 = cSrc0;
1361
257
      } 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
257
    }
1374
1375
    //----- write destination pixel
1376
425
    *destColorPtr++ = cResult0;
1377
425
    *destAlphaPtr++ = aResult;
1378
1379
425
    cSrcPtr += cSrcStride;
1380
425
    ++alphaPtr;
1381
425
  }
1382
1383
425
  updateModX(lastX);
1384
425
}
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
10.8k
            SplashColorPtr cSrcPtr) {
1392
10.8k
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1393
10.8k
  Guchar cSrc0, cSrc1, cSrc2;
1394
10.8k
  Guchar cDest0, cDest1, cDest2;
1395
10.8k
  Guchar cResult0, cResult1, cResult2;
1396
10.8k
  SplashColorPtr destColorPtr;
1397
10.8k
  Guchar *destAlphaPtr;
1398
10.8k
  int cSrcStride, x, lastX;
1399
1400
10.8k
  if (cSrcPtr) {
1401
355
    cSrcStride = 3;
1402
10.5k
  } else {
1403
10.5k
    cSrcPtr = pipe->cSrcVal;
1404
10.5k
    cSrcStride = 0;
1405
10.5k
  }
1406
12.1k
  for (; x0 <= x1; ++x0) {
1407
10.8k
    if (*alphaPtr) {
1408
9.61k
      break;
1409
9.61k
    }
1410
1.27k
    cSrcPtr += cSrcStride;
1411
1.27k
    ++alphaPtr;
1412
1.27k
  }
1413
10.8k
  if (x0 > x1) {
1414
1.27k
    return;
1415
1.27k
  }
1416
9.61k
  updateModX(x0);
1417
9.61k
  updateModY(y);
1418
9.61k
  lastX = x0;
1419
1420
9.61k
  useBitmapRow(y);
1421
1422
9.61k
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1423
9.61k
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1424
1425
19.2k
  for (x = x0; x <= x1; ++x) {
1426
1427
    //----- alpha
1428
9.61k
    alpha = *alphaPtr;
1429
9.61k
    if (!alpha) {
1430
0
      destColorPtr += 3;
1431
0
      ++destAlphaPtr;
1432
0
      cSrcPtr += cSrcStride;
1433
0
      ++alphaPtr;
1434
0
      continue;
1435
0
    }
1436
9.61k
    lastX = x;
1437
1438
    //----- source color
1439
9.61k
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
1440
9.61k
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
1441
9.61k
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
1442
1443
    //----- source alpha
1444
9.61k
    aSrc = alpha;
1445
1446
    //----- special case for aSrc = 255
1447
9.61k
    if (aSrc == 255) {
1448
7.21k
      aResult = 255;
1449
7.21k
      cResult0 = cSrc0;
1450
7.21k
      cResult1 = cSrc1;
1451
7.21k
      cResult2 = cSrc2;
1452
7.21k
    } else {
1453
1454
      //----- read destination alpha
1455
2.39k
      aDest = *destAlphaPtr;
1456
1457
      //----- special case for aDest = 0
1458
2.39k
      if (aDest == 0) {
1459
494
  aResult = aSrc;
1460
494
  cResult0 = cSrc0;
1461
494
  cResult1 = cSrc1;
1462
494
  cResult2 = cSrc2;
1463
1.90k
      } else {
1464
1465
  //----- read destination pixel
1466
1.90k
  cDest0 = destColorPtr[0];
1467
1.90k
  cDest1 = destColorPtr[1];
1468
1.90k
  cDest2 = destColorPtr[2];
1469
1470
  //----- result alpha and non-isolated group element correction
1471
1.90k
  aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
1472
1.90k
  alphaI = aResult;
1473
1474
  //----- result color
1475
1.90k
  cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
1476
1.90k
  cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
1477
1.90k
  cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
1478
1.90k
      }
1479
2.39k
    }
1480
1481
    //----- write destination pixel
1482
9.61k
    destColorPtr[0] = cResult0;
1483
9.61k
    destColorPtr[1] = cResult1;
1484
9.61k
    destColorPtr[2] = cResult2;
1485
9.61k
    destColorPtr += 3;
1486
9.61k
    *destAlphaPtr++ = aResult;
1487
1488
9.61k
    cSrcPtr += cSrcStride;
1489
9.61k
    ++alphaPtr;
1490
9.61k
  }
1491
1492
9.61k
  updateModX(lastX);
1493
9.61k
}
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
7
             SplashColorPtr cSrcPtr) {
1611
7
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1612
7
  Guchar cSrc0, cSrc1, cSrc2, cSrc3;
1613
7
  Guchar cDest0, cDest1, cDest2, cDest3;
1614
7
  Guchar cResult0, cResult1, cResult2, cResult3;
1615
7
  SplashColorPtr destColorPtr;
1616
7
  Guchar *destAlphaPtr;
1617
7
  int cSrcStride, x, lastX;
1618
1619
7
  if (cSrcPtr) {
1620
7
    cSrcStride = 4;
1621
7
  } else {
1622
0
    cSrcPtr = pipe->cSrcVal;
1623
0
    cSrcStride = 0;
1624
0
  }
1625
7
  for (; x0 <= x1; ++x0) {
1626
7
    if (*alphaPtr) {
1627
7
      break;
1628
7
    }
1629
0
    cSrcPtr += cSrcStride;
1630
0
    ++alphaPtr;
1631
0
  }
1632
7
  if (x0 > x1) {
1633
0
    return;
1634
0
  }
1635
7
  updateModX(x0);
1636
7
  updateModY(y);
1637
7
  lastX = x0;
1638
1639
7
  useBitmapRow(y);
1640
1641
7
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 4 * x0];
1642
7
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
1643
1644
14
  for (x = x0; x <= x1; ++x) {
1645
1646
    //----- alpha
1647
7
    alpha = *alphaPtr;
1648
7
    if (!alpha) {
1649
0
      destColorPtr += 4;
1650
0
      ++destAlphaPtr;
1651
0
      cSrcPtr += cSrcStride;
1652
0
      ++alphaPtr;
1653
0
      continue;
1654
0
    }
1655
7
    lastX = x;
1656
1657
    //----- read destination pixel
1658
7
    cDest0 = destColorPtr[0];
1659
7
    cDest1 = destColorPtr[1];
1660
7
    cDest2 = destColorPtr[2];
1661
7
    cDest3 = destColorPtr[3];
1662
7
    aDest = *destAlphaPtr;
1663
1664
    //----- overprint
1665
7
    cSrc0 = state->cmykTransferC[cSrcPtr[0]];
1666
7
    cSrc1 = state->cmykTransferM[cSrcPtr[1]];
1667
7
    cSrc2 = state->cmykTransferY[cSrcPtr[2]];
1668
7
    cSrc3 = state->cmykTransferK[cSrcPtr[3]];
1669
1670
    //----- source alpha
1671
7
    aSrc = alpha;
1672
1673
    //----- special case for aSrc = 255
1674
7
    if (aSrc == 255) {
1675
7
      aResult = 255;
1676
7
      cResult0 = cSrc0;
1677
7
      cResult1 = cSrc1;
1678
7
      cResult2 = cSrc2;
1679
7
      cResult3 = cSrc3;
1680
7
    } 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
7
    destColorPtr[0] = cResult0;
1705
7
    destColorPtr[1] = cResult1;
1706
7
    destColorPtr[2] = cResult2;
1707
7
    destColorPtr[3] = cResult3;
1708
7
    destColorPtr += 4;
1709
7
    *destAlphaPtr++ = aResult;
1710
1711
7
    cSrcPtr += cSrcStride;
1712
7
    ++alphaPtr;
1713
7
  }
1714
1715
7
  updateModX(lastX);
1716
7
}
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
812
                                      SplashColorPtr cSrcPtr) {
1726
812
  Guchar alpha, aSrc, cSrc0, cDest0, cResult0;
1727
812
  SplashColorPtr destColorPtr;
1728
812
  int cSrcStride, x, lastX;
1729
1730
812
  if (cSrcPtr) {
1731
264
    cSrcStride = 1;
1732
548
  } else {
1733
548
    cSrcPtr = pipe->cSrcVal;
1734
548
    cSrcStride = 0;
1735
548
  }
1736
10.8k
  for (; x0 <= x1; ++x0) {
1737
10.7k
    if (*alphaPtr) {
1738
694
      break;
1739
694
    }
1740
10.0k
    cSrcPtr += cSrcStride;
1741
10.0k
    ++alphaPtr;
1742
10.0k
  }
1743
812
  if (x0 > x1) {
1744
118
    return;
1745
118
  }
1746
694
  updateModX(x0);
1747
694
  updateModY(y);
1748
694
  lastX = x0;
1749
1750
694
  useBitmapRow(y);
1751
1752
694
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
1753
1754
40.4k
  for (x = x0; x <= x1; ++x) {
1755
1756
    //----- alpha
1757
39.7k
    alpha = *alphaPtr;
1758
39.7k
    if (!alpha) {
1759
0
      ++destColorPtr;
1760
0
      cSrcPtr += cSrcStride;
1761
0
      ++alphaPtr;
1762
0
      continue;
1763
0
    }
1764
39.7k
    lastX = x;
1765
1766
    //----- source color
1767
39.7k
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
1768
1769
    //----- source alpha
1770
39.7k
    aSrc = alpha;
1771
1772
    //----- special case for aSrc = 255
1773
39.7k
    if (aSrc == 255) {
1774
10.2k
      cResult0 = cSrc0;
1775
29.4k
    } else {
1776
1777
      //----- read destination pixel
1778
29.4k
      cDest0 = *destColorPtr;
1779
1780
      //----- result color
1781
29.4k
      cResult0 = div255((255 - aSrc) * cDest0 + aSrc * cSrc0);
1782
29.4k
    }
1783
1784
    //----- write destination pixel
1785
39.7k
    *destColorPtr++ = cResult0;
1786
1787
39.7k
    cSrcPtr += cSrcStride;
1788
39.7k
    ++alphaPtr;
1789
39.7k
  }
1790
1791
694
  updateModX(lastX);
1792
694
}
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
434
         SplashColorPtr cSrcPtr) {
1968
434
  Guchar alpha, aSrc, aDest, alphaI, aResult;
1969
434
  Guchar cSrc0, cSrc1, cSrc2;
1970
434
  Guchar cDest0, cDest1, cDest2;
1971
434
  Guchar cResult0, cResult1, cResult2;
1972
434
  SplashColorPtr destColorPtr;
1973
434
  Guchar *destAlphaPtr;
1974
434
  int cSrcStride, x, lastX;
1975
1976
434
  if (cSrcPtr) {
1977
7
    cSrcStride = 3;
1978
427
  } else {
1979
427
    cSrcPtr = pipe->cSrcVal;
1980
427
    cSrcStride = 0;
1981
427
  }
1982
456
  for (; x0 <= x1; ++x0) {
1983
434
    if (*alphaPtr) {
1984
412
      break;
1985
412
    }
1986
22
    cSrcPtr += cSrcStride;
1987
22
    ++alphaPtr;
1988
22
  }
1989
434
  if (x0 > x1) {
1990
22
    return;
1991
22
  }
1992
412
  updateModX(x0);
1993
412
  updateModY(y);
1994
412
  lastX = x0;
1995
1996
412
  useBitmapRow(y);
1997
1998
412
  destColorPtr = &bitmap->data[y * bitmap->rowSize + 3 * x0];
1999
412
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2000
2001
824
  for (x = x0; x <= x1; ++x) {
2002
2003
    //----- alpha
2004
412
    alpha = *alphaPtr;
2005
412
    if (!alpha) {
2006
0
      destColorPtr += 3;
2007
0
      ++destAlphaPtr;
2008
0
      cSrcPtr += cSrcStride;
2009
0
      ++alphaPtr;
2010
0
      continue;
2011
0
    }
2012
412
    lastX = x;
2013
2014
    //----- read destination pixel
2015
412
    cDest0 = destColorPtr[0];
2016
412
    cDest1 = destColorPtr[1];
2017
412
    cDest2 = destColorPtr[2];
2018
412
    aDest = *destAlphaPtr;
2019
2020
    //----- source color
2021
412
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2022
412
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2023
412
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2024
2025
    //----- source alpha
2026
412
    aSrc = div255(pipe->aInput * alpha);
2027
2028
    //----- result alpha and non-isolated group element correction
2029
412
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2030
412
    alphaI = aResult;
2031
2032
    //----- result color
2033
412
    if (alphaI == 0) {
2034
212
      cResult0 = 0;
2035
212
      cResult1 = 0;
2036
212
      cResult2 = 0;
2037
212
    } else {
2038
200
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2039
200
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2040
200
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2041
200
    }
2042
2043
    //----- write destination pixel
2044
412
    destColorPtr[0] = cResult0;
2045
412
    destColorPtr[1] = cResult1;
2046
412
    destColorPtr[2] = cResult2;
2047
412
    destColorPtr += 3;
2048
412
    *destAlphaPtr++ = aResult;
2049
2050
412
    cSrcPtr += cSrcStride;
2051
412
    ++alphaPtr;
2052
412
  }
2053
2054
412
  updateModX(lastX);
2055
412
}
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
67
          SplashColorPtr cSrcPtr) {
2284
67
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2285
67
  Guchar cSrc0, cDest0, cResult0;
2286
67
  SplashColorPtr destColorPtr;
2287
67
  Guchar *destAlphaPtr;
2288
67
  SplashColorPtr softMaskPtr;
2289
67
  int cSrcStride, x, lastX;
2290
2291
67
  if (cSrcPtr) {
2292
0
    cSrcStride = 1;
2293
67
  } else {
2294
67
    cSrcPtr = pipe->cSrcVal;
2295
67
    cSrcStride = 0;
2296
67
  }
2297
97
  for (; x0 <= x1; ++x0) {
2298
67
    if (*alphaPtr) {
2299
37
      break;
2300
37
    }
2301
30
    cSrcPtr += cSrcStride;
2302
30
    ++alphaPtr;
2303
30
  }
2304
67
  if (x0 > x1) {
2305
30
    return;
2306
30
  }
2307
37
  updateModX(x0);
2308
37
  updateModY(y);
2309
37
  lastX = x0;
2310
2311
37
  useBitmapRow(y);
2312
2313
37
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0];
2314
37
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2315
37
  softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
2316
2317
74
  for (x = x0; x <= x1; ++x) {
2318
2319
    //----- alpha
2320
37
    alpha = *alphaPtr;
2321
37
    if (!alpha) {
2322
0
      ++destColorPtr;
2323
0
      ++destAlphaPtr;
2324
0
      ++softMaskPtr;
2325
0
      cSrcPtr += cSrcStride;
2326
0
      ++alphaPtr;
2327
0
      continue;
2328
0
    }
2329
37
    lastX = x;
2330
2331
    //----- read source color
2332
37
    cSrc0 = state->grayTransfer[cSrcPtr[0]];
2333
2334
    //----- source alpha
2335
37
    aSrc = div255(*softMaskPtr++ * alpha);
2336
2337
    //----- special case for aSrc = 255
2338
37
    if (aSrc == 255) {
2339
0
      aResult = 255;
2340
0
      cResult0 = cSrc0;
2341
37
    } else {
2342
2343
      //----- read destination alpha
2344
37
      aDest = *destAlphaPtr;
2345
2346
      //----- special case for aDest = 0
2347
37
      if (aDest == 0) {
2348
37
        aResult = aSrc;
2349
37
        cResult0 = cSrc0;
2350
37
      } else {
2351
2352
        //----- read destination pixel
2353
0
        cDest0 = destColorPtr[0];
2354
2355
        //----- result alpha and non-isolated group element correction
2356
0
        aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2357
0
        alphaI = aResult;
2358
2359
        //----- result color
2360
0
        cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2361
0
      }
2362
37
    }
2363
2364
    //----- write destination pixel
2365
37
    destColorPtr[0] = cResult0;
2366
37
    ++destColorPtr;
2367
37
    *destAlphaPtr++ = aResult;
2368
2369
37
    cSrcPtr += cSrcStride;
2370
37
    ++alphaPtr;
2371
37
  }
2372
2373
37
  updateModX(lastX);
2374
37
}
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
193
         SplashColorPtr cSrcPtr) {
2385
193
  Guchar alpha, aSrc, aDest, alphaI, aResult;
2386
193
  Guchar cSrc0, cSrc1, cSrc2;
2387
193
  Guchar cDest0, cDest1, cDest2;
2388
193
  Guchar cResult0, cResult1, cResult2;
2389
193
  SplashColorPtr destColorPtr;
2390
193
  Guchar *destAlphaPtr;
2391
193
  SplashColorPtr softMaskPtr;
2392
193
  int cSrcStride, x, lastX;
2393
2394
193
  if (cSrcPtr) {
2395
39
    cSrcStride = 3;
2396
154
  } else {
2397
154
    cSrcPtr = pipe->cSrcVal;
2398
154
    cSrcStride = 0;
2399
154
  }
2400
244
  for (; x0 <= x1; ++x0) {
2401
193
    if (*alphaPtr) {
2402
142
      break;
2403
142
    }
2404
51
    cSrcPtr += cSrcStride;
2405
51
    ++alphaPtr;
2406
51
  }
2407
193
  if (x0 > x1) {
2408
51
    return;
2409
51
  }
2410
142
  updateModX(x0);
2411
142
  updateModY(y);
2412
142
  lastX = x0;
2413
2414
142
  useBitmapRow(y);
2415
2416
142
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 3];
2417
142
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2418
142
  softMaskPtr = &state->softMask->data[y * state->softMask->rowSize + x0];
2419
2420
284
  for (x = x0; x <= x1; ++x) {
2421
2422
    //----- alpha
2423
142
    alpha = *alphaPtr;
2424
142
    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
142
    lastX = x;
2433
2434
    //----- read source color
2435
142
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2436
142
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2437
142
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2438
2439
    //----- source alpha
2440
142
    aSrc = div255(*softMaskPtr++ * alpha);
2441
2442
    //----- special case for aSrc = 255
2443
142
    if (aSrc == 255) {
2444
19
      aResult = 255;
2445
19
      cResult0 = cSrc0;
2446
19
      cResult1 = cSrc1;
2447
19
      cResult2 = cSrc2;
2448
123
    } else {
2449
2450
      //----- read destination alpha
2451
123
      aDest = *destAlphaPtr;
2452
2453
      //----- special case for aDest = 0
2454
123
      if (aDest == 0) {
2455
102
        aResult = aSrc;
2456
102
        cResult0 = cSrc0;
2457
102
        cResult1 = cSrc1;
2458
102
        cResult2 = cSrc2;
2459
102
      } else {
2460
2461
        //----- read destination pixel
2462
21
        cDest0 = destColorPtr[0];
2463
21
        cDest1 = destColorPtr[1];
2464
21
        cDest2 = destColorPtr[2];
2465
2466
        //----- result alpha and non-isolated group element correction
2467
21
        aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2468
21
        alphaI = aResult;
2469
2470
        //----- result color
2471
21
        cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2472
21
        cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2473
21
        cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2474
21
      }
2475
123
    }
2476
2477
    //----- write destination pixel
2478
142
    destColorPtr[0] = cResult0;
2479
142
    destColorPtr[1] = cResult1;
2480
142
    destColorPtr[2] = cResult2;
2481
142
    destColorPtr += 3;
2482
142
    *destAlphaPtr++ = aResult;
2483
2484
142
    cSrcPtr += cSrcStride;
2485
142
    ++alphaPtr;
2486
142
  }
2487
2488
142
  updateModX(lastX);
2489
142
}
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
13.4k
             SplashColorPtr cSrcPtr) {
2822
13.4k
  Guchar alpha, aSrc, aDest, alphaI, alpha0, aResult;
2823
13.4k
  Guchar cSrc0, cSrc1, cSrc2;
2824
13.4k
  Guchar cDest0, cDest1, cDest2;
2825
13.4k
  Guchar cResult0, cResult1, cResult2;
2826
13.4k
  SplashColorPtr destColorPtr;
2827
13.4k
  Guchar *destAlphaPtr;
2828
13.4k
  Guchar *alpha0Ptr;
2829
13.4k
  int cSrcStride, x, lastX;
2830
2831
13.4k
  if (cSrcPtr) {
2832
0
    cSrcStride = 3;
2833
13.4k
  } else {
2834
13.4k
    cSrcPtr = pipe->cSrcVal;
2835
13.4k
    cSrcStride = 0;
2836
13.4k
  }
2837
13.9k
  for (; x0 <= x1; ++x0) {
2838
13.4k
    if (*alphaPtr) {
2839
12.8k
      break;
2840
12.8k
    }
2841
544
    cSrcPtr += cSrcStride;
2842
544
    ++alphaPtr;
2843
544
  }
2844
13.4k
  if (x0 > x1) {
2845
544
    return;
2846
544
  }
2847
12.8k
  updateModX(x0);
2848
12.8k
  updateModY(y);
2849
12.8k
  lastX = x0;
2850
2851
12.8k
  useBitmapRow(y);
2852
2853
12.8k
  destColorPtr = &bitmap->data[y * bitmap->rowSize + x0 * 3];
2854
12.8k
  destAlphaPtr = &bitmap->alpha[y * bitmap->alphaRowSize + x0];
2855
12.8k
  if (alpha0Bitmap) {
2856
0
    alpha0Ptr = &alpha0Bitmap->alpha[y * alpha0Bitmap->alphaRowSize + x0];
2857
12.8k
  } else {
2858
12.8k
    alpha0Ptr = &parent->bitmap->alpha[
2859
12.8k
        (parentOffsetY + y) * parent->bitmap->alphaRowSize
2860
12.8k
        + (parentOffsetX + x0)];
2861
12.8k
  }
2862
2863
25.7k
  for (x = x0; x <= x1; ++x) {
2864
2865
    //----- alpha
2866
12.8k
    alpha = *alphaPtr;
2867
12.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
12.8k
    lastX = x;
2876
2877
    //----- read destination pixel
2878
12.8k
    cDest0 = destColorPtr[0];
2879
12.8k
    cDest1 = destColorPtr[1];
2880
12.8k
    cDest2 = destColorPtr[2];
2881
12.8k
    aDest = *destAlphaPtr;
2882
2883
    //----- source color
2884
12.8k
    cSrc0 = state->rgbTransferR[cSrcPtr[0]];
2885
12.8k
    cSrc1 = state->rgbTransferG[cSrcPtr[1]];
2886
12.8k
    cSrc2 = state->rgbTransferB[cSrcPtr[2]];
2887
2888
    //----- source alpha
2889
12.8k
    aSrc = div255(pipe->aInput * alpha);
2890
2891
    //----- result alpha and non-isolated group element correction
2892
12.8k
    aResult = (Guchar)(aSrc + aDest - div255(aSrc * aDest));
2893
12.8k
    alpha0 = *alpha0Ptr++;
2894
12.8k
    alphaI = (Guchar)(aResult + alpha0 - div255(aResult * alpha0));
2895
2896
    //----- result color
2897
12.8k
    if (alphaI == 0) {
2898
3
      cResult0 = 0;
2899
3
      cResult1 = 0;
2900
3
      cResult2 = 0;
2901
12.8k
    } else {
2902
12.8k
      cResult0 = (Guchar)(((alphaI - aSrc) * cDest0 + aSrc * cSrc0) / alphaI);
2903
12.8k
      cResult1 = (Guchar)(((alphaI - aSrc) * cDest1 + aSrc * cSrc1) / alphaI);
2904
12.8k
      cResult2 = (Guchar)(((alphaI - aSrc) * cDest2 + aSrc * cSrc2) / alphaI);
2905
12.8k
    }
2906
2907
    //----- write destination pixel
2908
12.8k
    destColorPtr[0] = cResult0;
2909
12.8k
    destColorPtr[1] = cResult1;
2910
12.8k
    destColorPtr[2] = cResult2;
2911
12.8k
    destColorPtr += 3;
2912
12.8k
    *destAlphaPtr++ = aResult;
2913
2914
12.8k
    cSrcPtr += cSrcStride;
2915
12.8k
    ++alphaPtr;
2916
12.8k
  } // for (x ...)
2917
2918
12.8k
  updateModX(lastX);
2919
12.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
44.3k
void Splash::useBitmapRow(int y) {
3151
44.3k
  if (!deferredInit) {
3152
27.8k
    return;
3153
27.8k
  }
3154
3155
16.5k
  int y0, y1;
3156
16.5k
  if (deferredInitYMin > deferredInitYMax) {
3157
4.06k
    y0 = y1 = y;
3158
4.06k
    deferredInitYMin = deferredInitYMax = y;
3159
12.4k
  } else if (y < deferredInitYMin) {
3160
0
    y0 = y;
3161
0
    y1 = deferredInitYMin - 1;
3162
0
    deferredInitYMin = y;
3163
12.4k
  } else if (y > deferredInitYMax) {
3164
0
    y0 = deferredInitYMax + 1;
3165
0
    y1 = y;
3166
0
    deferredInitYMax = y;
3167
12.4k
  } else {
3168
12.4k
    return;
3169
12.4k
  }
3170
3171
4.06k
  parent->useBitmapRow(parentOffsetY + y);
3172
4.06k
  if (state->inNonIsolatedGroup) {
3173
5.30k
    for (int yy = y0; yy <= y1; ++yy) {
3174
2.65k
      copyParentRowColor(yy);
3175
2.65k
    }
3176
2.65k
  } else {
3177
2.81k
    for (int yy = y0; yy <= y1; ++yy) {
3178
1.40k
      zeroRowColor(yy);
3179
1.40k
    }
3180
1.40k
  }
3181
4.06k
  if (bitmap->alpha) {
3182
8.12k
    for (int yy = y0; yy <= y1; ++yy) {
3183
4.06k
      zeroRowAlpha(yy);
3184
4.06k
    }
3185
4.06k
  }
3186
4.06k
  if (alpha0Bitmap) {
3187
0
    for (int yy = y0; yy <= y1; ++yy) {
3188
0
      computeAlpha0Row(yy);
3189
0
    }
3190
0
  }
3191
4.06k
}
3192
3193
2.65k
void Splash::copyParentRowColor(int y) {
3194
2.65k
  if (parent->bitmap->mode != bitmap->mode) {
3195
0
    return;
3196
0
  }
3197
2.65k
  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
2.65k
  } else {
3220
2.65k
    SplashColorPtr p = &bitmap->data[y * bitmap->rowSize];
3221
2.65k
    SplashColorPtr q =
3222
2.65k
        &parent->bitmap->data[(parentOffsetY + y) * parent->bitmap->rowSize
3223
2.65k
            + bitmapComps * parentOffsetX];
3224
2.65k
    memcpy(p, q, bitmapComps * bitmap->width);
3225
2.65k
  }
3226
2.65k
}
3227
3228
1.40k
void Splash::zeroRowColor(int y) {
3229
1.40k
  memset(&bitmap->data[y * bitmap->rowSize], 0,
3230
1.40k
   bitmap->rowSize < 0 ? -bitmap->rowSize : bitmap->rowSize);
3231
1.40k
}
3232
3233
4.06k
void Splash::zeroRowAlpha(int y) {
3234
4.06k
  memset(&bitmap->alpha[y * bitmap->alphaRowSize], 0, bitmap->width);
3235
4.06k
}
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
48.5M
            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
48.5M
  *xo = xi * matrix[0] + yi * matrix[2] + matrix[4];
3271
48.5M
  *yo = xi * matrix[1] + yi * matrix[3] + matrix[5];
3272
48.5M
}
3273
3274
//------------------------------------------------------------------------
3275
// SplashImageCache
3276
//------------------------------------------------------------------------
3277
3278
397k
SplashImageCache::SplashImageCache() {
3279
397k
  tag = NULL;
3280
397k
  width = 0;
3281
397k
  height = 0;
3282
397k
  mode = splashModeRGB8;
3283
397k
  alpha = gFalse;
3284
397k
  interpolate = gFalse;
3285
397k
  colorData = NULL;
3286
397k
  alphaData = NULL;
3287
397k
  refCount = 1;
3288
397k
}
3289
3290
397k
SplashImageCache::~SplashImageCache() {
3291
397k
  if (tag) {
3292
137
    delete tag;
3293
137
  }
3294
397k
  gfree(colorData);
3295
397k
  gfree(alphaData);
3296
397k
}
3297
3298
GBool SplashImageCache::match(GString *aTag, int aWidth, int aHeight,
3299
            SplashColorMode aMode, GBool aAlpha,
3300
1.10k
            GBool aInterpolate) {
3301
1.10k
  return aTag && tag && !aTag->cmp(tag) &&
3302
153
   aWidth == width && aHeight == height &&
3303
153
   aMode == mode && aAlpha == alpha &&
3304
153
   aInterpolate == interpolate;
3305
1.10k
}
3306
3307
void SplashImageCache::reset(GString *aTag, int aWidth, int aHeight,
3308
           SplashColorMode aMode, GBool aAlpha,
3309
955
           GBool aInterpolate) {
3310
955
  if (tag) {
3311
206
    delete tag;
3312
206
  }
3313
955
  if (aTag) {
3314
343
    tag = aTag->copy();
3315
612
  } else {
3316
612
    tag = NULL;
3317
612
  }
3318
955
  width = aWidth;
3319
955
  height = aHeight;
3320
955
  mode = aMode;
3321
955
  alpha = aAlpha;
3322
955
  interpolate = aInterpolate; 
3323
955
  gfree(colorData);
3324
955
  colorData = NULL;
3325
955
  gfree(alphaData);
3326
955
  alphaData = NULL;
3327
955
}
3328
3329
7.45k
void SplashImageCache::incRefCount() {
3330
7.45k
  ++refCount;
3331
7.45k
}
3332
3333
404k
void SplashImageCache::decRefCount() {
3334
404k
  --refCount;
3335
404k
  if (refCount == 0) {
3336
397k
    delete this;
3337
397k
  }
3338
404k
}
3339
3340
//------------------------------------------------------------------------
3341
// ImageScaler
3342
//------------------------------------------------------------------------
3343
3344
// Abstract base class.
3345
class ImageScaler {
3346
public:
3347
3348
932
  ImageScaler() {}
3349
932
  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.86k
  virtual Guchar *colorData() { return colorLine; }
3375
30
  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
932
           GBool aInterpolate) {
3436
932
  colorTmpBuf0 = NULL;
3437
932
  colorTmpBuf1 = NULL;
3438
932
  colorTmpBuf2 = NULL;
3439
932
  alphaTmpBuf0 = NULL;
3440
932
  alphaTmpBuf1 = NULL;
3441
932
  alphaTmpBuf2 = NULL;
3442
932
  colorAccBuf = NULL;
3443
932
  alphaAccBuf = NULL;
3444
932
  colorLine = NULL;
3445
932
  alphaLine = NULL;
3446
3447
932
  src = aSrc;
3448
932
  srcData = aSrcData;
3449
932
  srcWidth = aSrcWidth;
3450
932
  srcHeight = aSrcHeight;
3451
932
  scaledWidth = aScaledWidth;
3452
932
  scaledHeight = aScaledHeight;
3453
932
  nComps = aNComps;
3454
932
  hasAlpha = aHasAlpha;
3455
3456
  // select scaling function; allocate buffers
3457
932
  if (scaledHeight <= srcHeight) {
3458
    // vertical downscaling
3459
932
    yp = srcHeight / scaledHeight;
3460
932
    yq = srcHeight % scaledHeight;
3461
932
    yt = 0;
3462
932
    colorTmpBuf0 = (Guchar *)gmallocn(srcWidth, nComps);
3463
932
    colorAccBuf = (Guint *)gmallocn(srcWidth, nComps * (int)sizeof(Guint));
3464
932
    if (hasAlpha) {
3465
15
      alphaTmpBuf0 = (Guchar *)gmalloc(srcWidth);
3466
15
      alphaAccBuf = (Guint *)gmallocn(srcWidth, sizeof(Guint));
3467
15
    }
3468
932
    if (scaledWidth <= srcWidth) {
3469
932
      scalingFunc = &BasicImageScaler::vertDownscaleHorizDownscale;
3470
932
    } else {
3471
0
      if (aInterpolate) {
3472
0
  scalingFunc = &BasicImageScaler::vertDownscaleHorizUpscaleInterp;
3473
0
      } else {
3474
0
  scalingFunc = &BasicImageScaler::vertDownscaleHorizUpscaleNoInterp;
3475
0
      }
3476
0
    }
3477
932
  } 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
932
  if (scaledWidth <= srcWidth) {
3515
932
    xp = srcWidth / scaledWidth;
3516
932
    xq = srcWidth % scaledWidth;
3517
932
  } 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
932
  colorLine = (Guchar *)gmallocn(scaledWidth, nComps);
3525
932
  if (hasAlpha) {
3526
15
    alphaLine = (Guchar *)gmalloc(scaledWidth);
3527
15
  }
3528
932
}
3529
3530
932
BasicImageScaler::~BasicImageScaler() {
3531
932
  gfree(colorTmpBuf0);
3532
932
  gfree(colorTmpBuf1);
3533
932
  gfree(colorTmpBuf2);
3534
932
  gfree(alphaTmpBuf0);
3535
932
  gfree(alphaTmpBuf1);
3536
932
  gfree(alphaTmpBuf2);
3537
932
  gfree(colorAccBuf);
3538
932
  gfree(alphaAccBuf);
3539
932
  gfree(colorLine);
3540
932
  gfree(alphaLine);
3541
932
}
3542
3543
932
void BasicImageScaler::nextLine() {
3544
932
  (this->*scalingFunc)();
3545
932
}
3546
3547
932
void BasicImageScaler::vertDownscaleHorizDownscale() {
3548
  //--- vert downscale
3549
932
  int yStep = yp;
3550
932
  yt += yq;
3551
932
  if (yt >= scaledHeight) {
3552
0
    yt -= scaledHeight;
3553
0
    ++yStep;
3554
0
  }
3555
932
  memset(colorAccBuf, 0, (srcWidth * nComps) * sizeof(Guint));
3556
932
  if (hasAlpha) {
3557
15
    memset(alphaAccBuf, 0, srcWidth * sizeof(Guint));
3558
15
  }
3559
932
  int nRowComps = srcWidth * nComps;
3560
99.1k
  for (int i = 0; i < yStep; ++i) {
3561
98.1k
    (*src)(srcData, colorTmpBuf0, alphaTmpBuf0);
3562
62.1M
    for (int j = 0; j < nRowComps; ++j) {
3563
62.0M
      colorAccBuf[j] += colorTmpBuf0[j];
3564
62.0M
    }
3565
98.1k
    if (hasAlpha) {
3566
820k
      for (int j = 0; j < srcWidth; ++j) {
3567
810k
  alphaAccBuf[j] += alphaTmpBuf0[j];
3568
810k
      }
3569
10.4k
    }
3570
98.1k
  }
3571
3572
  //--- horiz downscale
3573
932
  int colorAcc[splashMaxColorComps];
3574
932
  int xt = 0;
3575
932
  int unscaledColorIdx = 0;
3576
932
  int unscaledAlphaIdx = 0;
3577
932
  int scaledColorIdx = 0;
3578
3579
1.86k
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
3580
932
    int xStep = xp;
3581
932
    xt += xq;
3582
932
    if (xt >= scaledWidth) {
3583
0
      xt -= scaledWidth;
3584
0
      ++xStep;
3585
0
    }
3586
3587
3.04k
    for (int j = 0; j < nComps; ++j) {
3588
2.11k
      colorAcc[j] = 0;
3589
2.11k
    }
3590
88.0k
    for (int i = 0; i < xStep; ++i) {
3591
283k
      for (int j = 0; j < nComps; ++j) {
3592
196k
  colorAcc[j] += colorAccBuf[unscaledColorIdx + j];
3593
196k
      }
3594
87.0k
      unscaledColorIdx += nComps;
3595
87.0k
    }
3596
932
    int nPixels = yStep * xStep;
3597
3.04k
    for (int j = 0; j < nComps; ++j) {
3598
2.11k
      colorLine[scaledColorIdx + j] = (Guchar)(colorAcc[j] / nPixels);
3599
2.11k
    }
3600
932
    scaledColorIdx += nComps;
3601
3602
932
    if (hasAlpha) {
3603
15
      int alphaAcc = 0;
3604
1.03k
      for (int i = 0; i < xStep; ++i) {
3605
1.02k
  alphaAcc += alphaAccBuf[unscaledAlphaIdx];
3606
1.02k
  ++unscaledAlphaIdx;
3607
1.02k
      }
3608
15
      alphaLine[scaledIdx] = (Guchar)(alphaAcc / nPixels);
3609
15
    }
3610
932
  }
3611
932
}
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
932
  BasicImageScaler(aSrc, aSrcData, aSrcWidth, aSrcHeight, aNComps, aHasAlpha,
4011
932
       aScaledWidth, aScaledHeight, aInterpolate)
4012
932
{
4013
932
  colorPtr = aColorCache;
4014
932
  alphaPtr = aAlphaCache;
4015
932
}
4016
4017
932
void SavingImageScaler::nextLine() {
4018
932
  BasicImageScaler::nextLine();
4019
932
  memcpy(colorPtr, colorData(), scaledWidth * nComps);
4020
932
  colorPtr += scaledWidth * nComps;
4021
932
  if (hasAlpha) {
4022
15
    memcpy(alphaPtr, alphaData(), scaledWidth);
4023
15
    alphaPtr += scaledWidth;
4024
15
  }
4025
932
}
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
533k
  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
788
         GBool aInterpolate, GBool aAntialias) {
4152
788
  tmpBuf0 = NULL;
4153
788
  tmpBuf1 = NULL;
4154
788
  tmpBuf2 = NULL;
4155
788
  accBuf = NULL;
4156
788
  line = NULL;
4157
4158
788
  src = aSrc;
4159
788
  srcData = aSrcData;
4160
788
  srcWidth = aSrcWidth;
4161
788
  srcHeight = aSrcHeight;
4162
788
  scaledWidth = aScaledWidth;
4163
788
  scaledHeight = aScaledHeight;
4164
4165
  // select scaling function; allocate buffers
4166
788
  if (scaledHeight <= srcHeight) {
4167
    // vertical downscaling
4168
657
    yp = srcHeight / scaledHeight;
4169
657
    yq = srcHeight % scaledHeight;
4170
657
    yt = 0;
4171
657
    tmpBuf0 = (Guchar *)gmalloc(srcWidth);
4172
657
    accBuf = (Guint *)gmallocn(srcWidth, sizeof(Guint));
4173
657
    if (scaledWidth <= srcWidth) {
4174
595
      if (!aAntialias) {
4175
0
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizDownscaleThresh;
4176
595
      } else {
4177
595
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizDownscale;
4178
595
      }
4179
595
    } else {
4180
62
      if (!aAntialias) {
4181
0
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizUpscaleThresh;
4182
62
      } else if (aInterpolate) {
4183
0
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizUpscaleInterp;
4184
62
      } else {
4185
62
  scalingFunc = &ImageMaskScaler::vertDownscaleHorizUpscaleNoInterp;
4186
62
      }
4187
62
    }
4188
657
  } else {
4189
    // vertical upscaling
4190
131
    yp = scaledHeight / srcHeight;
4191
131
    yq = scaledHeight % srcHeight;
4192
131
    yt = 0;
4193
131
    yn = 0;
4194
131
    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
131
    } 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
131
    } else {
4214
131
      tmpBuf0 = (Guchar *)gmalloc(srcWidth);
4215
131
      if (scaledWidth <= srcWidth) {
4216
75
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizDownscaleNoInterp;
4217
75
      } else {
4218
56
  scalingFunc = &ImageMaskScaler::vertUpscaleHorizUpscaleNoInterp;
4219
56
      }
4220
131
    }
4221
131
  }
4222
788
  if (scaledWidth <= srcWidth) {
4223
670
    xp = srcWidth / scaledWidth;
4224
670
    xq = srcWidth % scaledWidth;
4225
670
  } else {
4226
118
    xp = scaledWidth / srcWidth;
4227
118
    xq = scaledWidth % srcWidth;
4228
118
    if (aInterpolate) {
4229
0
      xInvScale = (SplashCoord)srcWidth / (SplashCoord)scaledWidth;
4230
0
    }
4231
118
  }
4232
788
  line = (Guchar *)gmalloc(scaledWidth);
4233
788
}
4234
4235
788
ImageMaskScaler::~ImageMaskScaler() {
4236
788
  gfree(tmpBuf0);
4237
788
  gfree(tmpBuf1);
4238
788
  gfree(tmpBuf2);
4239
788
  gfree(accBuf);
4240
788
  gfree(line);
4241
788
}
4242
4243
533k
void ImageMaskScaler::nextLine() {
4244
533k
  (this->*scalingFunc)();
4245
533k
}
4246
4247
24.3k
void ImageMaskScaler::vertDownscaleHorizDownscale() {
4248
  //--- vert downscale
4249
24.3k
  int yStep = yp;
4250
24.3k
  yt += yq;
4251
24.3k
  if (yt >= scaledHeight) {
4252
14.5k
    yt -= scaledHeight;
4253
14.5k
    ++yStep;
4254
14.5k
  }
4255
24.3k
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4256
119k
  for (int i = 0; i < yStep; ++i) {
4257
95.2k
    (*src)(srcData, tmpBuf0);
4258
338M
    for (int j = 0; j < srcWidth; ++j) {
4259
338M
      accBuf[j] += tmpBuf0[j];
4260
338M
    }
4261
95.2k
  }
4262
4263
  //--- horiz downscale
4264
24.3k
  int acc;
4265
24.3k
  int xt = 0;
4266
24.3k
  int unscaledIdx = 0;
4267
2.54M
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4268
2.52M
    int xStep = xp;
4269
2.52M
    xt += xq;
4270
2.52M
    if (xt >= scaledWidth) {
4271
0
      xt -= scaledWidth;
4272
0
      ++xStep;
4273
0
    }
4274
4275
2.52M
    acc = 0;
4276
17.5M
    for (int i = 0; i < xStep; ++i) {
4277
15.0M
      acc += accBuf[unscaledIdx];
4278
15.0M
      ++unscaledIdx;
4279
15.0M
    }
4280
2.52M
    line[scaledIdx] = (Guchar)((255 * acc) / (xStep * yStep));
4281
2.52M
  }
4282
24.3k
}
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
26.4k
void ImageMaskScaler::vertDownscaleHorizUpscaleNoInterp() {
4322
  //--- vert downscale
4323
26.4k
  int yStep = yp;
4324
26.4k
  yt += yq;
4325
26.4k
  if (yt >= scaledHeight) {
4326
0
    yt -= scaledHeight;
4327
0
    ++yStep;
4328
0
  }
4329
26.4k
  memset(accBuf, 0, srcWidth * sizeof(Guint));
4330
53.1k
  for (int i = 0; i < yStep; ++i) {
4331
26.6k
    (*src)(srcData, tmpBuf0);
4332
1.50M
    for (int j = 0; j < srcWidth; ++j) {
4333
1.47M
      accBuf[j] += tmpBuf0[j];
4334
1.47M
    }
4335
26.6k
  }
4336
4337
  //--- horiz upscale
4338
26.4k
  int xt = 0;
4339
26.4k
  int scaledIdx = 0;
4340
1.50M
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
4341
1.47M
    int xStep = xp;
4342
1.47M
    xt += xq;
4343
1.47M
    if (xt >= srcWidth) {
4344
127k
      xt -= srcWidth;
4345
127k
      ++xStep;
4346
127k
    }
4347
1.47M
    Guchar buf = (Guchar)((255 * accBuf[srcIdx]) / yStep);
4348
134M
    for (int i = 0; i < xStep; ++i) {
4349
132M
      line[scaledIdx] = buf;
4350
132M
      ++scaledIdx;
4351
132M
    }
4352
1.47M
  }
4353
26.4k
}
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
475k
void ImageMaskScaler::vertUpscaleHorizDownscaleNoInterp() {
4426
  //--- vert upscale
4427
475k
  if (yn == 0) {
4428
7.22k
    yn = yp;
4429
7.22k
    yt += yq;
4430
7.22k
    if (yt >= srcHeight) {
4431
3.38k
      yt -= srcHeight;
4432
3.38k
      ++yn;
4433
3.38k
    }
4434
7.22k
    (*src)(srcData, tmpBuf0);
4435
7.22k
  }
4436
475k
  --yn;
4437
4438
  //--- horiz downscale
4439
475k
  int acc;
4440
475k
  int xt = 0;
4441
475k
  int unscaledIdx = 0;
4442
1.00M
  for (int scaledIdx = 0; scaledIdx < scaledWidth; ++scaledIdx) {
4443
527k
    int xStep = xp;
4444
527k
    xt += xq;
4445
527k
    if (xt >= scaledWidth) {
4446
0
      xt -= scaledWidth;
4447
0
      ++xStep;
4448
0
    }
4449
4450
527k
    acc = 0;
4451
3.94M
    for (int i = 0; i < xStep; ++i) {
4452
3.41M
      acc += tmpBuf0[unscaledIdx];
4453
3.41M
      ++unscaledIdx;
4454
3.41M
    }
4455
527k
    line[scaledIdx] = (Guchar)((255 * acc) / xStep);
4456
527k
  }
4457
475k
}
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
7.52k
void ImageMaskScaler::vertUpscaleHorizUpscaleNoInterp() {
4547
  //--- vert upscale
4548
7.52k
  if (yn == 0) {
4549
4.44k
    yn = yp;
4550
4.44k
    yt += yq;
4551
4.44k
    if (yt >= srcHeight) {
4552
689
      yt -= srcHeight;
4553
689
      ++yn;
4554
689
    }
4555
4.44k
    (*src)(srcData, tmpBuf0);
4556
4.44k
  }
4557
7.52k
  --yn;
4558
4559
  //--- horiz upscale
4560
7.52k
  int xt = 0;
4561
7.52k
  int scaledIdx = 0;
4562
64.3k
  for (int srcIdx = 0; srcIdx < srcWidth; ++srcIdx) {
4563
56.8k
    int xStep = xp;
4564
56.8k
    xt += xq;
4565
56.8k
    if (xt >= srcWidth) {
4566
34.7k
      xt -= srcWidth;
4567
34.7k
      ++xStep;
4568
34.7k
    }
4569
56.8k
    Guchar buf = (Guchar)(255 * tmpBuf0[srcIdx]);
4570
12.2M
    for (int i = 0; i < xStep; ++i) {
4571
12.1M
      line[scaledIdx] = buf;
4572
12.1M
      ++scaledIdx;
4573
12.1M
    }
4574
56.8k
  }
4575
7.52k
}
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
398k
         SplashScreenParams *screenParams) {
4637
398k
  bitmap = bitmapA;
4638
398k
  bitmapComps = splashColorModeNComps[bitmap->mode];
4639
398k
  vectorAntialias = vectorAntialiasA;
4640
398k
  state = new SplashState(bitmap->width, bitmap->height, vectorAntialias,
4641
398k
        screenParams);
4642
398k
  scanBuf = (Guchar *)gmalloc(bitmap->width);
4643
398k
  if (bitmap->mode == splashModeMono1) {
4644
128
    scanBuf2 = (Guchar *)gmalloc(bitmap->width);
4645
398k
  } else {
4646
398k
    scanBuf2 = NULL;
4647
398k
  }
4648
398k
  parent = NULL;
4649
398k
  deferredInit = gFalse;
4650
398k
  alpha0Bitmap = NULL;
4651
398k
  shapeBitmap = NULL;
4652
398k
  overprintMaskBitmap = NULL;
4653
398k
  minLineWidth = 0;
4654
398k
  clearModRegion();
4655
398k
  debugMode = gFalse;
4656
4657
398k
  if (imageCacheA) {
4658
1.57k
    imageCache = imageCacheA;
4659
1.57k
    imageCache->incRefCount();
4660
397k
  } else {
4661
397k
    imageCache = new SplashImageCache();
4662
397k
  }
4663
398k
}
4664
4665
Splash::Splash(SplashBitmap *bitmapA, GBool vectorAntialiasA,
4666
5.87k
         SplashImageCache *imageCacheA, SplashScreen *screenA) {
4667
5.87k
  bitmap = bitmapA;
4668
5.87k
  bitmapComps = splashColorModeNComps[bitmap->mode];
4669
5.87k
  vectorAntialias = vectorAntialiasA;
4670
5.87k
  state = new SplashState(bitmap->width, bitmap->height, vectorAntialias,
4671
5.87k
        screenA);
4672
5.87k
  scanBuf = (Guchar *)gmalloc(bitmap->width);
4673
5.87k
  if (bitmap->mode == splashModeMono1) {
4674
0
    scanBuf2 = (Guchar *)gmalloc(bitmap->width);
4675
5.87k
  } else {
4676
5.87k
    scanBuf2 = NULL;
4677
5.87k
  }
4678
5.87k
  parent = NULL;
4679
5.87k
  deferredInit = gFalse;
4680
5.87k
  alpha0Bitmap = NULL;
4681
5.87k
  shapeBitmap = NULL;
4682
5.87k
  overprintMaskBitmap = NULL;
4683
5.87k
  minLineWidth = 0;
4684
5.87k
  clearModRegion();
4685
5.87k
  debugMode = gFalse;
4686
4687
5.87k
  if (imageCacheA) {
4688
5.87k
    imageCache = imageCacheA;
4689
5.87k
    imageCache->incRefCount();
4690
5.87k
  } else {
4691
0
    imageCache = new SplashImageCache();
4692
0
  }
4693
5.87k
}
4694
4695
404k
Splash::~Splash() {
4696
404k
  imageCache->decRefCount();
4697
4698
404k
  while (state->next) {
4699
86
    restoreState();
4700
86
  }
4701
404k
  delete state;
4702
404k
  gfree(scanBuf);
4703
404k
  gfree(scanBuf2);
4704
404k
}
4705
4706
//------------------------------------------------------------------------
4707
// state read
4708
//------------------------------------------------------------------------
4709
4710
40.8k
SplashCoord *Splash::getMatrix() {
4711
40.8k
  return state->matrix;
4712
40.8k
}
4713
4714
4.52k
SplashPattern *Splash::getStrokePattern() {
4715
4.52k
  return state->strokePattern;
4716
4.52k
}
4717
4718
4.52k
SplashPattern *Splash::getFillPattern() {
4719
4.52k
  return state->fillPattern;
4720
4.52k
}
4721
4722
5.87k
SplashScreen *Splash::getScreen() {
4723
5.87k
  return state->screen;
4724
5.87k
}
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
4.54k
SplashCoord Splash::getLineWidth() {
4739
4.54k
  return state->lineWidth;
4740
4.54k
}
4741
4742
4.54k
int Splash::getLineCap() {
4743
4.54k
  return state->lineCap;
4744
4.54k
}
4745
4746
4.54k
int Splash::getLineJoin() {
4747
4.54k
  return state->lineJoin;
4748
4.54k
}
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
4.54k
SplashCoord *Splash::getLineDash() {
4759
4.54k
  return state->lineDash;
4760
4.54k
}
4761
4762
4.54k
int Splash::getLineDashLength() {
4763
4.54k
  return state->lineDashLength;
4764
4.54k
}
4765
4766
4.54k
SplashCoord Splash::getLineDashPhase() {
4767
4.54k
  return state->lineDashPhase;
4768
4.54k
}
4769
4770
1.33M
SplashStrokeAdjustMode Splash::getStrokeAdjust() {
4771
1.33M
  return state->strokeAdjust;
4772
1.33M
}
4773
4774
34.0k
SplashClip *Splash::getClip() {
4775
34.0k
  return state->clip;
4776
34.0k
}
4777
4778
6.94k
SplashBitmap *Splash::getSoftMask() {
4779
6.94k
  return state->softMask;
4780
6.94k
}
4781
4782
9.85k
GBool Splash::getInNonIsolatedGroup() {
4783
9.85k
  return state->inNonIsolatedGroup;
4784
9.85k
}
4785
4786
6.93k
GBool Splash::getInKnockoutGroup() {
4787
6.93k
  return state->inKnockoutGroup;
4788
6.93k
}
4789
4790
//------------------------------------------------------------------------
4791
// state write
4792
//------------------------------------------------------------------------
4793
4794
1.99M
void Splash::setMatrix(SplashCoord *matrix) {
4795
1.99M
  memcpy(state->matrix, matrix, 6 * sizeof(SplashCoord));
4796
1.99M
}
4797
4798
941k
void Splash::setStrokePattern(SplashPattern *strokePattern) {
4799
941k
  state->setStrokePattern(strokePattern);
4800
941k
}
4801
4802
1.94M
void Splash::setFillPattern(SplashPattern *fillPattern) {
4803
1.94M
  state->setFillPattern(fillPattern);
4804
1.94M
}
4805
4806
0
void Splash::setScreen(SplashScreen *screen) {
4807
0
  state->setScreen(screen);
4808
0
}
4809
4810
4.33k
void Splash::setBlendFunc(SplashBlendFunc func) {
4811
4.33k
  state->blendFunc = func;
4812
4.33k
}
4813
4814
5.63k
void Splash::setStrokeAlpha(SplashCoord alpha) {
4815
5.63k
  state->strokeAlpha = alpha;
4816
5.63k
}
4817
4818
5.79k
void Splash::setFillAlpha(SplashCoord alpha) {
4819
5.79k
  state->fillAlpha = alpha;
4820
5.79k
}
4821
4822
536k
void Splash::setLineWidth(SplashCoord lineWidth) {
4823
536k
  state->lineWidth = lineWidth;
4824
536k
}
4825
4826
809k
void Splash::setLineCap(int lineCap) {
4827
809k
  if (lineCap >= 0 && lineCap <= 2) {
4828
809k
    state->lineCap = lineCap;
4829
809k
  } else {
4830
0
    state->lineCap = 0;
4831
0
  }
4832
809k
}
4833
4834
785k
void Splash::setLineJoin(int lineJoin) {
4835
785k
  if (lineJoin >= 0 && lineJoin <= 2) {
4836
785k
    state->lineJoin = lineJoin;
4837
785k
  } else {
4838
0
    state->lineJoin = 0;
4839
0
  }
4840
785k
}
4841
4842
791k
void Splash::setMiterLimit(SplashCoord miterLimit) {
4843
791k
  state->miterLimit = miterLimit;
4844
791k
}
4845
4846
384k
void Splash::setFlatness(SplashCoord flatness) {
4847
384k
  if (flatness < 1) {
4848
0
    state->flatness = 1;
4849
384k
  } else {
4850
384k
    state->flatness = flatness;
4851
384k
  }
4852
384k
}
4853
4854
void Splash::setLineDash(SplashCoord *lineDash, int lineDashLength,
4855
824k
       SplashCoord lineDashPhase) {
4856
824k
  state->setLineDash(lineDash, lineDashLength, lineDashPhase);
4857
824k
}
4858
4859
3.07M
void Splash::setStrokeAdjust(SplashStrokeAdjustMode strokeAdjust) {
4860
3.07M
  state->strokeAdjust = strokeAdjust;
4861
3.07M
}
4862
4863
3.14k
void Splash::setAlphaIsShape(GBool alphaIsShape) {
4864
3.14k
  state->alphaIsShape = alphaIsShape;
4865
3.14k
}
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
203k
SplashError Splash::clipToPath(SplashPath *path, GBool eo) {
4878
203k
  return state->clipToPath(path, eo);
4879
203k
}
4880
4881
9.64k
void Splash::setSoftMask(SplashBitmap *softMask, GBool deleteBitmap) {
4882
9.64k
  state->setSoftMask(softMask, deleteBitmap);
4883
9.64k
}
4884
4885
void Splash::setInTransparencyGroup(Splash *parentA,
4886
            int parentOffsetXA, int parentOffsetYA,
4887
            SplashAlphaBitmap *alpha0BitmapA,
4888
            SplashAlphaBitmap *shapeBitmapA,
4889
4.40k
            GBool nonIsolated, GBool knockout) {
4890
4.40k
  parent = parentA;
4891
4.40k
  parentOffsetX = parentOffsetXA;
4892
4.40k
  parentOffsetY = parentOffsetYA;
4893
4.40k
  deferredInit = gTrue;
4894
4.40k
  deferredInitYMin = 1;
4895
4.40k
  deferredInitYMax = 0;
4896
4.40k
  alpha0Bitmap = alpha0BitmapA;
4897
4.40k
  shapeBitmap = shapeBitmapA;
4898
4.40k
  state->inNonIsolatedGroup = nonIsolated;
4899
4.40k
  state->inKnockoutGroup = knockout;
4900
4.40k
}
4901
4902
1.28k
void Splash::forceDeferredInit(int y, int h) {
4903
1.28k
  useBitmapRow(y);
4904
1.28k
  useBitmapRow(y + h - 1);
4905
1.28k
}
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
6.83k
GBool Splash::checkTransparentRect(int x, int y, int w, int h) {
4911
6.83k
  if (state->inNonIsolatedGroup) {
4912
0
    return gFalse;
4913
0
  }
4914
4915
4916
6.83k
  if (!bitmap->alpha) {
4917
0
    return gFalse;
4918
0
  }
4919
6.83k
  int yy0, yy1;
4920
6.83k
  if (deferredInit) {
4921
    // all transparency groups initialize alpha to zero, so anything
4922
    // outside of groupDestInit[YMin,YMax] will have alpha=0
4923
18
    yy0 = (y > deferredInitYMin) ? y : deferredInitYMin;
4924
18
    yy1 = (y + h - 1 < deferredInitYMax) ? y + h - 1 : deferredInitYMax;
4925
6.81k
  } else {
4926
6.81k
    yy0 = y;
4927
6.81k
    yy1 = y + h - 1;
4928
6.81k
  }
4929
6.83k
  Guchar *alphaP = &bitmap->alpha[yy0 * bitmap->alphaRowSize + x];
4930
10.9k
  for (int yy = yy0; yy <= yy1; ++yy) {
4931
10.8k
    for (int xx = 0; xx < w; ++xx) {
4932
6.81k
      if (alphaP[xx] != 0) {
4933
2.73k
  return gFalse;
4934
2.73k
      }
4935
6.81k
    }
4936
4.08k
    alphaP += bitmap->getAlphaRowSize();
4937
4.08k
  }
4938
4.09k
  return gTrue;
4939
6.83k
}
4940
4941
void Splash::setTransfer(Guchar *red, Guchar *green, Guchar *blue,
4942
529
       Guchar *gray) {
4943
529
  state->setTransfer(red, green, blue, gray);
4944
529
}
4945
4946
784k
void Splash::setOverprintMask(Guint overprintMask) {
4947
784k
  state->overprintMask = overprintMask;
4948
784k
}
4949
4950
4951
404k
void Splash::setEnablePathSimplification(GBool en) {
4952
404k
  state->enablePathSimplification = en;
4953
404k
}
4954
4955
//------------------------------------------------------------------------
4956
// state save/restore
4957
//------------------------------------------------------------------------
4958
4959
2.01M
void Splash::saveState() {
4960
2.01M
  SplashState *newState;
4961
4962
2.01M
  newState = state->copy();
4963
2.01M
  newState->next = state;
4964
2.01M
  state = newState;
4965
2.01M
}
4966
4967
2.12M
SplashError Splash::restoreState() {
4968
2.12M
  SplashState *oldState;
4969
4970
2.12M
  if (!state->next) {
4971
108k
    return splashErrNoSave;
4972
108k
  }
4973
2.01M
  oldState = state;
4974
2.01M
  state = state->next;
4975
2.01M
  delete oldState;
4976
2.01M
  return splashOk;
4977
2.12M
}
4978
4979
//------------------------------------------------------------------------
4980
// drawing operations
4981
//------------------------------------------------------------------------
4982
4983
397k
void Splash::clear(SplashColorPtr color, Guchar alpha) {
4984
397k
  SplashColorPtr row, p;
4985
397k
  Guchar mono;
4986
397k
  int x, y;
4987
4988
397k
  switch (bitmap->mode) {
4989
128
  case splashModeMono1:
4990
128
    mono = (color[0] & 0x80) ? 0xff : 0x00;
4991
128
    if (bitmap->rowSize < 0) {
4992
0
      memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
4993
0
       mono, -bitmap->rowSize * bitmap->height);
4994
128
    } else {
4995
128
      memset(bitmap->data, mono, bitmap->rowSize * bitmap->height);
4996
128
    }
4997
128
    break;
4998
140
  case splashModeMono8:
4999
140
    if (bitmap->rowSize < 0) {
5000
0
      memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
5001
0
       color[0], -bitmap->rowSize * bitmap->height);
5002
140
    } else {
5003
140
      memset(bitmap->data, color[0], bitmap->rowSize * bitmap->height);
5004
140
    }
5005
140
    break;
5006
397k
  case splashModeRGB8:
5007
397k
    if (color[0] == color[1] && color[1] == color[2]) {
5008
397k
      if (bitmap->rowSize < 0) {
5009
0
  memset(bitmap->data + bitmap->rowSize * (bitmap->height - 1),
5010
0
         color[0], -bitmap->rowSize * bitmap->height);
5011
397k
      } else {
5012
397k
  memset(bitmap->data, color[0], bitmap->rowSize * bitmap->height);
5013
397k
      }
5014
397k
    } 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
397k
    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
397k
#endif
5072
397k
  }
5073
5074
397k
  if (bitmap->alpha) {
5075
397k
    memset(bitmap->alpha, alpha, bitmap->alphaRowSize * bitmap->height);
5076
397k
  }
5077
5078
397k
  updateModX(0);
5079
397k
  updateModY(0);
5080
397k
  updateModX(bitmap->width - 1);
5081
397k
  updateModY(bitmap->height - 1);
5082
397k
}
5083
5084
111k
SplashError Splash::stroke(SplashPath *path) {
5085
111k
  SplashPath *path2, *dPath;
5086
111k
  SplashCoord w, w2, lineDashMax, lineDashTotal;
5087
111k
  int lineCap, lineJoin, i;
5088
5089
111k
  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
111k
  opClipRes = splashClipAllOutside;
5095
111k
  if (path->length == 0) {
5096
1.41k
    return splashErrEmptyPath;
5097
1.41k
  }
5098
109k
  if (pathAllOutside(path, gTrue)) {
5099
92.0k
    return splashOk;
5100
92.0k
  }
5101
17.7k
  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
17.7k
  double t0 = splashAbs(state->matrix[0]);
5113
17.7k
  double t1 = splashAbs(state->matrix[1]);
5114
17.7k
  double t2 = splashAbs(state->matrix[2]);
5115
17.7k
  double t3 = splashAbs(state->matrix[3]);
5116
17.7k
  double t01 = t0 * t0 + t1 * t1;
5117
17.7k
  double t23 = t2 * t2 + t3 * t3;
5118
17.7k
  w = sqrt((t01 > t23) ? t01 : t23);
5119
17.7k
  w2 = w * state->lineWidth;
5120
5121
  // construct the dashed path
5122
17.7k
  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
351
    lineDashTotal = 0;
5134
351
    lineDashMax = 0;
5135
2.95k
    for (i = 0; i < state->lineDashLength; ++i) {
5136
2.60k
      lineDashTotal += state->lineDash[i];
5137
2.60k
      if (state->lineDash[i] > lineDashMax) {
5138
521
  lineDashMax = state->lineDash[i];
5139
521
      }
5140
2.60k
    }
5141
    // Acrobat simply draws nothing if the dash array is [0]
5142
351
    if (lineDashTotal == 0) {
5143
15
      delete path2;
5144
15
      return splashOk;
5145
15
    }
5146
336
    if (w * lineDashMax > 0.1 &&
5147
326
  w * lineDashMax > 0.0001 * path->length) {
5148
326
      dPath = makeDashedPath(path2);
5149
326
      delete path2;
5150
326
      path2 = dPath;
5151
326
      if (path2->length == 0) {
5152
5
  delete path2;
5153
5
  return splashErrEmptyPath;
5154
5
      }
5155
326
    }
5156
336
  }
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
17.7k
  lineCap = state->lineCap;
5163
17.7k
  lineJoin = state->lineJoin;
5164
17.7k
  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
17.7k
  if (w > 0 && w2 < minLineWidth) {
5179
285
    strokeWide(path2, minLineWidth / w, splashLineCapButt, splashLineJoinBevel);
5180
17.4k
  } 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
17.4k
  } else {
5190
    // in gray and color modes, only use 0-width lines if the line
5191
    // width is explicitly set to 0
5192
17.4k
    if (state->lineWidth == 0) {
5193
1.19k
      strokeNarrow(path2);
5194
16.2k
    } else {
5195
16.2k
      strokeWide(path2, state->lineWidth, lineCap, lineJoin);
5196
16.2k
    }
5197
17.4k
  }
5198
5199
17.7k
  delete path2;
5200
17.7k
  return splashOk;
5201
17.7k
}
5202
5203
1.19k
void Splash::strokeNarrow(SplashPath *path) {
5204
1.19k
  SplashPipe pipe;
5205
1.19k
  SplashXPath *xPath;
5206
1.19k
  SplashXPathSeg *seg;
5207
1.19k
  int x0, x1, y0, y1, xa, xb, y;
5208
1.19k
  SplashCoord dxdy;
5209
1.19k
  SplashClipResult clipRes;
5210
1.19k
  int nClipRes[3];
5211
1.19k
  int i;
5212
5213
1.19k
  nClipRes[0] = nClipRes[1] = nClipRes[2] = 0;
5214
5215
1.19k
  xPath = new SplashXPath(path, state->matrix, state->flatness, gFalse,
5216
1.19k
        state->enablePathSimplification,
5217
1.19k
        state->strokeAdjust, state->clip);
5218
5219
1.19k
  pipeInit(&pipe, state->strokePattern,
5220
1.19k
     (Guchar)splashRound(state->strokeAlpha * 255),
5221
1.19k
     gFalse, gTrue, gFalse);
5222
5223
1.42M
  for (i = 0, seg = xPath->segs; i < xPath->length; ++i, ++seg) {
5224
1.42M
    if (seg->y0 <= seg->y1) {
5225
1.37M
      y0 = splashFloor(seg->y0);
5226
1.37M
      y1 = splashFloor(seg->y1);
5227
1.37M
      x0 = splashFloor(seg->x0);
5228
1.37M
      x1 = splashFloor(seg->x1);
5229
1.37M
    } else {
5230
42.9k
      y0 = splashFloor(seg->y1);
5231
42.9k
      y1 = splashFloor(seg->y0);
5232
42.9k
      x0 = splashFloor(seg->x1);
5233
42.9k
      x1 = splashFloor(seg->x0);
5234
42.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
1.42M
    if (y0 == y1 &&
5246
1.41M
  seg->y0 == seg->y1 &&
5247
1.37M
  state->clip->getIsSimple() &&
5248
838k
  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
1.42M
    } else if (x0 == x1 &&
5259
1.41M
  seg->x0 == seg->x1 &&
5260
1.41M
  state->clip->getIsSimple() &&
5261
876k
  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
1.42M
    if ((clipRes = state->clip->testRect(x0 <= x1 ? x0 : x1, y0,
5274
1.42M
           x0 <= x1 ? x1 : x0, y1,
5275
1.42M
           state->strokeAdjust))
5276
1.42M
  != splashClipAllOutside) {
5277
3.15k
      if (y0 == y1) {
5278
1.11k
  if (x0 <= x1) {
5279
983
    drawStrokeSpan(&pipe, x0, x1, y0, clipRes == splashClipAllInside);
5280
983
  } else {
5281
130
    drawStrokeSpan(&pipe, x1, x0, y0, clipRes == splashClipAllInside);
5282
130
  }
5283
2.04k
      } else if (x0 == x1) {
5284
642
  y = state->clip->getYMinI(state->strokeAdjust);
5285
642
  if (y0 < y) {
5286
547
    y0 = y;
5287
547
  }
5288
642
  y = state->clip->getYMaxI(state->strokeAdjust);
5289
642
  if (y1 > y) {
5290
642
    y1 = y;
5291
642
  }
5292
1.28k
  for (y = y0; y <= y1; ++y) {
5293
642
    drawStrokeSpan(&pipe, x0, x0, y, clipRes == splashClipAllInside);
5294
642
  }
5295
1.40k
      } else {
5296
1.40k
  dxdy = seg->dxdy;
5297
1.40k
  y = state->clip->getYMinI(state->strokeAdjust);
5298
1.40k
  if (y0 < y) {
5299
1.22k
    y0 = y;
5300
1.22k
    x0 = splashFloor(seg->x0 + ((SplashCoord)y0 - seg->y0) * dxdy);
5301
1.22k
  }
5302
1.40k
  y = state->clip->getYMaxI(state->strokeAdjust);
5303
1.40k
  if (y1 > y) {
5304
1.26k
    y1 = y;
5305
1.26k
    x1 = splashFloor(seg->x0 + ((SplashCoord)y1 + 1 - seg->y0) * dxdy);
5306
1.26k
  }
5307
1.40k
  if (x0 <= x1) {
5308
620
    xa = x0;
5309
1.24k
    for (y = y0; y <= y1; ++y) {
5310
620
      if (y < y1) {
5311
0
        xb = splashFloor(seg->x0 +
5312
0
             ((SplashCoord)y + 1 - seg->y0) * dxdy);
5313
620
      } else {
5314
620
        xb = x1 + 1;
5315
620
      }
5316
620
      if (xa == xb) {
5317
0
        drawStrokeSpan(&pipe, xa, xa, y, clipRes == splashClipAllInside);
5318
620
      } else {
5319
620
        drawStrokeSpan(&pipe, xa, xb - 1, y,
5320
620
           clipRes == splashClipAllInside);
5321
620
      }
5322
620
      xa = xb;
5323
620
    }
5324
782
  } else {
5325
782
    xa = x0;
5326
1.56k
    for (y = y0; y <= y1; ++y) {
5327
782
      if (y < y1) {
5328
0
        xb = splashFloor(seg->x0 +
5329
0
             ((SplashCoord)y + 1 - seg->y0) * dxdy);
5330
782
      } else {
5331
782
        xb = x1 - 1;
5332
782
      }
5333
782
      if (xa == xb) {
5334
0
        drawStrokeSpan(&pipe, xa, xa, y, clipRes == splashClipAllInside);
5335
782
      } else {
5336
782
        drawStrokeSpan(&pipe, xb + 1, xa, y,
5337
782
           clipRes == splashClipAllInside);
5338
782
      }
5339
782
      xa = xb;
5340
782
    }
5341
782
  }
5342
1.40k
      }
5343
3.15k
    }
5344
1.42M
    ++nClipRes[clipRes];
5345
1.42M
  }
5346
5347
1.19k
  if (nClipRes[splashClipPartial] ||
5348
704
      (nClipRes[splashClipAllInside] && nClipRes[splashClipAllOutside])) {
5349
704
    opClipRes = splashClipPartial;
5350
704
  } else if (nClipRes[splashClipAllInside]) {
5351
206
    opClipRes = splashClipAllInside;
5352
280
  } else {
5353
280
    opClipRes = splashClipAllOutside;
5354
280
  }
5355
5356
1.19k
  delete xPath;
5357
1.19k
}
5358
5359
void Splash::drawStrokeSpan(SplashPipe *pipe, int x0, int x1, int y,
5360
3.15k
          GBool noClip) {
5361
3.15k
  int x;
5362
5363
3.15k
  x = state->clip->getXMinI(state->strokeAdjust);
5364
3.15k
  if (x > x0) {
5365
890
    x0 = x;
5366
890
  }
5367
3.15k
  x = state->clip->getXMaxI(state->strokeAdjust);
5368
3.15k
  if (x < x1) {
5369
779
    x1 = x;
5370
779
  }
5371
3.15k
  if (x0 > x1) {
5372
188
    return;
5373
188
  }
5374
5.93k
  for (x = x0; x <= x1; ++x) {
5375
2.96k
    scanBuf[x] = 0xff;
5376
2.96k
  }
5377
2.96k
  if (!noClip) {
5378
2.26k
    if (!state->clip->clipSpanBinary(scanBuf, y, x0, x1, state->strokeAdjust)) {
5379
123
      return;
5380
123
    }
5381
2.26k
  }
5382
2.84k
  (this->*pipe->run)(pipe, x0, x1, y, scanBuf + x0, scanBuf + x0, NULL);
5383
2.84k
}
5384
5385
void Splash::strokeWide(SplashPath *path, SplashCoord w,
5386
16.5k
      int lineCap, int lineJoin) {
5387
16.5k
  SplashPath *path2;
5388
5389
16.5k
  path2 = makeStrokePath(path, w, lineCap, lineJoin, gFalse);
5390
16.5k
  fillWithPattern(path2, gFalse, state->strokePattern, state->strokeAlpha);
5391
16.5k
  delete path2;
5392
16.5k
}
5393
5394
SplashPath *Splash::flattenPath(SplashPath *path, SplashCoord *matrix,
5395
17.8k
        SplashCoord flatness) {
5396
17.8k
  SplashPath *fPath;
5397
17.8k
  SplashCoord flatness2;
5398
17.8k
  Guchar flag;
5399
17.8k
  int i;
5400
5401
17.8k
  fPath = new SplashPath();
5402
#if USE_FIXEDPOINT
5403
  flatness2 = flatness;
5404
#else
5405
17.8k
  flatness2 = flatness * flatness;
5406
17.8k
#endif
5407
17.8k
  i = 0;
5408
124k
  while (i < path->length) {
5409
107k
    flag = path->flags[i];
5410
107k
    if (flag & splashPathFirst) {
5411
27.9k
      fPath->moveTo(path->pts[i].x, path->pts[i].y);
5412
27.9k
      ++i;
5413
79.0k
    } else {
5414
79.0k
      if (flag & splashPathCurve) {
5415
8.39k
  flattenCurve(path->pts[i-1].x, path->pts[i-1].y,
5416
8.39k
         path->pts[i  ].x, path->pts[i  ].y,
5417
8.39k
         path->pts[i+1].x, path->pts[i+1].y,
5418
8.39k
         path->pts[i+2].x, path->pts[i+2].y,
5419
8.39k
         matrix, flatness2, fPath);
5420
8.39k
  i += 3;
5421
70.6k
      } else {
5422
70.6k
  fPath->lineTo(path->pts[i].x, path->pts[i].y);
5423
70.6k
  ++i;
5424
70.6k
      }
5425
79.0k
      if (path->flags[i-1] & splashPathClosed) {
5426
17.5k
  fPath->close();
5427
17.5k
      }
5428
79.0k
    }
5429
107k
  }
5430
17.8k
  return fPath;
5431
17.8k
}
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
8.39k
        SplashPath *fPath) {
5439
8.39k
  SplashCoord cx[splashMaxCurveSplits + 1][3];
5440
8.39k
  SplashCoord cy[splashMaxCurveSplits + 1][3];
5441
8.39k
  int cNext[splashMaxCurveSplits + 1];
5442
8.39k
  SplashCoord xl0, xl1, xl2, xr0, xr1, xr2, xr3, xx1, xx2, xh;
5443
8.39k
  SplashCoord yl0, yl1, yl2, yr0, yr1, yr2, yr3, yy1, yy2, yh;
5444
8.39k
  SplashCoord dx, dy, mx, my, tx, ty, d1, d2;
5445
8.39k
  int p1, p2, p3;
5446
5447
  // initial segment
5448
8.39k
  p1 = 0;
5449
8.39k
  p2 = splashMaxCurveSplits;
5450
8.39k
  cx[p1][0] = x0;  cy[p1][0] = y0;
5451
8.39k
  cx[p1][1] = x1;  cy[p1][1] = y1;
5452
8.39k
  cx[p1][2] = x2;  cy[p1][2] = y2;
5453
8.39k
  cx[p2][0] = x3;  cy[p2][0] = y3;
5454
8.39k
  cNext[p1] = p2;
5455
5456
14.5M
  while (p1 < splashMaxCurveSplits) {
5457
5458
    // get the next segment
5459
14.5M
    xl0 = cx[p1][0];  yl0 = cy[p1][0];
5460
14.5M
    xx1 = cx[p1][1];  yy1 = cy[p1][1];
5461
14.5M
    xx2 = cx[p1][2];  yy2 = cy[p1][2];
5462
14.5M
    p2 = cNext[p1];
5463
14.5M
    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
14.5M
    transform(matrix, (xl0 + xr3) * 0.5, (yl0 + yr3) * 0.5, &mx, &my);
5470
14.5M
    transform(matrix, xx1, yy1, &tx, &ty);
5471
#if USE_FIXEDPOINT
5472
    d1 = splashDist(tx, ty, mx, my);
5473
#else
5474
14.5M
    dx = tx - mx;
5475
14.5M
    dy = ty - my;
5476
14.5M
    d1 = dx*dx + dy*dy;
5477
14.5M
#endif
5478
14.5M
    transform(matrix, xx2, yy2, &tx, &ty);
5479
#if USE_FIXEDPOINT
5480
    d2 = splashDist(tx, ty, mx, my);
5481
#else
5482
14.5M
    dx = tx - mx;
5483
14.5M
    dy = ty - my;
5484
14.5M
    d2 = dx*dx + dy*dy;
5485
14.5M
#endif
5486
5487
    // if the curve is flat enough, or no more subdivisions are
5488
    // allowed, add the straight line segment
5489
14.5M
    if (p2 - p1 == 1 || (d1 <= flatness2 && d2 <= flatness2)) {
5490
7.26M
      fPath->lineTo(xr3, yr3);
5491
7.26M
      p1 = p2;
5492
5493
    // otherwise, subdivide the curve
5494
7.26M
    } else {
5495
7.26M
      xl1 = splashAvg(xl0, xx1);
5496
7.26M
      yl1 = splashAvg(yl0, yy1);
5497
7.26M
      xh = splashAvg(xx1, xx2);
5498
7.26M
      yh = splashAvg(yy1, yy2);
5499
7.26M
      xl2 = splashAvg(xl1, xh);
5500
7.26M
      yl2 = splashAvg(yl1, yh);
5501
7.26M
      xr2 = splashAvg(xx2, xr3);
5502
7.26M
      yr2 = splashAvg(yy2, yr3);
5503
7.26M
      xr1 = splashAvg(xh, xr2);
5504
7.26M
      yr1 = splashAvg(yh, yr2);
5505
7.26M
      xr0 = splashAvg(xl2, xr1);
5506
7.26M
      yr0 = splashAvg(yl2, yr1);
5507
      // add the new subdivision points
5508
7.26M
      p3 = (p1 + p2) / 2;
5509
7.26M
      cx[p1][1] = xl1;  cy[p1][1] = yl1;
5510
7.26M
      cx[p1][2] = xl2;  cy[p1][2] = yl2;
5511
7.26M
      cNext[p1] = p3;
5512
7.26M
      cx[p3][0] = xr0;  cy[p3][0] = yr0;
5513
7.26M
      cx[p3][1] = xr1;  cy[p3][1] = yr1;
5514
7.26M
      cx[p3][2] = xr2;  cy[p3][2] = yr2;
5515
7.26M
      cNext[p3] = p2;
5516
7.26M
    }
5517
14.5M
  }
5518
8.39k
}
5519
5520
326
SplashPath *Splash::makeDashedPath(SplashPath *path) {
5521
326
  SplashPath *dPath;
5522
326
  SplashCoord lineDashTotal;
5523
326
  SplashCoord lineDashStartPhase, lineDashDist, segLen;
5524
326
  SplashCoord x0, y0, x1, y1, xa, ya;
5525
326
  GBool lineDashStartOn, lineDashEndOn, lineDashOn, newPath;
5526
326
  int lineDashStartIdx, lineDashIdx, subpathStart, nDashes;
5527
326
  int i, j, k;
5528
5529
326
  lineDashTotal = 0;
5530
2.85k
  for (i = 0; i < state->lineDashLength; ++i) {
5531
2.53k
    lineDashTotal += state->lineDash[i];
5532
2.53k
  }
5533
  // Acrobat simply draws nothing if the dash array is [0]
5534
326
  if (lineDashTotal == 0) {
5535
0
    return new SplashPath();
5536
0
  }
5537
326
  lineDashStartPhase = state->lineDashPhase;
5538
326
  if (lineDashStartPhase > 0) {
5539
40
    i = splashFloor(lineDashStartPhase / lineDashTotal);
5540
40
    lineDashStartPhase -= lineDashTotal * i;
5541
286
  } else {
5542
286
    i = splashCeil(-lineDashStartPhase / lineDashTotal);
5543
286
    lineDashStartPhase += lineDashTotal * i;
5544
286
  }
5545
326
  lineDashStartOn = !((state->lineDashLength & 1) && (i & 1));
5546
326
  lineDashStartIdx = 0;
5547
326
  if (lineDashStartPhase > 0) {
5548
130
    while (lineDashStartPhase >= state->lineDash[lineDashStartIdx]) {
5549
90
      lineDashStartOn = !lineDashStartOn;
5550
90
      lineDashStartPhase -= state->lineDash[lineDashStartIdx];
5551
90
      if (++lineDashStartIdx == state->lineDashLength) {
5552
0
  lineDashStartIdx = 0;
5553
0
      }
5554
90
    }
5555
40
  }
5556
5557
326
  dPath = new SplashPath();
5558
5559
  // process each subpath
5560
326
  i = 0;
5561
928
  while (i < path->length) {
5562
5563
    // find the end of the subpath
5564
602
    for (j = i;
5565
42.2k
   j < path->length - 1 && !(path->flags[j] & splashPathLast);
5566
41.6k
   ++j) ;
5567
5568
    // initialize the dash parameters
5569
602
    lineDashOn = lineDashStartOn;
5570
602
    lineDashEndOn = lineDashStartOn;
5571
602
    lineDashIdx = lineDashStartIdx;
5572
602
    lineDashDist = state->lineDash[lineDashIdx] - lineDashStartPhase;
5573
602
    subpathStart = dPath->length;
5574
602
    nDashes = 0;
5575
5576
    // process each segment of the subpath
5577
602
    newPath = gTrue;
5578
42.2k
    for (k = i; k < j; ++k) {
5579
5580
      // grab the segment
5581
41.6k
      x0 = path->pts[k].x;
5582
41.6k
      y0 = path->pts[k].y;
5583
41.6k
      x1 = path->pts[k+1].x;
5584
41.6k
      y1 = path->pts[k+1].y;
5585
41.6k
      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
41.6k
      if (j == i+1 && segLen == 0) {
5591
31
  dPath->moveTo(x0, y0);
5592
31
  dPath->lineTo(x0, y0);
5593
31
      }
5594
5595
      // process the segment
5596
6.36M
      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
6.32M
  if (lineDashDist == 0) {
5606
915k
    if (lineDashOn) {
5607
398k
      if (newPath) {
5608
398k
        dPath->moveTo(x0, y0);
5609
398k
        newPath = gFalse;
5610
398k
        ++nDashes;
5611
398k
      }
5612
398k
      xa = x0 + ((SplashCoord)0.001 / segLen) * (x1 - x0);
5613
398k
      ya = y0 + ((SplashCoord)0.001 / segLen) * (y1 - y0);
5614
398k
      dPath->lineTo(xa, ya);
5615
398k
    }
5616
5617
5.41M
  } else if (lineDashDist >= segLen) {
5618
41.5k
    if (lineDashOn) {
5619
26.5k
      if (newPath) {
5620
4.37k
        dPath->moveTo(x0, y0);
5621
4.37k
        newPath = gFalse;
5622
4.37k
        ++nDashes;
5623
4.37k
      }
5624
26.5k
      dPath->lineTo(x1, y1);
5625
26.5k
    }
5626
41.5k
    lineDashDist -= segLen;
5627
41.5k
    segLen = 0;
5628
5629
5.37M
  } else {
5630
5.37M
    xa = x0 + (lineDashDist / segLen) * (x1 - x0);
5631
5.37M
    ya = y0 + (lineDashDist / segLen) * (y1 - y0);
5632
5.37M
    if (lineDashOn) {
5633
2.74M
      if (newPath) {
5634
2.74M
        dPath->moveTo(x0, y0);
5635
2.74M
        newPath = gFalse;
5636
2.74M
        ++nDashes;
5637
2.74M
      }
5638
2.74M
      dPath->lineTo(xa, ya);
5639
2.74M
    }
5640
5.37M
    x0 = xa;
5641
5.37M
    y0 = ya;
5642
5.37M
    segLen -= lineDashDist;
5643
5.37M
    lineDashDist = 0;
5644
5.37M
  }
5645
5646
6.32M
  lineDashEndOn = lineDashOn;
5647
5648
  // get the next entry in the dash array
5649
6.32M
  if (lineDashDist <= 0) {
5650
6.28M
    lineDashOn = !lineDashOn;
5651
6.28M
    if (++lineDashIdx == state->lineDashLength) {
5652
1.85M
      lineDashIdx = 0;
5653
1.85M
    }
5654
6.28M
    lineDashDist = state->lineDash[lineDashIdx];
5655
6.28M
    newPath = gTrue;
5656
6.28M
  }
5657
6.32M
      }
5658
41.6k
    }
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
602
    if ((path->flags[j] & splashPathClosed) &&
5664
143
  lineDashStartOn &&
5665
142
  lineDashEndOn) {
5666
116
      if (nDashes == 1) {
5667
39
  dPath->close();
5668
77
      } else if (nDashes > 1) {
5669
46
  k = subpathStart;
5670
57
  do {
5671
57
    ++k;
5672
57
    dPath->lineTo(dPath->pts[k].x, dPath->pts[k].y);
5673
57
  } while (!(dPath->flags[k] & splashPathLast));
5674
46
  ++k;
5675
46
  memmove(&dPath->pts[subpathStart], &dPath->pts[k],
5676
46
    (dPath->length - k) * sizeof(SplashPathPoint));
5677
46
  memmove(&dPath->flags[subpathStart], &dPath->flags[k],
5678
46
    (dPath->length - k) * sizeof(Guchar));
5679
46
  dPath->length -= k - subpathStart;
5680
46
  dPath->curSubpath -= k - subpathStart;
5681
46
      }
5682
116
    }
5683
5684
602
    i = j + 1;
5685
602
  }
5686
5687
326
  return dPath;
5688
326
}
5689
5690
1.00M
SplashError Splash::fill(SplashPath *path, GBool eo) {
5691
1.00M
  if (debugMode) {
5692
0
    printf("fill [eo:%d]:\n", eo);
5693
0
    dumpPath(path);
5694
0
  }
5695
1.00M
  if (path->length == 0) {
5696
2.41k
    return splashErrEmptyPath;
5697
2.41k
  }
5698
1.00M
  if (pathAllOutside(path, gFalse)) {
5699
970k
    opClipRes = splashClipAllOutside;
5700
970k
    return splashOk;
5701
970k
  }
5702
30.4k
  return fillWithPattern(path, eo, state->fillPattern, state->fillAlpha);
5703
1.00M
}
5704
5705
SplashError Splash::fillWithPattern(SplashPath *path, GBool eo,
5706
            SplashPattern *pattern,
5707
46.9k
            SplashCoord alpha) {
5708
46.9k
  SplashPipe pipe;
5709
46.9k
  SplashPath *path2;
5710
46.9k
  SplashXPath *xPath;
5711
46.9k
  SplashXPathScanner *scanner;
5712
46.9k
  int xMin, yMin, xMax, xMin2, xMax2, yMax, y, t;
5713
46.9k
  SplashClipResult clipRes;
5714
5715
46.9k
  path2 = tweakFillPath(path);
5716
5717
46.9k
  xPath = new SplashXPath(path2, state->matrix, state->flatness, gTrue,
5718
46.9k
        state->enablePathSimplification,
5719
46.9k
        state->strokeAdjust, state->clip);
5720
46.9k
  if (path2 != path) {
5721
1.37k
    delete path2;
5722
1.37k
  }
5723
46.9k
  xMin = xPath->getXMin();
5724
46.9k
  yMin = xPath->getYMin();
5725
46.9k
  xMax = xPath->getXMax();
5726
46.9k
  yMax = xPath->getYMax();
5727
46.9k
  if (xMin > xMax || yMin > yMax) {
5728
25
    delete xPath;
5729
25
    return splashOk;
5730
25
  }
5731
46.9k
  scanner = new SplashXPathScanner(xPath, eo, yMin, yMax);
5732
5733
  // check clipping
5734
46.9k
  if ((clipRes = state->clip->testRect(xMin, yMin, xMax, yMax,
5735
46.9k
               state->strokeAdjust))
5736
46.9k
      != splashClipAllOutside) {
5737
5738
26.7k
    if ((t = state->clip->getXMinI(state->strokeAdjust)) > xMin) {
5739
7.48k
      xMin = t;
5740
7.48k
    }
5741
26.7k
    if ((t = state->clip->getXMaxI(state->strokeAdjust)) < xMax) {
5742
11.0k
      xMax = t;
5743
11.0k
    }
5744
26.7k
    if ((t = state->clip->getYMinI(state->strokeAdjust)) > yMin) {
5745
18.8k
      yMin = t;
5746
18.8k
    }
5747
26.7k
    if ((t = state->clip->getYMaxI(state->strokeAdjust)) < yMax) {
5748
19.3k
      yMax = t;
5749
19.3k
    }
5750
26.7k
    if (xMin > xMax || yMin > yMax) {
5751
0
      delete scanner;
5752
0
      delete xPath;
5753
0
      return splashOk;
5754
0
    }
5755
5756
26.7k
    pipeInit(&pipe, pattern, (Guchar)splashRound(alpha * 255),
5757
26.7k
       gFalse, gTrue, gFalse);
5758
5759
    // draw the spans
5760
26.7k
    if (vectorAntialias) {
5761
45.5k
      for (y = yMin; y <= yMax; ++y) {
5762
22.7k
  scanner->getSpan(scanBuf, y, xMin, xMax, &xMin2, &xMax2);
5763
22.7k
  if (xMin2 <= xMax2) {
5764
18.2k
    if (clipRes != splashClipAllInside) {
5765
14.6k
      state->clip->clipSpan(scanBuf, y, xMin2, xMax2,
5766
14.6k
          state->strokeAdjust);
5767
14.6k
    }
5768
18.2k
    (this->*pipe.run)(&pipe, xMin2, xMax2, y,
5769
18.2k
          NULL, scanBuf + xMin2, NULL);
5770
18.2k
  }
5771
22.7k
      }
5772
22.7k
    } else {
5773
7.86k
      for (y = yMin; y <= yMax; ++y) {
5774
3.93k
  scanner->getSpanBinary(scanBuf, y, xMin, xMax, &xMin2, &xMax2);
5775
3.93k
  if (xMin2 <= xMax2) {
5776
3.93k
    if (clipRes != splashClipAllInside) {
5777
3.93k
      state->clip->clipSpanBinary(scanBuf, y, xMin2, xMax2,
5778
3.93k
          state->strokeAdjust);
5779
3.93k
    }
5780
3.93k
    (this->*pipe.run)(&pipe, xMin2, xMax2, y,
5781
3.93k
          NULL, scanBuf + xMin2, NULL);
5782
3.93k
  }
5783
3.93k
      }
5784
3.93k
    }
5785
26.7k
  }
5786
46.9k
  opClipRes = clipRes;
5787
5788
46.9k
  delete scanner;
5789
46.9k
  delete xPath;
5790
46.9k
  return splashOk;
5791
46.9k
}
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
46.9k
SplashPath *Splash::tweakFillPath(SplashPath *path) {
5805
46.9k
  SplashPath *path2;
5806
46.9k
  SplashCoord xx0, yy0, xx1, yy1, dx, dy, d, wx, wy, w;
5807
46.9k
  int n;
5808
5809
46.9k
  if (state->strokeAdjust == splashStrokeAdjustOff || path->hints) {
5810
20.7k
    return path;
5811
20.7k
  }
5812
5813
26.2k
  n = path->getLength();
5814
26.2k
  if (!((n == 2) ||
5815
25.9k
  (n == 3 &&
5816
844
   path->flags[1] == 0) ||
5817
25.0k
  (n == 4 &&
5818
1.26k
   path->flags[1] == 0 &&
5819
882
   path->flags[2] == 0) ||
5820
24.2k
  (n == 5 &&
5821
17.1k
   path->flags[1] == 0 &&
5822
17.0k
   path->flags[2] == 0 &&
5823
16.8k
   path->flags[3] == 0))) {
5824
7.31k
    return path;
5825
7.31k
  }
5826
5827
18.9k
  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
18.9k
  if (n == 2 ||
5832
18.6k
      (n == 3 && (path->flags[0] & splashPathClosed)) ||
5833
18.1k
      (n == 3 && (splashAbs(path->pts[0].x - path->pts[2].x) < 0.001 &&
5834
149
      splashAbs(path->pts[0].y - path->pts[2].y) < 0.001)) ||
5835
18.0k
      ((n == 4 ||
5836
17.2k
  (n == 5 && (path->flags[0] & splashPathClosed))) &&
5837
17.7k
       ((splashAbs(path->pts[0].x - path->pts[1].x) < 0.001 &&
5838
634
   splashAbs(path->pts[0].y - path->pts[1].y) < 0.001 &&
5839
620
   splashAbs(path->pts[2].x - path->pts[3].x) < 0.001 &&
5840
524
   splashAbs(path->pts[2].y - path->pts[3].y) < 0.001) ||
5841
17.2k
  (splashAbs(path->pts[0].x - path->pts[3].x) < 0.001 &&
5842
16.3k
   splashAbs(path->pts[0].y - path->pts[3].y) < 0.001 &&
5843
249
   splashAbs(path->pts[1].x - path->pts[2].x) < 0.001 &&
5844
1.37k
   splashAbs(path->pts[1].y - path->pts[2].y) < 0.001)))) {
5845
1.37k
    wx = state->matrix[0] + state->matrix[2];
5846
1.37k
    wy = state->matrix[1] + state->matrix[3];
5847
1.37k
    w = splashSqrt(wx*wx + wy*wy);
5848
1.37k
    if (w < 0.001) {
5849
428
      w = 0;
5850
949
    } else {
5851
      // min width is 0.1 -- this constant is minWidth * sqrt(2)
5852
949
      w = (SplashCoord)0.1414 / w;
5853
949
    }
5854
1.37k
    xx0 = path->pts[0].x;
5855
1.37k
    yy0 = path->pts[0].y;
5856
1.37k
    if (n <= 3) {
5857
808
      xx1 = path->pts[1].x;
5858
808
      yy1 = path->pts[1].y;
5859
808
    } else {
5860
569
      xx1 = path->pts[2].x;
5861
569
      yy1 = path->pts[2].y;
5862
569
    }
5863
1.37k
    dx = xx1 - xx0;
5864
1.37k
    dy = yy1 - yy0;
5865
1.37k
    d = splashSqrt(dx * dx + dy * dy);
5866
1.37k
    if (d < 0.001) {
5867
329
      d = 0;
5868
1.04k
    } else {
5869
1.04k
      d = w / d;
5870
1.04k
    }
5871
1.37k
    dx *= d;
5872
1.37k
    dy *= d;
5873
1.37k
    path2 = new SplashPath();
5874
1.37k
    path2->moveTo(xx0 + dy, yy0 - dx);
5875
1.37k
    path2->lineTo(xx1 + dy, yy1 - dx);
5876
1.37k
    path2->lineTo(xx1 - dy, yy1 + dx);
5877
1.37k
    path2->lineTo(xx0 - dy, yy0 + dx);
5878
1.37k
    path2->close(gTrue);
5879
1.37k
    path2->addStrokeAdjustHint(0, 2, 0, 4);
5880
1.37k
    path2->addStrokeAdjustHint(1, 3, 0, 4);
5881
5882
  // unclosed rectangle --> close and hint
5883
17.5k
  } else if (n == 4 && !(path->flags[0] & splashPathClosed)) {
5884
437
    path2->close(gTrue);
5885
437
    path2->addStrokeAdjustHint(0, 2, 0, 4);
5886
437
    path2->addStrokeAdjustHint(1, 3, 0, 4);
5887
5888
  // closed rectangle --> hint
5889
17.0k
  } else if (n == 5 && (path->flags[0] & splashPathClosed)) {
5890
16.5k
    path2->addStrokeAdjustHint(0, 2, 0, 4);
5891
16.5k
    path2->addStrokeAdjustHint(1, 3, 0, 4);
5892
16.5k
  }
5893
5894
18.9k
  return path2;
5895
26.2k
}
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.11M
GBool Splash::pathAllOutside(SplashPath *path, GBool stroke) {
5902
1.11M
  SplashCoord xMin1, yMin1, xMax1, yMax1;
5903
1.11M
  SplashCoord xMin2, yMin2, xMax2, yMax2;
5904
1.11M
  SplashCoord x, y;
5905
1.11M
  int i;
5906
5907
  //--- compute the path's bbox in user space
5908
1.11M
  xMin1 = xMax1 = path->pts[0].x;
5909
1.11M
  yMin1 = yMax1 = path->pts[0].y;
5910
12.8M
  for (i = 1; i < path->length; ++i) {
5911
11.6M
    if (path->pts[i].x < xMin1) {
5912
2.20M
      xMin1 = path->pts[i].x;
5913
9.48M
    } else if (path->pts[i].x > xMax1) {
5914
1.76M
      xMax1 = path->pts[i].x;
5915
1.76M
    }
5916
11.6M
    if (path->pts[i].y < yMin1) {
5917
2.93M
      yMin1 = path->pts[i].y;
5918
8.76M
    } else if (path->pts[i].y > yMax1) {
5919
2.76M
      yMax1 = path->pts[i].y;
5920
2.76M
    }
5921
11.6M
  }
5922
5923
  //--- allow for stroke width and miter limit
5924
1.11M
  if (stroke && state->lineWidth > 0) {
5925
107k
    SplashCoord w = state->lineWidth * 0.5;
5926
107k
    if (state->lineJoin == splashLineJoinMiter) {
5927
105k
      w *= state->miterLimit;
5928
105k
    }
5929
107k
    xMin1 -= w;
5930
107k
    yMin1 -= w;
5931
107k
    xMax1 += w;
5932
107k
    yMax1 += w;
5933
107k
  }
5934
5935
  //--- convert path bbox to device space
5936
1.11M
  transform(state->matrix, xMin1, yMin1, &x, &y);
5937
1.11M
  xMin2 = xMax2 = x;
5938
1.11M
  yMin2 = yMax2 = y;
5939
1.11M
  transform(state->matrix, xMin1, yMax1, &x, &y);
5940
1.11M
  if (x < xMin2) {
5941
7.28k
    xMin2 = x;
5942
1.10M
  } else if (x > xMax2) {
5943
4.96k
    xMax2 = x;
5944
4.96k
  }
5945
1.11M
  if (y < yMin2) {
5946
32.4k
    yMin2 = y;
5947
1.07M
  } else if (y > yMax2) {
5948
838k
    yMax2 = y;
5949
838k
  }
5950
1.11M
  transform(state->matrix, xMax1, yMin1, &x, &y);
5951
1.11M
  if (x < xMin2) {
5952
121k
    xMin2 = x;
5953
989k
  } else if (x > xMax2) {
5954
516k
    xMax2 = x;
5955
516k
  }
5956
1.11M
  if (y < yMin2) {
5957
2.26k
    yMin2 = y;
5958
1.10M
  } else if (y > yMax2) {
5959
1.97k
    yMax2 = y;
5960
1.97k
  }
5961
1.11M
  transform(state->matrix, xMax1, yMax1, &x, &y);
5962
1.11M
  if (x < xMin2) {
5963
5.20k
    xMin2 = x;
5964
1.10M
  } else if (x > xMax2) {
5965
2.77k
    xMax2 = x;
5966
2.77k
  }
5967
1.11M
  if (y < yMin2) {
5968
3.69k
    yMin2 = y;
5969
1.10M
  } else if (y > yMax2) {
5970
3.14k
    yMax2 = y;
5971
3.14k
  }
5972
5973
  //--- handle zero-width strokes
5974
1.11M
  if (stroke && state->lineWidth == 0) {
5975
1.83k
    xMin1 -= 1;
5976
1.83k
    yMin1 -= 1;
5977
1.83k
    xMax1 += 1;
5978
1.83k
    yMax1 += 1;
5979
1.83k
  }
5980
5981
  //--- check against the clip rect
5982
1.11M
  return xMin2 > state->clip->getXMax() ||
5983
603k
         xMax2 < state->clip->getXMin() ||
5984
455k
         yMin2 > state->clip->getYMax() ||
5985
144k
         yMax2 < state->clip->getYMin();
5986
1.11M
}
5987
5988
SplashError Splash::fillChar(SplashCoord x, SplashCoord y,
5989
488k
           Guint c, SplashFont *font) {
5990
488k
  SplashGlyphBitmap glyph;
5991
488k
  SplashCoord xt, yt;
5992
488k
  int x0, y0, xFrac, yFrac;
5993
488k
  SplashError err;
5994
5995
488k
  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
488k
  transform(state->matrix, x, y, &xt, &yt);
6000
488k
  x0 = splashFloor(xt);
6001
488k
  xFrac = splashFloor((xt - x0) * splashFontFraction);
6002
488k
  y0 = splashFloor(yt);
6003
488k
  yFrac = splashFloor((yt - y0) * splashFontFraction);
6004
488k
  if (!font->getGlyph(c, xFrac, yFrac, &glyph)) {
6005
468k
    return splashErrNoGlyph;
6006
468k
  }
6007
20.0k
  err = fillGlyph2(x0, y0, &glyph);
6008
20.0k
  if (glyph.freeData) {
6009
11.1k
    gfree(glyph.data);
6010
11.1k
  }
6011
20.0k
  return err;
6012
488k
}
6013
6014
SplashError Splash::fillGlyph(SplashCoord x, SplashCoord y,
6015
1.38k
            SplashGlyphBitmap *glyph) {
6016
1.38k
  SplashCoord xt, yt;
6017
1.38k
  int x0, y0;
6018
6019
1.38k
  transform(state->matrix, x, y, &xt, &yt);
6020
1.38k
  x0 = splashFloor(xt);
6021
1.38k
  y0 = splashFloor(yt);
6022
1.38k
  return fillGlyph2(x0, y0, glyph);
6023
1.38k
}
6024
6025
21.4k
SplashError Splash::fillGlyph2(int x0, int y0, SplashGlyphBitmap *glyph) {
6026
21.4k
  SplashPipe pipe;
6027
21.4k
  SplashClipResult clipRes;
6028
21.4k
  Guchar alpha;
6029
21.4k
  Guchar *p;
6030
21.4k
  int xMin, yMin, xMax, yMax;
6031
21.4k
  int x, y, xg, yg, xx, t;
6032
6033
21.4k
  xg = x0 - glyph->x;
6034
21.4k
  yg = y0 - glyph->y;
6035
21.4k
  xMin = xg;
6036
21.4k
  xMax = xg + glyph->w - 1;
6037
21.4k
  yMin = yg;
6038
21.4k
  yMax = yg + glyph->h - 1;
6039
21.4k
  if ((clipRes = state->clip->testRect(xMin, yMin, xMax, yMax,
6040
21.4k
               state->strokeAdjust))
6041
21.4k
      != splashClipAllOutside) {
6042
22
    pipeInit(&pipe, state->fillPattern,
6043
22
       (Guchar)splashRound(state->fillAlpha * 255),
6044
22
       gFalse, gTrue, gFalse);
6045
22
    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
22
    } else {
6069
22
      if ((t = state->clip->getXMinI(state->strokeAdjust)) > xMin) {
6070
22
  xMin = t;
6071
22
      }
6072
22
      if ((t = state->clip->getXMaxI(state->strokeAdjust)) < xMax) {
6073
22
  xMax = t;
6074
22
      }
6075
22
      if ((t = state->clip->getYMinI(state->strokeAdjust)) > yMin) {
6076
22
  yMin = t;
6077
22
      }
6078
22
      if ((t = state->clip->getYMaxI(state->strokeAdjust)) < yMax) {
6079
22
  yMax = t;
6080
22
      }
6081
22
      if (xMin <= xMax && yMin <= yMax) {
6082
22
  if (glyph->aa) {
6083
44
    for (y = yMin; y <= yMax; ++y) {
6084
22
      p = glyph->data + (y - yg) * glyph->w + (xMin - xg);
6085
22
      memcpy(scanBuf + xMin, p, xMax - xMin + 1);
6086
22
      state->clip->clipSpan(scanBuf, y, xMin, xMax,
6087
22
          state->strokeAdjust);
6088
22
      (this->*pipe.run)(&pipe, xMin, xMax, y,
6089
22
            NULL, scanBuf + xMin, NULL);
6090
22
    }
6091
22
  } 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
22
      }
6116
22
    }
6117
22
  }
6118
21.4k
  opClipRes = clipRes;
6119
6120
21.4k
  return splashOk;
6121
21.4k
}
6122
6123
void Splash::getImageBounds(SplashCoord xyMin, SplashCoord xyMax,
6124
83.7k
          int *xyMinI, int *xyMaxI) {
6125
83.7k
  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
24.8k
    if (xyMin < -1e9) {
6140
12.4k
      xyMin = -1e9;
6141
12.4k
    } else if (xyMin > 1e9) {
6142
3.58k
      xyMin = 1e9;
6143
3.58k
    }
6144
24.8k
    if (xyMax < -1e9) {
6145
11.4k
      xyMax = -1e9;
6146
13.3k
    } else if (xyMax > 1e9) {
6147
7.79k
      xyMax = 1e9;
6148
7.79k
    }
6149
24.8k
#endif
6150
24.8k
    *xyMinI = splashFloor(xyMin);
6151
24.8k
    *xyMaxI = splashFloor(xyMax);
6152
24.8k
    if (*xyMaxI <= *xyMinI) {
6153
19.6k
      *xyMaxI = *xyMinI + 1;
6154
19.6k
    }
6155
58.9k
  } else {
6156
58.9k
    splashStrokeAdjust(xyMin, xyMax, xyMinI, xyMaxI, state->strokeAdjust);
6157
58.9k
  }
6158
83.7k
}
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
39.6k
          GBool antialias) {
6171
39.6k
  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
39.6k
  if (!splashCheckDet(mat[0], mat[1], mat[2], mat[3], 0.000001)) {
6179
2.71k
    return splashErrSingularMatrix;
6180
2.71k
  }
6181
6182
  //--- compute image bbox, check clipping
6183
36.9k
  GBool flipsOnly = splashAbs(mat[1]) <= 0.0001 && splashAbs(mat[2]) <= 0.0001;
6184
36.9k
  GBool rot90Only = splashAbs(mat[0]) <= 0.0001 && splashAbs(mat[3]) <= 0.0001;
6185
36.9k
  GBool horizFlip = gFalse;
6186
36.9k
  GBool vertFlip = gFalse;
6187
36.9k
  int xMin, yMin, xMax, yMax;
6188
36.9k
  if (flipsOnly) {
6189
19.7k
    horizFlip = mat[0] < 0;
6190
19.7k
    vertFlip = mat[3] < 0;
6191
19.7k
    if (horizFlip) {
6192
16.5k
      getImageBounds(mat[0] + mat[4], mat[4], &xMin, &xMax);
6193
16.5k
    } else {
6194
3.21k
      getImageBounds(mat[4], mat[0] + mat[4], &xMin, &xMax);
6195
3.21k
    }
6196
19.7k
    if (vertFlip) {
6197
10.1k
      getImageBounds(mat[3] + mat[5], mat[5], &yMin, &yMax);
6198
10.1k
    } else {
6199
9.60k
      getImageBounds(mat[5], mat[3] + mat[5], &yMin, &yMax);
6200
9.60k
    }
6201
19.7k
  } else if (rot90Only) {
6202
6.22k
    horizFlip = mat[2] < 0;
6203
6.22k
    vertFlip = mat[1] < 0;
6204
6.22k
    if (horizFlip) {
6205
530
      getImageBounds(mat[2] + mat[4], mat[4], &xMin, &xMax);
6206
5.69k
    } else {
6207
5.69k
      getImageBounds(mat[4], mat[2] + mat[4], &xMin, &xMax);
6208
5.69k
    }
6209
6.22k
    if (vertFlip) {
6210
1.87k
      getImageBounds(mat[1] + mat[5], mat[5], &yMin, &yMax);
6211
4.35k
    } else {
6212
4.35k
      getImageBounds(mat[5], mat[1] + mat[5], &yMin, &yMax);
6213
4.35k
    }
6214
10.9k
  } else {
6215
10.9k
    int xx = splashRound(mat[4]);   // (0,0)
6216
10.9k
    int yy = splashRound(mat[5]);
6217
10.9k
    xMin = xMax = xx;
6218
10.9k
    yMin = yMax = yy;
6219
10.9k
    xx = splashRound(mat[0] + mat[4]);    // (1,0)
6220
10.9k
    yy = splashRound(mat[1] + mat[5]);
6221
10.9k
    if (xx < xMin) {
6222
300
      xMin = xx;
6223
10.6k
    } else if (xx > xMax) {
6224
824
      xMax = xx;
6225
824
    }
6226
10.9k
    if (yy < yMin) {
6227
224
      yMin = yy;
6228
10.7k
    } else if (yy > yMax) {
6229
303
      yMax = yy;
6230
303
    }
6231
10.9k
    xx = splashRound(mat[2] + mat[4]);    // (0,1)
6232
10.9k
    yy = splashRound(mat[3] + mat[5]);
6233
10.9k
    if (xx < xMin) {
6234
420
      xMin = xx;
6235
10.5k
    } else if (xx > xMax) {
6236
252
      xMax = xx;
6237
252
    }
6238
10.9k
    if (yy < yMin) {
6239
134
      yMin = yy;
6240
10.8k
    } else if (yy > yMax) {
6241
217
      yMax = yy;
6242
217
    }
6243
10.9k
    xx = splashRound(mat[0] + mat[2] + mat[4]); // (1,1)
6244
10.9k
    yy = splashRound(mat[1] + mat[3] + mat[5]);
6245
10.9k
    if (xx < xMin) {
6246
519
      xMin = xx;
6247
10.4k
    } else if (xx > xMax) {
6248
1.57k
      xMax = xx;
6249
1.57k
    }
6250
10.9k
    if (yy < yMin) {
6251
780
      yMin = yy;
6252
10.2k
    } else if (yy > yMax) {
6253
167
      yMax = yy;
6254
167
    }
6255
10.9k
    if (xMax <= xMin) {
6256
7.75k
      xMax = xMin + 1;
6257
7.75k
    }
6258
10.9k
    if (yMax <= yMin) {
6259
9.44k
      yMax = yMin + 1;
6260
9.44k
    }
6261
10.9k
  }
6262
36.9k
  SplashClipResult clipRes =
6263
36.9k
      state->clip->testRect(xMin, yMin, xMax - 1, yMax - 1,
6264
36.9k
          state->strokeAdjust);
6265
36.9k
  if (clipRes == splashClipAllOutside) {
6266
36.1k
    return splashOk;
6267
36.1k
  }
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
798
  int clipW = state->clip->getXMaxI(state->strokeAdjust)
6274
798
        - state->clip->getXMinI(state->strokeAdjust);
6275
798
  int clipH = state->clip->getYMaxI(state->strokeAdjust)
6276
798
        - state->clip->getYMinI(state->strokeAdjust);
6277
798
  GBool veryLarge = ((xMax - xMin) / 8 > clipW && xMax - xMin > 1000) ||
6278
192
                    ((yMax - yMin) / 8 > clipH && yMax - yMin > 1000);
6279
6280
  //--- set up the SplashDrawImageMaskRowData object and the pipes
6281
798
  SplashDrawImageMaskRowData dd;
6282
798
  pipeInit(&dd.pipe, state->fillPattern,
6283
798
     (Guchar)splashRound(state->fillAlpha * 255),
6284
798
     gFalse, gTrue, gFalse);
6285
6286
  //--- choose the drawRow function
6287
798
  SplashDrawImageMaskRowFunc drawRowFunc;
6288
798
  if (clipRes == splashClipAllInside) {
6289
128
    drawRowFunc = &Splash::drawImageMaskRowNoClip;
6290
670
  } else {
6291
670
    if (antialias) {
6292
670
      drawRowFunc = &Splash::drawImageMaskRowClipAA;
6293
670
    } else {
6294
0
      drawRowFunc = &Splash::drawImageMaskRowClipNoAA;
6295
0
    }
6296
670
  }
6297
6298
  //--- horizontal/vertical flips only
6299
798
  if (flipsOnly && !veryLarge) {
6300
128
    int scaledWidth = xMax - xMin;
6301
128
    int scaledHeight = yMax - yMin;
6302
128
    ImageMaskScaler scaler(src, srcData, w, h,
6303
128
         scaledWidth, scaledHeight, interpolate, antialias);
6304
128
    Guchar *tmpLine = NULL;
6305
128
    if (horizFlip) {
6306
0
      tmpLine = (Guchar *)gmalloc(scaledWidth);
6307
0
    }
6308
128
    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
128
    } else {
6324
128
      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
128
      } else {            // top-down
6332
37.6k
  for (int y = 0; y < scaledHeight; ++y) {
6333
37.5k
    scaler.nextLine();
6334
37.5k
    (this->*drawRowFunc)(&dd, scaler.data(),
6335
37.5k
             xMin, yMin + y, scaledWidth);
6336
37.5k
  }
6337
128
      }
6338
128
    }
6339
128
    gfree(tmpLine);
6340
6341
  //--- 90/270 rotation
6342
670
  } 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
670
  } else {
6385
    // estimate of size of scaled image
6386
670
    int scaledWidth = splashRound(splashSqrt(mat[0] * mat[0]
6387
670
               + mat[1] * mat[1]));
6388
670
    int scaledHeight = splashRound(splashSqrt(mat[2] * mat[2]
6389
670
                + mat[3] * mat[3]));
6390
670
    if (scaledWidth < 1) {
6391
648
      scaledWidth = 1;
6392
648
    }
6393
670
    if (scaledHeight < 1) {
6394
584
      scaledHeight = 1;
6395
584
    }
6396
670
    GBool downscaling = gTrue;
6397
670
    if (scaledWidth >= w && scaledHeight >= h) {
6398
10
      downscaling = gFalse;
6399
10
      scaledWidth = w;
6400
10
      scaledHeight = h;
6401
10
    }
6402
6403
    // compute mapping from device space to scaled image space
6404
670
    SplashCoord mat1[6];
6405
670
    mat1[0] = mat[0] / scaledWidth;
6406
670
    mat1[1] = mat[1] / scaledWidth;
6407
670
    mat1[2] = mat[2] / scaledHeight;
6408
670
    mat1[3] = mat[3] / scaledHeight;
6409
670
    mat1[4] = mat[4];
6410
670
    mat1[5] = mat[5];
6411
670
    SplashCoord det = mat1[0] * mat1[3] - mat1[1] * mat1[2];
6412
670
    if (splashAbs(det) < 1e-6) {
6413
      // this should be caught by the singular matrix check in drawImage
6414
0
      return splashErrSingularMatrix;
6415
0
    }
6416
670
    SplashCoord invMat[6];
6417
670
    invMat[0] = mat1[3] / det;
6418
670
    invMat[1] = -mat1[1] / det;
6419
670
    invMat[2] = -mat1[2] / det;
6420
670
    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
670
    invMat[4] = (mat1[2] * mat1[5] - mat1[3] * mat1[4]) / det
6425
670
                + (invMat[0] + invMat[2]) * 0.5;
6426
670
    invMat[5] = (mat1[1] * mat1[4] - mat1[0] * mat1[5]) / det
6427
670
                + (invMat[1] + invMat[3]) * 0.5;
6428
6429
    // if downscaling: store the downscaled image mask
6430
    // if upscaling: store the unscaled image mask
6431
670
    Guchar *scaledMask = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
6432
670
    if (downscaling) {
6433
660
      ImageMaskScaler scaler(src, srcData, w, h,
6434
660
           scaledWidth, scaledHeight, interpolate, antialias);
6435
660
      Guchar *ptr = scaledMask;
6436
496k
      for (int y = 0; y < scaledHeight; ++y) {
6437
495k
  scaler.nextLine();
6438
495k
  memcpy(ptr, scaler.data(), scaledWidth);
6439
495k
  ptr += scaledWidth;
6440
495k
      }
6441
660
    } else {
6442
10
      Guchar *ptr = scaledMask;
6443
1.06k
      for (int y = 0; y < scaledHeight; ++y) {
6444
1.05k
  (*src)(srcData, ptr);
6445
2.10k
  for (int x = 0; x < scaledWidth; ++x) {
6446
1.05k
    *ptr = (Guchar)(*ptr * 255);
6447
1.05k
    ++ptr;
6448
1.05k
  }
6449
1.05k
      }
6450
10
    }
6451
6452
    // draw it
6453
670
    if (interpolate && antialias) {
6454
23
      drawImageMaskArbitraryInterp(scaledMask,
6455
23
           &dd, drawRowFunc, invMat,
6456
23
           scaledWidth, scaledHeight,
6457
23
           xMin, yMin, xMax, yMax);
6458
647
    } else {
6459
647
      drawImageMaskArbitraryNoInterp(scaledMask,
6460
647
             &dd, drawRowFunc, invMat,
6461
647
             scaledWidth, scaledHeight,
6462
647
             xMin, yMin, xMax, yMax);
6463
647
    }
6464
6465
    // free the downscaled/unscaled image
6466
670
    gfree(scaledMask);
6467
670
  }
6468
6469
798
  return splashOk;
6470
798
}
6471
6472
void Splash::drawImageMaskArbitraryNoInterp(
6473
           Guchar *scaledMask,
6474
           SplashDrawImageMaskRowData *dd,
6475
           SplashDrawImageMaskRowFunc drawRowFunc,
6476
           SplashCoord *invMat,
6477
           int scaledWidth, int scaledHeight,
6478
647
           int xMin, int yMin, int xMax, int yMax) {
6479
647
  int tt = state->clip->getXMinI(state->strokeAdjust);
6480
647
  if (tt > xMin) {
6481
61
    xMin = tt;
6482
61
  }
6483
647
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
6484
647
  if (tt < xMax) {
6485
642
    xMax = tt;
6486
642
  }
6487
647
  tt = state->clip->getYMinI(state->strokeAdjust);
6488
647
  if (tt > yMin) {
6489
174
    yMin = tt;
6490
174
  }
6491
647
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
6492
647
  if (tt < yMax) {
6493
642
    yMax = tt;
6494
642
  }
6495
647
  if (xMax <= xMin || yMax <= yMin) {
6496
0
    return;
6497
0
  }
6498
6499
647
  Guchar *buf = (Guchar *)gmalloc(xMax - xMin);
6500
6501
1.82k
  for (int y = yMin; y < yMax; ++y) {
6502
1.17k
    int rowMin = xMax;
6503
1.17k
    int rowMax = 0;
6504
51.3k
    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
50.1k
      int xx = splashFloor((SplashCoord)x * invMat[0]
6508
50.1k
         + (SplashCoord)y * invMat[2] + invMat[4]);
6509
50.1k
      int yy = splashFloor((SplashCoord)x * invMat[1]
6510
50.1k
         + (SplashCoord)y * invMat[3] + invMat[5]);
6511
50.1k
      if (xx >= 0 && xx < scaledWidth &&
6512
49.9k
    yy >= 0 && yy < scaledHeight) {
6513
49.8k
  Guchar *p = scaledMask + (yy * (SplashBitmapRowSize)scaledWidth + xx);
6514
49.8k
  Guchar *q = buf + (x - xMin);
6515
49.8k
  *q = *p;
6516
49.8k
  if (x < rowMin) {
6517
884
    rowMin = x;
6518
884
  }
6519
49.8k
  rowMax = x + 1;
6520
49.8k
      }
6521
50.1k
    }
6522
1.17k
    if (rowMin < rowMax) {
6523
884
      (this->*drawRowFunc)(dd, buf + (rowMin - xMin),
6524
884
         rowMin, y, rowMax - rowMin);
6525
884
    }
6526
1.17k
  }
6527
6528
647
  gfree(buf);
6529
647
}
6530
6531
void Splash::drawImageMaskArbitraryInterp(
6532
           Guchar *scaledMask,
6533
           SplashDrawImageMaskRowData *dd,
6534
           SplashDrawImageMaskRowFunc drawRowFunc,
6535
           SplashCoord *invMat,
6536
           int scaledWidth, int scaledHeight,
6537
23
           int xMin, int yMin, int xMax, int yMax) {
6538
23
  int tt = state->clip->getXMinI(state->strokeAdjust);
6539
23
  if (tt > xMin) {
6540
11
    xMin = tt;
6541
11
  }
6542
23
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
6543
23
  if (tt < xMax) {
6544
20
    xMax = tt;
6545
20
  }
6546
23
  tt = state->clip->getYMinI(state->strokeAdjust);
6547
23
  if (tt > yMin) {
6548
12
    yMin = tt;
6549
12
  }
6550
23
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
6551
23
  if (tt < yMax) {
6552
22
    yMax = tt;
6553
22
  }
6554
23
  if (xMax <= xMin || yMax <= yMin) {
6555
0
    return;
6556
0
  }
6557
6558
23
  Guchar *buf = (Guchar *)gmalloc(xMax - xMin);
6559
6560
53
  for (int y = yMin; y < yMax; ++y) {
6561
30
    int rowMin = xMax;
6562
30
    int rowMax = 0;
6563
70
    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
40
      SplashCoord xs = (SplashCoord)x * invMat[0]
6567
40
                 + (SplashCoord)y * invMat[2] + invMat[4];
6568
40
      SplashCoord ys = (SplashCoord)x * invMat[1]
6569
40
                 + (SplashCoord)y * invMat[3] + invMat[5];
6570
40
      int x0 = splashFloor(xs - 0.5);
6571
40
      int x1 = x0 + 1;
6572
40
      int y0 = splashFloor(ys - 0.5);
6573
40
      int y1 = y0 + 1;
6574
40
      if (x1 >= 0 && x0 < scaledWidth && y1 >= 0 && y0 < scaledHeight) {
6575
40
  SplashCoord sx0 = (SplashCoord)x1 + 0.5 - xs;
6576
40
  SplashCoord sx1 = (SplashCoord)1 - sx0;
6577
40
  SplashCoord sy0 = (SplashCoord)y1 + 0.5 - ys;
6578
40
  SplashCoord sy1 = (SplashCoord)1 - sy0;
6579
40
  if (x0 < 0) {
6580
20
    x0 = 0;
6581
20
  }
6582
40
  if (x1 >= scaledWidth) {
6583
20
    x1 = scaledWidth - 1;
6584
20
  }
6585
40
  if (y0 < 0) {
6586
3
    y0 = 0;
6587
3
  }
6588
40
  if (y1 >= scaledHeight) {
6589
37
    y1 = scaledHeight - 1;
6590
37
  }
6591
40
  Guchar *p00 = scaledMask + (y0 * (SplashBitmapRowSize)scaledWidth + x0);
6592
40
  Guchar *p10 = scaledMask + (y0 * (SplashBitmapRowSize)scaledWidth + x1);
6593
40
  Guchar *p01 = scaledMask + (y1 * (SplashBitmapRowSize)scaledWidth + x0);
6594
40
  Guchar *p11 = scaledMask + (y1 * (SplashBitmapRowSize)scaledWidth + x1);
6595
40
  Guchar *q = buf + (x - xMin);
6596
40
  *q = (Guchar)(int)(sx0 * (sy0 * (int)*p00 + sy1 * (int)*p01) +
6597
40
         sx1 * (sy0 * (int)*p10 + sy1 * (int)*p11));
6598
40
  if (x < rowMin) {
6599
30
    rowMin = x;
6600
30
  }
6601
40
  rowMax = x + 1;
6602
40
      }
6603
40
    }
6604
30
    if (rowMin < rowMax) {
6605
30
      (this->*drawRowFunc)(dd, buf + (rowMin - xMin),
6606
30
         rowMin, y, rowMax - rowMin);
6607
30
    }
6608
30
  }
6609
6610
23
  gfree(buf);
6611
23
}
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
37.5k
            int x, int y, int width) {
6628
37.5k
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
6629
37.5k
        NULL, maskData, NULL);
6630
37.5k
}
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
914
            int x, int y, int width) {
6659
914
  if (y < 0 || y >= bitmap->height) {
6660
0
    return;
6661
0
  }
6662
914
  if (x < 0) {
6663
0
    maskData -= x;
6664
0
    width += x;
6665
0
    x = 0;
6666
0
  }
6667
914
  if (x + width > bitmap->width) {
6668
0
    width = bitmap->width - x;
6669
0
  }
6670
914
  if (width <= 0) {
6671
0
    return;
6672
0
  }
6673
914
  memcpy(scanBuf + x, maskData, width);
6674
914
  state->clip->clipSpan(scanBuf, y, x, x + width - 1, state->strokeAdjust);
6675
914
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
6676
914
        NULL, scanBuf + x, NULL);
6677
914
}
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
21.0k
            GBool interpolate) {
6690
21.0k
  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
21.0k
  GBool ok = gFalse;
6698
21.0k
  int nComps = 0;
6699
21.0k
  switch (bitmap->mode) {
6700
0
  case splashModeMono1:
6701
1.43k
  case splashModeMono8:
6702
1.43k
    ok = srcMode == splashModeMono8;
6703
1.43k
    nComps = 1;
6704
1.43k
    break;
6705
19.4k
  case splashModeRGB8:
6706
19.4k
  case splashModeBGR8:
6707
19.4k
    ok = srcMode == splashModeRGB8;
6708
19.4k
    nComps = 3;
6709
19.4k
    break;
6710
0
#if SPLASH_CMYK
6711
211
  case splashModeCMYK8:
6712
211
    ok = srcMode == splashModeCMYK8;
6713
211
    nComps = 4;
6714
211
    break;
6715
0
#endif
6716
0
  default:
6717
0
    ok = gFalse;
6718
0
    break;
6719
21.0k
  }
6720
21.0k
  if (!ok) {
6721
0
    return splashErrModeMismatch;
6722
0
  }
6723
6724
  //--- check for singular matrix
6725
21.0k
  if (!splashCheckDet(mat[0], mat[1], mat[2], mat[3], 0.000001)) {
6726
875
    return splashErrSingularMatrix;
6727
875
  }
6728
6729
  //--- compute image bbox, check clipping
6730
20.1k
  GBool flipsOnly = splashAbs(mat[1]) <= 0.0001 && splashAbs(mat[2]) <= 0.0001;
6731
20.1k
  GBool rot90Only = splashAbs(mat[0]) <= 0.0001 && splashAbs(mat[3]) <= 0.0001;
6732
20.1k
  GBool horizFlip = gFalse;
6733
20.1k
  GBool vertFlip = gFalse;
6734
20.1k
  int xMin, yMin, xMax, yMax;
6735
20.1k
  if (flipsOnly) {
6736
12.5k
    horizFlip = mat[0] < 0;
6737
12.5k
    vertFlip = mat[3] < 0;
6738
12.5k
    if (horizFlip) {
6739
6.96k
      getImageBounds(mat[0] + mat[4], mat[4], &xMin, &xMax);
6740
6.96k
    } else {
6741
5.61k
      getImageBounds(mat[4], mat[0] + mat[4], &xMin, &xMax);
6742
5.61k
    }
6743
12.5k
    if (vertFlip) {
6744
6.72k
      getImageBounds(mat[3] + mat[5], mat[5], &yMin, &yMax);
6745
6.72k
    } else {
6746
5.85k
      getImageBounds(mat[5], mat[3] + mat[5], &yMin, &yMax);
6747
5.85k
    }
6748
12.5k
  } else if (rot90Only) {
6749
3.33k
    horizFlip = mat[2] < 0;
6750
3.33k
    vertFlip = mat[1] < 0;
6751
3.33k
    if (horizFlip) {
6752
730
      getImageBounds(mat[2] + mat[4], mat[4], &xMin, &xMax);
6753
2.60k
    } else {
6754
2.60k
      getImageBounds(mat[4], mat[2] + mat[4], &xMin, &xMax);
6755
2.60k
    }
6756
3.33k
    if (vertFlip) {
6757
1.64k
      getImageBounds(mat[1] + mat[5], mat[5], &yMin, &yMax);
6758
1.68k
    } else {
6759
1.68k
      getImageBounds(mat[5], mat[1] + mat[5], &yMin, &yMax);
6760
1.68k
    }
6761
4.29k
  } else {
6762
4.29k
    int xx = splashRound(mat[4]);   // (0,0)
6763
4.29k
    int yy = splashRound(mat[5]);
6764
4.29k
    xMin = xMax = xx;
6765
4.29k
    yMin = yMax = yy;
6766
4.29k
    xx = splashRound(mat[0] + mat[4]);    // (1,0)
6767
4.29k
    yy = splashRound(mat[1] + mat[5]);
6768
4.29k
    if (xx < xMin) {
6769
172
      xMin = xx;
6770
4.12k
    } else if (xx > xMax) {
6771
768
      xMax = xx;
6772
768
    }
6773
4.29k
    if (yy < yMin) {
6774
189
      yMin = yy;
6775
4.10k
    } else if (yy > yMax) {
6776
394
      yMax = yy;
6777
394
    }
6778
4.29k
    xx = splashRound(mat[2] + mat[4]);    // (0,1)
6779
4.29k
    yy = splashRound(mat[3] + mat[5]);
6780
4.29k
    if (xx < xMin) {
6781
951
      xMin = xx;
6782
3.34k
    } else if (xx > xMax) {
6783
192
      xMax = xx;
6784
192
    }
6785
4.29k
    if (yy < yMin) {
6786
647
      yMin = yy;
6787
3.64k
    } else if (yy > yMax) {
6788
183
      yMax = yy;
6789
183
    }
6790
4.29k
    xx = splashRound(mat[0] + mat[2] + mat[4]); // (1,1)
6791
4.29k
    yy = splashRound(mat[1] + mat[3] + mat[5]);
6792
4.29k
    if (xx < xMin) {
6793
279
      xMin = xx;
6794
4.01k
    } else if (xx > xMax) {
6795
726
      xMax = xx;
6796
726
    }
6797
4.29k
    if (yy < yMin) {
6798
386
      yMin = yy;
6799
3.91k
    } else if (yy > yMax) {
6800
37
      yMax = yy;
6801
37
    }
6802
4.29k
    if (xMax <= xMin) {
6803
2.08k
      xMax = xMin + 1;
6804
2.08k
    }
6805
4.29k
    if (yMax <= yMin) {
6806
3.00k
      yMax = yMin + 1;
6807
3.00k
    }
6808
4.29k
  }
6809
20.1k
  SplashClipResult clipRes =
6810
20.1k
      state->clip->testRect(xMin, yMin, xMax - 1, yMax - 1,
6811
20.1k
          state->strokeAdjust);
6812
20.1k
  if (clipRes == splashClipAllOutside) {
6813
19.0k
    return splashOk;
6814
19.0k
  }
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.10k
  int clipW = state->clip->getXMaxI(state->strokeAdjust)
6821
1.10k
        - state->clip->getXMinI(state->strokeAdjust);
6822
1.10k
  int clipH = state->clip->getYMaxI(state->strokeAdjust)
6823
1.10k
        - state->clip->getYMinI(state->strokeAdjust);
6824
1.10k
  GBool veryLarge = ((xMax - xMin) / 8 > clipW && xMax - xMin > 1000) ||
6825
169
                    ((yMax - yMin) / 8 > clipH && yMax - yMin > 1000);
6826
6827
  //--- set up the SplashDrawImageRowData object and the pipes
6828
1.10k
  SplashDrawImageRowData dd;
6829
1.10k
  dd.nComps = nComps;
6830
1.10k
  dd.srcAlpha = srcAlpha;
6831
1.10k
  pipeInit(&dd.pipe, NULL,
6832
1.10k
     (Guchar)splashRound(state->fillAlpha * 255),
6833
1.10k
     gFalse, clipRes != splashClipAllInside || srcAlpha,
6834
1.10k
     gFalse, gFalse, state->alphaIsShape);
6835
6836
  //--- choose the drawRow function
6837
1.10k
  SplashDrawImageRowFunc drawRowFunc;
6838
1.10k
  if (clipRes == splashClipAllInside) {
6839
0
    if (srcAlpha) {
6840
0
      drawRowFunc = &Splash::drawImageRowNoClipAlpha;
6841
0
    } else {
6842
0
      drawRowFunc = &Splash::drawImageRowNoClipNoAlpha;
6843
0
    }
6844
1.10k
  } else {
6845
1.10k
    if (srcAlpha) {
6846
22
      if (vectorAntialias) {
6847
22
  drawRowFunc = &Splash::drawImageRowClipAlphaAA;
6848
22
      } else {
6849
0
  drawRowFunc = &Splash::drawImageRowClipAlphaNoAA;
6850
0
      }
6851
1.08k
    } else {
6852
1.08k
      if (vectorAntialias) {
6853
1.08k
  drawRowFunc = &Splash::drawImageRowClipNoAlphaAA;
6854
1.08k
      } else {
6855
0
  drawRowFunc = &Splash::drawImageRowClipNoAlphaNoAA;
6856
0
      }
6857
1.08k
    }
6858
1.10k
  }
6859
6860
  //--- horizontal/vertical flips only
6861
1.10k
  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.10k
  } 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.10k
  } else {
6993
    // estimate of size of scaled image
6994
1.10k
    int scaledWidth = splashRound(splashSqrt(mat[0] * mat[0]
6995
1.10k
               + mat[1] * mat[1]));
6996
1.10k
    int scaledHeight = splashRound(splashSqrt(mat[2] * mat[2]
6997
1.10k
                + mat[3] * mat[3]));
6998
1.10k
    if (scaledWidth < 1) {
6999
1.10k
      scaledWidth = 1;
7000
1.10k
    }
7001
1.10k
    if (scaledHeight < 1) {
7002
1.10k
      scaledHeight = 1;
7003
1.10k
    }
7004
1.10k
    if (scaledWidth >= w && scaledHeight >= h) {
7005
23
      scaledWidth = w;
7006
23
      scaledHeight = h;
7007
23
    }
7008
7009
    // compute mapping from device space to scaled image space
7010
1.10k
    SplashCoord mat1[6];
7011
1.10k
    mat1[0] = mat[0] / scaledWidth;
7012
1.10k
    mat1[1] = mat[1] / scaledWidth;
7013
1.10k
    mat1[2] = mat[2] / scaledHeight;
7014
1.10k
    mat1[3] = mat[3] / scaledHeight;
7015
1.10k
    mat1[4] = mat[4];
7016
1.10k
    mat1[5] = mat[5];
7017
1.10k
    SplashCoord det = mat1[0] * mat1[3] - mat1[1] * mat1[2];
7018
1.10k
    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.10k
    SplashCoord invMat[6];
7023
1.10k
    invMat[0] = mat1[3] / det;
7024
1.10k
    invMat[1] = -mat1[1] / det;
7025
1.10k
    invMat[2] = -mat1[2] / det;
7026
1.10k
    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.10k
    invMat[4] = (mat1[2] * mat1[5] - mat1[3] * mat1[4]) / det
7031
1.10k
                + (invMat[0] + invMat[2]) * 0.5;
7032
1.10k
    invMat[5] = (mat1[1] * mat1[4] - mat1[0] * mat1[5]) / det
7033
1.10k
                + (invMat[1] + invMat[3]) * 0.5;
7034
7035
1.10k
    Guchar *scaledColor, *scaledAlpha;
7036
1.10k
    GBool freeScaledImage;
7037
1.10k
    getScaledImage(imageTag, src, srcData, w, h, nComps,
7038
1.10k
       scaledWidth, scaledHeight, srcMode, srcAlpha, interpolate,
7039
1.10k
       &scaledColor, &scaledAlpha, &freeScaledImage);
7040
7041
    // draw it
7042
1.10k
    if (interpolate) {
7043
78
      drawImageArbitraryInterp(scaledColor, scaledAlpha,
7044
78
             &dd, drawRowFunc, invMat,
7045
78
             scaledWidth, scaledHeight,
7046
78
             xMin, yMin, xMax, yMax,
7047
78
             nComps, srcAlpha);
7048
1.03k
    } else {
7049
1.03k
      drawImageArbitraryNoInterp(scaledColor, scaledAlpha,
7050
1.03k
         &dd, drawRowFunc, invMat,
7051
1.03k
         scaledWidth, scaledHeight,
7052
1.03k
         xMin, yMin, xMax, yMax,
7053
1.03k
         nComps, srcAlpha);
7054
1.03k
    }
7055
7056
    // free the downscaled/unscaled image
7057
1.10k
    if (freeScaledImage) {
7058
0
      gfree(scaledColor);
7059
0
      gfree(scaledAlpha);
7060
0
    }
7061
1.10k
  }
7062
7063
1.10k
  return splashOk;
7064
1.10k
}
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.10k
          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.10k
  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.10k
  } else {
7170
1.10k
    if (!imageCache->match(imageTag, scaledWidth, scaledHeight,
7171
1.10k
        srcMode, srcAlpha, interpolate) ||
7172
955
  !imageCache->colorData) {
7173
955
      imageCache->reset(imageTag, scaledWidth, scaledHeight,
7174
955
      srcMode, srcAlpha, interpolate);
7175
955
      int lineSize;
7176
955
      if (scaledWidth < INT_MAX / nComps) {
7177
955
  lineSize = scaledWidth * nComps;
7178
955
      } else {
7179
0
  lineSize = -1;
7180
0
      }
7181
955
      imageCache->colorData = (Guchar *)gmallocn64(scaledHeight, lineSize);
7182
955
      if (srcAlpha) {
7183
15
  imageCache->alphaData = (Guchar *)gmallocn64(scaledHeight, scaledWidth);
7184
15
      }
7185
955
      if (scaledWidth == w && scaledHeight == h) {
7186
23
  Guchar *colorPtr = imageCache->colorData;
7187
23
  Guchar *alphaPtr = imageCache->alphaData;
7188
46
  for (int y = 0; y < scaledHeight; ++y) {
7189
23
    (*src)(srcData, colorPtr, alphaPtr);
7190
23
    colorPtr += scaledWidth * nComps;
7191
23
    if (srcAlpha) {
7192
0
      alphaPtr += scaledWidth;
7193
0
    }
7194
23
  }
7195
932
      } else {
7196
932
  SavingImageScaler scaler(src, srcData, w, h, nComps, srcAlpha,
7197
932
         scaledWidth, scaledHeight, interpolate,
7198
932
         imageCache->colorData, imageCache->alphaData);
7199
932
  Guchar *colorPtr = imageCache->colorData;
7200
932
  Guchar *alphaPtr = imageCache->alphaData;
7201
1.86k
  for (int y = 0; y < scaledHeight; ++y) {
7202
932
    scaler.nextLine();
7203
932
    memcpy(colorPtr, scaler.colorData(), scaledWidth * nComps);
7204
932
    colorPtr += scaledWidth * nComps;
7205
932
    if (srcAlpha) {
7206
15
      memcpy(alphaPtr, scaler.alphaData(), scaledWidth);
7207
15
      alphaPtr += scaledWidth;
7208
15
    }
7209
932
  }
7210
932
      }
7211
955
    }
7212
1.10k
    *scaledColor = imageCache->colorData;
7213
1.10k
    *scaledAlpha = imageCache->alphaData;
7214
1.10k
    *freeScaledImage = gFalse;
7215
1.10k
  }
7216
1.10k
}
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
1.03k
          int nComps, GBool srcAlpha) {
7226
1.03k
  int tt = state->clip->getXMinI(state->strokeAdjust);
7227
1.03k
  if (tt > xMin) {
7228
132
    xMin = tt;
7229
132
  }
7230
1.03k
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
7231
1.03k
  if (tt < xMax) {
7232
1.02k
    xMax = tt;
7233
1.02k
  }
7234
1.03k
  tt = state->clip->getYMinI(state->strokeAdjust);
7235
1.03k
  if (tt > yMin) {
7236
283
    yMin = tt;
7237
283
  }
7238
1.03k
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
7239
1.03k
  if (tt < yMax) {
7240
1.03k
    yMax = tt;
7241
1.03k
  }
7242
1.03k
  if (xMax <= xMin || yMax <= yMin) {
7243
0
    return;
7244
0
  }
7245
7246
1.03k
  Guchar *colorBuf = (Guchar *)gmallocn(xMax - xMin, nComps);
7247
1.03k
  Guchar *alphaBuf = NULL;
7248
1.03k
  if (srcAlpha) {
7249
22
    alphaBuf = (Guchar *)gmalloc(xMax - xMin);
7250
22
  }
7251
7252
2.06k
  for (int y = yMin; y < yMax; ++y) {
7253
1.03k
    int rowMin = xMax;
7254
1.03k
    int rowMax = 0;
7255
2.06k
    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
1.03k
      int xx = splashFloor((SplashCoord)x * invMat[0]
7259
1.03k
         + (SplashCoord)y * invMat[2] + invMat[4]);
7260
1.03k
      int yy = splashFloor((SplashCoord)x * invMat[1]
7261
1.03k
         + (SplashCoord)y * invMat[3] + invMat[5]);
7262
1.03k
      if (xx >= 0 && xx < scaledWidth &&
7263
768
    yy >= 0 && yy < scaledHeight) {
7264
689
  Guchar *p = scaledColor +
7265
689
                (yy * (SplashBitmapRowSize)scaledWidth + xx) * nComps;
7266
689
  Guchar *q = colorBuf + (x - xMin) * nComps;
7267
2.23k
  for (int i = 0; i < nComps; ++i) {
7268
1.54k
    *q++ = *p++;
7269
1.54k
  }
7270
689
  if (srcAlpha) {
7271
22
    alphaBuf[x - xMin] =
7272
22
        scaledAlpha[yy * (SplashBitmapRowSize)scaledWidth + xx];
7273
22
  }
7274
689
  if (x < rowMin) {
7275
689
    rowMin = x;
7276
689
  }
7277
689
  rowMax = x + 1;
7278
689
      }
7279
1.03k
    }
7280
1.03k
    if (rowMin < rowMax) {
7281
689
      (this->*drawRowFunc)(dd,
7282
689
         colorBuf + (rowMin - xMin) * nComps,
7283
689
         alphaBuf + (rowMin - xMin),
7284
689
         rowMin, y, rowMax - rowMin);
7285
689
    }
7286
1.03k
  }
7287
7288
1.03k
  gfree(colorBuf);
7289
1.03k
  gfree(alphaBuf);
7290
1.03k
}
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
78
              int nComps, GBool srcAlpha) {
7299
78
  int tt = state->clip->getXMinI(state->strokeAdjust);
7300
78
  if (tt > xMin) {
7301
31
    xMin = tt;
7302
31
  }
7303
78
  tt = state->clip->getXMaxI(state->strokeAdjust) + 1;
7304
78
  if (tt < xMax) {
7305
78
    xMax = tt;
7306
78
  }
7307
78
  tt = state->clip->getYMinI(state->strokeAdjust);
7308
78
  if (tt > yMin) {
7309
46
    yMin = tt;
7310
46
  }
7311
78
  tt = state->clip->getYMaxI(state->strokeAdjust) + 1;
7312
78
  if (tt < yMax) {
7313
78
    yMax = tt;
7314
78
  }
7315
78
  if (xMax <= xMin || yMax <= yMin) {
7316
0
    return;
7317
0
  }
7318
7319
78
  Guchar *colorBuf = (Guchar *)gmallocn(xMax - xMin, nComps);
7320
78
  Guchar *alphaBuf = NULL;
7321
78
  if (srcAlpha) {
7322
0
    alphaBuf = (Guchar *)gmalloc(xMax - xMin);
7323
0
  }
7324
7325
156
  for (int y = yMin; y < yMax; ++y) {
7326
78
    int rowMin = xMax;
7327
78
    int rowMax = 0;
7328
156
    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
78
      SplashCoord xs = (SplashCoord)x * invMat[0]
7332
78
                 + (SplashCoord)y * invMat[2] + invMat[4];
7333
78
      SplashCoord ys = (SplashCoord)x * invMat[1]
7334
78
                 + (SplashCoord)y * invMat[3] + invMat[5];
7335
78
      int x0 = splashFloor(xs - 0.5);
7336
78
      int x1 = x0 + 1;
7337
78
      int y0 = splashFloor(ys - 0.5);
7338
78
      int y1 = y0 + 1;
7339
78
      if (x1 >= 0 && x0 < scaledWidth && y1 >= 0 && y0 < scaledHeight) {
7340
78
  SplashCoord sx0 = (SplashCoord)x1 + 0.5 - xs;
7341
78
  SplashCoord sx1 = (SplashCoord)1 - sx0;
7342
78
  SplashCoord sy0 = (SplashCoord)y1 + 0.5 - ys;
7343
78
  SplashCoord sy1 = (SplashCoord)1 - sy0;
7344
78
  if (x0 < 0) {
7345
0
    x0 = 0;
7346
0
  }
7347
78
  if (x1 >= scaledWidth) {
7348
78
    x1 = scaledWidth - 1;
7349
78
  }
7350
78
  if (y0 < 0) {
7351
1
    y0 = 0;
7352
1
  }
7353
78
  if (y1 >= scaledHeight) {
7354
77
    y1 = scaledHeight - 1;
7355
77
  }
7356
78
  Guchar *p00 = scaledColor +
7357
78
                  (y0 * (SplashBitmapRowSize)scaledWidth + x0) * nComps;
7358
78
  Guchar *p10 = scaledColor +
7359
78
                  (y0 * (SplashBitmapRowSize)scaledWidth + x1) * nComps;
7360
78
  Guchar *p01 = scaledColor +
7361
78
                  (y1 * (SplashBitmapRowSize)scaledWidth + x0) * nComps;
7362
78
  Guchar *p11 = scaledColor +
7363
78
                  (y1 * (SplashBitmapRowSize)scaledWidth + x1) * nComps;
7364
78
  Guchar *q = colorBuf + (x - xMin) * nComps;
7365
312
  for (int i = 0; i < nComps; ++i) {
7366
234
    *q++ = (Guchar)(int)(sx0 * (sy0 * (int)*p00++ + sy1 * (int)*p01++) +
7367
234
             sx1 * (sy0 * (int)*p10++ + sy1 * (int)*p11++));
7368
234
  }
7369
78
  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
78
  if (x < rowMin) {
7379
78
    rowMin = x;
7380
78
  }
7381
78
  rowMax = x + 1;
7382
78
      }
7383
78
    }
7384
78
    if (rowMin < rowMax) {
7385
78
      (this->*drawRowFunc)(dd,
7386
78
         colorBuf + (rowMin - xMin) * nComps,
7387
78
         alphaBuf + (rowMin - xMin),
7388
78
         rowMin, y, rowMax - rowMin);
7389
78
    }
7390
78
  }
7391
7392
78
  gfree(colorBuf);
7393
78
  gfree(alphaBuf);
7394
78
}
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
745
               int x, int y, int width) {
7465
745
  if (y < 0 || y >= bitmap->height) {
7466
0
    return;
7467
0
  }
7468
745
  if (x < 0) {
7469
0
    colorData -= x * data->nComps;
7470
0
    width += x;
7471
0
    x = 0;
7472
0
  }
7473
745
  if (x + width > bitmap->width) {
7474
0
    width = bitmap->width - x;
7475
0
  }
7476
745
  if (width <= 0) {
7477
0
    return;
7478
0
  }
7479
745
  memset(scanBuf + x, 0xff, width);
7480
745
  state->clip->clipSpan(scanBuf, y, x, x + width - 1, state->strokeAdjust);
7481
745
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7482
745
        NULL, scanBuf + x, colorData);
7483
745
}
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
22
             int x, int y, int width) {
7515
22
  if (y < 0 || y >= bitmap->height) {
7516
0
    return;
7517
0
  }
7518
22
  if (x < 0) {
7519
0
    colorData -= x * data->nComps;
7520
0
    alphaData -= x;
7521
0
    width += x;
7522
0
    x = 0;
7523
0
  }
7524
22
  if (x + width > bitmap->width) {
7525
0
    width = bitmap->width - x;
7526
0
  }
7527
22
  if (width <= 0) {
7528
0
    return;
7529
0
  }
7530
22
  memcpy(scanBuf + x, alphaData, width);
7531
22
  state->clip->clipSpan(scanBuf, y, x, x + width - 1, state->strokeAdjust);
7532
22
  (this->*data->pipe.run)(&data->pipe, x, x + width - 1, y,
7533
22
        NULL, scanBuf + x, colorData);
7534
22
}
7535
7536
SplashError Splash::composite(SplashBitmap *src, SplashAlphaBitmap *srcShape,
7537
            int xSrc, int ySrc, int xDest, int yDest,
7538
2.97k
            int w, int h, GBool noClip, GBool nonIsolated) {
7539
2.97k
  SplashPipe pipe;
7540
2.97k
  Guchar *mono1Ptr, *lineBuf, *linePtr;
7541
2.97k
  Guchar mono1Mask, b;
7542
2.97k
  int x0, x1, x, y0, y1, y, t;
7543
7544
2.97k
  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
2.97k
  GBool usesShape = srcShape && shapeBitmap && src->alpha;
7551
2.97k
  GBool usesAlpha = !noClip || src->alpha;
7552
2.97k
  pipeInit(&pipe, NULL,
7553
2.97k
     (Guchar)splashRound(state->fillAlpha * 255),
7554
2.97k
     usesShape, usesAlpha, nonIsolated);
7555
2.97k
  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
2.97k
  } 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
2.97k
  } else { // src->mode not mono1 or BGR8
7842
2.97k
    if (noClip) {
7843
78
      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
78
      } else if (usesAlpha) {
7854
156
  for (y = 0; y < h; ++y) {
7855
78
    (this->*pipe.run)(&pipe, xDest, xDest + w - 1, yDest + y,
7856
78
          NULL,
7857
78
          src->alpha +
7858
78
            (ySrc + y) * src->alphaRowSize + xSrc,
7859
78
          src->data + (ySrc + y) * src->rowSize +
7860
78
          xSrc * bitmapComps);
7861
78
  }
7862
78
      } 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
2.90k
    } else {
7871
2.90k
      x0 = xDest;
7872
2.90k
      if ((t = state->clip->getXMinI(state->strokeAdjust)) > x0) {
7873
623
  x0 = t;
7874
623
      }
7875
2.90k
      x1 = xDest + w;
7876
2.90k
      if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < x1) {
7877
933
  x1 = t;
7878
933
      }
7879
2.90k
      y0 = yDest;
7880
2.90k
      if ((t = state->clip->getYMinI(state->strokeAdjust)) > y0) {
7881
956
  y0 = t;
7882
956
      }
7883
2.90k
      y1 = yDest + h;
7884
2.90k
      if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < y1) {
7885
958
  y1 = t;
7886
958
      }
7887
2.90k
      if (x0 < x1 && y0 < y1) {
7888
1.38k
  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.38k
  } else if (usesAlpha) {
7906
2.76k
    for (y = y0; y < y1; ++y) {
7907
1.38k
      memcpy(scanBuf + x0,
7908
1.38k
       src->alpha + (ySrc + y - yDest) * src->alphaRowSize + 
7909
1.38k
         (xSrc + x0 - xDest),
7910
1.38k
       x1 - x0);
7911
1.38k
      state->clip->clipSpan(scanBuf, y, x0, x1 - 1, state->strokeAdjust);
7912
1.38k
      (this->*pipe.run)(&pipe, x0, x1 - 1, y,
7913
1.38k
            NULL,
7914
1.38k
            scanBuf + x0,
7915
1.38k
            src->data +
7916
1.38k
              (ySrc + y - yDest) * src->rowSize +
7917
1.38k
              (xSrc + x0 - xDest) * bitmapComps);
7918
1.38k
    }
7919
1.38k
  } 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.38k
      }
7932
2.90k
    }
7933
2.97k
  }
7934
7935
2.97k
  return splashOk;
7936
2.97k
}
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.00k
void Splash::compositeBackground(SplashColorPtr color) {
8067
1.00k
  SplashColorPtr p;
8068
1.00k
  Guchar *q;
8069
1.00k
  Guchar alpha, alpha1, c, color0, color1, color2, mask;
8070
1.00k
#if SPLASH_CMYK
8071
1.00k
  Guchar color3;
8072
1.00k
#endif
8073
1.00k
  int x, y;
8074
8075
1.00k
  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
733
  case splashModeMono8:
8108
733
    color0 = color[0];
8109
1.46k
    for (y = 0; y < bitmap->height; ++y) {
8110
733
      p = &bitmap->data[y * bitmap->rowSize];
8111
733
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8112
1.46k
      for (x = 0; x < bitmap->width; ++x) {
8113
733
  alpha = *q++;
8114
733
  if (alpha == 0) {
8115
330
    p[0] = color0;
8116
403
  } else if (alpha != 255) {
8117
257
    alpha1 = (Guchar)(255 - alpha);
8118
257
    p[0] = div255(alpha1 * color0 + alpha * p[0]);
8119
257
  }
8120
733
  ++p;
8121
733
      }
8122
733
    }
8123
733
    break;
8124
39
  case splashModeRGB8:
8125
39
  case splashModeBGR8:
8126
39
    color0 = color[0];
8127
39
    color1 = color[1];
8128
39
    color2 = color[2];
8129
78
    for (y = 0; y < bitmap->height; ++y) {
8130
39
      p = &bitmap->data[y * bitmap->rowSize];
8131
39
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8132
78
      for (x = 0; x < bitmap->width; ++x) {
8133
39
  alpha = *q++;
8134
39
  if (alpha == 0) {
8135
39
    p[0] = color0;
8136
39
    p[1] = color1;
8137
39
    p[2] = color2;
8138
39
  } 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
39
  p += 3;
8145
39
      }
8146
39
    }
8147
39
    break;
8148
0
#if SPLASH_CMYK
8149
230
  case splashModeCMYK8:
8150
230
    color0 = color[0];
8151
230
    color1 = color[1];
8152
230
    color2 = color[2];
8153
230
    color3 = color[3];
8154
460
    for (y = 0; y < bitmap->height; ++y) {
8155
230
      p = &bitmap->data[y * bitmap->rowSize];
8156
230
      q = &bitmap->alpha[y * bitmap->alphaRowSize];
8157
460
      for (x = 0; x < bitmap->width; ++x) {
8158
230
  alpha = *q++;
8159
230
  if (alpha == 0) {
8160
223
    p[0] = color0;
8161
223
    p[1] = color1;
8162
223
    p[2] = color2;
8163
223
    p[3] = color3;
8164
223
  } 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
230
  p += 4;
8172
230
      }
8173
230
    }
8174
230
    break;
8175
1.00k
#endif
8176
1.00k
  }
8177
1.00k
  memset(bitmap->alpha, 255, bitmap->alphaRowSize * bitmap->height);
8178
1.00k
}
8179
8180
SplashPath *Splash::makeStrokePath(SplashPath *path, SplashCoord w,
8181
           int lineCap, int lineJoin,
8182
16.6k
           GBool flatten) {
8183
16.6k
  SplashPath *pathIn, *dashPath, *pathOut;
8184
16.6k
  SplashCoord d, dx, dy, wdx, wdy, dxNext, dyNext, wdxNext, wdyNext;
8185
16.6k
  SplashCoord crossprod, dotprod, miter, m;
8186
16.6k
  SplashCoord angle, angleNext, dAngle, xc, yc;
8187
16.6k
  SplashCoord dxJoin, dyJoin, dJoin, kappa;
8188
16.6k
  SplashCoord cx1, cy1, cx2, cy2, cx3, cy3, cx4, cy4;
8189
16.6k
  GBool first, last, closed;
8190
16.6k
  int subpathStart0, subpathStart1, seg, i0, i1, j0, j1, k0, k1;
8191
16.6k
  int left0, left1, left2, right0, right1, right2, join0, join1, join2;
8192
16.6k
  int leftFirst, rightFirst, firstPt;
8193
8194
16.6k
  pathOut = new SplashPath();
8195
8196
16.6k
  if (path->length == 0) {
8197
0
    return pathOut;
8198
0
  }
8199
8200
16.6k
  if (flatten) {
8201
117
    pathIn = flattenPath(path, state->matrix, state->flatness);
8202
117
    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
16.5k
  } else {
8212
16.5k
    pathIn = path;
8213
16.5k
  }
8214
8215
16.6k
  subpathStart0 = subpathStart1 = 0; // make gcc happy
8216
16.6k
  seg = 0; // make gcc happy
8217
16.6k
  closed = gFalse; // make gcc happy
8218
16.6k
  left0 = left1 = right0 = right1 = join0 = join1 = 0; // make gcc happy
8219
16.6k
  leftFirst = rightFirst = firstPt = 0; // make gcc happy
8220
8221
16.6k
  i0 = 0;
8222
16.6k
  for (i1 = i0;
8223
22.7k
       !(pathIn->flags[i1] & splashPathLast) &&
8224
19.1k
   i1 + 1 < pathIn->length &&
8225
19.1k
   pathIn->pts[i1+1].x == pathIn->pts[i1].x &&
8226
7.72k
   pathIn->pts[i1+1].y == pathIn->pts[i1].y;
8227
16.6k
       ++i1) ;
8228
8229
11.5M
  while (i1 < pathIn->length) {
8230
11.5M
    if ((first = pathIn->flags[i0] & splashPathFirst)) {
8231
3.10M
      subpathStart0 = i0;
8232
3.10M
      subpathStart1 = i1;
8233
3.10M
      seg = 0;
8234
3.10M
      closed = pathIn->flags[i0] & splashPathClosed;
8235
3.10M
    }
8236
11.5M
    j0 = i1 + 1;
8237
11.5M
    if (j0 < pathIn->length) {
8238
11.5M
      for (j1 = j0;
8239
12.1M
     !(pathIn->flags[j1] & splashPathLast) &&
8240
9.02M
       j1 + 1 < pathIn->length &&
8241
9.02M
       pathIn->pts[j1+1].x == pathIn->pts[j1].x &&
8242
718k
       pathIn->pts[j1+1].y == pathIn->pts[j1].y;
8243
11.5M
     ++j1) ;
8244
11.5M
    } else {
8245
16.6k
      j1 = j0;
8246
16.6k
    }
8247
11.5M
    if (pathIn->flags[i1] & splashPathLast) {
8248
3.10M
      if (first && lineCap == splashLineCapRound) {
8249
  // special case: zero-length subpath with round line caps -->
8250
  // draw a circle
8251
840
  pathOut->moveTo(pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8252
840
      pathIn->pts[i0].y);
8253
840
  pathOut->curveTo(pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8254
840
       pathIn->pts[i0].y + bezierCircle2 * w,
8255
840
       pathIn->pts[i0].x + bezierCircle2 * w,
8256
840
       pathIn->pts[i0].y + (SplashCoord)0.5 * w,
8257
840
       pathIn->pts[i0].x,
8258
840
       pathIn->pts[i0].y + (SplashCoord)0.5 * w);
8259
840
  pathOut->curveTo(pathIn->pts[i0].x - bezierCircle2 * w,
8260
840
       pathIn->pts[i0].y + (SplashCoord)0.5 * w,
8261
840
       pathIn->pts[i0].x - (SplashCoord)0.5 * w,
8262
840
       pathIn->pts[i0].y + bezierCircle2 * w,
8263
840
       pathIn->pts[i0].x - (SplashCoord)0.5 * w,
8264
840
       pathIn->pts[i0].y);
8265
840
  pathOut->curveTo(pathIn->pts[i0].x - (SplashCoord)0.5 * w,
8266
840
       pathIn->pts[i0].y - bezierCircle2 * w,
8267
840
       pathIn->pts[i0].x - bezierCircle2 * w,
8268
840
       pathIn->pts[i0].y - (SplashCoord)0.5 * w,
8269
840
       pathIn->pts[i0].x,
8270
840
       pathIn->pts[i0].y - (SplashCoord)0.5 * w);
8271
840
  pathOut->curveTo(pathIn->pts[i0].x + bezierCircle2 * w,
8272
840
       pathIn->pts[i0].y - (SplashCoord)0.5 * w,
8273
840
       pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8274
840
       pathIn->pts[i0].y - bezierCircle2 * w,
8275
840
       pathIn->pts[i0].x + (SplashCoord)0.5 * w,
8276
840
       pathIn->pts[i0].y);
8277
840
  pathOut->close();
8278
840
      }
8279
3.10M
      i0 = j0;
8280
3.10M
      i1 = j1;
8281
3.10M
      continue;
8282
3.10M
    }
8283
8.47M
    last = pathIn->flags[j1] & splashPathLast;
8284
8.47M
    if (last) {
8285
2.57M
      k0 = subpathStart1 + 1;
8286
5.89M
    } else {
8287
5.89M
      k0 = j1 + 1;
8288
5.89M
    }
8289
8.47M
    for (k1 = k0;
8290
8.51M
   !(pathIn->flags[k1] & splashPathLast) &&
8291
5.94M
     k1 + 1 < pathIn->length &&
8292
5.94M
     pathIn->pts[k1+1].x == pathIn->pts[k1].x &&
8293
146k
     pathIn->pts[k1+1].y == pathIn->pts[k1].y;
8294
8.47M
   ++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
8.47M
    d = (SplashCoord)1 / splashDist(pathIn->pts[i1].x, pathIn->pts[i1].y,
8306
8.47M
            pathIn->pts[j0].x, pathIn->pts[j0].y);
8307
8.47M
    dx = d * (pathIn->pts[j0].x - pathIn->pts[i1].x);
8308
8.47M
    dy = d * (pathIn->pts[j0].y - pathIn->pts[i1].y);
8309
8.47M
#endif
8310
8.47M
    wdx = (SplashCoord)0.5 * w * dx;
8311
8.47M
    wdy = (SplashCoord)0.5 * w * dy;
8312
8313
    // draw the start cap
8314
8.47M
    if (i0 == subpathStart0) {
8315
2.57M
      firstPt = pathOut->length;
8316
2.57M
    }
8317
8.47M
    if (first && !closed) {
8318
2.56M
      switch (lineCap) {
8319
2.51M
      case splashLineCapButt:
8320
2.51M
  pathOut->moveTo(pathIn->pts[i0].x - wdy, pathIn->pts[i0].y + wdx);
8321
2.51M
  pathOut->lineTo(pathIn->pts[i0].x + wdy, pathIn->pts[i0].y - wdx);
8322
2.51M
  break;
8323
50.8k
      case splashLineCapRound:
8324
50.8k
  pathOut->moveTo(pathIn->pts[i0].x - wdy, pathIn->pts[i0].y + wdx);
8325
50.8k
  pathOut->curveTo(pathIn->pts[i0].x - wdy - bezierCircle * wdx,
8326
50.8k
       pathIn->pts[i0].y + wdx - bezierCircle * wdy,
8327
50.8k
       pathIn->pts[i0].x - wdx - bezierCircle * wdy,
8328
50.8k
       pathIn->pts[i0].y - wdy + bezierCircle * wdx,
8329
50.8k
       pathIn->pts[i0].x - wdx,
8330
50.8k
       pathIn->pts[i0].y - wdy);
8331
50.8k
  pathOut->curveTo(pathIn->pts[i0].x - wdx + bezierCircle * wdy,
8332
50.8k
       pathIn->pts[i0].y - wdy - bezierCircle * wdx,
8333
50.8k
       pathIn->pts[i0].x + wdy - bezierCircle * wdx,
8334
50.8k
       pathIn->pts[i0].y - wdx - bezierCircle * wdy,
8335
50.8k
       pathIn->pts[i0].x + wdy,
8336
50.8k
       pathIn->pts[i0].y - wdx);
8337
50.8k
  break;
8338
1.54k
      case splashLineCapProjecting:
8339
1.54k
  pathOut->moveTo(pathIn->pts[i0].x - wdx - wdy,
8340
1.54k
      pathIn->pts[i0].y + wdx - wdy);
8341
1.54k
  pathOut->lineTo(pathIn->pts[i0].x - wdx + wdy,
8342
1.54k
      pathIn->pts[i0].y - wdx - wdy);
8343
1.54k
  break;
8344
2.56M
      }
8345
5.91M
    } else {
8346
5.91M
      pathOut->moveTo(pathIn->pts[i0].x - wdy, pathIn->pts[i0].y + wdx);
8347
5.91M
      pathOut->lineTo(pathIn->pts[i0].x + wdy, pathIn->pts[i0].y - wdx);
8348
5.91M
    }
8349
8350
    // draw the left side of the segment rectangle and the end cap
8351
8.47M
    left2 = pathOut->length - 1;
8352
8.47M
    if (last && !closed) {
8353
2.56M
      switch (lineCap) {
8354
2.51M
      case splashLineCapButt:
8355
2.51M
  pathOut->lineTo(pathIn->pts[j0].x + wdy, pathIn->pts[j0].y - wdx);
8356
2.51M
  pathOut->lineTo(pathIn->pts[j0].x - wdy, pathIn->pts[j0].y + wdx);
8357
2.51M
  break;
8358
50.8k
      case splashLineCapRound:
8359
50.8k
  pathOut->lineTo(pathIn->pts[j0].x + wdy, pathIn->pts[j0].y - wdx);
8360
50.8k
  pathOut->curveTo(pathIn->pts[j0].x + wdy + bezierCircle * wdx,
8361
50.8k
       pathIn->pts[j0].y - wdx + bezierCircle * wdy,
8362
50.8k
       pathIn->pts[j0].x + wdx + bezierCircle * wdy,
8363
50.8k
       pathIn->pts[j0].y + wdy - bezierCircle * wdx,
8364
50.8k
       pathIn->pts[j0].x + wdx,
8365
50.8k
       pathIn->pts[j0].y + wdy);
8366
50.8k
  pathOut->curveTo(pathIn->pts[j0].x + wdx - bezierCircle * wdy,
8367
50.8k
       pathIn->pts[j0].y + wdy + bezierCircle * wdx,
8368
50.8k
       pathIn->pts[j0].x - wdy + bezierCircle * wdx,
8369
50.8k
       pathIn->pts[j0].y + wdx + bezierCircle * wdy,
8370
50.8k
       pathIn->pts[j0].x - wdy,
8371
50.8k
       pathIn->pts[j0].y + wdx);
8372
50.8k
  break;
8373
1.54k
      case splashLineCapProjecting:
8374
1.54k
  pathOut->lineTo(pathIn->pts[j0].x + wdy + wdx,
8375
1.54k
      pathIn->pts[j0].y - wdx + wdy);
8376
1.54k
  pathOut->lineTo(pathIn->pts[j0].x - wdy + wdx,
8377
1.54k
      pathIn->pts[j0].y + wdx + wdy);
8378
1.54k
  break;
8379
2.56M
      }
8380
5.91M
    } else {
8381
5.91M
      pathOut->lineTo(pathIn->pts[j0].x + wdy, pathIn->pts[j0].y - wdx);
8382
5.91M
      pathOut->lineTo(pathIn->pts[j0].x - wdy, pathIn->pts[j0].y + wdx);
8383
5.91M
    }
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
8.47M
    right2 = pathOut->length - 1;
8389
8.47M
    pathOut->close(state->strokeAdjust != splashStrokeAdjustOff);
8390
8391
    // draw the join
8392
8.47M
    join2 = pathOut->length;
8393
8.47M
    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
5.91M
      d = (SplashCoord)1 / splashDist(pathIn->pts[j1].x, pathIn->pts[j1].y,
8405
5.91M
              pathIn->pts[k0].x, pathIn->pts[k0].y);
8406
5.91M
      dxNext = d * (pathIn->pts[k0].x - pathIn->pts[j1].x);
8407
5.91M
      dyNext = d * (pathIn->pts[k0].y - pathIn->pts[j1].y);
8408
5.91M
#endif
8409
5.91M
      wdxNext = (SplashCoord)0.5 * w * dxNext;
8410
5.91M
      wdyNext = (SplashCoord)0.5 * w * dyNext;
8411
8412
      // compute the join parameters
8413
5.91M
      crossprod = dx * dyNext - dy * dxNext;
8414
5.91M
      dotprod = -(dx * dxNext + dy * dyNext);
8415
5.91M
      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.20k
  miter = (state->miterLimit + 1) * (state->miterLimit + 1);
8423
9.20k
  m = 0;
8424
5.90M
      } else {
8425
5.90M
  miter = (SplashCoord)2 / ((SplashCoord)1 - dotprod);
8426
5.90M
  if (miter < 1) {
8427
    // this can happen because of floating point inaccuracies
8428
1.73k
    miter = 1;
8429
1.73k
  }
8430
5.90M
  m = splashSqrt(miter - 1);
8431
5.90M
      }
8432
8433
      // round join
8434
5.91M
      if (lineJoin == splashLineJoinRound) {
8435
  // join angle < 180
8436
682k
  if (crossprod < 0) {
8437
268k
    angle = atan2((double)dx, (double)-dy);
8438
268k
    angleNext = atan2((double)dxNext, (double)-dyNext);
8439
268k
    if (angle < angleNext) {
8440
688
      angle += 2 * M_PI;
8441
688
    }
8442
268k
    dAngle = (angle  - angleNext) / M_PI;
8443
268k
    if (dAngle < 0.501) {
8444
      // span angle is <= 90 degrees -> draw a single arc
8445
268k
      kappa = dAngle * bezierCircle * w;
8446
268k
      cx1 = pathIn->pts[j0].x - wdy + kappa * dx;
8447
268k
      cy1 = pathIn->pts[j0].y + wdx + kappa * dy;
8448
268k
      cx2 = pathIn->pts[j0].x - wdyNext - kappa * dxNext;
8449
268k
      cy2 = pathIn->pts[j0].y + wdxNext - kappa * dyNext;
8450
268k
      pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8451
268k
      pathOut->lineTo(pathIn->pts[j0].x - wdyNext,
8452
268k
          pathIn->pts[j0].y + wdxNext);
8453
268k
      pathOut->curveTo(cx2, cy2, cx1, cy1,
8454
268k
           pathIn->pts[j0].x - wdy,
8455
268k
           pathIn->pts[j0].y + wdx);
8456
268k
    } else {
8457
      // span angle is > 90 degrees -> split into two arcs
8458
440
      dJoin = splashDist(-wdy, wdx, -wdyNext, wdxNext);
8459
440
      if (dJoin > 0) {
8460
440
        dxJoin = (-wdyNext + wdy) / dJoin;
8461
440
        dyJoin = (wdxNext - wdx) / dJoin;
8462
440
        xc = pathIn->pts[j0].x
8463
440
       + (SplashCoord)0.5 * w
8464
440
         * cos((double)((SplashCoord)0.5 * (angle + angleNext)));
8465
440
        yc = pathIn->pts[j0].y
8466
440
       + (SplashCoord)0.5 * w
8467
440
         * sin((double)((SplashCoord)0.5 * (angle + angleNext)));
8468
440
        kappa = dAngle * bezierCircle2 * w;
8469
440
        cx1 = pathIn->pts[j0].x - wdy + kappa * dx;
8470
440
        cy1 = pathIn->pts[j0].y + wdx + kappa * dy;
8471
440
        cx2 = xc - kappa * dxJoin;
8472
440
        cy2 = yc - kappa * dyJoin;
8473
440
        cx3 = xc + kappa * dxJoin;
8474
440
        cy3 = yc + kappa * dyJoin;
8475
440
        cx4 = pathIn->pts[j0].x - wdyNext - kappa * dxNext;
8476
440
        cy4 = pathIn->pts[j0].y + wdxNext - kappa * dyNext;
8477
440
        pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8478
440
        pathOut->lineTo(pathIn->pts[j0].x - wdyNext,
8479
440
            pathIn->pts[j0].y + wdxNext);
8480
440
        pathOut->curveTo(cx4, cy4, cx3, cy3, xc, yc);
8481
440
        pathOut->curveTo(cx2, cy2, cx1, cy1,
8482
440
             pathIn->pts[j0].x - wdy,
8483
440
             pathIn->pts[j0].y + wdx);
8484
440
      }
8485
440
    }
8486
8487
  // join angle >= 180
8488
414k
  } else {
8489
414k
    angle = atan2((double)-dx, (double)dy);
8490
414k
    angleNext = atan2((double)-dxNext, (double)dyNext);
8491
414k
    if (angleNext < angle) {
8492
502
      angleNext += 2 * M_PI;
8493
502
    }
8494
414k
    dAngle = (angleNext - angle) / M_PI;
8495
414k
    if (dAngle < 0.501) {
8496
      // span angle is <= 90 degrees -> draw a single arc
8497
413k
      kappa = dAngle * bezierCircle * w;
8498
413k
        cx1 = pathIn->pts[j0].x + wdy + kappa * dx;
8499
413k
        cy1 = pathIn->pts[j0].y - wdx + kappa * dy;
8500
413k
        cx2 = pathIn->pts[j0].x + wdyNext - kappa * dxNext;
8501
413k
        cy2 = pathIn->pts[j0].y - wdxNext - kappa * dyNext;
8502
413k
        pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8503
413k
        pathOut->lineTo(pathIn->pts[j0].x + wdy,
8504
413k
            pathIn->pts[j0].y - wdx);
8505
413k
        pathOut->curveTo(cx1, cy1, cx2, cy2,
8506
413k
             pathIn->pts[j0].x + wdyNext,
8507
413k
             pathIn->pts[j0].y - wdxNext);
8508
413k
    } else {
8509
      // span angle is > 90 degrees -> split into two arcs
8510
695
      dJoin = splashDist(wdy, -wdx, wdyNext, -wdxNext);
8511
695
      if (dJoin > 0) {
8512
671
        dxJoin = (wdyNext - wdy) / dJoin;
8513
671
        dyJoin = (-wdxNext + wdx) / dJoin;
8514
671
        xc = pathIn->pts[j0].x
8515
671
       + (SplashCoord)0.5 * w
8516
671
         * cos((double)((SplashCoord)0.5 * (angle + angleNext)));
8517
671
        yc = pathIn->pts[j0].y
8518
671
       + (SplashCoord)0.5 * w
8519
671
         * sin((double)((SplashCoord)0.5 * (angle + angleNext)));
8520
671
        kappa = dAngle * bezierCircle2 * w;
8521
671
        cx1 = pathIn->pts[j0].x + wdy + kappa * dx;
8522
671
        cy1 = pathIn->pts[j0].y - wdx + kappa * dy;
8523
671
        cx2 = xc - kappa * dxJoin;
8524
671
        cy2 = yc - kappa * dyJoin;
8525
671
        cx3 = xc + kappa * dxJoin;
8526
671
        cy3 = yc + kappa * dyJoin;
8527
671
        cx4 = pathIn->pts[j0].x + wdyNext - kappa * dxNext;
8528
671
        cy4 = pathIn->pts[j0].y - wdxNext - kappa * dyNext;
8529
671
        pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8530
671
        pathOut->lineTo(pathIn->pts[j0].x + wdy,
8531
671
            pathIn->pts[j0].y - wdx);
8532
671
        pathOut->curveTo(cx1, cy1, cx2, cy2, xc, yc);
8533
671
        pathOut->curveTo(cx3, cy3, cx4, cy4,
8534
671
             pathIn->pts[j0].x + wdyNext,
8535
671
             pathIn->pts[j0].y - wdxNext);
8536
671
      }
8537
695
    }
8538
414k
  }
8539
8540
5.22M
      } else {
8541
5.22M
  pathOut->moveTo(pathIn->pts[j0].x, pathIn->pts[j0].y);
8542
8543
  // join angle < 180
8544
5.22M
  if (crossprod < 0) {
8545
1.85M
    pathOut->lineTo(pathIn->pts[j0].x - wdyNext,
8546
1.85M
        pathIn->pts[j0].y + wdxNext);
8547
    // miter join inside limit
8548
1.85M
    if (lineJoin == splashLineJoinMiter &&
8549
1.59M
        splashSqrt(miter) <= state->miterLimit) {
8550
1.41M
      pathOut->lineTo(pathIn->pts[j0].x - wdy + wdx * m,
8551
1.41M
          pathIn->pts[j0].y + wdx + wdy * m);
8552
1.41M
      pathOut->lineTo(pathIn->pts[j0].x - wdy,
8553
1.41M
          pathIn->pts[j0].y + wdx);
8554
    // bevel join or miter join outside limit
8555
1.41M
    } else {
8556
434k
      pathOut->lineTo(pathIn->pts[j0].x - wdy,
8557
434k
          pathIn->pts[j0].y + wdx);
8558
434k
    }
8559
8560
  // join angle >= 180
8561
3.37M
  } else {
8562
3.37M
    pathOut->lineTo(pathIn->pts[j0].x + wdy,
8563
3.37M
        pathIn->pts[j0].y - wdx);
8564
    // miter join inside limit
8565
3.37M
    if (lineJoin == splashLineJoinMiter &&
8566
3.05M
        splashSqrt(miter) <= state->miterLimit) {
8567
2.80M
      pathOut->lineTo(pathIn->pts[j0].x + wdy + wdx * m,
8568
2.80M
          pathIn->pts[j0].y - wdx + wdy * m);
8569
2.80M
      pathOut->lineTo(pathIn->pts[j0].x + wdyNext,
8570
2.80M
          pathIn->pts[j0].y - wdxNext);
8571
    // bevel join or miter join outside limit
8572
2.80M
    } else {
8573
566k
      pathOut->lineTo(pathIn->pts[j0].x + wdyNext,
8574
566k
          pathIn->pts[j0].y - wdxNext);
8575
566k
    }
8576
3.37M
  }
8577
5.22M
      }
8578
8579
5.91M
      pathOut->close();
8580
5.91M
    }
8581
8582
    // add stroke adjustment hints
8583
8.47M
    if (state->strokeAdjust != splashStrokeAdjustOff) {
8584
8585
      // subpath with one segment
8586
8.42M
      if (seg == 0 && last) {
8587
2.55M
  switch (lineCap) {
8588
2.50M
  case splashLineCapButt:
8589
2.50M
    pathOut->addStrokeAdjustHint(firstPt, left2 + 1,
8590
2.50M
               firstPt, pathOut->length - 1);
8591
2.50M
    break;
8592
644
  case splashLineCapProjecting:
8593
644
    pathOut->addStrokeAdjustHint(firstPt, left2 + 1,
8594
644
               firstPt, pathOut->length - 1, gTrue);
8595
644
    break;
8596
50.7k
  case splashLineCapRound:
8597
50.7k
    break;
8598
2.55M
  }
8599
2.55M
  pathOut->addStrokeAdjustHint(left2, right2,
8600
2.55M
             firstPt, pathOut->length - 1);
8601
5.86M
      } else {
8602
8603
  // start of subpath
8604
5.86M
  if (seg == 1) {
8605
8606
    // start cap
8607
17.1k
    if (!closed) {
8608
7.44k
      switch (lineCap) {
8609
6.45k
      case splashLineCapButt:
8610
6.45k
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8611
6.45k
             firstPt, firstPt + 1);
8612
6.45k
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8613
6.45k
             right1 + 1, right1 + 1);
8614
6.45k
        break;
8615
836
      case splashLineCapProjecting:
8616
836
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8617
836
             firstPt, firstPt + 1, gTrue);
8618
836
        pathOut->addStrokeAdjustHint(firstPt, left1 + 1,
8619
836
             right1 + 1, right1 + 1, gTrue);
8620
836
        break;
8621
154
      case splashLineCapRound:
8622
154
        break;
8623
7.44k
      }
8624
7.44k
    }
8625
8626
    // first segment
8627
17.1k
    pathOut->addStrokeAdjustHint(left1, right1, firstPt, left2);
8628
17.1k
    pathOut->addStrokeAdjustHint(left1, right1, right2 + 1, right2 + 1);
8629
17.1k
  }
8630
8631
  // middle of subpath
8632
5.86M
  if (seg > 1) {
8633
5.83M
    pathOut->addStrokeAdjustHint(left1, right1, left0 + 1, right0);
8634
5.83M
    pathOut->addStrokeAdjustHint(left1, right1, join0, left2);
8635
5.83M
    pathOut->addStrokeAdjustHint(left1, right1, right2 + 1, right2 + 1);
8636
5.83M
  }
8637
8638
  // end of subpath
8639
5.86M
  if (last) {
8640
8641
17.1k
    if (closed) {
8642
      // first segment
8643
9.74k
      pathOut->addStrokeAdjustHint(leftFirst, rightFirst,
8644
9.74k
           left2 + 1, right2);
8645
9.74k
      pathOut->addStrokeAdjustHint(leftFirst, rightFirst,
8646
9.74k
           join2, pathOut->length - 1);
8647
8648
      // last segment
8649
9.74k
      pathOut->addStrokeAdjustHint(left2, right2,
8650
9.74k
           left1 + 1, right1);
8651
9.74k
      pathOut->addStrokeAdjustHint(left2, right2,
8652
9.74k
           join1, pathOut->length - 1);
8653
9.74k
      pathOut->addStrokeAdjustHint(left2, right2,
8654
9.74k
           leftFirst - 1, leftFirst);
8655
9.74k
      pathOut->addStrokeAdjustHint(left2, right2,
8656
9.74k
           rightFirst + 1, rightFirst + 1);
8657
8658
9.74k
    } else {
8659
8660
      // last segment
8661
7.44k
      pathOut->addStrokeAdjustHint(left2, right2,
8662
7.44k
           left1 + 1, right1);
8663
7.44k
      pathOut->addStrokeAdjustHint(left2, right2,
8664
7.44k
           join1, pathOut->length - 1);
8665
8666
      // end cap
8667
7.44k
      switch (lineCap) {
8668
6.45k
      case splashLineCapButt:
8669
6.45k
        pathOut->addStrokeAdjustHint(left2 - 1, left2 + 1,
8670
6.45k
             left2 + 1, left2 + 2);
8671
6.45k
        break;
8672
836
      case splashLineCapProjecting:
8673
836
        pathOut->addStrokeAdjustHint(left2 - 1, left2 + 1,
8674
836
             left2 + 1, left2 + 2, gTrue);
8675
836
        break;
8676
154
      case splashLineCapRound:
8677
154
        break;
8678
7.44k
      }
8679
7.44k
    }
8680
17.1k
  }
8681
5.86M
      }
8682
8683
8.42M
      left0 = left1;
8684
8.42M
      left1 = left2;
8685
8.42M
      right0 = right1;
8686
8.42M
      right1 = right2;
8687
8.42M
      join0 = join1;
8688
8.42M
      join1 = join2;
8689
8.42M
      if (seg == 0) {
8690
2.57M
  leftFirst = left2;
8691
2.57M
  rightFirst = right2;
8692
2.57M
      }
8693
8.42M
    }
8694
8695
8.47M
    i0 = j0;
8696
8.47M
    i1 = j1;
8697
8.47M
    ++seg;
8698
8.47M
  }
8699
8700
16.6k
  if (pathIn != path) {
8701
117
    delete pathIn;
8702
117
  }
8703
8704
16.6k
  return pathOut;
8705
16.6k
}
8706
8707
SplashClipResult Splash::limitRectToClipRect(int *xMin, int *yMin,
8708
1.03k
               int *xMax, int *yMax) {
8709
1.03k
  int t;
8710
8711
1.03k
  if ((t = state->clip->getXMinI(state->strokeAdjust)) > *xMin) {
8712
865
    *xMin = t;
8713
865
  }
8714
1.03k
  if ((t = state->clip->getXMaxI(state->strokeAdjust) + 1) < *xMax) {
8715
227
    *xMax = t;
8716
227
  }
8717
1.03k
  if ((t = state->clip->getYMinI(state->strokeAdjust)) > *yMin) {
8718
847
    *yMin = t;
8719
847
  }
8720
1.03k
  if ((t = state->clip->getYMaxI(state->strokeAdjust) + 1) < *yMax) {
8721
188
    *yMax = t;
8722
188
  }
8723
1.03k
  if (*xMin >= *xMax || *yMin >= *yMax) {
8724
922
    return splashClipAllOutside;
8725
922
  }
8726
113
  return state->clip->testRect(*xMin, *yMin, *xMax - 1, *yMax - 1,
8727
113
             state->strokeAdjust);
8728
1.03k
}
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