Sanity check the values of TRUE and FALSE.
[pdfium.git] / core / src / fxge / dib / fx_dib_transform.cpp
1 // Copyright 2014 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
6
7 #include "../../../include/fxge/fx_dib.h"
8 #include "dib_int.h"
9
10 const int SDP_Table[513] = {
11     256, 256, 256, 256, 256, 256, 256, 256, 256, 255, 255, 255, 255, 255, 255,
12     254, 254, 254, 254, 253, 253, 253, 252, 252, 252, 251, 251, 251, 250, 250,
13     249, 249, 249, 248, 248, 247, 247, 246, 246, 245, 244, 244, 243, 243, 242,
14     242, 241, 240, 240, 239, 238, 238, 237, 236, 236, 235, 234, 233, 233, 232,
15     231, 230, 230, 229, 228, 227, 226, 226, 225, 224, 223, 222, 221, 220, 219,
16     218, 218, 217, 216, 215, 214, 213, 212, 211, 210, 209, 208, 207, 206, 205,
17     204, 203, 202, 201, 200, 199, 198, 196, 195, 194, 193, 192, 191, 190, 189,
18     188, 186, 185, 184, 183, 182, 181, 179, 178, 177, 176, 175, 173, 172, 171,
19     170, 169, 167, 166, 165, 164, 162, 161, 160, 159, 157, 156, 155, 154, 152,
20     151, 150, 149, 147, 146, 145, 143, 142, 141, 140, 138, 137, 136, 134, 133,
21     132, 130, 129, 128, 126, 125, 124, 122, 121, 120, 119, 117, 116, 115, 113,
22     112, 111, 109, 108, 107, 105, 104, 103, 101, 100, 99,  97,  96,  95,  93,
23     92,  91,  89,  88,  87,  85,  84,  83,  81,  80,  79,  77,  76,  75,  73,
24     72,  71,  69,  68,  67,  66,  64,  63,  62,  60,  59,  58,  57,  55,  54,
25     53,  52,  50,  49,  48,  47,  45,  44,  43,  42,  40,  39,  38,  37,  36,
26     34,  33,  32,  31,  30,  28,  27,  26,  25,  24,  23,  21,  20,  19,  18,
27     17,  16,  15,  14,  13,  11,  10,  9,   8,   7,   6,   5,   4,   3,   2,
28     1,   0,   0,   -1,  -2,  -3,  -4,  -5,  -6,  -7,  -7,  -8,  -9,  -10, -11,
29     -12, -12, -13, -14, -15, -15, -16, -17, -17, -18, -19, -19, -20, -21, -21,
30     -22, -22, -23, -24, -24, -25, -25, -26, -26, -27, -27, -27, -28, -28, -29,
31     -29, -30, -30, -30, -31, -31, -31, -32, -32, -32, -33, -33, -33, -33, -34,
32     -34, -34, -34, -35, -35, -35, -35, -35, -36, -36, -36, -36, -36, -36, -36,
33     -36, -36, -37, -37, -37, -37, -37, -37, -37, -37, -37, -37, -37, -37, -37,
34     -37, -37, -37, -37, -37, -37, -37, -37, -36, -36, -36, -36, -36, -36, -36,
35     -36, -36, -35, -35, -35, -35, -35, -35, -34, -34, -34, -34, -34, -33, -33,
36     -33, -33, -33, -32, -32, -32, -32, -31, -31, -31, -31, -30, -30, -30, -30,
37     -29, -29, -29, -29, -28, -28, -28, -27, -27, -27, -27, -26, -26, -26, -25,
38     -25, -25, -24, -24, -24, -23, -23, -23, -22, -22, -22, -22, -21, -21, -21,
39     -20, -20, -20, -19, -19, -19, -18, -18, -18, -17, -17, -17, -16, -16, -16,
40     -15, -15, -15, -14, -14, -14, -13, -13, -13, -12, -12, -12, -11, -11, -11,
41     -10, -10, -10, -9,  -9,  -9,  -9,  -8,  -8,  -8,  -7,  -7,  -7,  -7,  -6,
42     -6,  -6,  -6,  -5,  -5,  -5,  -5,  -4,  -4,  -4,  -4,  -3,  -3,  -3,  -3,
43     -3,  -2,  -2,  -2,  -2,  -2,  -1,  -1,  -1,  -1,  -1,  -1,  0,   0,   0,
44     0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,   0,
45     0,   0,   0,
46 };
47 class CFX_BilinearMatrix : public CPDF_FixedMatrix {
48  public:
49   CFX_BilinearMatrix(const CFX_AffineMatrix& src, int bits)
50       : CPDF_FixedMatrix(src, bits) {}
51   inline void Transform(int x,
52                         int y,
53                         int& x1,
54                         int& y1,
55                         int& res_x,
56                         int& res_y) {
57     x1 = a * x + c * y + e + base / 2;
58     y1 = b * x + d * y + f + base / 2;
59     res_x = x1 % base;
60     res_y = y1 % base;
61     if (res_x < 0 && res_x > -base) {
62       res_x = base + res_x;
63     }
64     if (res_y < 0 && res_x > -base) {
65       res_y = base + res_y;
66     }
67     x1 /= base;
68     y1 /= base;
69   }
70 };
71 CFX_DIBitmap* CFX_DIBSource::SwapXY(FX_BOOL bXFlip,
72                                     FX_BOOL bYFlip,
73                                     const FX_RECT* pDestClip) const {
74   FX_RECT dest_clip(0, 0, m_Height, m_Width);
75   if (pDestClip) {
76     dest_clip.Intersect(*pDestClip);
77   }
78   if (dest_clip.IsEmpty()) {
79     return NULL;
80   }
81   CFX_DIBitmap* pTransBitmap = new CFX_DIBitmap;
82   int result_height = dest_clip.Height(), result_width = dest_clip.Width();
83   if (!pTransBitmap->Create(result_width, result_height, GetFormat())) {
84     delete pTransBitmap;
85     return NULL;
86   }
87   pTransBitmap->CopyPalette(m_pPalette);
88   int dest_pitch = pTransBitmap->GetPitch();
89   uint8_t* dest_buf = pTransBitmap->GetBuffer();
90   int row_start = bXFlip ? m_Height - dest_clip.right : dest_clip.left;
91   int row_end = bXFlip ? m_Height - dest_clip.left : dest_clip.right;
92   int col_start = bYFlip ? m_Width - dest_clip.bottom : dest_clip.top;
93   int col_end = bYFlip ? m_Width - dest_clip.top : dest_clip.bottom;
94   if (GetBPP() == 1) {
95     FXSYS_memset(dest_buf, 0xff, dest_pitch * result_height);
96     for (int row = row_start; row < row_end; row++) {
97       const uint8_t* src_scan = GetScanline(row);
98       int dest_col = (bXFlip ? dest_clip.right - (row - row_start) - 1 : row) -
99                      dest_clip.left;
100       uint8_t* dest_scan = dest_buf;
101       if (bYFlip) {
102         dest_scan += (result_height - 1) * dest_pitch;
103       }
104       int dest_step = bYFlip ? -dest_pitch : dest_pitch;
105       for (int col = col_start; col < col_end; col++) {
106         if (!(src_scan[col / 8] & (1 << (7 - col % 8)))) {
107           dest_scan[dest_col / 8] &= ~(1 << (7 - dest_col % 8));
108         }
109         dest_scan += dest_step;
110       }
111     }
112   } else {
113     int nBytes = GetBPP() / 8;
114     int dest_step = bYFlip ? -dest_pitch : dest_pitch;
115     if (nBytes == 3) {
116       dest_step -= 2;
117     }
118     for (int row = row_start; row < row_end; row++) {
119       int dest_col = (bXFlip ? dest_clip.right - (row - row_start) - 1 : row) -
120                      dest_clip.left;
121       uint8_t* dest_scan = dest_buf + dest_col * nBytes;
122       if (bYFlip) {
123         dest_scan += (result_height - 1) * dest_pitch;
124       }
125       if (nBytes == 4) {
126         FX_DWORD* src_scan = (FX_DWORD*)GetScanline(row) + col_start;
127         for (int col = col_start; col < col_end; col++) {
128           *(FX_DWORD*)dest_scan = *src_scan++;
129           dest_scan += dest_step;
130         }
131       } else {
132         const uint8_t* src_scan = GetScanline(row) + col_start * nBytes;
133         if (nBytes == 1)
134           for (int col = col_start; col < col_end; col++) {
135             *dest_scan = *src_scan++;
136             dest_scan += dest_step;
137           }
138         else
139           for (int col = col_start; col < col_end; col++) {
140             *dest_scan++ = *src_scan++;
141             *dest_scan++ = *src_scan++;
142             *dest_scan = *src_scan++;
143             dest_scan += dest_step;
144           }
145       }
146     }
147   }
148   if (m_pAlphaMask) {
149     dest_pitch = pTransBitmap->m_pAlphaMask->GetPitch();
150     dest_buf = pTransBitmap->m_pAlphaMask->GetBuffer();
151     int dest_step = bYFlip ? -dest_pitch : dest_pitch;
152     for (int row = row_start; row < row_end; row++) {
153       int dest_col = (bXFlip ? dest_clip.right - (row - row_start) - 1 : row) -
154                      dest_clip.left;
155       uint8_t* dest_scan = dest_buf + dest_col;
156       if (bYFlip) {
157         dest_scan += (result_height - 1) * dest_pitch;
158       }
159       const uint8_t* src_scan = m_pAlphaMask->GetScanline(row) + col_start;
160       for (int col = col_start; col < col_end; col++) {
161         *dest_scan = *src_scan++;
162         dest_scan += dest_step;
163       }
164     }
165   }
166   return pTransBitmap;
167 }
168 #define FIX16_005 0.05f
169 FX_RECT FXDIB_SwapClipBox(FX_RECT& clip,
170                           int width,
171                           int height,
172                           FX_BOOL bFlipX,
173                           FX_BOOL bFlipY) {
174   FX_RECT rect;
175   if (bFlipY) {
176     rect.left = height - clip.top;
177     rect.right = height - clip.bottom;
178   } else {
179     rect.left = clip.top;
180     rect.right = clip.bottom;
181   }
182   if (bFlipX) {
183     rect.top = width - clip.left;
184     rect.bottom = width - clip.right;
185   } else {
186     rect.top = clip.left;
187     rect.bottom = clip.right;
188   }
189   rect.Normalize();
190   return rect;
191 }
192 CFX_DIBitmap* CFX_DIBSource::TransformTo(const CFX_AffineMatrix* pDestMatrix,
193                                          int& result_left,
194                                          int& result_top,
195                                          FX_DWORD flags,
196                                          const FX_RECT* pDestClip) const {
197   CFX_ImageTransformer transformer;
198   transformer.Start(this, pDestMatrix, flags, pDestClip);
199   transformer.Continue(NULL);
200   result_left = transformer.m_ResultLeft;
201   result_top = transformer.m_ResultTop;
202   CFX_DIBitmap* pTransformed = transformer.m_Storer.Detach();
203   return pTransformed;
204 }
205 CFX_DIBitmap* CFX_DIBSource::StretchTo(int dest_width,
206                                        int dest_height,
207                                        FX_DWORD flags,
208                                        const FX_RECT* pClip) const {
209   FX_RECT clip_rect(0, 0, FXSYS_abs(dest_width), FXSYS_abs(dest_height));
210   if (pClip) {
211     clip_rect.Intersect(*pClip);
212   }
213   if (clip_rect.IsEmpty()) {
214     return NULL;
215   }
216   if (dest_width == m_Width && dest_height == m_Height) {
217     return Clone(&clip_rect);
218   }
219   CFX_ImageStretcher stretcher;
220   CFX_BitmapStorer storer;
221   if (stretcher.Start(&storer, this, dest_width, dest_height, clip_rect,
222                       flags)) {
223     stretcher.Continue(NULL);
224   }
225   return storer.Detach();
226 }
227 CFX_ImageTransformer::CFX_ImageTransformer() {
228   m_Status = 0;
229   m_pMatrix = NULL;
230 }
231 CFX_ImageTransformer::~CFX_ImageTransformer() {}
232 FX_BOOL CFX_ImageTransformer::Start(const CFX_DIBSource* pSrc,
233                                     const CFX_AffineMatrix* pDestMatrix,
234                                     int flags,
235                                     const FX_RECT* pDestClip) {
236   m_pMatrix = (CFX_AffineMatrix*)pDestMatrix;
237   CFX_FloatRect unit_rect = pDestMatrix->GetUnitRect();
238   FX_RECT result_rect = unit_rect.GetClosestRect();
239   FX_RECT result_clip = result_rect;
240   if (pDestClip) {
241     result_clip.Intersect(*pDestClip);
242   }
243   if (result_clip.IsEmpty()) {
244     return FALSE;
245   }
246   m_ResultLeft = result_clip.left;
247   m_ResultTop = result_clip.top;
248   m_ResultWidth = result_clip.Width();
249   m_ResultHeight = result_clip.Height();
250   m_Flags = flags;
251   if (FXSYS_fabs(pDestMatrix->a) < FXSYS_fabs(pDestMatrix->b) / 20 &&
252       FXSYS_fabs(pDestMatrix->d) < FXSYS_fabs(pDestMatrix->c) / 20 &&
253       FXSYS_fabs(pDestMatrix->a) < 0.5f && FXSYS_fabs(pDestMatrix->d) < 0.5f) {
254     int dest_width = result_rect.Width();
255     int dest_height = result_rect.Height();
256     result_clip.Offset(-result_rect.left, -result_rect.top);
257     result_clip = FXDIB_SwapClipBox(result_clip, dest_width, dest_height,
258                                     pDestMatrix->c > 0, pDestMatrix->b < 0);
259     m_Stretcher.Start(&m_Storer, pSrc, dest_height, dest_width, result_clip,
260                       flags);
261     m_Status = 1;
262     return TRUE;
263   }
264   if (FXSYS_fabs(pDestMatrix->b) < FIX16_005 &&
265       FXSYS_fabs(pDestMatrix->c) < FIX16_005) {
266     int dest_width = pDestMatrix->a > 0 ? (int)FXSYS_ceil(pDestMatrix->a)
267                                         : (int)FXSYS_floor(pDestMatrix->a);
268     int dest_height = pDestMatrix->d > 0 ? (int)-FXSYS_ceil(pDestMatrix->d)
269                                          : (int)-FXSYS_floor(pDestMatrix->d);
270     result_clip.Offset(-result_rect.left, -result_rect.top);
271     m_Stretcher.Start(&m_Storer, pSrc, dest_width, dest_height, result_clip,
272                       flags);
273     m_Status = 2;
274     return TRUE;
275   }
276   int stretch_width =
277       (int)FXSYS_ceil(FXSYS_sqrt2(pDestMatrix->a, pDestMatrix->b));
278   int stretch_height =
279       (int)FXSYS_ceil(FXSYS_sqrt2(pDestMatrix->c, pDestMatrix->d));
280   CFX_AffineMatrix stretch2dest(1.0f, 0.0f, 0.0f, -1.0f, 0.0f,
281                                 (FX_FLOAT)(stretch_height));
282   stretch2dest.Concat(
283       pDestMatrix->a / stretch_width, pDestMatrix->b / stretch_width,
284       pDestMatrix->c / stretch_height, pDestMatrix->d / stretch_height,
285       pDestMatrix->e, pDestMatrix->f);
286   m_dest2stretch.SetReverse(stretch2dest);
287   CFX_FloatRect clip_rect_f(result_clip);
288   clip_rect_f.Transform(&m_dest2stretch);
289   m_StretchClip = clip_rect_f.GetOutterRect();
290   m_StretchClip.Intersect(0, 0, stretch_width, stretch_height);
291   m_Stretcher.Start(&m_Storer, pSrc, stretch_width, stretch_height,
292                     m_StretchClip, flags);
293   m_Status = 3;
294   return TRUE;
295 }
296 uint8_t _bilinear_interpol(const uint8_t* buf,
297                            int row_offset_l,
298                            int row_offset_r,
299                            int src_col_l,
300                            int src_col_r,
301                            int res_x,
302                            int res_y,
303                            int bpp,
304                            int c_offset) {
305   int i_resx = 255 - res_x;
306   int col_bpp_l = src_col_l * bpp;
307   int col_bpp_r = src_col_r * bpp;
308   const uint8_t* buf_u = buf + row_offset_l + c_offset;
309   const uint8_t* buf_d = buf + row_offset_r + c_offset;
310   const uint8_t* src_pos0 = buf_u + col_bpp_l;
311   const uint8_t* src_pos1 = buf_u + col_bpp_r;
312   const uint8_t* src_pos2 = buf_d + col_bpp_l;
313   const uint8_t* src_pos3 = buf_d + col_bpp_r;
314   uint8_t r_pos_0 = (*src_pos0 * i_resx + *src_pos1 * res_x) >> 8;
315   uint8_t r_pos_1 = (*src_pos2 * i_resx + *src_pos3 * res_x) >> 8;
316   return (r_pos_0 * (255 - res_y) + r_pos_1 * res_y) >> 8;
317 }
318 uint8_t _bicubic_interpol(const uint8_t* buf,
319                           int pitch,
320                           int pos_pixel[],
321                           int u_w[],
322                           int v_w[],
323                           int res_x,
324                           int res_y,
325                           int bpp,
326                           int c_offset) {
327   int s_result = 0;
328   for (int i = 0; i < 4; i++) {
329     int a_result = 0;
330     for (int j = 0; j < 4; j++) {
331       a_result += u_w[j] * (*(uint8_t*)(buf + pos_pixel[i + 4] * pitch +
332                                         pos_pixel[j] * bpp + c_offset));
333     }
334     s_result += a_result * v_w[i];
335   }
336   s_result >>= 16;
337   return (uint8_t)(s_result < 0 ? 0 : s_result > 255 ? 255 : s_result);
338 }
339 void _bicubic_get_pos_weight(int pos_pixel[],
340                              int u_w[],
341                              int v_w[],
342                              int src_col_l,
343                              int src_row_l,
344                              int res_x,
345                              int res_y,
346                              int stretch_width,
347                              int stretch_height) {
348   pos_pixel[0] = src_col_l - 1;
349   pos_pixel[1] = src_col_l;
350   pos_pixel[2] = src_col_l + 1;
351   pos_pixel[3] = src_col_l + 2;
352   pos_pixel[4] = src_row_l - 1;
353   pos_pixel[5] = src_row_l;
354   pos_pixel[6] = src_row_l + 1;
355   pos_pixel[7] = src_row_l + 2;
356   for (int i = 0; i < 4; i++) {
357     if (pos_pixel[i] < 0) {
358       pos_pixel[i] = 0;
359     }
360     if (pos_pixel[i] >= stretch_width) {
361       pos_pixel[i] = stretch_width - 1;
362     }
363     if (pos_pixel[i + 4] < 0) {
364       pos_pixel[i + 4] = 0;
365     }
366     if (pos_pixel[i + 4] >= stretch_height) {
367       pos_pixel[i + 4] = stretch_height - 1;
368     }
369   }
370   u_w[0] = SDP_Table[256 + res_x];
371   u_w[1] = SDP_Table[res_x];
372   u_w[2] = SDP_Table[256 - res_x];
373   u_w[3] = SDP_Table[512 - res_x];
374   v_w[0] = SDP_Table[256 + res_y];
375   v_w[1] = SDP_Table[res_y];
376   v_w[2] = SDP_Table[256 - res_y];
377   v_w[3] = SDP_Table[512 - res_y];
378 }
379 FXDIB_Format _GetTransformedFormat(const CFX_DIBSource* pDrc) {
380   FXDIB_Format format = pDrc->GetFormat();
381   if (pDrc->IsAlphaMask()) {
382     format = FXDIB_8bppMask;
383   } else if (format >= 1025) {
384     format = FXDIB_Cmyka;
385   } else if (format <= 32 || format == FXDIB_Argb) {
386     format = FXDIB_Argb;
387   } else {
388     format = FXDIB_Rgba;
389   }
390   return format;
391 }
392 FX_BOOL CFX_ImageTransformer::Continue(IFX_Pause* pPause) {
393   if (m_Status == 1) {
394     if (m_Stretcher.Continue(pPause)) {
395       return TRUE;
396     }
397     if (m_Storer.GetBitmap()) {
398       m_Storer.Replace(
399           m_Storer.GetBitmap()->SwapXY(m_pMatrix->c > 0, m_pMatrix->b < 0));
400     }
401     return FALSE;
402   }
403   if (m_Status == 2) {
404     return m_Stretcher.Continue(pPause);
405   }
406   if (m_Status != 3) {
407     return FALSE;
408   }
409   if (m_Stretcher.Continue(pPause)) {
410     return TRUE;
411   }
412   int stretch_width = m_StretchClip.Width();
413   int stretch_height = m_StretchClip.Height();
414   if (m_Storer.GetBitmap() == NULL) {
415     return FALSE;
416   }
417   const uint8_t* stretch_buf = m_Storer.GetBitmap()->GetBuffer();
418   const uint8_t* stretch_buf_mask = NULL;
419   if (m_Storer.GetBitmap()->m_pAlphaMask) {
420     stretch_buf_mask = m_Storer.GetBitmap()->m_pAlphaMask->GetBuffer();
421   }
422   int stretch_pitch = m_Storer.GetBitmap()->GetPitch();
423   CFX_DIBitmap* pTransformed = new CFX_DIBitmap;
424   FXDIB_Format transformF = _GetTransformedFormat(m_Stretcher.m_pSource);
425   if (!pTransformed->Create(m_ResultWidth, m_ResultHeight, transformF)) {
426     delete pTransformed;
427     return FALSE;
428   }
429   pTransformed->Clear(0);
430   if (pTransformed->m_pAlphaMask) {
431     pTransformed->m_pAlphaMask->Clear(0);
432   }
433   CFX_AffineMatrix result2stretch(1.0f, 0.0f, 0.0f, 1.0f,
434                                   (FX_FLOAT)(m_ResultLeft),
435                                   (FX_FLOAT)(m_ResultTop));
436   result2stretch.Concat(m_dest2stretch);
437   result2stretch.TranslateI(-m_StretchClip.left, -m_StretchClip.top);
438   if (stretch_buf_mask == NULL && pTransformed->m_pAlphaMask) {
439     pTransformed->m_pAlphaMask->Clear(0xff000000);
440   } else if (pTransformed->m_pAlphaMask) {
441     int stretch_pitch_mask = m_Storer.GetBitmap()->m_pAlphaMask->GetPitch();
442     if (!(m_Flags & FXDIB_DOWNSAMPLE) && !(m_Flags & FXDIB_BICUBIC_INTERPOL)) {
443       CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
444       for (int row = 0; row < m_ResultHeight; row++) {
445         uint8_t* dest_pos_mask =
446             (uint8_t*)pTransformed->m_pAlphaMask->GetScanline(row);
447         for (int col = 0; col < m_ResultWidth; col++) {
448           int src_col_l, src_row_l, res_x, res_y;
449           result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
450                                        res_y);
451           if (src_col_l >= 0 && src_col_l <= stretch_width && src_row_l >= 0 &&
452               src_row_l <= stretch_height) {
453             if (src_col_l == stretch_width) {
454               src_col_l--;
455             }
456             if (src_row_l == stretch_height) {
457               src_row_l--;
458             }
459             int src_col_r = src_col_l + 1;
460             int src_row_r = src_row_l + 1;
461             if (src_col_r == stretch_width) {
462               src_col_r--;
463             }
464             if (src_row_r == stretch_height) {
465               src_row_r--;
466             }
467             int row_offset_l = src_row_l * stretch_pitch_mask;
468             int row_offset_r = src_row_r * stretch_pitch_mask;
469             *dest_pos_mask =
470                 _bilinear_interpol(stretch_buf_mask, row_offset_l, row_offset_r,
471                                    src_col_l, src_col_r, res_x, res_y, 1, 0);
472           }
473           dest_pos_mask++;
474         }
475       }
476     } else if (m_Flags & FXDIB_BICUBIC_INTERPOL) {
477       CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
478       for (int row = 0; row < m_ResultHeight; row++) {
479         uint8_t* dest_pos_mask =
480             (uint8_t*)pTransformed->m_pAlphaMask->GetScanline(row);
481         for (int col = 0; col < m_ResultWidth; col++) {
482           int src_col_l, src_row_l, res_x, res_y;
483           result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
484                                        res_y);
485           if (src_col_l >= 0 && src_col_l <= stretch_width && src_row_l >= 0 &&
486               src_row_l <= stretch_height) {
487             int pos_pixel[8];
488             int u_w[4], v_w[4];
489             if (src_col_l == stretch_width) {
490               src_col_l--;
491             }
492             if (src_row_l == stretch_height) {
493               src_row_l--;
494             }
495             _bicubic_get_pos_weight(pos_pixel, u_w, v_w, src_col_l, src_row_l,
496                                     res_x, res_y, stretch_width,
497                                     stretch_height);
498             *dest_pos_mask =
499                 _bicubic_interpol(stretch_buf_mask, stretch_pitch_mask,
500                                   pos_pixel, u_w, v_w, res_x, res_y, 1, 0);
501           }
502           dest_pos_mask++;
503         }
504       }
505     } else {
506       CPDF_FixedMatrix result2stretch_fix(result2stretch, 8);
507       for (int row = 0; row < m_ResultHeight; row++) {
508         uint8_t* dest_pos_mask =
509             (uint8_t*)pTransformed->m_pAlphaMask->GetScanline(row);
510         for (int col = 0; col < m_ResultWidth; col++) {
511           int src_col, src_row;
512           result2stretch_fix.Transform(col, row, src_col, src_row);
513           if (src_col >= 0 && src_col <= stretch_width && src_row >= 0 &&
514               src_row <= stretch_height) {
515             if (src_col == stretch_width) {
516               src_col--;
517             }
518             if (src_row == stretch_height) {
519               src_row--;
520             }
521             *dest_pos_mask =
522                 stretch_buf_mask[src_row * stretch_pitch_mask + src_col];
523           }
524           dest_pos_mask++;
525         }
526       }
527     }
528   }
529   if (m_Storer.GetBitmap()->IsAlphaMask()) {
530     if (!(m_Flags & FXDIB_DOWNSAMPLE) && !(m_Flags & FXDIB_BICUBIC_INTERPOL)) {
531       CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
532       for (int row = 0; row < m_ResultHeight; row++) {
533         uint8_t* dest_scan = (uint8_t*)pTransformed->GetScanline(row);
534         for (int col = 0; col < m_ResultWidth; col++) {
535           int src_col_l, src_row_l, res_x, res_y;
536           result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
537                                        res_y);
538           if (src_col_l >= 0 && src_col_l <= stretch_width && src_row_l >= 0 &&
539               src_row_l <= stretch_height) {
540             if (src_col_l == stretch_width) {
541               src_col_l--;
542             }
543             if (src_row_l == stretch_height) {
544               src_row_l--;
545             }
546             int src_col_r = src_col_l + 1;
547             int src_row_r = src_row_l + 1;
548             if (src_col_r == stretch_width) {
549               src_col_r--;
550             }
551             if (src_row_r == stretch_height) {
552               src_row_r--;
553             }
554             int row_offset_l = src_row_l * stretch_pitch;
555             int row_offset_r = src_row_r * stretch_pitch;
556             *dest_scan =
557                 _bilinear_interpol(stretch_buf, row_offset_l, row_offset_r,
558                                    src_col_l, src_col_r, res_x, res_y, 1, 0);
559           }
560           dest_scan++;
561         }
562       }
563     } else if (m_Flags & FXDIB_BICUBIC_INTERPOL) {
564       CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
565       for (int row = 0; row < m_ResultHeight; row++) {
566         uint8_t* dest_scan = (uint8_t*)pTransformed->GetScanline(row);
567         for (int col = 0; col < m_ResultWidth; col++) {
568           int src_col_l, src_row_l, res_x, res_y;
569           result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
570                                        res_y);
571           if (src_col_l >= 0 && src_col_l <= stretch_width && src_row_l >= 0 &&
572               src_row_l <= stretch_height) {
573             int pos_pixel[8];
574             int u_w[4], v_w[4];
575             if (src_col_l == stretch_width) {
576               src_col_l--;
577             }
578             if (src_row_l == stretch_height) {
579               src_row_l--;
580             }
581             _bicubic_get_pos_weight(pos_pixel, u_w, v_w, src_col_l, src_row_l,
582                                     res_x, res_y, stretch_width,
583                                     stretch_height);
584             *dest_scan =
585                 _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel, u_w,
586                                   v_w, res_x, res_y, 1, 0);
587           }
588           dest_scan++;
589         }
590       }
591     } else {
592       CPDF_FixedMatrix result2stretch_fix(result2stretch, 8);
593       for (int row = 0; row < m_ResultHeight; row++) {
594         uint8_t* dest_scan = (uint8_t*)pTransformed->GetScanline(row);
595         for (int col = 0; col < m_ResultWidth; col++) {
596           int src_col, src_row;
597           result2stretch_fix.Transform(col, row, src_col, src_row);
598           if (src_col >= 0 && src_col <= stretch_width && src_row >= 0 &&
599               src_row <= stretch_height) {
600             if (src_col == stretch_width) {
601               src_col--;
602             }
603             if (src_row == stretch_height) {
604               src_row--;
605             }
606             const uint8_t* src_pixel =
607                 stretch_buf + stretch_pitch * src_row + src_col;
608             *dest_scan = *src_pixel;
609           }
610           dest_scan++;
611         }
612       }
613     }
614   } else {
615     int Bpp = m_Storer.GetBitmap()->GetBPP() / 8;
616     int destBpp = pTransformed->GetBPP() / 8;
617     if (Bpp == 1) {
618       FX_DWORD argb[256];
619       FX_ARGB* pPal = m_Storer.GetBitmap()->GetPalette();
620       if (pPal) {
621         for (int i = 0; i < 256; i++) {
622           argb[i] = pPal[i];
623         }
624       } else {
625         if (m_Storer.GetBitmap()->IsCmykImage())
626           for (int i = 0; i < 256; i++) {
627             argb[i] = 255 - i;
628           }
629         else
630           for (int i = 0; i < 256; i++) {
631             argb[i] = 0xff000000 | (i * 0x010101);
632           }
633       }
634       if (!(m_Flags & FXDIB_DOWNSAMPLE) &&
635           !(m_Flags & FXDIB_BICUBIC_INTERPOL)) {
636         CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
637         for (int row = 0; row < m_ResultHeight; row++) {
638           uint8_t* dest_pos = (uint8_t*)pTransformed->GetScanline(row);
639           for (int col = 0; col < m_ResultWidth; col++) {
640             int src_col_l, src_row_l, res_x, res_y;
641             result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
642                                          res_y);
643             if (src_col_l >= 0 && src_col_l <= stretch_width &&
644                 src_row_l >= 0 && src_row_l <= stretch_height) {
645               if (src_col_l == stretch_width) {
646                 src_col_l--;
647               }
648               if (src_row_l == stretch_height) {
649                 src_row_l--;
650               }
651               int src_col_r = src_col_l + 1;
652               int src_row_r = src_row_l + 1;
653               if (src_col_r == stretch_width) {
654                 src_col_r--;
655               }
656               if (src_row_r == stretch_height) {
657                 src_row_r--;
658               }
659               int row_offset_l = src_row_l * stretch_pitch;
660               int row_offset_r = src_row_r * stretch_pitch;
661               FX_DWORD r_bgra_cmyk = argb[_bilinear_interpol(
662                   stretch_buf, row_offset_l, row_offset_r, src_col_l, src_col_r,
663                   res_x, res_y, 1, 0)];
664               if (transformF == FXDIB_Rgba) {
665                 dest_pos[0] = (uint8_t)(r_bgra_cmyk >> 24);
666                 dest_pos[1] = (uint8_t)(r_bgra_cmyk >> 16);
667                 dest_pos[2] = (uint8_t)(r_bgra_cmyk >> 8);
668               } else {
669                 *(FX_DWORD*)dest_pos = r_bgra_cmyk;
670               }
671             }
672             dest_pos += destBpp;
673           }
674         }
675       } else if (m_Flags & FXDIB_BICUBIC_INTERPOL) {
676         CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
677         for (int row = 0; row < m_ResultHeight; row++) {
678           uint8_t* dest_pos = (uint8_t*)pTransformed->GetScanline(row);
679           for (int col = 0; col < m_ResultWidth; col++) {
680             int src_col_l, src_row_l, res_x, res_y;
681             result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
682                                          res_y);
683             if (src_col_l >= 0 && src_col_l <= stretch_width &&
684                 src_row_l >= 0 && src_row_l <= stretch_height) {
685               int pos_pixel[8];
686               int u_w[4], v_w[4];
687               if (src_col_l == stretch_width) {
688                 src_col_l--;
689               }
690               if (src_row_l == stretch_height) {
691                 src_row_l--;
692               }
693               _bicubic_get_pos_weight(pos_pixel, u_w, v_w, src_col_l, src_row_l,
694                                       res_x, res_y, stretch_width,
695                                       stretch_height);
696               FX_DWORD r_bgra_cmyk =
697                   argb[_bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel,
698                                          u_w, v_w, res_x, res_y, 1, 0)];
699               if (transformF == FXDIB_Rgba) {
700                 dest_pos[0] = (uint8_t)(r_bgra_cmyk >> 24);
701                 dest_pos[1] = (uint8_t)(r_bgra_cmyk >> 16);
702                 dest_pos[2] = (uint8_t)(r_bgra_cmyk >> 8);
703               } else {
704                 *(FX_DWORD*)dest_pos = r_bgra_cmyk;
705               }
706             }
707             dest_pos += destBpp;
708           }
709         }
710       } else {
711         CPDF_FixedMatrix result2stretch_fix(result2stretch, 8);
712         for (int row = 0; row < m_ResultHeight; row++) {
713           uint8_t* dest_pos = (uint8_t*)pTransformed->GetScanline(row);
714           for (int col = 0; col < m_ResultWidth; col++) {
715             int src_col, src_row;
716             result2stretch_fix.Transform(col, row, src_col, src_row);
717             if (src_col >= 0 && src_col <= stretch_width && src_row >= 0 &&
718                 src_row <= stretch_height) {
719               if (src_col == stretch_width) {
720                 src_col--;
721               }
722               if (src_row == stretch_height) {
723                 src_row--;
724               }
725               FX_DWORD r_bgra_cmyk =
726                   argb[stretch_buf[src_row * stretch_pitch + src_col]];
727               if (transformF == FXDIB_Rgba) {
728                 dest_pos[0] = (uint8_t)(r_bgra_cmyk >> 24);
729                 dest_pos[1] = (uint8_t)(r_bgra_cmyk >> 16);
730                 dest_pos[2] = (uint8_t)(r_bgra_cmyk >> 8);
731               } else {
732                 *(FX_DWORD*)dest_pos = r_bgra_cmyk;
733               }
734             }
735             dest_pos += destBpp;
736           }
737         }
738       }
739     } else {
740       FX_BOOL bHasAlpha = m_Storer.GetBitmap()->HasAlpha();
741       int destBpp = pTransformed->GetBPP() / 8;
742       if (!(m_Flags & FXDIB_DOWNSAMPLE) &&
743           !(m_Flags & FXDIB_BICUBIC_INTERPOL)) {
744         CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
745         for (int row = 0; row < m_ResultHeight; row++) {
746           uint8_t* dest_pos = (uint8_t*)pTransformed->GetScanline(row);
747           for (int col = 0; col < m_ResultWidth; col++) {
748             int src_col_l, src_row_l, res_x, res_y, r_pos_k_r = 0;
749             result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
750                                          res_y);
751             if (src_col_l >= 0 && src_col_l <= stretch_width &&
752                 src_row_l >= 0 && src_row_l <= stretch_height) {
753               if (src_col_l == stretch_width) {
754                 src_col_l--;
755               }
756               if (src_row_l == stretch_height) {
757                 src_row_l--;
758               }
759               int src_col_r = src_col_l + 1;
760               int src_row_r = src_row_l + 1;
761               if (src_col_r == stretch_width) {
762                 src_col_r--;
763               }
764               if (src_row_r == stretch_height) {
765                 src_row_r--;
766               }
767               int row_offset_l = src_row_l * stretch_pitch;
768               int row_offset_r = src_row_r * stretch_pitch;
769               uint8_t r_pos_red_y_r = _bilinear_interpol(
770                   stretch_buf, row_offset_l, row_offset_r, src_col_l, src_col_r,
771                   res_x, res_y, Bpp, 2);
772               uint8_t r_pos_green_m_r = _bilinear_interpol(
773                   stretch_buf, row_offset_l, row_offset_r, src_col_l, src_col_r,
774                   res_x, res_y, Bpp, 1);
775               uint8_t r_pos_blue_c_r = _bilinear_interpol(
776                   stretch_buf, row_offset_l, row_offset_r, src_col_l, src_col_r,
777                   res_x, res_y, Bpp, 0);
778               if (bHasAlpha) {
779                 if (transformF != FXDIB_Argb) {
780                   if (transformF == FXDIB_Rgba) {
781                     dest_pos[0] = r_pos_blue_c_r;
782                     dest_pos[1] = r_pos_green_m_r;
783                     dest_pos[2] = r_pos_red_y_r;
784                   } else {
785                     r_pos_k_r = _bilinear_interpol(
786                         stretch_buf, row_offset_l, row_offset_r, src_col_l,
787                         src_col_r, res_x, res_y, Bpp, 3);
788                     *(FX_DWORD*)dest_pos =
789                         FXCMYK_TODIB(CmykEncode(r_pos_blue_c_r, r_pos_green_m_r,
790                                                 r_pos_red_y_r, r_pos_k_r));
791                   }
792                 } else {
793                   uint8_t r_pos_a_r = _bilinear_interpol(
794                       stretch_buf, row_offset_l, row_offset_r, src_col_l,
795                       src_col_r, res_x, res_y, Bpp, 3);
796                   *(FX_DWORD*)dest_pos = FXARGB_TODIB(
797                       FXARGB_MAKE(r_pos_a_r, r_pos_red_y_r, r_pos_green_m_r,
798                                   r_pos_blue_c_r));
799                 }
800               } else {
801                 r_pos_k_r = 0xff;
802                 if (transformF == FXDIB_Cmyka) {
803                   r_pos_k_r = _bilinear_interpol(
804                       stretch_buf, row_offset_l, row_offset_r, src_col_l,
805                       src_col_r, res_x, res_y, Bpp, 3);
806                   *(FX_DWORD*)dest_pos =
807                       FXCMYK_TODIB(CmykEncode(r_pos_blue_c_r, r_pos_green_m_r,
808                                               r_pos_red_y_r, r_pos_k_r));
809                 } else {
810                   *(FX_DWORD*)dest_pos = FXARGB_TODIB(
811                       FXARGB_MAKE(r_pos_k_r, r_pos_red_y_r, r_pos_green_m_r,
812                                   r_pos_blue_c_r));
813                 }
814               }
815             }
816             dest_pos += destBpp;
817           }
818         }
819       } else if (m_Flags & FXDIB_BICUBIC_INTERPOL) {
820         CFX_BilinearMatrix result2stretch_fix(result2stretch, 8);
821         for (int row = 0; row < m_ResultHeight; row++) {
822           uint8_t* dest_pos = (uint8_t*)pTransformed->GetScanline(row);
823           for (int col = 0; col < m_ResultWidth; col++) {
824             int src_col_l, src_row_l, res_x, res_y, r_pos_k_r = 0;
825             result2stretch_fix.Transform(col, row, src_col_l, src_row_l, res_x,
826                                          res_y);
827             if (src_col_l >= 0 && src_col_l <= stretch_width &&
828                 src_row_l >= 0 && src_row_l <= stretch_height) {
829               int pos_pixel[8];
830               int u_w[4], v_w[4];
831               if (src_col_l == stretch_width) {
832                 src_col_l--;
833               }
834               if (src_row_l == stretch_height) {
835                 src_row_l--;
836               }
837               _bicubic_get_pos_weight(pos_pixel, u_w, v_w, src_col_l, src_row_l,
838                                       res_x, res_y, stretch_width,
839                                       stretch_height);
840               uint8_t r_pos_red_y_r =
841                   _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel, u_w,
842                                     v_w, res_x, res_y, Bpp, 2);
843               uint8_t r_pos_green_m_r =
844                   _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel, u_w,
845                                     v_w, res_x, res_y, Bpp, 1);
846               uint8_t r_pos_blue_c_r =
847                   _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel, u_w,
848                                     v_w, res_x, res_y, Bpp, 0);
849               if (bHasAlpha) {
850                 if (transformF != FXDIB_Argb) {
851                   if (transformF == FXDIB_Rgba) {
852                     dest_pos[0] = r_pos_blue_c_r;
853                     dest_pos[1] = r_pos_green_m_r;
854                     dest_pos[2] = r_pos_red_y_r;
855                   } else {
856                     r_pos_k_r =
857                         _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel,
858                                           u_w, v_w, res_x, res_y, Bpp, 3);
859                     *(FX_DWORD*)dest_pos =
860                         FXCMYK_TODIB(CmykEncode(r_pos_blue_c_r, r_pos_green_m_r,
861                                                 r_pos_red_y_r, r_pos_k_r));
862                   }
863                 } else {
864                   uint8_t r_pos_a_r =
865                       _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel,
866                                         u_w, v_w, res_x, res_y, Bpp, 3);
867                   *(FX_DWORD*)dest_pos = FXARGB_TODIB(
868                       FXARGB_MAKE(r_pos_a_r, r_pos_red_y_r, r_pos_green_m_r,
869                                   r_pos_blue_c_r));
870                 }
871               } else {
872                 r_pos_k_r = 0xff;
873                 if (transformF == FXDIB_Cmyka) {
874                   r_pos_k_r =
875                       _bicubic_interpol(stretch_buf, stretch_pitch, pos_pixel,
876                                         u_w, v_w, res_x, res_y, Bpp, 3);
877                   *(FX_DWORD*)dest_pos =
878                       FXCMYK_TODIB(CmykEncode(r_pos_blue_c_r, r_pos_green_m_r,
879                                               r_pos_red_y_r, r_pos_k_r));
880                 } else {
881                   *(FX_DWORD*)dest_pos = FXARGB_TODIB(
882                       FXARGB_MAKE(r_pos_k_r, r_pos_red_y_r, r_pos_green_m_r,
883                                   r_pos_blue_c_r));
884                 }
885               }
886             }
887             dest_pos += destBpp;
888           }
889         }
890       } else {
891         CPDF_FixedMatrix result2stretch_fix(result2stretch, 8);
892         for (int row = 0; row < m_ResultHeight; row++) {
893           uint8_t* dest_pos = (uint8_t*)pTransformed->GetScanline(row);
894           for (int col = 0; col < m_ResultWidth; col++) {
895             int src_col, src_row;
896             result2stretch_fix.Transform(col, row, src_col, src_row);
897             if (src_col >= 0 && src_col <= stretch_width && src_row >= 0 &&
898                 src_row <= stretch_height) {
899               if (src_col == stretch_width) {
900                 src_col--;
901               }
902               if (src_row == stretch_height) {
903                 src_row--;
904               }
905               const uint8_t* src_pos =
906                   stretch_buf + src_row * stretch_pitch + src_col * Bpp;
907               if (bHasAlpha) {
908                 if (transformF != FXDIB_Argb) {
909                   if (transformF == FXDIB_Rgba) {
910                     dest_pos[0] = src_pos[0];
911                     dest_pos[1] = src_pos[1];
912                     dest_pos[2] = src_pos[2];
913                   } else {
914                     *(FX_DWORD*)dest_pos = FXCMYK_TODIB(CmykEncode(
915                         src_pos[0], src_pos[1], src_pos[2], src_pos[3]));
916                   }
917                 } else {
918                   *(FX_DWORD*)dest_pos = FXARGB_TODIB(FXARGB_MAKE(
919                       src_pos[3], src_pos[2], src_pos[1], src_pos[0]));
920                 }
921               } else {
922                 if (transformF == FXDIB_Cmyka) {
923                   *(FX_DWORD*)dest_pos = FXCMYK_TODIB(CmykEncode(
924                       src_pos[0], src_pos[1], src_pos[2], src_pos[3]));
925                 } else {
926                   *(FX_DWORD*)dest_pos = FXARGB_TODIB(
927                       FXARGB_MAKE(0xff, src_pos[2], src_pos[1], src_pos[0]));
928                 }
929               }
930             }
931             dest_pos += destBpp;
932           }
933         }
934       }
935     }
936   }
937   m_Storer.Replace(pTransformed);
938   return FALSE;
939 }