• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 "core/include/fxge/fx_dib.h"
8 
9 #include <limits.h>
10 
11 #include "core/include/fxge/fx_ge.h"
12 #include "core/include/fxcodec/fx_codec.h"
13 #include "dib_int.h"
14 
15 FX_BOOL ConvertBuffer(FXDIB_Format dest_format,
16                       uint8_t* dest_buf,
17                       int dest_pitch,
18                       int width,
19                       int height,
20                       const CFX_DIBSource* pSrcBitmap,
21                       int src_left,
22                       int src_top,
23                       FX_DWORD*& pal,
24                       void* pIccTransform);
CmykDecode(FX_DWORD cmyk,int & c,int & m,int & y,int & k)25 void CmykDecode(FX_DWORD cmyk, int& c, int& m, int& y, int& k) {
26   c = FXSYS_GetCValue(cmyk);
27   m = FXSYS_GetMValue(cmyk);
28   y = FXSYS_GetYValue(cmyk);
29   k = FXSYS_GetKValue(cmyk);
30 }
ArgbDecode(FX_DWORD argb,int & a,int & r,int & g,int & b)31 void ArgbDecode(FX_DWORD argb, int& a, int& r, int& g, int& b) {
32   a = FXARGB_A(argb);
33   r = FXARGB_R(argb);
34   g = FXARGB_G(argb);
35   b = FXARGB_B(argb);
36 }
ArgbDecode(FX_DWORD argb,int & a,FX_COLORREF & rgb)37 void ArgbDecode(FX_DWORD argb, int& a, FX_COLORREF& rgb) {
38   a = FXARGB_A(argb);
39   rgb = FXSYS_RGB(FXARGB_R(argb), FXARGB_G(argb), FXARGB_B(argb));
40 }
ArgbEncode(int a,FX_COLORREF rgb)41 FX_DWORD ArgbEncode(int a, FX_COLORREF rgb) {
42   return FXARGB_MAKE(a, FXSYS_GetRValue(rgb), FXSYS_GetGValue(rgb),
43                      FXSYS_GetBValue(rgb));
44 }
CFX_DIBSource()45 CFX_DIBSource::CFX_DIBSource() {
46   m_bpp = 0;
47   m_AlphaFlag = 0;
48   m_Width = m_Height = 0;
49   m_Pitch = 0;
50   m_pPalette = NULL;
51   m_pAlphaMask = NULL;
52 }
~CFX_DIBSource()53 CFX_DIBSource::~CFX_DIBSource() {
54   FX_Free(m_pPalette);
55   delete m_pAlphaMask;
56 }
CFX_DIBitmap()57 CFX_DIBitmap::CFX_DIBitmap() {
58   m_bExtBuf = FALSE;
59   m_pBuffer = NULL;
60   m_pPalette = NULL;
61 }
62 #define _MAX_OOM_LIMIT_ 12000000
Create(int width,int height,FXDIB_Format format,uint8_t * pBuffer,int pitch)63 FX_BOOL CFX_DIBitmap::Create(int width,
64                              int height,
65                              FXDIB_Format format,
66                              uint8_t* pBuffer,
67                              int pitch) {
68   m_pBuffer = NULL;
69   m_bpp = (uint8_t)format;
70   m_AlphaFlag = (uint8_t)(format >> 8);
71   m_Width = m_Height = m_Pitch = 0;
72   if (width <= 0 || height <= 0 || pitch < 0) {
73     return FALSE;
74   }
75   if ((INT_MAX - 31) / width < (format & 0xff)) {
76     return FALSE;
77   }
78   if (!pitch) {
79     pitch = (width * (format & 0xff) + 31) / 32 * 4;
80   }
81   if ((1 << 30) / pitch < height) {
82     return FALSE;
83   }
84   if (pBuffer) {
85     m_pBuffer = pBuffer;
86     m_bExtBuf = TRUE;
87   } else {
88     int size = pitch * height + 4;
89     int oomlimit = _MAX_OOM_LIMIT_;
90     if (oomlimit >= 0 && size >= oomlimit) {
91       m_pBuffer = FX_TryAlloc(uint8_t, size);
92       if (!m_pBuffer) {
93         return FALSE;
94       }
95     } else {
96       m_pBuffer = FX_Alloc(uint8_t, size);
97     }
98   }
99   m_Width = width;
100   m_Height = height;
101   m_Pitch = pitch;
102   if (HasAlpha() && format != FXDIB_Argb) {
103     FX_BOOL ret = TRUE;
104     ret = BuildAlphaMask();
105     if (!ret) {
106       if (!m_bExtBuf) {
107         FX_Free(m_pBuffer);
108         m_pBuffer = NULL;
109         m_Width = m_Height = m_Pitch = 0;
110         return FALSE;
111       }
112     }
113   }
114   return TRUE;
115 }
Copy(const CFX_DIBSource * pSrc)116 FX_BOOL CFX_DIBitmap::Copy(const CFX_DIBSource* pSrc) {
117   if (m_pBuffer) {
118     return FALSE;
119   }
120   if (!Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat())) {
121     return FALSE;
122   }
123   CopyPalette(pSrc->GetPalette());
124   CopyAlphaMask(pSrc->m_pAlphaMask);
125   for (int row = 0; row < pSrc->GetHeight(); row++) {
126     FXSYS_memcpy(m_pBuffer + row * m_Pitch, pSrc->GetScanline(row), m_Pitch);
127   }
128   return TRUE;
129 }
~CFX_DIBitmap()130 CFX_DIBitmap::~CFX_DIBitmap() {
131   if (!m_bExtBuf) {
132     FX_Free(m_pBuffer);
133   }
134   m_pBuffer = NULL;
135 }
TakeOver(CFX_DIBitmap * pSrcBitmap)136 void CFX_DIBitmap::TakeOver(CFX_DIBitmap* pSrcBitmap) {
137   if (!m_bExtBuf) {
138     FX_Free(m_pBuffer);
139   }
140   FX_Free(m_pPalette);
141   delete m_pAlphaMask;
142   m_pBuffer = pSrcBitmap->m_pBuffer;
143   m_pPalette = pSrcBitmap->m_pPalette;
144   m_pAlphaMask = pSrcBitmap->m_pAlphaMask;
145   pSrcBitmap->m_pBuffer = NULL;
146   pSrcBitmap->m_pPalette = NULL;
147   pSrcBitmap->m_pAlphaMask = NULL;
148   m_bpp = pSrcBitmap->m_bpp;
149   m_bExtBuf = pSrcBitmap->m_bExtBuf;
150   m_AlphaFlag = pSrcBitmap->m_AlphaFlag;
151   m_Width = pSrcBitmap->m_Width;
152   m_Height = pSrcBitmap->m_Height;
153   m_Pitch = pSrcBitmap->m_Pitch;
154 }
Clone(const FX_RECT * pClip) const155 CFX_DIBitmap* CFX_DIBSource::Clone(const FX_RECT* pClip) const {
156   FX_RECT rect(0, 0, m_Width, m_Height);
157   if (pClip) {
158     rect.Intersect(*pClip);
159     if (rect.IsEmpty()) {
160       return NULL;
161     }
162   }
163   CFX_DIBitmap* pNewBitmap = new CFX_DIBitmap;
164   if (!pNewBitmap->Create(rect.Width(), rect.Height(), GetFormat())) {
165     delete pNewBitmap;
166     return NULL;
167   }
168   pNewBitmap->CopyPalette(m_pPalette);
169   pNewBitmap->CopyAlphaMask(m_pAlphaMask, pClip);
170   if (GetBPP() == 1 && rect.left % 8 != 0) {
171     int left_shift = rect.left % 32;
172     int right_shift = 32 - left_shift;
173     int dword_count = pNewBitmap->m_Pitch / 4;
174     for (int row = rect.top; row < rect.bottom; row++) {
175       FX_DWORD* src_scan = (FX_DWORD*)GetScanline(row) + rect.left / 32;
176       FX_DWORD* dest_scan = (FX_DWORD*)pNewBitmap->GetScanline(row - rect.top);
177       for (int i = 0; i < dword_count; i++) {
178         dest_scan[i] =
179             (src_scan[i] << left_shift) | (src_scan[i + 1] >> right_shift);
180       }
181     }
182   } else {
183     int copy_len = (pNewBitmap->GetWidth() * pNewBitmap->GetBPP() + 7) / 8;
184     if (m_Pitch < (FX_DWORD)copy_len) {
185       copy_len = m_Pitch;
186     }
187     for (int row = rect.top; row < rect.bottom; row++) {
188       const uint8_t* src_scan = GetScanline(row) + rect.left * m_bpp / 8;
189       uint8_t* dest_scan = (uint8_t*)pNewBitmap->GetScanline(row - rect.top);
190       FXSYS_memcpy(dest_scan, src_scan, copy_len);
191     }
192   }
193   return pNewBitmap;
194 }
BuildPalette()195 void CFX_DIBSource::BuildPalette() {
196   if (m_pPalette) {
197     return;
198   }
199   if (GetBPP() == 1) {
200     m_pPalette = FX_Alloc(FX_DWORD, 2);
201     if (IsCmykImage()) {
202       m_pPalette[0] = 0xff;
203       m_pPalette[1] = 0;
204     } else {
205       m_pPalette[0] = 0xff000000;
206       m_pPalette[1] = 0xffffffff;
207     }
208   } else if (GetBPP() == 8) {
209     m_pPalette = FX_Alloc(FX_DWORD, 256);
210     if (IsCmykImage()) {
211       for (int i = 0; i < 256; i++) {
212         m_pPalette[i] = 0xff - i;
213       }
214     } else {
215       for (int i = 0; i < 256; i++) {
216         m_pPalette[i] = 0xff000000 | (i * 0x10101);
217       }
218     }
219   }
220 }
BuildAlphaMask()221 FX_BOOL CFX_DIBSource::BuildAlphaMask() {
222   if (m_pAlphaMask) {
223     return TRUE;
224   }
225   m_pAlphaMask = new CFX_DIBitmap;
226   if (!m_pAlphaMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
227     delete m_pAlphaMask;
228     m_pAlphaMask = NULL;
229     return FALSE;
230   }
231   FXSYS_memset(m_pAlphaMask->GetBuffer(), 0xff,
232                m_pAlphaMask->GetHeight() * m_pAlphaMask->GetPitch());
233   return TRUE;
234 }
GetPaletteEntry(int index) const235 FX_DWORD CFX_DIBSource::GetPaletteEntry(int index) const {
236   ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
237   if (m_pPalette) {
238     return m_pPalette[index];
239   }
240   if (IsCmykImage()) {
241     if (GetBPP() == 1) {
242       return index ? 0 : 0xff;
243     }
244     return 0xff - index;
245   }
246   if (GetBPP() == 1) {
247     return index ? 0xffffffff : 0xff000000;
248   }
249   return index * 0x10101 | 0xff000000;
250 }
SetPaletteEntry(int index,FX_DWORD color)251 void CFX_DIBSource::SetPaletteEntry(int index, FX_DWORD color) {
252   ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
253   if (!m_pPalette) {
254     BuildPalette();
255   }
256   m_pPalette[index] = color;
257 }
FindPalette(FX_DWORD color) const258 int CFX_DIBSource::FindPalette(FX_DWORD color) const {
259   ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
260   if (!m_pPalette) {
261     if (IsCmykImage()) {
262       if (GetBPP() == 1) {
263         return ((uint8_t)color == 0xff) ? 0 : 1;
264       }
265       return 0xff - (uint8_t)color;
266     }
267     if (GetBPP() == 1) {
268       return ((uint8_t)color == 0xff) ? 1 : 0;
269     }
270     return (uint8_t)color;
271   }
272   int palsize = (1 << GetBPP());
273   for (int i = 0; i < palsize; i++)
274     if (m_pPalette[i] == color) {
275       return i;
276     }
277   return -1;
278 }
Clear(FX_DWORD color)279 void CFX_DIBitmap::Clear(FX_DWORD color) {
280   if (!m_pBuffer) {
281     return;
282   }
283   switch (GetFormat()) {
284     case FXDIB_1bppMask:
285       FXSYS_memset(m_pBuffer, (color & 0xff000000) ? 0xff : 0,
286                    m_Pitch * m_Height);
287       break;
288     case FXDIB_1bppRgb: {
289       int index = FindPalette(color);
290       FXSYS_memset(m_pBuffer, index ? 0xff : 0, m_Pitch * m_Height);
291       break;
292     }
293     case FXDIB_8bppMask:
294       FXSYS_memset(m_pBuffer, color >> 24, m_Pitch * m_Height);
295       break;
296     case FXDIB_8bppRgb: {
297       int index = FindPalette(color);
298       FXSYS_memset(m_pBuffer, index, m_Pitch * m_Height);
299       break;
300     }
301     case FXDIB_Rgb:
302     case FXDIB_Rgba: {
303       int a, r, g, b;
304       ArgbDecode(color, a, r, g, b);
305       if (r == g && g == b) {
306         FXSYS_memset(m_pBuffer, r, m_Pitch * m_Height);
307       } else {
308         int byte_pos = 0;
309         for (int col = 0; col < m_Width; col++) {
310           m_pBuffer[byte_pos++] = b;
311           m_pBuffer[byte_pos++] = g;
312           m_pBuffer[byte_pos++] = r;
313         }
314         for (int row = 1; row < m_Height; row++) {
315           FXSYS_memcpy(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);
316         }
317       }
318       break;
319     }
320     case FXDIB_Rgb32:
321     case FXDIB_Argb: {
322       color = IsCmykImage() ? FXCMYK_TODIB(color) : FXARGB_TODIB(color);
323       for (int i = 0; i < m_Width; i++) {
324         ((FX_DWORD*)m_pBuffer)[i] = color;
325       }
326       for (int row = 1; row < m_Height; row++) {
327         FXSYS_memcpy(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);
328       }
329       break;
330     }
331     default:
332       break;
333   }
334 }
GetOverlapRect(int & dest_left,int & dest_top,int & width,int & height,int src_width,int src_height,int & src_left,int & src_top,const CFX_ClipRgn * pClipRgn)335 void CFX_DIBSource::GetOverlapRect(int& dest_left,
336                                    int& dest_top,
337                                    int& width,
338                                    int& height,
339                                    int src_width,
340                                    int src_height,
341                                    int& src_left,
342                                    int& src_top,
343                                    const CFX_ClipRgn* pClipRgn) {
344   if (width == 0 || height == 0) {
345     return;
346   }
347   ASSERT(width > 0 && height > 0);
348   if (dest_left > m_Width || dest_top > m_Height) {
349     width = 0;
350     height = 0;
351     return;
352   }
353   int x_offset = dest_left - src_left;
354   int y_offset = dest_top - src_top;
355   FX_RECT src_rect(src_left, src_top, src_left + width, src_top + height);
356   FX_RECT src_bound(0, 0, src_width, src_height);
357   src_rect.Intersect(src_bound);
358   FX_RECT dest_rect(src_rect.left + x_offset, src_rect.top + y_offset,
359                     src_rect.right + x_offset, src_rect.bottom + y_offset);
360   FX_RECT dest_bound(0, 0, m_Width, m_Height);
361   dest_rect.Intersect(dest_bound);
362   if (pClipRgn) {
363     dest_rect.Intersect(pClipRgn->GetBox());
364   }
365   dest_left = dest_rect.left;
366   dest_top = dest_rect.top;
367   src_left = dest_left - x_offset;
368   src_top = dest_top - y_offset;
369   width = dest_rect.right - dest_rect.left;
370   height = dest_rect.bottom - dest_rect.top;
371 }
TransferBitmap(int dest_left,int dest_top,int width,int height,const CFX_DIBSource * pSrcBitmap,int src_left,int src_top,void * pIccTransform)372 FX_BOOL CFX_DIBitmap::TransferBitmap(int dest_left,
373                                      int dest_top,
374                                      int width,
375                                      int height,
376                                      const CFX_DIBSource* pSrcBitmap,
377                                      int src_left,
378                                      int src_top,
379                                      void* pIccTransform) {
380   if (!m_pBuffer) {
381     return FALSE;
382   }
383   GetOverlapRect(dest_left, dest_top, width, height, pSrcBitmap->GetWidth(),
384                  pSrcBitmap->GetHeight(), src_left, src_top, NULL);
385   if (width == 0 || height == 0) {
386     return TRUE;
387   }
388   FXDIB_Format dest_format = GetFormat();
389   FXDIB_Format src_format = pSrcBitmap->GetFormat();
390   if (dest_format == src_format && !pIccTransform) {
391     if (GetBPP() == 1) {
392       for (int row = 0; row < height; row++) {
393         uint8_t* dest_scan = m_pBuffer + (dest_top + row) * m_Pitch;
394         const uint8_t* src_scan = pSrcBitmap->GetScanline(src_top + row);
395         for (int col = 0; col < width; col++) {
396           if (src_scan[(src_left + col) / 8] &
397               (1 << (7 - (src_left + col) % 8))) {
398             dest_scan[(dest_left + col) / 8] |= 1
399                                                 << (7 - (dest_left + col) % 8);
400           } else {
401             dest_scan[(dest_left + col) / 8] &=
402                 ~(1 << (7 - (dest_left + col) % 8));
403           }
404         }
405       }
406     } else {
407       int Bpp = GetBPP() / 8;
408       for (int row = 0; row < height; row++) {
409         uint8_t* dest_scan =
410             m_pBuffer + (dest_top + row) * m_Pitch + dest_left * Bpp;
411         const uint8_t* src_scan =
412             pSrcBitmap->GetScanline(src_top + row) + src_left * Bpp;
413         FXSYS_memcpy(dest_scan, src_scan, width * Bpp);
414       }
415     }
416   } else {
417     if (m_pPalette) {
418       return FALSE;
419     }
420     if (m_bpp == 8) {
421       dest_format = FXDIB_8bppMask;
422     }
423     uint8_t* dest_buf =
424         m_pBuffer + dest_top * m_Pitch + dest_left * GetBPP() / 8;
425     FX_DWORD* d_plt = NULL;
426     if (!ConvertBuffer(dest_format, dest_buf, m_Pitch, width, height,
427                        pSrcBitmap, src_left, src_top, d_plt, pIccTransform)) {
428       return FALSE;
429     }
430   }
431   return TRUE;
432 }
TransferMask(int dest_left,int dest_top,int width,int height,const CFX_DIBSource * pMask,FX_DWORD color,int src_left,int src_top,int alpha_flag,void * pIccTransform)433 FX_BOOL CFX_DIBitmap::TransferMask(int dest_left,
434                                    int dest_top,
435                                    int width,
436                                    int height,
437                                    const CFX_DIBSource* pMask,
438                                    FX_DWORD color,
439                                    int src_left,
440                                    int src_top,
441                                    int alpha_flag,
442                                    void* pIccTransform) {
443   if (!m_pBuffer) {
444     return FALSE;
445   }
446   ASSERT(HasAlpha() && (m_bpp >= 24));
447   ASSERT(pMask->IsAlphaMask());
448   if (!HasAlpha() || !pMask->IsAlphaMask() || m_bpp < 24) {
449     return FALSE;
450   }
451   GetOverlapRect(dest_left, dest_top, width, height, pMask->GetWidth(),
452                  pMask->GetHeight(), src_left, src_top, NULL);
453   if (width == 0 || height == 0) {
454     return TRUE;
455   }
456   int src_bpp = pMask->GetBPP();
457   int alpha;
458   FX_DWORD dst_color;
459   if (alpha_flag >> 8) {
460     alpha = alpha_flag & 0xff;
461     dst_color = FXCMYK_TODIB(color);
462   } else {
463     alpha = FXARGB_A(color);
464     dst_color = FXARGB_TODIB(color);
465   }
466   uint8_t* color_p = (uint8_t*)&dst_color;
467   if (pIccTransform && CFX_GEModule::Get()->GetCodecModule() &&
468       CFX_GEModule::Get()->GetCodecModule()->GetIccModule()) {
469     ICodec_IccModule* pIccModule =
470         CFX_GEModule::Get()->GetCodecModule()->GetIccModule();
471     pIccModule->TranslateScanline(pIccTransform, color_p, color_p, 1);
472   } else {
473     if (alpha_flag >> 8 && !IsCmykImage())
474       AdobeCMYK_to_sRGB1(FXSYS_GetCValue(color), FXSYS_GetMValue(color),
475                          FXSYS_GetYValue(color), FXSYS_GetKValue(color),
476                          color_p[2], color_p[1], color_p[0]);
477     else if (!(alpha_flag >> 8) && IsCmykImage()) {
478       return FALSE;
479     }
480   }
481   if (!IsCmykImage()) {
482     color_p[3] = (uint8_t)alpha;
483   }
484   if (GetFormat() == FXDIB_Argb) {
485     for (int row = 0; row < height; row++) {
486       FX_DWORD* dest_pos =
487           (FX_DWORD*)(m_pBuffer + (dest_top + row) * m_Pitch + dest_left * 4);
488       const uint8_t* src_scan = pMask->GetScanline(src_top + row);
489       if (src_bpp == 1) {
490         for (int col = 0; col < width; col++) {
491           int src_bitpos = src_left + col;
492           if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {
493             *dest_pos = dst_color;
494           } else {
495             *dest_pos = 0;
496           }
497           dest_pos++;
498         }
499       } else {
500         src_scan += src_left;
501         dst_color = FXARGB_TODIB(dst_color);
502         dst_color &= 0xffffff;
503         for (int col = 0; col < width; col++) {
504           FXARGB_SETDIB(dest_pos++,
505                         dst_color | ((alpha * (*src_scan++) / 255) << 24));
506         }
507       }
508     }
509   } else {
510     int comps = m_bpp / 8;
511     for (int row = 0; row < height; row++) {
512       uint8_t* dest_color_pos =
513           m_pBuffer + (dest_top + row) * m_Pitch + dest_left * comps;
514       uint8_t* dest_alpha_pos =
515           (uint8_t*)m_pAlphaMask->GetScanline(dest_top + row) + dest_left;
516       const uint8_t* src_scan = pMask->GetScanline(src_top + row);
517       if (src_bpp == 1) {
518         for (int col = 0; col < width; col++) {
519           int src_bitpos = src_left + col;
520           if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {
521             FXSYS_memcpy(dest_color_pos, color_p, comps);
522             *dest_alpha_pos = 0xff;
523           } else {
524             FXSYS_memset(dest_color_pos, 0, comps);
525             *dest_alpha_pos = 0;
526           }
527           dest_color_pos += comps;
528           dest_alpha_pos++;
529         }
530       } else {
531         src_scan += src_left;
532         for (int col = 0; col < width; col++) {
533           FXSYS_memcpy(dest_color_pos, color_p, comps);
534           dest_color_pos += comps;
535           *dest_alpha_pos++ = (alpha * (*src_scan++) / 255);
536         }
537       }
538     }
539   }
540   return TRUE;
541 }
CopyPalette(const FX_DWORD * pSrc,FX_DWORD size)542 void CFX_DIBSource::CopyPalette(const FX_DWORD* pSrc, FX_DWORD size) {
543   if (!pSrc || GetBPP() > 8) {
544     FX_Free(m_pPalette);
545     m_pPalette = NULL;
546   } else {
547     FX_DWORD pal_size = 1 << GetBPP();
548     if (!m_pPalette) {
549       m_pPalette = FX_Alloc(FX_DWORD, pal_size);
550     }
551     if (pal_size > size) {
552       pal_size = size;
553     }
554     FXSYS_memcpy(m_pPalette, pSrc, pal_size * sizeof(FX_DWORD));
555   }
556 }
GetPalette(FX_DWORD * pal,int alpha) const557 void CFX_DIBSource::GetPalette(FX_DWORD* pal, int alpha) const {
558   ASSERT(GetBPP() <= 8 && !IsCmykImage());
559   if (GetBPP() == 1) {
560     pal[0] =
561         ((m_pPalette ? m_pPalette[0] : 0xff000000) & 0xffffff) | (alpha << 24);
562     pal[1] =
563         ((m_pPalette ? m_pPalette[1] : 0xffffffff) & 0xffffff) | (alpha << 24);
564     return;
565   }
566   if (m_pPalette) {
567     for (int i = 0; i < 256; i++) {
568       pal[i] = (m_pPalette[i] & 0x00ffffff) | (alpha << 24);
569     }
570   } else {
571     for (int i = 0; i < 256; i++) {
572       pal[i] = (i * 0x10101) | (alpha << 24);
573     }
574   }
575 }
GetAlphaMask(const FX_RECT * pClip) const576 CFX_DIBitmap* CFX_DIBSource::GetAlphaMask(const FX_RECT* pClip) const {
577   ASSERT(GetFormat() == FXDIB_Argb);
578   FX_RECT rect(0, 0, m_Width, m_Height);
579   if (pClip) {
580     rect.Intersect(*pClip);
581     if (rect.IsEmpty()) {
582       return NULL;
583     }
584   }
585   CFX_DIBitmap* pMask = new CFX_DIBitmap;
586   if (!pMask->Create(rect.Width(), rect.Height(), FXDIB_8bppMask)) {
587     delete pMask;
588     return NULL;
589   }
590   for (int row = rect.top; row < rect.bottom; row++) {
591     const uint8_t* src_scan = GetScanline(row) + rect.left * 4 + 3;
592     uint8_t* dest_scan = (uint8_t*)pMask->GetScanline(row - rect.top);
593     for (int col = rect.left; col < rect.right; col++) {
594       *dest_scan++ = *src_scan;
595       src_scan += 4;
596     }
597   }
598   return pMask;
599 }
CopyAlphaMask(const CFX_DIBSource * pAlphaMask,const FX_RECT * pClip)600 FX_BOOL CFX_DIBSource::CopyAlphaMask(const CFX_DIBSource* pAlphaMask,
601                                      const FX_RECT* pClip) {
602   if (!HasAlpha() || GetFormat() == FXDIB_Argb) {
603     return FALSE;
604   }
605   if (pAlphaMask) {
606     FX_RECT rect(0, 0, pAlphaMask->m_Width, pAlphaMask->m_Height);
607     if (pClip) {
608       rect.Intersect(*pClip);
609       if (rect.IsEmpty() || rect.Width() != m_Width ||
610           rect.Height() != m_Height) {
611         return FALSE;
612       }
613     } else {
614       if (pAlphaMask->m_Width != m_Width || pAlphaMask->m_Height != m_Height) {
615         return FALSE;
616       }
617     }
618     for (int row = 0; row < m_Height; row++)
619       FXSYS_memcpy((void*)m_pAlphaMask->GetScanline(row),
620                    pAlphaMask->GetScanline(row + rect.top) + rect.left,
621                    m_pAlphaMask->m_Pitch);
622   } else {
623     m_pAlphaMask->Clear(0xff000000);
624   }
625   return TRUE;
626 }
627 const int g_ChannelOffset[] = {0, 2, 1, 0, 0, 1, 2, 3, 3};
LoadChannel(FXDIB_Channel destChannel,const CFX_DIBSource * pSrcBitmap,FXDIB_Channel srcChannel)628 FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel,
629                                   const CFX_DIBSource* pSrcBitmap,
630                                   FXDIB_Channel srcChannel) {
631   if (!m_pBuffer) {
632     return FALSE;
633   }
634   CFX_DIBSource* pSrcClone = (CFX_DIBSource*)pSrcBitmap;
635   CFX_DIBitmap* pDst = this;
636   int destOffset, srcOffset;
637   if (srcChannel == FXDIB_Alpha) {
638     if (!pSrcBitmap->HasAlpha() && !pSrcBitmap->IsAlphaMask()) {
639       return FALSE;
640     }
641     if (pSrcBitmap->GetBPP() == 1) {
642       pSrcClone = pSrcBitmap->CloneConvert(FXDIB_8bppMask);
643       if (!pSrcClone) {
644         return FALSE;
645       }
646     }
647     if (pSrcBitmap->GetFormat() == FXDIB_Argb) {
648       srcOffset = 3;
649     } else {
650       srcOffset = 0;
651     }
652   } else {
653     if (pSrcBitmap->IsAlphaMask()) {
654       return FALSE;
655     }
656     if (pSrcBitmap->GetBPP() < 24) {
657       if (pSrcBitmap->IsCmykImage()) {
658         pSrcClone = pSrcBitmap->CloneConvert(
659             (FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x20));
660       } else {
661         pSrcClone = pSrcBitmap->CloneConvert(
662             (FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x18));
663       }
664       if (!pSrcClone) {
665         return FALSE;
666       }
667     }
668     srcOffset = g_ChannelOffset[srcChannel];
669   }
670   if (destChannel == FXDIB_Alpha) {
671     if (IsAlphaMask()) {
672       if (!ConvertFormat(FXDIB_8bppMask)) {
673         if (pSrcClone != pSrcBitmap) {
674           delete pSrcClone;
675         }
676         return FALSE;
677       }
678       destOffset = 0;
679     } else {
680       destOffset = 0;
681       if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
682         if (pSrcClone != pSrcBitmap) {
683           delete pSrcClone;
684         }
685         return FALSE;
686       }
687       if (GetFormat() == FXDIB_Argb) {
688         destOffset = 3;
689       }
690     }
691   } else {
692     if (IsAlphaMask()) {
693       if (pSrcClone != pSrcBitmap) {
694         delete pSrcClone;
695       }
696       return FALSE;
697     }
698     if (GetBPP() < 24) {
699       if (HasAlpha()) {
700         if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
701           if (pSrcClone != pSrcBitmap) {
702             delete pSrcClone;
703           }
704           return FALSE;
705         }
706       } else
707 #if _FXM_PLATFORM_ == _FXM_PLATFORM_APPLE_
708           if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {
709 #else
710           if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {
711 #endif
712         if (pSrcClone != pSrcBitmap) {
713           delete pSrcClone;
714         }
715         return FALSE;
716       }
717     }
718     destOffset = g_ChannelOffset[destChannel];
719   }
720   if (srcChannel == FXDIB_Alpha && pSrcClone->m_pAlphaMask) {
721     CFX_DIBitmap* pAlphaMask = pSrcClone->m_pAlphaMask;
722     if (pSrcClone->GetWidth() != m_Width ||
723         pSrcClone->GetHeight() != m_Height) {
724       if (pAlphaMask) {
725         pAlphaMask = pAlphaMask->StretchTo(m_Width, m_Height);
726         if (!pAlphaMask) {
727           if (pSrcClone != pSrcBitmap) {
728             delete pSrcClone;
729           }
730           return FALSE;
731         }
732       }
733     }
734     if (pSrcClone != pSrcBitmap) {
735       pSrcClone->m_pAlphaMask = NULL;
736       delete pSrcClone;
737     }
738     pSrcClone = pAlphaMask;
739     srcOffset = 0;
740   } else if (pSrcClone->GetWidth() != m_Width ||
741              pSrcClone->GetHeight() != m_Height) {
742     CFX_DIBitmap* pSrcMatched = pSrcClone->StretchTo(m_Width, m_Height);
743     if (pSrcClone != pSrcBitmap) {
744       delete pSrcClone;
745     }
746     if (!pSrcMatched) {
747       return FALSE;
748     }
749     pSrcClone = pSrcMatched;
750   }
751   if (destChannel == FXDIB_Alpha && m_pAlphaMask) {
752     pDst = m_pAlphaMask;
753     destOffset = 0;
754   }
755   int srcBytes = pSrcClone->GetBPP() / 8;
756   int destBytes = pDst->GetBPP() / 8;
757   for (int row = 0; row < m_Height; row++) {
758     uint8_t* dest_pos = (uint8_t*)pDst->GetScanline(row) + destOffset;
759     const uint8_t* src_pos = pSrcClone->GetScanline(row) + srcOffset;
760     for (int col = 0; col < m_Width; col++) {
761       *dest_pos = *src_pos;
762       dest_pos += destBytes;
763       src_pos += srcBytes;
764     }
765   }
766   if (pSrcClone != pSrcBitmap && pSrcClone != pSrcBitmap->m_pAlphaMask) {
767     delete pSrcClone;
768   }
769   return TRUE;
770 }
771 FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, int value) {
772   if (!m_pBuffer) {
773     return FALSE;
774   }
775   int destOffset;
776   if (destChannel == FXDIB_Alpha) {
777     if (IsAlphaMask()) {
778       if (!ConvertFormat(FXDIB_8bppMask)) {
779         return FALSE;
780       }
781       destOffset = 0;
782     } else {
783       destOffset = 0;
784       if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
785         return FALSE;
786       }
787       if (GetFormat() == FXDIB_Argb) {
788         destOffset = 3;
789       }
790     }
791   } else {
792     if (IsAlphaMask()) {
793       return FALSE;
794     }
795     if (GetBPP() < 24) {
796       if (HasAlpha()) {
797         if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
798           return FALSE;
799         }
800       } else
801 #if _FXM_PLATFORM_ == _FXM_PLATFORM_APPLE_
802           if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {
803         return FALSE;
804       }
805 #else
806           if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {
807         return FALSE;
808       }
809 #endif
810     }
811     destOffset = g_ChannelOffset[destChannel];
812   }
813   int Bpp = GetBPP() / 8;
814   if (Bpp == 1) {
815     FXSYS_memset(m_pBuffer, value, m_Height * m_Pitch);
816     return TRUE;
817   }
818   if (destChannel == FXDIB_Alpha && m_pAlphaMask) {
819     FXSYS_memset(m_pAlphaMask->GetBuffer(), value,
820                  m_pAlphaMask->GetHeight() * m_pAlphaMask->GetPitch());
821     return TRUE;
822   }
823   for (int row = 0; row < m_Height; row++) {
824     uint8_t* scan_line = m_pBuffer + row * m_Pitch + destOffset;
825     for (int col = 0; col < m_Width; col++) {
826       *scan_line = value;
827       scan_line += Bpp;
828     }
829   }
830   return TRUE;
831 }
832 FX_BOOL CFX_DIBitmap::MultiplyAlpha(const CFX_DIBSource* pSrcBitmap) {
833   if (!m_pBuffer) {
834     return FALSE;
835   }
836   ASSERT(pSrcBitmap->IsAlphaMask());
837   if (!pSrcBitmap->IsAlphaMask()) {
838     return FALSE;
839   }
840   if (!IsAlphaMask() && !HasAlpha()) {
841     return LoadChannel(FXDIB_Alpha, pSrcBitmap, FXDIB_Alpha);
842   }
843   CFX_DIBitmap* pSrcClone = (CFX_DIBitmap*)pSrcBitmap;
844   if (pSrcBitmap->GetWidth() != m_Width ||
845       pSrcBitmap->GetHeight() != m_Height) {
846     pSrcClone = pSrcBitmap->StretchTo(m_Width, m_Height);
847     if (!pSrcClone) {
848       return FALSE;
849     }
850   }
851   if (IsAlphaMask()) {
852     if (!ConvertFormat(FXDIB_8bppMask)) {
853       if (pSrcClone != pSrcBitmap) {
854         delete pSrcClone;
855       }
856       return FALSE;
857     }
858     for (int row = 0; row < m_Height; row++) {
859       uint8_t* dest_scan = m_pBuffer + m_Pitch * row;
860       uint8_t* src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;
861       if (pSrcClone->GetBPP() == 1) {
862         for (int col = 0; col < m_Width; col++) {
863           if (!((1 << (7 - col % 8)) & src_scan[col / 8])) {
864             dest_scan[col] = 0;
865           }
866         }
867       } else {
868         for (int col = 0; col < m_Width; col++) {
869           *dest_scan = (*dest_scan) * src_scan[col] / 255;
870           dest_scan++;
871         }
872       }
873     }
874   } else {
875     if (GetFormat() == FXDIB_Argb) {
876       if (pSrcClone->GetBPP() == 1) {
877         if (pSrcClone != pSrcBitmap) {
878           delete pSrcClone;
879         }
880         return FALSE;
881       }
882       for (int row = 0; row < m_Height; row++) {
883         uint8_t* dest_scan = m_pBuffer + m_Pitch * row + 3;
884         uint8_t* src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;
885         for (int col = 0; col < m_Width; col++) {
886           *dest_scan = (*dest_scan) * src_scan[col] / 255;
887           dest_scan += 4;
888         }
889       }
890     } else {
891       m_pAlphaMask->MultiplyAlpha(pSrcClone);
892     }
893   }
894   if (pSrcClone != pSrcBitmap) {
895     delete pSrcClone;
896   }
897   return TRUE;
898 }
899 FX_BOOL CFX_DIBitmap::GetGrayData(void* pIccTransform) {
900   if (!m_pBuffer) {
901     return FALSE;
902   }
903   switch (GetFormat()) {
904     case FXDIB_1bppRgb: {
905       if (!m_pPalette) {
906         return FALSE;
907       }
908       uint8_t gray[2];
909       for (int i = 0; i < 2; i++) {
910         int r = (uint8_t)(m_pPalette[i] >> 16);
911         int g = (uint8_t)(m_pPalette[i] >> 8);
912         int b = (uint8_t)m_pPalette[i];
913         gray[i] = (uint8_t)FXRGB2GRAY(r, g, b);
914       }
915       CFX_DIBitmap* pMask = new CFX_DIBitmap;
916       if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
917         delete pMask;
918         return FALSE;
919       }
920       FXSYS_memset(pMask->GetBuffer(), gray[0], pMask->GetPitch() * m_Height);
921       for (int row = 0; row < m_Height; row++) {
922         uint8_t* src_pos = m_pBuffer + row * m_Pitch;
923         uint8_t* dest_pos = (uint8_t*)pMask->GetScanline(row);
924         for (int col = 0; col < m_Width; col++) {
925           if (src_pos[col / 8] & (1 << (7 - col % 8))) {
926             *dest_pos = gray[1];
927           }
928           dest_pos++;
929         }
930       }
931       TakeOver(pMask);
932       delete pMask;
933       break;
934     }
935     case FXDIB_8bppRgb: {
936       if (!m_pPalette) {
937         return FALSE;
938       }
939       uint8_t gray[256];
940       for (int i = 0; i < 256; i++) {
941         int r = (uint8_t)(m_pPalette[i] >> 16);
942         int g = (uint8_t)(m_pPalette[i] >> 8);
943         int b = (uint8_t)m_pPalette[i];
944         gray[i] = (uint8_t)FXRGB2GRAY(r, g, b);
945       }
946       CFX_DIBitmap* pMask = new CFX_DIBitmap;
947       if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
948         delete pMask;
949         return FALSE;
950       }
951       for (int row = 0; row < m_Height; row++) {
952         uint8_t* dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
953         uint8_t* src_pos = m_pBuffer + row * m_Pitch;
954         for (int col = 0; col < m_Width; col++) {
955           *dest_pos++ = gray[*src_pos++];
956         }
957       }
958       TakeOver(pMask);
959       delete pMask;
960       break;
961     }
962     case FXDIB_Rgb: {
963       CFX_DIBitmap* pMask = new CFX_DIBitmap;
964       if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
965         delete pMask;
966         return FALSE;
967       }
968       for (int row = 0; row < m_Height; row++) {
969         uint8_t* src_pos = m_pBuffer + row * m_Pitch;
970         uint8_t* dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
971         for (int col = 0; col < m_Width; col++) {
972           *dest_pos++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);
973           src_pos += 3;
974         }
975       }
976       TakeOver(pMask);
977       delete pMask;
978       break;
979     }
980     case FXDIB_Rgb32: {
981       CFX_DIBitmap* pMask = new CFX_DIBitmap;
982       if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
983         delete pMask;
984         return FALSE;
985       }
986       for (int row = 0; row < m_Height; row++) {
987         uint8_t* src_pos = m_pBuffer + row * m_Pitch;
988         uint8_t* dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
989         for (int col = 0; col < m_Width; col++) {
990           *dest_pos++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);
991           src_pos += 4;
992         }
993       }
994       TakeOver(pMask);
995       delete pMask;
996       break;
997     }
998     default:
999       return FALSE;
1000   }
1001   return TRUE;
1002 }
1003 FX_BOOL CFX_DIBitmap::MultiplyAlpha(int alpha) {
1004   if (!m_pBuffer) {
1005     return FALSE;
1006   }
1007   switch (GetFormat()) {
1008     case FXDIB_1bppMask:
1009       if (!ConvertFormat(FXDIB_8bppMask)) {
1010         return FALSE;
1011       }
1012       MultiplyAlpha(alpha);
1013       break;
1014     case FXDIB_8bppMask: {
1015       for (int row = 0; row < m_Height; row++) {
1016         uint8_t* scan_line = m_pBuffer + row * m_Pitch;
1017         for (int col = 0; col < m_Width; col++) {
1018           scan_line[col] = scan_line[col] * alpha / 255;
1019         }
1020       }
1021       break;
1022     }
1023     case FXDIB_Argb: {
1024       for (int row = 0; row < m_Height; row++) {
1025         uint8_t* scan_line = m_pBuffer + row * m_Pitch + 3;
1026         for (int col = 0; col < m_Width; col++) {
1027           *scan_line = (*scan_line) * alpha / 255;
1028           scan_line += 4;
1029         }
1030       }
1031       break;
1032     }
1033     default:
1034       if (HasAlpha()) {
1035         m_pAlphaMask->MultiplyAlpha(alpha);
1036       } else if (IsCmykImage()) {
1037         if (!ConvertFormat((FXDIB_Format)(GetFormat() | 0x0200))) {
1038           return FALSE;
1039         }
1040         m_pAlphaMask->MultiplyAlpha(alpha);
1041       } else {
1042         if (!ConvertFormat(FXDIB_Argb)) {
1043           return FALSE;
1044         }
1045         MultiplyAlpha(alpha);
1046       }
1047       break;
1048   }
1049   return TRUE;
1050 }
1051 FX_DWORD CFX_DIBitmap::GetPixel(int x, int y) const {
1052   if (!m_pBuffer) {
1053     return 0;
1054   }
1055   uint8_t* pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;
1056   switch (GetFormat()) {
1057     case FXDIB_1bppMask: {
1058       if ((*pos) & (1 << (7 - x % 8))) {
1059         return 0xff000000;
1060       }
1061       return 0;
1062     }
1063     case FXDIB_1bppRgb: {
1064       if ((*pos) & (1 << (7 - x % 8))) {
1065         return m_pPalette ? m_pPalette[1] : 0xffffffff;
1066       }
1067       return m_pPalette ? m_pPalette[0] : 0xff000000;
1068     }
1069     case FXDIB_8bppMask:
1070       return (*pos) << 24;
1071     case FXDIB_8bppRgb:
1072       return m_pPalette ? m_pPalette[*pos] : (0xff000000 | ((*pos) * 0x10101));
1073     case FXDIB_Rgb:
1074     case FXDIB_Rgba:
1075     case FXDIB_Rgb32:
1076       return FXARGB_GETDIB(pos) | 0xff000000;
1077     case FXDIB_Argb:
1078       return FXARGB_GETDIB(pos);
1079     default:
1080       break;
1081   }
1082   return 0;
1083 }
1084 void CFX_DIBitmap::SetPixel(int x, int y, FX_DWORD color) {
1085   if (!m_pBuffer) {
1086     return;
1087   }
1088   if (x < 0 || x >= m_Width || y < 0 || y >= m_Height) {
1089     return;
1090   }
1091   uint8_t* pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;
1092   switch (GetFormat()) {
1093     case FXDIB_1bppMask:
1094       if (color >> 24) {
1095         *pos |= 1 << (7 - x % 8);
1096       } else {
1097         *pos &= ~(1 << (7 - x % 8));
1098       }
1099       break;
1100     case FXDIB_1bppRgb:
1101       if (m_pPalette) {
1102         if (color == m_pPalette[1]) {
1103           *pos |= 1 << (7 - x % 8);
1104         } else {
1105           *pos &= ~(1 << (7 - x % 8));
1106         }
1107       } else {
1108         if (color == 0xffffffff) {
1109           *pos |= 1 << (7 - x % 8);
1110         } else {
1111           *pos &= ~(1 << (7 - x % 8));
1112         }
1113       }
1114       break;
1115     case FXDIB_8bppMask:
1116       *pos = (uint8_t)(color >> 24);
1117       break;
1118     case FXDIB_8bppRgb: {
1119       if (m_pPalette) {
1120         for (int i = 0; i < 256; i++) {
1121           if (m_pPalette[i] == color) {
1122             *pos = (uint8_t)i;
1123             return;
1124           }
1125         }
1126         *pos = 0;
1127       } else {
1128         *pos = FXRGB2GRAY(FXARGB_R(color), FXARGB_G(color), FXARGB_B(color));
1129       }
1130       break;
1131     }
1132     case FXDIB_Rgb:
1133     case FXDIB_Rgb32: {
1134       int alpha = FXARGB_A(color);
1135       pos[0] = (FXARGB_B(color) * alpha + pos[0] * (255 - alpha)) / 255;
1136       pos[1] = (FXARGB_G(color) * alpha + pos[1] * (255 - alpha)) / 255;
1137       pos[2] = (FXARGB_R(color) * alpha + pos[2] * (255 - alpha)) / 255;
1138       break;
1139     }
1140     case FXDIB_Rgba: {
1141       pos[0] = FXARGB_B(color);
1142       pos[1] = FXARGB_G(color);
1143       pos[2] = FXARGB_R(color);
1144       break;
1145     }
1146     case FXDIB_Argb:
1147       FXARGB_SETDIB(pos, color);
1148       break;
1149     default:
1150       break;
1151   }
1152 }
1153 void CFX_DIBitmap::DownSampleScanline(int line,
1154                                       uint8_t* dest_scan,
1155                                       int dest_bpp,
1156                                       int dest_width,
1157                                       FX_BOOL bFlipX,
1158                                       int clip_left,
1159                                       int clip_width) const {
1160   if (!m_pBuffer) {
1161     return;
1162   }
1163   int src_Bpp = m_bpp / 8;
1164   uint8_t* scanline = m_pBuffer + line * m_Pitch;
1165   if (src_Bpp == 0) {
1166     for (int i = 0; i < clip_width; i++) {
1167       FX_DWORD dest_x = clip_left + i;
1168       FX_DWORD src_x = dest_x * m_Width / dest_width;
1169       if (bFlipX) {
1170         src_x = m_Width - src_x - 1;
1171       }
1172       src_x %= m_Width;
1173       dest_scan[i] = (scanline[src_x / 8] & (1 << (7 - src_x % 8))) ? 255 : 0;
1174     }
1175   } else if (src_Bpp == 1) {
1176     for (int i = 0; i < clip_width; i++) {
1177       FX_DWORD dest_x = clip_left + i;
1178       FX_DWORD src_x = dest_x * m_Width / dest_width;
1179       if (bFlipX) {
1180         src_x = m_Width - src_x - 1;
1181       }
1182       src_x %= m_Width;
1183       int dest_pos = i;
1184       if (m_pPalette) {
1185         if (!IsCmykImage()) {
1186           dest_pos *= 3;
1187           FX_ARGB argb = m_pPalette[scanline[src_x]];
1188           dest_scan[dest_pos] = FXARGB_B(argb);
1189           dest_scan[dest_pos + 1] = FXARGB_G(argb);
1190           dest_scan[dest_pos + 2] = FXARGB_R(argb);
1191         } else {
1192           dest_pos *= 4;
1193           FX_CMYK cmyk = m_pPalette[scanline[src_x]];
1194           dest_scan[dest_pos] = FXSYS_GetCValue(cmyk);
1195           dest_scan[dest_pos + 1] = FXSYS_GetMValue(cmyk);
1196           dest_scan[dest_pos + 2] = FXSYS_GetYValue(cmyk);
1197           dest_scan[dest_pos + 3] = FXSYS_GetKValue(cmyk);
1198         }
1199       } else {
1200         dest_scan[dest_pos] = scanline[src_x];
1201       }
1202     }
1203   } else {
1204     for (int i = 0; i < clip_width; i++) {
1205       FX_DWORD dest_x = clip_left + i;
1206       FX_DWORD src_x =
1207           bFlipX ? (m_Width - dest_x * m_Width / dest_width - 1) * src_Bpp
1208                  : (dest_x * m_Width / dest_width) * src_Bpp;
1209       src_x %= m_Width * src_Bpp;
1210       int dest_pos = i * src_Bpp;
1211       for (int b = 0; b < src_Bpp; b++) {
1212         dest_scan[dest_pos + b] = scanline[src_x + b];
1213       }
1214     }
1215   }
1216 }
1217 FX_BOOL CFX_DIBitmap::ConvertColorScale(FX_DWORD forecolor,
1218                                         FX_DWORD backcolor) {
1219   ASSERT(!IsAlphaMask());
1220   if (!m_pBuffer || IsAlphaMask()) {
1221     return FALSE;
1222   }
1223   int fc, fm, fy, fk, bc, bm, by, bk;
1224   int fr, fg, fb, br, bg, bb;
1225   FX_BOOL isCmykImage = IsCmykImage();
1226   if (isCmykImage) {
1227     fc = FXSYS_GetCValue(forecolor);
1228     fm = FXSYS_GetMValue(forecolor);
1229     fy = FXSYS_GetYValue(forecolor);
1230     fk = FXSYS_GetKValue(forecolor);
1231     bc = FXSYS_GetCValue(backcolor);
1232     bm = FXSYS_GetMValue(backcolor);
1233     by = FXSYS_GetYValue(backcolor);
1234     bk = FXSYS_GetKValue(backcolor);
1235   } else {
1236     fr = FXSYS_GetRValue(forecolor);
1237     fg = FXSYS_GetGValue(forecolor);
1238     fb = FXSYS_GetBValue(forecolor);
1239     br = FXSYS_GetRValue(backcolor);
1240     bg = FXSYS_GetGValue(backcolor);
1241     bb = FXSYS_GetBValue(backcolor);
1242   }
1243   if (m_bpp <= 8) {
1244     if (isCmykImage) {
1245       if (forecolor == 0xff && backcolor == 0 && !m_pPalette) {
1246         return TRUE;
1247       }
1248     } else if (forecolor == 0 && backcolor == 0xffffff && !m_pPalette) {
1249       return TRUE;
1250     }
1251     if (!m_pPalette) {
1252       BuildPalette();
1253     }
1254     int size = 1 << m_bpp;
1255     if (isCmykImage) {
1256       for (int i = 0; i < size; i++) {
1257         uint8_t b, g, r;
1258         AdobeCMYK_to_sRGB1(FXSYS_GetCValue(m_pPalette[i]),
1259                            FXSYS_GetMValue(m_pPalette[i]),
1260                            FXSYS_GetYValue(m_pPalette[i]),
1261                            FXSYS_GetKValue(m_pPalette[i]), r, g, b);
1262         int gray = 255 - FXRGB2GRAY(r, g, b);
1263         m_pPalette[i] = CmykEncode(
1264             bc + (fc - bc) * gray / 255, bm + (fm - bm) * gray / 255,
1265             by + (fy - by) * gray / 255, bk + (fk - bk) * gray / 255);
1266       }
1267     } else
1268       for (int i = 0; i < size; i++) {
1269         int gray = FXRGB2GRAY(FXARGB_R(m_pPalette[i]), FXARGB_G(m_pPalette[i]),
1270                               FXARGB_B(m_pPalette[i]));
1271         m_pPalette[i] = FXARGB_MAKE(0xff, br + (fr - br) * gray / 255,
1272                                     bg + (fg - bg) * gray / 255,
1273                                     bb + (fb - bb) * gray / 255);
1274       }
1275     return TRUE;
1276   }
1277   if (isCmykImage) {
1278     if (forecolor == 0xff && backcolor == 0x00) {
1279       for (int row = 0; row < m_Height; row++) {
1280         uint8_t* scanline = m_pBuffer + row * m_Pitch;
1281         for (int col = 0; col < m_Width; col++) {
1282           uint8_t b, g, r;
1283           AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],
1284                              r, g, b);
1285           *scanline++ = 0;
1286           *scanline++ = 0;
1287           *scanline++ = 0;
1288           *scanline++ = 255 - FXRGB2GRAY(r, g, b);
1289         }
1290       }
1291       return TRUE;
1292     }
1293   } else if (forecolor == 0 && backcolor == 0xffffff) {
1294     for (int row = 0; row < m_Height; row++) {
1295       uint8_t* scanline = m_pBuffer + row * m_Pitch;
1296       int gap = m_bpp / 8 - 2;
1297       for (int col = 0; col < m_Width; col++) {
1298         int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);
1299         *scanline++ = gray;
1300         *scanline++ = gray;
1301         *scanline = gray;
1302         scanline += gap;
1303       }
1304     }
1305     return TRUE;
1306   }
1307   if (isCmykImage) {
1308     for (int row = 0; row < m_Height; row++) {
1309       uint8_t* scanline = m_pBuffer + row * m_Pitch;
1310       for (int col = 0; col < m_Width; col++) {
1311         uint8_t b, g, r;
1312         AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],
1313                            r, g, b);
1314         int gray = 255 - FXRGB2GRAY(r, g, b);
1315         *scanline++ = bc + (fc - bc) * gray / 255;
1316         *scanline++ = bm + (fm - bm) * gray / 255;
1317         *scanline++ = by + (fy - by) * gray / 255;
1318         *scanline++ = bk + (fk - bk) * gray / 255;
1319       }
1320     }
1321   } else {
1322     for (int row = 0; row < m_Height; row++) {
1323       uint8_t* scanline = m_pBuffer + row * m_Pitch;
1324       int gap = m_bpp / 8 - 2;
1325       for (int col = 0; col < m_Width; col++) {
1326         int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);
1327         *scanline++ = bb + (fb - bb) * gray / 255;
1328         *scanline++ = bg + (fg - bg) * gray / 255;
1329         *scanline = br + (fr - br) * gray / 255;
1330         scanline += gap;
1331       }
1332     }
1333   }
1334   return TRUE;
1335 }
1336 FX_BOOL CFX_DIBitmap::DitherFS(const FX_DWORD* pPalette,
1337                                int pal_size,
1338                                const FX_RECT* pRect) {
1339   if (!m_pBuffer) {
1340     return FALSE;
1341   }
1342   if (m_bpp != 8 && m_pPalette && m_AlphaFlag != 0) {
1343     return FALSE;
1344   }
1345   if (m_Width < 4 && m_Height < 4) {
1346     return FALSE;
1347   }
1348   FX_RECT rect(0, 0, m_Width, m_Height);
1349   if (pRect) {
1350     rect.Intersect(*pRect);
1351   }
1352   uint8_t translate[256];
1353   for (int i = 0; i < 256; i++) {
1354     int err2 = 65536;
1355     for (int j = 0; j < pal_size; j++) {
1356       uint8_t entry = (uint8_t)pPalette[j];
1357       int err = (int)entry - i;
1358       if (err * err < err2) {
1359         err2 = err * err;
1360         translate[i] = entry;
1361       }
1362     }
1363   }
1364   for (int row = rect.top; row < rect.bottom; row++) {
1365     uint8_t* scan = m_pBuffer + row * m_Pitch;
1366     uint8_t* next_scan = m_pBuffer + (row + 1) * m_Pitch;
1367     for (int col = rect.left; col < rect.right; col++) {
1368       int src_pixel = scan[col];
1369       int dest_pixel = translate[src_pixel];
1370       scan[col] = (uint8_t)dest_pixel;
1371       int error = -dest_pixel + src_pixel;
1372       if (col < rect.right - 1) {
1373         int src = scan[col + 1];
1374         src += error * 7 / 16;
1375         if (src > 255) {
1376           scan[col + 1] = 255;
1377         } else if (src < 0) {
1378           scan[col + 1] = 0;
1379         } else {
1380           scan[col + 1] = src;
1381         }
1382       }
1383       if (col < rect.right - 1 && row < rect.bottom - 1) {
1384         int src = next_scan[col + 1];
1385         src += error * 1 / 16;
1386         if (src > 255) {
1387           next_scan[col + 1] = 255;
1388         } else if (src < 0) {
1389           next_scan[col + 1] = 0;
1390         } else {
1391           next_scan[col + 1] = src;
1392         }
1393       }
1394       if (row < rect.bottom - 1) {
1395         int src = next_scan[col];
1396         src += error * 5 / 16;
1397         if (src > 255) {
1398           next_scan[col] = 255;
1399         } else if (src < 0) {
1400           next_scan[col] = 0;
1401         } else {
1402           next_scan[col] = src;
1403         }
1404       }
1405       if (col > rect.left && row < rect.bottom - 1) {
1406         int src = next_scan[col - 1];
1407         src += error * 3 / 16;
1408         if (src > 255) {
1409           next_scan[col - 1] = 255;
1410         } else if (src < 0) {
1411           next_scan[col - 1] = 0;
1412         } else {
1413           next_scan[col - 1] = src;
1414         }
1415       }
1416     }
1417   }
1418   return TRUE;
1419 }
1420 CFX_DIBitmap* CFX_DIBSource::FlipImage(FX_BOOL bXFlip, FX_BOOL bYFlip) const {
1421   CFX_DIBitmap* pFlipped = new CFX_DIBitmap;
1422   if (!pFlipped->Create(m_Width, m_Height, GetFormat())) {
1423     delete pFlipped;
1424     return NULL;
1425   }
1426   pFlipped->CopyPalette(m_pPalette);
1427   uint8_t* pDestBuffer = pFlipped->GetBuffer();
1428   int Bpp = m_bpp / 8;
1429   for (int row = 0; row < m_Height; row++) {
1430     const uint8_t* src_scan = GetScanline(row);
1431     uint8_t* dest_scan =
1432         pDestBuffer + m_Pitch * (bYFlip ? (m_Height - row - 1) : row);
1433     if (!bXFlip) {
1434       FXSYS_memcpy(dest_scan, src_scan, m_Pitch);
1435       continue;
1436     }
1437     if (m_bpp == 1) {
1438       FXSYS_memset(dest_scan, 0, m_Pitch);
1439       for (int col = 0; col < m_Width; col++)
1440         if (src_scan[col / 8] & (1 << (7 - col % 8))) {
1441           int dest_col = m_Width - col - 1;
1442           dest_scan[dest_col / 8] |= (1 << (7 - dest_col % 8));
1443         }
1444     } else {
1445       dest_scan += (m_Width - 1) * Bpp;
1446       if (Bpp == 1) {
1447         for (int col = 0; col < m_Width; col++) {
1448           *dest_scan = *src_scan;
1449           dest_scan--;
1450           src_scan++;
1451         }
1452       } else if (Bpp == 3) {
1453         for (int col = 0; col < m_Width; col++) {
1454           dest_scan[0] = src_scan[0];
1455           dest_scan[1] = src_scan[1];
1456           dest_scan[2] = src_scan[2];
1457           dest_scan -= 3;
1458           src_scan += 3;
1459         }
1460       } else {
1461         ASSERT(Bpp == 4);
1462         for (int col = 0; col < m_Width; col++) {
1463           *(FX_DWORD*)dest_scan = *(FX_DWORD*)src_scan;
1464           dest_scan -= 4;
1465           src_scan += 4;
1466         }
1467       }
1468     }
1469   }
1470   if (m_pAlphaMask) {
1471     pDestBuffer = pFlipped->m_pAlphaMask->GetBuffer();
1472     FX_DWORD dest_pitch = pFlipped->m_pAlphaMask->GetPitch();
1473     for (int row = 0; row < m_Height; row++) {
1474       const uint8_t* src_scan = m_pAlphaMask->GetScanline(row);
1475       uint8_t* dest_scan =
1476           pDestBuffer + dest_pitch * (bYFlip ? (m_Height - row - 1) : row);
1477       if (!bXFlip) {
1478         FXSYS_memcpy(dest_scan, src_scan, dest_pitch);
1479         continue;
1480       }
1481       dest_scan += (m_Width - 1);
1482       for (int col = 0; col < m_Width; col++) {
1483         *dest_scan = *src_scan;
1484         dest_scan--;
1485         src_scan++;
1486       }
1487     }
1488   }
1489   return pFlipped;
1490 }
1491 CFX_DIBExtractor::CFX_DIBExtractor(const CFX_DIBSource* pSrc) {
1492   m_pBitmap = NULL;
1493   if (pSrc->GetBuffer()) {
1494     m_pBitmap = new CFX_DIBitmap;
1495     if (!m_pBitmap->Create(pSrc->GetWidth(), pSrc->GetHeight(),
1496                            pSrc->GetFormat(), pSrc->GetBuffer())) {
1497       delete m_pBitmap;
1498       m_pBitmap = NULL;
1499       return;
1500     }
1501     m_pBitmap->CopyPalette(pSrc->GetPalette());
1502     m_pBitmap->CopyAlphaMask(pSrc->m_pAlphaMask);
1503   } else {
1504     m_pBitmap = pSrc->Clone();
1505   }
1506 }
1507 CFX_DIBExtractor::~CFX_DIBExtractor() {
1508   delete m_pBitmap;
1509 }
1510 CFX_FilteredDIB::CFX_FilteredDIB() {
1511   m_pScanline = NULL;
1512   m_pSrc = NULL;
1513 }
1514 CFX_FilteredDIB::~CFX_FilteredDIB() {
1515   if (m_bAutoDropSrc) {
1516     delete m_pSrc;
1517   }
1518   FX_Free(m_pScanline);
1519 }
1520 void CFX_FilteredDIB::LoadSrc(const CFX_DIBSource* pSrc, FX_BOOL bAutoDropSrc) {
1521   m_pSrc = pSrc;
1522   m_bAutoDropSrc = bAutoDropSrc;
1523   m_Width = pSrc->GetWidth();
1524   m_Height = pSrc->GetHeight();
1525   FXDIB_Format format = GetDestFormat();
1526   m_bpp = (uint8_t)format;
1527   m_AlphaFlag = (uint8_t)(format >> 8);
1528   m_Pitch = (m_Width * (format & 0xff) + 31) / 32 * 4;
1529   m_pPalette = GetDestPalette();
1530   m_pScanline = FX_Alloc(uint8_t, m_Pitch);
1531 }
1532 const uint8_t* CFX_FilteredDIB::GetScanline(int line) const {
1533   TranslateScanline(m_pScanline, m_pSrc->GetScanline(line));
1534   return m_pScanline;
1535 }
1536 void CFX_FilteredDIB::DownSampleScanline(int line,
1537                                          uint8_t* dest_scan,
1538                                          int dest_bpp,
1539                                          int dest_width,
1540                                          FX_BOOL bFlipX,
1541                                          int clip_left,
1542                                          int clip_width) const {
1543   m_pSrc->DownSampleScanline(line, dest_scan, dest_bpp, dest_width, bFlipX,
1544                              clip_left, clip_width);
1545   TranslateDownSamples(dest_scan, dest_scan, clip_width, dest_bpp);
1546 }
1547 CFX_ImageRenderer::CFX_ImageRenderer() {
1548   m_Status = 0;
1549   m_pTransformer = NULL;
1550   m_bRgbByteOrder = FALSE;
1551   m_BlendType = FXDIB_BLEND_NORMAL;
1552 }
1553 CFX_ImageRenderer::~CFX_ImageRenderer() {
1554   delete m_pTransformer;
1555 }
1556 FX_BOOL CFX_ImageRenderer::Start(CFX_DIBitmap* pDevice,
1557                                  const CFX_ClipRgn* pClipRgn,
1558                                  const CFX_DIBSource* pSource,
1559                                  int bitmap_alpha,
1560                                  FX_DWORD mask_color,
1561                                  const CFX_Matrix* pMatrix,
1562                                  FX_DWORD dib_flags,
1563                                  FX_BOOL bRgbByteOrder,
1564                                  int alpha_flag,
1565                                  void* pIccTransform,
1566                                  int blend_type) {
1567   m_Matrix = *pMatrix;
1568   CFX_FloatRect image_rect_f = m_Matrix.GetUnitRect();
1569   FX_RECT image_rect = image_rect_f.GetOutterRect();
1570   m_ClipBox = pClipRgn ? pClipRgn->GetBox() : FX_RECT(0, 0, pDevice->GetWidth(),
1571                                                       pDevice->GetHeight());
1572   m_ClipBox.Intersect(image_rect);
1573   if (m_ClipBox.IsEmpty()) {
1574     return FALSE;
1575   }
1576   m_pDevice = pDevice;
1577   m_pClipRgn = pClipRgn;
1578   m_MaskColor = mask_color;
1579   m_BitmapAlpha = bitmap_alpha;
1580   m_Matrix = *pMatrix;
1581   m_Flags = dib_flags;
1582   m_AlphaFlag = alpha_flag;
1583   m_pIccTransform = pIccTransform;
1584   m_bRgbByteOrder = bRgbByteOrder;
1585   m_BlendType = blend_type;
1586   FX_BOOL ret = TRUE;
1587   if ((FXSYS_fabs(m_Matrix.b) >= 0.5f || m_Matrix.a == 0) ||
1588       (FXSYS_fabs(m_Matrix.c) >= 0.5f || m_Matrix.d == 0)) {
1589     if (FXSYS_fabs(m_Matrix.a) < FXSYS_fabs(m_Matrix.b) / 20 &&
1590         FXSYS_fabs(m_Matrix.d) < FXSYS_fabs(m_Matrix.c) / 20 &&
1591         FXSYS_fabs(m_Matrix.a) < 0.5f && FXSYS_fabs(m_Matrix.d) < 0.5f) {
1592       int dest_width = image_rect.Width();
1593       int dest_height = image_rect.Height();
1594       FX_RECT bitmap_clip = m_ClipBox;
1595       bitmap_clip.Offset(-image_rect.left, -image_rect.top);
1596       bitmap_clip = FXDIB_SwapClipBox(bitmap_clip, dest_width, dest_height,
1597                                       m_Matrix.c > 0, m_Matrix.b < 0);
1598       m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color, m_ClipBox,
1599                          TRUE, m_Matrix.c > 0, m_Matrix.b < 0, m_bRgbByteOrder,
1600                          alpha_flag, pIccTransform, m_BlendType);
1601       if (!m_Stretcher.Start(&m_Composer, pSource, dest_height, dest_width,
1602                              bitmap_clip, dib_flags)) {
1603         return FALSE;
1604       }
1605       m_Status = 1;
1606       return TRUE;
1607     }
1608     m_Status = 2;
1609     m_pTransformer = new CFX_ImageTransformer;
1610     m_pTransformer->Start(pSource, &m_Matrix, dib_flags, &m_ClipBox);
1611     return TRUE;
1612   }
1613   int dest_width = image_rect.Width();
1614   if (m_Matrix.a < 0) {
1615     dest_width = -dest_width;
1616   }
1617   int dest_height = image_rect.Height();
1618   if (m_Matrix.d > 0) {
1619     dest_height = -dest_height;
1620   }
1621   if (dest_width == 0 || dest_height == 0) {
1622     return FALSE;
1623   }
1624   FX_RECT bitmap_clip = m_ClipBox;
1625   bitmap_clip.Offset(-image_rect.left, -image_rect.top);
1626   m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color, m_ClipBox,
1627                      FALSE, FALSE, FALSE, m_bRgbByteOrder, alpha_flag,
1628                      pIccTransform, m_BlendType);
1629   m_Status = 1;
1630   ret = m_Stretcher.Start(&m_Composer, pSource, dest_width, dest_height,
1631                           bitmap_clip, dib_flags);
1632   return ret;
1633 }
1634 FX_BOOL CFX_ImageRenderer::Continue(IFX_Pause* pPause) {
1635   if (m_Status == 1) {
1636     return m_Stretcher.Continue(pPause);
1637   }
1638   if (m_Status == 2) {
1639     if (m_pTransformer->Continue(pPause)) {
1640       return TRUE;
1641     }
1642     CFX_DIBitmap* pBitmap = m_pTransformer->m_Storer.Detach();
1643     if (!pBitmap) {
1644       return FALSE;
1645     }
1646     if (!pBitmap->GetBuffer()) {
1647       delete pBitmap;
1648       return FALSE;
1649     }
1650     if (pBitmap->IsAlphaMask()) {
1651       if (m_BitmapAlpha != 255) {
1652         if (m_AlphaFlag >> 8) {
1653           m_AlphaFlag =
1654               (((uint8_t)((m_AlphaFlag & 0xff) * m_BitmapAlpha / 255)) |
1655                ((m_AlphaFlag >> 8) << 8));
1656         } else {
1657           m_MaskColor = FXARGB_MUL_ALPHA(m_MaskColor, m_BitmapAlpha);
1658         }
1659       }
1660       m_pDevice->CompositeMask(m_pTransformer->m_ResultLeft,
1661                                m_pTransformer->m_ResultTop, pBitmap->GetWidth(),
1662                                pBitmap->GetHeight(), pBitmap, m_MaskColor, 0, 0,
1663                                m_BlendType, m_pClipRgn, m_bRgbByteOrder,
1664                                m_AlphaFlag, m_pIccTransform);
1665     } else {
1666       if (m_BitmapAlpha != 255) {
1667         pBitmap->MultiplyAlpha(m_BitmapAlpha);
1668       }
1669       m_pDevice->CompositeBitmap(
1670           m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,
1671           pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, 0, 0, m_BlendType,
1672           m_pClipRgn, m_bRgbByteOrder, m_pIccTransform);
1673     }
1674     delete pBitmap;
1675     return FALSE;
1676   }
1677   return FALSE;
1678 }
1679 CFX_BitmapStorer::CFX_BitmapStorer() {
1680   m_pBitmap = NULL;
1681 }
1682 CFX_BitmapStorer::~CFX_BitmapStorer() {
1683   delete m_pBitmap;
1684 }
1685 CFX_DIBitmap* CFX_BitmapStorer::Detach() {
1686   CFX_DIBitmap* pBitmap = m_pBitmap;
1687   m_pBitmap = NULL;
1688   return pBitmap;
1689 }
1690 void CFX_BitmapStorer::Replace(CFX_DIBitmap* pBitmap) {
1691   delete m_pBitmap;
1692   m_pBitmap = pBitmap;
1693 }
1694 void CFX_BitmapStorer::ComposeScanline(int line,
1695                                        const uint8_t* scanline,
1696                                        const uint8_t* scan_extra_alpha) {
1697   uint8_t* dest_buf = (uint8_t*)m_pBitmap->GetScanline(line);
1698   uint8_t* dest_alpha_buf =
1699       m_pBitmap->m_pAlphaMask
1700           ? (uint8_t*)m_pBitmap->m_pAlphaMask->GetScanline(line)
1701           : NULL;
1702   if (dest_buf) {
1703     FXSYS_memcpy(dest_buf, scanline, m_pBitmap->GetPitch());
1704   }
1705   if (dest_alpha_buf) {
1706     FXSYS_memcpy(dest_alpha_buf, scan_extra_alpha,
1707                  m_pBitmap->m_pAlphaMask->GetPitch());
1708   }
1709 }
1710 FX_BOOL CFX_BitmapStorer::SetInfo(int width,
1711                                   int height,
1712                                   FXDIB_Format src_format,
1713                                   FX_DWORD* pSrcPalette) {
1714   m_pBitmap = new CFX_DIBitmap;
1715   if (!m_pBitmap->Create(width, height, src_format)) {
1716     delete m_pBitmap;
1717     m_pBitmap = NULL;
1718     return FALSE;
1719   }
1720   if (pSrcPalette) {
1721     m_pBitmap->CopyPalette(pSrcPalette);
1722   }
1723   return TRUE;
1724 }
1725