Merge to XFA: Use stdint.h types throughout PDFium.
[pdfium.git] / core / src / fxge / dib / fx_dib_main.cpp
index 0953f2d..74c4276 100644 (file)
-// Copyright 2014 PDFium Authors. All rights reserved.\r
-// Use of this source code is governed by a BSD-style license that can be\r
-// found in the LICENSE file.\r
\r
-// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com\r
-\r
-#include "../../../include/fxge/fx_dib.h"\r
-#include "../../../include/fxge/fx_ge.h"\r
-#include "../../../include/fxcodec/fx_codec.h"\r
-#include "dib_int.h"\r
-#include <limits.h>\r
-FX_BOOL ConvertBuffer(FXDIB_Format dest_format, FX_LPBYTE dest_buf, int dest_pitch, int width, int height,\r
-                      const CFX_DIBSource* pSrcBitmap, int src_left, int src_top, FX_DWORD*& pal, void* pIccTransform);\r
-void CmykDecode(FX_DWORD cmyk, int& c, int& m, int& y, int& k)\r
-{\r
-    c = FXSYS_GetCValue(cmyk);\r
-    m = FXSYS_GetMValue(cmyk);\r
-    y = FXSYS_GetYValue(cmyk);\r
-    k = FXSYS_GetKValue(cmyk);\r
-}\r
-void ArgbDecode(FX_DWORD argb, int& a, int& r, int& g, int& b)\r
-{\r
-    a = FXARGB_A(argb);\r
-    r = FXARGB_R(argb);\r
-    g = FXARGB_G(argb);\r
-    b = FXARGB_B(argb);\r
-}\r
-void ArgbDecode(FX_DWORD argb, int& a, FX_COLORREF& rgb)\r
-{\r
-    a = FXARGB_A(argb);\r
-    rgb = FXSYS_RGB(FXARGB_R(argb), FXARGB_G(argb), FXARGB_B(argb));\r
-}\r
-FX_DWORD ArgbEncode(int a, FX_COLORREF rgb)\r
-{\r
-    return FXARGB_MAKE(a, FXSYS_GetRValue(rgb), FXSYS_GetGValue(rgb), FXSYS_GetBValue(rgb));\r
-}\r
-CFX_DIBSource::CFX_DIBSource()\r
-{\r
-    m_bpp = 0;\r
-    m_AlphaFlag = 0;\r
-    m_Width = m_Height = 0;\r
-    m_Pitch = 0;\r
-    m_pPalette = NULL;\r
-    m_pAlphaMask = NULL;\r
-}\r
-CFX_DIBSource::~CFX_DIBSource()\r
-{\r
-    if (m_pPalette) {\r
-        FX_Free(m_pPalette);\r
-    }\r
-    if (m_pAlphaMask) {\r
-        delete m_pAlphaMask;\r
-    }\r
-}\r
-CFX_DIBitmap::CFX_DIBitmap()\r
-{\r
-    m_bExtBuf = FALSE;\r
-    m_pBuffer = NULL;\r
-    m_pPalette = NULL;\r
-}\r
-#define _MAX_OOM_LIMIT_        12000000\r
-FX_BOOL CFX_DIBitmap::Create(int width, int height, FXDIB_Format format, FX_LPBYTE pBuffer, int pitch)\r
-{\r
-    m_pBuffer = NULL;\r
-    m_bpp = (FX_BYTE)format;\r
-    m_AlphaFlag = (FX_BYTE)(format >> 8);\r
-    m_Width = m_Height = m_Pitch = 0;\r
-    if (width <= 0 || height <= 0 || pitch < 0) {\r
-        return FALSE;\r
-    }\r
-    if ((INT_MAX - 31) / width < (format & 0xff)) {\r
-        return FALSE;\r
-    }\r
-    if (!pitch) {\r
-        pitch = (width * (format & 0xff) + 31) / 32 * 4;\r
-    }\r
-    if ((1 << 30) / pitch < height) {\r
-        return FALSE;\r
-    }\r
-    if (pBuffer) {\r
-        m_pBuffer = pBuffer;\r
-        m_bExtBuf = TRUE;\r
-    } else {\r
-        int size = pitch * height + 4;\r
-        int oomlimit = _MAX_OOM_LIMIT_;\r
-        if (oomlimit >= 0 && size >= oomlimit) {\r
-            m_pBuffer = FX_AllocNL(FX_BYTE, size);\r
-        } else {\r
-            m_pBuffer = FX_Alloc(FX_BYTE, size);\r
-        }\r
-        if (m_pBuffer == NULL) {\r
-            return FALSE;\r
-        }\r
-        FXSYS_memset32(m_pBuffer, 0, sizeof (FX_BYTE) * size);\r
-    }\r
-    m_Width = width;\r
-    m_Height = height;\r
-    m_Pitch = pitch;\r
-    if (HasAlpha() && format != FXDIB_Argb) {\r
-        FX_BOOL ret = TRUE;\r
-        ret = BuildAlphaMask();\r
-        if (!ret) {\r
-            if (!m_bExtBuf && m_pBuffer) {\r
-                FX_Free(m_pBuffer);\r
-                m_pBuffer = NULL;\r
-                m_Width = m_Height = m_Pitch = 0;\r
-                return FALSE;\r
-            }\r
-        }\r
-    }\r
-    return TRUE;\r
-}\r
-FX_BOOL CFX_DIBitmap::Copy(const CFX_DIBSource* pSrc)\r
-{\r
-    if (m_pBuffer) {\r
-        return FALSE;\r
-    }\r
-    if (!Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat())) {\r
-        return FALSE;\r
-    }\r
-    CopyPalette(pSrc->GetPalette());\r
-    CopyAlphaMask(pSrc->m_pAlphaMask);\r
-    for (int row = 0; row < pSrc->GetHeight(); row ++) {\r
-        FXSYS_memcpy32(m_pBuffer + row * m_Pitch, pSrc->GetScanline(row), m_Pitch);\r
-    }\r
-    return TRUE;\r
-}\r
-CFX_DIBitmap::~CFX_DIBitmap()\r
-{\r
-    if (m_pBuffer && !m_bExtBuf) {\r
-        FX_Free(m_pBuffer);\r
-    }\r
-    m_pBuffer = NULL;\r
-}\r
-void CFX_DIBitmap::TakeOver(CFX_DIBitmap* pSrcBitmap)\r
-{\r
-    if (m_pBuffer && !m_bExtBuf) {\r
-        FX_Free(m_pBuffer);\r
-    }\r
-    if (m_pPalette) {\r
-        FX_Free(m_pPalette);\r
-    }\r
-    if (m_pAlphaMask) {\r
-        delete m_pAlphaMask;\r
-    }\r
-    m_pBuffer = pSrcBitmap->m_pBuffer;\r
-    m_pPalette = pSrcBitmap->m_pPalette;\r
-    m_pAlphaMask = pSrcBitmap->m_pAlphaMask;\r
-    pSrcBitmap->m_pBuffer = NULL;\r
-    pSrcBitmap->m_pPalette = NULL;\r
-    pSrcBitmap->m_pAlphaMask = NULL;\r
-    m_bpp = pSrcBitmap->m_bpp;\r
-    m_bExtBuf = pSrcBitmap->m_bExtBuf;\r
-    m_AlphaFlag = pSrcBitmap->m_AlphaFlag;\r
-    m_Width = pSrcBitmap->m_Width;\r
-    m_Height = pSrcBitmap->m_Height;\r
-    m_Pitch = pSrcBitmap->m_Pitch;\r
-}\r
-CFX_DIBitmap* CFX_DIBSource::Clone(const FX_RECT* pClip) const\r
-{\r
-    FX_RECT rect(0, 0, m_Width, m_Height);\r
-    if (pClip) {\r
-        rect.Intersect(*pClip);\r
-        if (rect.IsEmpty()) {\r
-            return NULL;\r
-        }\r
-    }\r
-    CFX_DIBitmap* pNewBitmap = FX_NEW CFX_DIBitmap;\r
-    if (!pNewBitmap) {\r
-        return NULL;\r
-    }\r
-    if (!pNewBitmap->Create(rect.Width(), rect.Height(), GetFormat())) {\r
-        delete pNewBitmap;\r
-        return NULL;\r
-    }\r
-    pNewBitmap->CopyPalette(m_pPalette);\r
-    pNewBitmap->CopyAlphaMask(m_pAlphaMask, pClip);\r
-    if (GetBPP() == 1 && rect.left % 8 != 0) {\r
-        int left_shift = rect.left % 32;\r
-        int right_shift = 32 - left_shift;\r
-        int dword_count = pNewBitmap->m_Pitch / 4;\r
-        for (int row = rect.top; row < rect.bottom; row ++) {\r
-            FX_DWORD* src_scan = (FX_DWORD*)GetScanline(row) + rect.left / 32;\r
-            FX_DWORD* dest_scan = (FX_DWORD*)pNewBitmap->GetScanline(row - rect.top);\r
-            for (int i = 0; i < dword_count; i ++) {\r
-                dest_scan[i] = (src_scan[i] << left_shift) | (src_scan[i + 1] >> right_shift);\r
-            }\r
-        }\r
-    } else {\r
-        int copy_len = (pNewBitmap->GetWidth() * pNewBitmap->GetBPP() + 7) / 8;\r
-        if (m_Pitch < (FX_DWORD)copy_len) {\r
-            copy_len = m_Pitch;\r
-        }\r
-        for (int row = rect.top; row < rect.bottom; row ++) {\r
-            FX_LPCBYTE src_scan = GetScanline(row) + rect.left * m_bpp / 8;\r
-            FX_LPBYTE dest_scan = (FX_LPBYTE)pNewBitmap->GetScanline(row - rect.top);\r
-            FXSYS_memcpy32(dest_scan, src_scan, copy_len);\r
-        }\r
-    }\r
-    return pNewBitmap;\r
-}\r
-void CFX_DIBSource::BuildPalette()\r
-{\r
-    if (m_pPalette) {\r
-        return;\r
-    }\r
-    if (GetBPP() == 1) {\r
-        m_pPalette = FX_Alloc(FX_DWORD, 2);\r
-        if (!m_pPalette) {\r
-            return;\r
-        }\r
-        if(IsCmykImage()) {\r
-            m_pPalette[0] = 0xff;\r
-            m_pPalette[1] = 0;\r
-        } else {\r
-            m_pPalette[0] = 0xff000000;\r
-            m_pPalette[1] = 0xffffffff;\r
-        }\r
-    } else if (GetBPP() == 8) {\r
-        m_pPalette = FX_Alloc(FX_DWORD, 256);\r
-        if (!m_pPalette) {\r
-            return;\r
-        }\r
-        if(IsCmykImage()) {\r
-            for (int i = 0; i < 256; i ++) {\r
-                m_pPalette[i] = 0xff - i;\r
-            }\r
-        } else {\r
-            for (int i = 0; i < 256; i ++) {\r
-                m_pPalette[i] = 0xff000000 | (i * 0x10101);\r
-            }\r
-        }\r
-    }\r
-}\r
-FX_BOOL CFX_DIBSource::BuildAlphaMask()\r
-{\r
-    if (m_pAlphaMask) {\r
-        return TRUE;\r
-    }\r
-    m_pAlphaMask = FX_NEW CFX_DIBitmap;\r
-    if (!m_pAlphaMask) {\r
-        return FALSE;\r
-    }\r
-    if (!m_pAlphaMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {\r
-        delete m_pAlphaMask;\r
-        m_pAlphaMask = NULL;\r
-        return FALSE;\r
-    }\r
-    FXSYS_memset8(m_pAlphaMask->GetBuffer(), 0xff, m_pAlphaMask->GetHeight()*m_pAlphaMask->GetPitch());\r
-    return TRUE;\r
-}\r
-FX_DWORD CFX_DIBSource::GetPaletteEntry(int index) const\r
-{\r
-    ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());\r
-    if (m_pPalette) {\r
-        return m_pPalette[index];\r
-    }\r
-    if (IsCmykImage()) {\r
-        if (GetBPP() == 1) {\r
-            return index ? 0 : 0xff;\r
-        }\r
-        return 0xff - index;\r
-    }\r
-    if (GetBPP() == 1) {\r
-        return index ? 0xffffffff : 0xff000000;\r
-    }\r
-    return index * 0x10101 | 0xff000000;\r
-}\r
-void CFX_DIBSource::SetPaletteEntry(int index, FX_DWORD color)\r
-{\r
-    ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());\r
-    if (m_pPalette == NULL) {\r
-        BuildPalette();\r
-    }\r
-    m_pPalette[index] = color;\r
-}\r
-int CFX_DIBSource::FindPalette(FX_DWORD color) const\r
-{\r
-    ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());\r
-    if (m_pPalette == NULL) {\r
-        if (IsCmykImage()) {\r
-            if (GetBPP() == 1) {\r
-                return ((FX_BYTE)color == 0xff) ? 0 : 1;\r
-            }\r
-            return 0xff - (FX_BYTE)color;\r
-        }\r
-        if (GetBPP() == 1) {\r
-            return ((FX_BYTE)color == 0xff) ? 1 : 0;\r
-        }\r
-        return (FX_BYTE)color;\r
-    }\r
-    int palsize = (1 << GetBPP());\r
-    for (int i = 0; i < palsize; i ++)\r
-        if (m_pPalette[i] == color) {\r
-            return i;\r
-        }\r
-    return -1;\r
-}\r
-void CFX_DIBitmap::Clear(FX_DWORD color)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return;\r
-    }\r
-    switch (GetFormat()) {\r
-        case FXDIB_1bppMask:\r
-            FXSYS_memset8(m_pBuffer, (color & 0xff000000) ? 0xff : 0, m_Pitch * m_Height);\r
-            break;\r
-        case FXDIB_1bppRgb: {\r
-                int index = FindPalette(color);\r
-                FXSYS_memset8(m_pBuffer, index ? 0xff : 0, m_Pitch * m_Height);\r
-                break;\r
-            }\r
-        case FXDIB_8bppMask:\r
-            FXSYS_memset8(m_pBuffer, color >> 24, m_Pitch * m_Height);\r
-            break;\r
-        case FXDIB_8bppRgb: {\r
-                int index = FindPalette(color);\r
-                FXSYS_memset8(m_pBuffer, index, m_Pitch * m_Height);\r
-                break;\r
-            }\r
-        case FXDIB_Rgb:\r
-        case FXDIB_Rgba: {\r
-                int a, r, g, b;\r
-                ArgbDecode(color, a, r, g, b);\r
-                if (r == g && g == b) {\r
-                    FXSYS_memset8(m_pBuffer, r, m_Pitch * m_Height);\r
-                } else {\r
-                    int byte_pos = 0;\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        m_pBuffer[byte_pos++] = b;\r
-                        m_pBuffer[byte_pos++] = g;\r
-                        m_pBuffer[byte_pos++] = r;\r
-                    }\r
-                    for (int row = 1; row < m_Height; row ++) {\r
-                        FXSYS_memcpy32(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);\r
-                    }\r
-                }\r
-                break;\r
-            }\r
-        case FXDIB_Rgb32:\r
-        case FXDIB_Argb: {\r
-                color = IsCmykImage() ? FXCMYK_TODIB(color) : FXARGB_TODIB(color);\r
-                for (int i = 0; i < m_Width; i ++) {\r
-                    ((FX_DWORD*)m_pBuffer)[i] = color;\r
-                }\r
-                for (int row = 1; row < m_Height; row ++) {\r
-                    FXSYS_memcpy32(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);\r
-                }\r
-                break;\r
-            }\r
-        default:\r
-            break;\r
-    }\r
-}\r
-void CFX_DIBSource::GetOverlapRect(int& dest_left, int& dest_top, int& width, int& height,\r
-                                   int src_width, int src_height, int& src_left, int& src_top,\r
-                                   const CFX_ClipRgn* pClipRgn)\r
-{\r
-    if (width == 0 || height == 0) {\r
-        return;\r
-    }\r
-    ASSERT(width > 0 && height > 0);\r
-    if (dest_left > m_Width || dest_top > m_Height) {\r
-        width = 0;\r
-        height = 0;\r
-        return;\r
-    }\r
-    int x_offset = dest_left - src_left;\r
-    int y_offset = dest_top - src_top;\r
-    FX_RECT src_rect(src_left, src_top, src_left + width, src_top + height);\r
-    FX_RECT src_bound(0, 0, src_width, src_height);\r
-    src_rect.Intersect(src_bound);\r
-    FX_RECT dest_rect(src_rect.left + x_offset, src_rect.top + y_offset,\r
-                      src_rect.right + x_offset, src_rect.bottom + y_offset);\r
-    FX_RECT dest_bound(0, 0, m_Width, m_Height);\r
-    dest_rect.Intersect(dest_bound);\r
-    if (pClipRgn) {\r
-        dest_rect.Intersect(pClipRgn->GetBox());\r
-    }\r
-    dest_left = dest_rect.left;\r
-    dest_top = dest_rect.top;\r
-    src_left = dest_left - x_offset;\r
-    src_top = dest_top - y_offset;\r
-    width = dest_rect.right - dest_rect.left;\r
-    height = dest_rect.bottom - dest_rect.top;\r
-}\r
-FX_BOOL CFX_DIBitmap::TransferBitmap(int dest_left, int dest_top, int width, int height,\r
-                                     const CFX_DIBSource* pSrcBitmap, int src_left, int src_top, void* pIccTransform)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    GetOverlapRect(dest_left, dest_top, width, height, pSrcBitmap->GetWidth(), pSrcBitmap->GetHeight(), src_left, src_top, NULL);\r
-    if (width == 0 || height == 0) {\r
-        return TRUE;\r
-    }\r
-    FXDIB_Format dest_format = GetFormat();\r
-    FXDIB_Format src_format = pSrcBitmap->GetFormat();\r
-    if (dest_format == src_format && pIccTransform == NULL) {\r
-        if (GetBPP() == 1) {\r
-            for (int row = 0; row < height; row ++) {\r
-                FX_LPBYTE dest_scan = m_pBuffer + (dest_top + row) * m_Pitch;\r
-                FX_LPCBYTE src_scan = pSrcBitmap->GetScanline(src_top + row);\r
-                for (int col = 0; col < width; col ++) {\r
-                    if (src_scan[(src_left + col) / 8] & (1 << (7 - (src_left + col) % 8))) {\r
-                        dest_scan[(dest_left + col) / 8] |= 1 << (7 - (dest_left + col) % 8);\r
-                    } else {\r
-                        dest_scan[(dest_left + col) / 8] &= ~(1 << (7 - (dest_left + col) % 8));\r
-                    }\r
-                }\r
-            }\r
-        } else {\r
-            int Bpp = GetBPP() / 8;\r
-            for (int row = 0; row < height; row ++) {\r
-                FX_LPBYTE dest_scan = m_pBuffer + (dest_top + row) * m_Pitch + dest_left * Bpp;\r
-                FX_LPCBYTE src_scan = pSrcBitmap->GetScanline(src_top + row) + src_left * Bpp;\r
-                FXSYS_memcpy32(dest_scan, src_scan, width * Bpp);\r
-            }\r
-        }\r
-    } else {\r
-        if (m_pPalette) {\r
-            return FALSE;\r
-        }\r
-        if (m_bpp == 8) {\r
-            dest_format = FXDIB_8bppMask;\r
-        }\r
-        FX_LPBYTE dest_buf = m_pBuffer + dest_top * m_Pitch + dest_left * GetBPP() / 8;\r
-        FX_DWORD* d_plt = NULL;\r
-        if(!ConvertBuffer(dest_format, dest_buf, m_Pitch, width, height, pSrcBitmap, src_left, src_top, d_plt, pIccTransform)) {\r
-            return FALSE;\r
-        }\r
-    }\r
-    return TRUE;\r
-}\r
-#ifndef _FPDFAPI_MINI_\r
-FX_BOOL CFX_DIBitmap::TransferMask(int dest_left, int dest_top, int width, int height,\r
-                                   const CFX_DIBSource* pMask, FX_DWORD color, int src_left, int src_top, int alpha_flag, void* pIccTransform)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    ASSERT(HasAlpha() && (m_bpp >= 24));\r
-    ASSERT(pMask->IsAlphaMask());\r
-    if (!HasAlpha() || !pMask->IsAlphaMask() || m_bpp < 24) {\r
-        return FALSE;\r
-    }\r
-    GetOverlapRect(dest_left, dest_top, width, height, pMask->GetWidth(), pMask->GetHeight(), src_left, src_top, NULL);\r
-    if (width == 0 || height == 0) {\r
-        return TRUE;\r
-    }\r
-    int src_pitch = pMask->GetPitch();\r
-    int src_bpp = pMask->GetBPP();\r
-    int alpha;\r
-    FX_DWORD dst_color;\r
-    if (alpha_flag >> 8) {\r
-        alpha = alpha_flag & 0xff;\r
-        dst_color = FXCMYK_TODIB(color);\r
-    } else {\r
-        alpha = FXARGB_A(color);\r
-        dst_color = FXARGB_TODIB(color);\r
-    }\r
-    FX_LPBYTE color_p = (FX_LPBYTE)&dst_color;\r
-    if (pIccTransform && CFX_GEModule::Get()->GetCodecModule() && CFX_GEModule::Get()->GetCodecModule()->GetIccModule()) {\r
-        ICodec_IccModule* pIccModule = CFX_GEModule::Get()->GetCodecModule()->GetIccModule();\r
-        pIccModule->TranslateScanline(pIccTransform, color_p, color_p, 1);\r
-    } else {\r
-        if (alpha_flag >> 8 && !IsCmykImage())\r
-            AdobeCMYK_to_sRGB1(FXSYS_GetCValue(color), FXSYS_GetMValue(color), FXSYS_GetYValue(color), FXSYS_GetKValue(color),\r
-                               color_p[2], color_p[1], color_p[0]);\r
-        else if (!(alpha_flag >> 8) && IsCmykImage()) {\r
-            return FALSE;\r
-        }\r
-    }\r
-    if(!IsCmykImage()) {\r
-        color_p[3] = (FX_BYTE)alpha;\r
-    }\r
-    if (GetFormat() == FXDIB_Argb) {\r
-        for (int row = 0; row < height; row ++) {\r
-            FX_DWORD* dest_pos = (FX_DWORD*)(m_pBuffer + (dest_top + row) * m_Pitch + dest_left * 4);\r
-            FX_LPCBYTE src_scan = pMask->GetScanline(src_top + row);\r
-            if (src_bpp == 1) {\r
-                for (int col = 0; col < width; col ++) {\r
-                    int src_bitpos = src_left + col;\r
-                    if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {\r
-                        *dest_pos = dst_color;\r
-                    } else {\r
-                        *dest_pos = 0;\r
-                    }\r
-                    dest_pos ++;\r
-                }\r
-            } else {\r
-                src_scan += src_left;\r
-                dst_color = FXARGB_TODIB(dst_color);\r
-                dst_color &= 0xffffff;\r
-                for (int col = 0; col < width; col ++) {\r
-                    FXARGB_SETDIB(dest_pos++, dst_color | ((alpha * (*src_scan++) / 255) << 24));\r
-                }\r
-            }\r
-        }\r
-    } else {\r
-        int comps = m_bpp / 8;\r
-        for (int row = 0; row < height; row ++) {\r
-            FX_LPBYTE dest_color_pos = m_pBuffer + (dest_top + row) * m_Pitch + dest_left * comps;\r
-            FX_LPBYTE dest_alpha_pos = (FX_LPBYTE)m_pAlphaMask->GetScanline(dest_top + row) + dest_left;\r
-            FX_LPCBYTE src_scan = pMask->GetScanline(src_top + row);\r
-            if (src_bpp == 1) {\r
-                for (int col = 0; col < width; col ++) {\r
-                    int src_bitpos = src_left + col;\r
-                    if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {\r
-                        FXSYS_memcpy32(dest_color_pos, color_p, comps);\r
-                        *dest_alpha_pos = 0xff;\r
-                    } else {\r
-                        FXSYS_memset32(dest_color_pos, 0, comps);\r
-                        *dest_alpha_pos = 0;\r
-                    }\r
-                    dest_color_pos += comps;\r
-                    dest_alpha_pos ++;\r
-                }\r
-            } else {\r
-                src_scan += src_left;\r
-                for (int col = 0; col < width; col ++) {\r
-                    FXSYS_memcpy32(dest_color_pos, color_p, comps);\r
-                    dest_color_pos += comps;\r
-                    *dest_alpha_pos++ = (alpha * (*src_scan++) / 255);\r
-                }\r
-            }\r
-        }\r
-    }\r
-    return TRUE;\r
-}\r
-#endif\r
-void CFX_DIBSource::CopyPalette(const FX_DWORD* pSrc, FX_DWORD size)\r
-{\r
-    if (pSrc == NULL || GetBPP() > 8) {\r
-        if (m_pPalette) {\r
-            FX_Free(m_pPalette);\r
-        }\r
-        m_pPalette = NULL;\r
-    } else {\r
-        FX_DWORD pal_size = 1 << GetBPP();\r
-        if (m_pPalette == NULL) {\r
-            m_pPalette = FX_Alloc(FX_DWORD, pal_size);\r
-        }\r
-        if (!m_pPalette) {\r
-            return;\r
-        }\r
-        if (pal_size > size) {\r
-            pal_size = size;\r
-        }\r
-        FXSYS_memcpy32(m_pPalette, pSrc, pal_size * sizeof(FX_DWORD));\r
-    }\r
-}\r
-void CFX_DIBSource::GetPalette(FX_DWORD* pal, int alpha) const\r
-{\r
-    ASSERT(GetBPP() <= 8 && !IsCmykImage());\r
-    if (GetBPP() == 1) {\r
-        pal[0] = ((m_pPalette ? m_pPalette[0] : 0xff000000) & 0xffffff) | (alpha << 24);\r
-        pal[1] = ((m_pPalette ? m_pPalette[1] : 0xffffffff) & 0xffffff) | (alpha << 24);\r
-        return;\r
-    }\r
-    if (m_pPalette) {\r
-        for (int i = 0; i < 256; i ++) {\r
-            pal[i] = (m_pPalette[i] & 0x00ffffff) | (alpha << 24);\r
-        }\r
-    } else {\r
-        for (int i = 0; i < 256; i ++) {\r
-            pal[i] = (i * 0x10101) | (alpha << 24);\r
-        }\r
-    }\r
-}\r
-CFX_DIBitmap* CFX_DIBSource::GetAlphaMask(const FX_RECT* pClip) const\r
-{\r
-    ASSERT(GetFormat() == FXDIB_Argb);\r
-    FX_RECT rect(0, 0, m_Width, m_Height);\r
-    if (pClip) {\r
-        rect.Intersect(*pClip);\r
-        if (rect.IsEmpty()) {\r
-            return NULL;\r
-        }\r
-    }\r
-    CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;\r
-    if (!pMask) {\r
-        return NULL;\r
-    }\r
-    if (!pMask->Create(rect.Width(), rect.Height(), FXDIB_8bppMask)) {\r
-        delete pMask;\r
-        return NULL;\r
-    }\r
-    for (int row = rect.top; row < rect.bottom; row ++) {\r
-        FX_LPCBYTE src_scan = GetScanline(row) + rect.left * 4 + 3;\r
-        FX_LPBYTE dest_scan = (FX_LPBYTE)pMask->GetScanline(row - rect.top);\r
-        for (int col = rect.left; col < rect.right; col ++) {\r
-            *dest_scan ++ = *src_scan;\r
-            src_scan += 4;\r
-        }\r
-    }\r
-    return pMask;\r
-}\r
-FX_BOOL CFX_DIBSource::CopyAlphaMask(const CFX_DIBSource* pAlphaMask, const FX_RECT* pClip)\r
-{\r
-    if (!HasAlpha() || GetFormat() == FXDIB_Argb) {\r
-        return FALSE;\r
-    }\r
-    if (pAlphaMask) {\r
-        FX_RECT rect(0, 0, pAlphaMask->m_Width, pAlphaMask->m_Height);\r
-        if (pClip) {\r
-            rect.Intersect(*pClip);\r
-            if (rect.IsEmpty() || rect.Width() != m_Width || rect.Height() != m_Height) {\r
-                return FALSE;\r
-            }\r
-        } else {\r
-            if (pAlphaMask->m_Width != m_Width || pAlphaMask->m_Height != m_Height) {\r
-                return FALSE;\r
-            }\r
-        }\r
-        for (int row = 0; row < m_Height; row ++)\r
-            FXSYS_memcpy32((void*)m_pAlphaMask->GetScanline(row),\r
-                           pAlphaMask->GetScanline(row + rect.top) + rect.left, m_pAlphaMask->m_Pitch);\r
-    } else {\r
-        m_pAlphaMask->Clear(0xff000000);\r
-    }\r
-    return TRUE;\r
-}\r
-const int g_ChannelOffset[] = {0, 2, 1, 0, 0, 1, 2, 3, 3};\r
-FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, const CFX_DIBSource* pSrcBitmap, FXDIB_Channel srcChannel)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    CFX_DIBSource* pSrcClone = (CFX_DIBSource*)pSrcBitmap;\r
-    CFX_DIBitmap* pDst = this;\r
-    int destOffset, srcOffset;\r
-    if (srcChannel == FXDIB_Alpha) {\r
-        if (!pSrcBitmap->HasAlpha() && !pSrcBitmap->IsAlphaMask()) {\r
-            return FALSE;\r
-        }\r
-        if (pSrcBitmap->GetBPP() == 1) {\r
-            pSrcClone = pSrcBitmap->CloneConvert(FXDIB_8bppMask);\r
-            if (pSrcClone == NULL) {\r
-                return FALSE;\r
-            }\r
-        }\r
-        if(pSrcBitmap->GetFormat() == FXDIB_Argb) {\r
-            srcOffset = 3;\r
-        } else {\r
-            srcOffset = 0;\r
-        }\r
-    } else {\r
-        if (pSrcBitmap->IsAlphaMask()) {\r
-            return FALSE;\r
-        }\r
-        if (pSrcBitmap->GetBPP() < 24) {\r
-            if (pSrcBitmap->IsCmykImage()) {\r
-                pSrcClone = pSrcBitmap->CloneConvert((FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x20));\r
-            } else {\r
-                pSrcClone = pSrcBitmap->CloneConvert((FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x18));\r
-            }\r
-            if (pSrcClone == NULL) {\r
-                return FALSE;\r
-            }\r
-        }\r
-        srcOffset = g_ChannelOffset[srcChannel];\r
-    }\r
-    if (destChannel == FXDIB_Alpha) {\r
-        if (IsAlphaMask()) {\r
-            if(!ConvertFormat(FXDIB_8bppMask)) {\r
-                if (pSrcClone != pSrcBitmap) {\r
-                    delete pSrcClone;\r
-                }\r
-                return FALSE;\r
-            }\r
-            destOffset = 0;\r
-        } else {\r
-            destOffset = 0;\r
-            if(!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {\r
-                if (pSrcClone != pSrcBitmap) {\r
-                    delete pSrcClone;\r
-                }\r
-                return FALSE;\r
-            }\r
-            if (GetFormat() == FXDIB_Argb) {\r
-                destOffset = 3;\r
-            }\r
-        }\r
-    } else {\r
-        if (IsAlphaMask()) {\r
-            if (pSrcClone != pSrcBitmap) {\r
-                delete pSrcClone;\r
-            }\r
-            return FALSE;\r
-        }\r
-        if (GetBPP() < 24) {\r
-            if (HasAlpha()) {\r
-                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {\r
-                    if (pSrcClone != pSrcBitmap) {\r
-                        delete pSrcClone;\r
-                    }\r
-                    return FALSE;\r
-                }\r
-            } else\r
-#if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_\r
-                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {\r
-#else\r
-                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {\r
-#endif\r
-                    if (pSrcClone != pSrcBitmap) {\r
-                        delete pSrcClone;\r
-                    }\r
-                    return FALSE;\r
-                }\r
-        }\r
-        destOffset = g_ChannelOffset[destChannel];\r
-    }\r
-    if (srcChannel == FXDIB_Alpha && pSrcClone->m_pAlphaMask) {\r
-        CFX_DIBitmap* pAlphaMask = pSrcClone->m_pAlphaMask;\r
-        if (pSrcClone->GetWidth() != m_Width || pSrcClone->GetHeight() != m_Height) {\r
-            if (pAlphaMask) {\r
-                pAlphaMask = pAlphaMask->StretchTo(m_Width, m_Height);\r
-                if (pAlphaMask == NULL) {\r
-                    if (pSrcClone != pSrcBitmap) {\r
-                        delete pSrcClone;\r
-                    }\r
-                    return FALSE;\r
-                }\r
-            }\r
-        }\r
-        if (pSrcClone != pSrcBitmap) {\r
-            pSrcClone->m_pAlphaMask = NULL;\r
-            delete pSrcClone;\r
-        }\r
-        pSrcClone = pAlphaMask;\r
-        srcOffset = 0;\r
-    } else if (pSrcClone->GetWidth() != m_Width || pSrcClone->GetHeight() != m_Height) {\r
-        CFX_DIBitmap* pSrcMatched = pSrcClone->StretchTo(m_Width, m_Height);\r
-        if (pSrcClone != pSrcBitmap) {\r
-            delete pSrcClone;\r
-        }\r
-        if (pSrcMatched == NULL) {\r
-            return FALSE;\r
-        }\r
-        pSrcClone = pSrcMatched;\r
-    }\r
-    if (destChannel == FXDIB_Alpha && m_pAlphaMask) {\r
-        pDst = m_pAlphaMask;\r
-        destOffset = 0;\r
-    }\r
-    int srcBytes = pSrcClone->GetBPP() / 8;\r
-    int destBytes = pDst->GetBPP() / 8;\r
-    for (int row = 0; row < m_Height; row ++) {\r
-        FX_LPBYTE dest_pos = (FX_LPBYTE)pDst->GetScanline(row) + destOffset;\r
-        FX_LPCBYTE src_pos = pSrcClone->GetScanline(row) + srcOffset;\r
-        for (int col = 0; col < m_Width; col ++) {\r
-            *dest_pos = *src_pos;\r
-            dest_pos += destBytes;\r
-            src_pos += srcBytes;\r
-        }\r
-    }\r
-    if (pSrcClone != pSrcBitmap && pSrcClone != pSrcBitmap->m_pAlphaMask) {\r
-        delete pSrcClone;\r
-    }\r
-    return TRUE;\r
-}\r
-FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, int value)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    int destOffset;\r
-    if (destChannel == FXDIB_Alpha) {\r
-        if (IsAlphaMask()) {\r
-            if(!ConvertFormat(FXDIB_8bppMask)) {\r
-                return FALSE;\r
-            }\r
-            destOffset = 0;\r
-        } else {\r
-            destOffset = 0;\r
-            if(!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {\r
-                return FALSE;\r
-            }\r
-            if (GetFormat() == FXDIB_Argb) {\r
-                destOffset = 3;\r
-            }\r
-        }\r
-    } else {\r
-        if (IsAlphaMask()) {\r
-            return FALSE;\r
-        }\r
-        if (GetBPP() < 24) {\r
-            if (HasAlpha()) {\r
-                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {\r
-                    return FALSE;\r
-                }\r
-            } else\r
-#if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_\r
-                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {\r
-                    return FALSE;\r
-                }\r
-#else\r
-                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {\r
-                    return FALSE;\r
-                }\r
-#endif\r
-        }\r
-        destOffset = g_ChannelOffset[destChannel];\r
-    }\r
-    int Bpp = GetBPP() / 8;\r
-    if (Bpp == 1) {\r
-        FXSYS_memset8(m_pBuffer, value, m_Height * m_Pitch);\r
-        return TRUE;\r
-    }\r
-    if (destChannel == FXDIB_Alpha && m_pAlphaMask) {\r
-        FXSYS_memset8(m_pAlphaMask->GetBuffer(), value, m_pAlphaMask->GetHeight()*m_pAlphaMask->GetPitch());\r
-        return TRUE;\r
-    }\r
-    for (int row = 0; row < m_Height; row ++) {\r
-        FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch + destOffset;\r
-        for (int col = 0; col < m_Width; col ++) {\r
-            *scan_line = value;\r
-            scan_line += Bpp;\r
-        }\r
-    }\r
-    return TRUE;\r
-}\r
-FX_BOOL CFX_DIBitmap::MultiplyAlpha(const CFX_DIBSource* pSrcBitmap)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    ASSERT(pSrcBitmap->IsAlphaMask());\r
-    if (!pSrcBitmap->IsAlphaMask()) {\r
-        return FALSE;\r
-    }\r
-    if (!IsAlphaMask() && !HasAlpha()) {\r
-        return LoadChannel(FXDIB_Alpha, pSrcBitmap, FXDIB_Alpha);\r
-    }\r
-    CFX_DIBitmap* pSrcClone = (CFX_DIBitmap*)pSrcBitmap;\r
-    if (pSrcBitmap->GetWidth() != m_Width || pSrcBitmap->GetHeight() != m_Height) {\r
-        pSrcClone = pSrcBitmap->StretchTo(m_Width, m_Height);\r
-        ASSERT(pSrcClone != NULL);\r
-        if (pSrcClone == NULL) {\r
-            return FALSE;\r
-        }\r
-    }\r
-    if (IsAlphaMask()) {\r
-        if(!ConvertFormat(FXDIB_8bppMask)) {\r
-            if (pSrcClone != pSrcBitmap) {\r
-                delete pSrcClone;\r
-            }\r
-            return FALSE;\r
-        }\r
-        for (int row = 0; row < m_Height; row ++) {\r
-            FX_LPBYTE dest_scan = m_pBuffer + m_Pitch * row;\r
-            FX_LPBYTE src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;\r
-            if (pSrcClone->GetBPP() == 1) {\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    if (!((1 << (7 - col % 8)) & src_scan[col / 8])) {\r
-                        dest_scan[col] = 0;\r
-                    }\r
-                }\r
-            } else {\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    *dest_scan = (*dest_scan) * src_scan[col] / 255;\r
-                    dest_scan ++;\r
-                }\r
-            }\r
-        }\r
-    } else {\r
-        if(GetFormat() == FXDIB_Argb) {\r
-            if (pSrcClone->GetBPP() == 1) {\r
-                if (pSrcClone != pSrcBitmap) {\r
-                    delete pSrcClone;\r
-                }\r
-                return FALSE;\r
-            }\r
-            for (int row = 0; row < m_Height; row ++) {\r
-                FX_LPBYTE dest_scan = m_pBuffer + m_Pitch * row + 3;\r
-                FX_LPBYTE src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    *dest_scan = (*dest_scan) * src_scan[col] / 255;\r
-                    dest_scan += 4;\r
-                }\r
-            }\r
-        } else {\r
-            m_pAlphaMask->MultiplyAlpha(pSrcClone);\r
-        }\r
-    }\r
-    if (pSrcClone != pSrcBitmap) {\r
-        delete pSrcClone;\r
-    }\r
-    return TRUE;\r
-}\r
-FX_BOOL CFX_DIBitmap::GetGrayData(void* pIccTransform)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    switch (GetFormat()) {\r
-        case FXDIB_1bppRgb: {\r
-                if (m_pPalette == NULL) {\r
-                    return FALSE;\r
-                }\r
-                FX_BYTE gray[2];\r
-                for (int i = 0; i < 2; i ++) {\r
-                    int r = (FX_BYTE)(m_pPalette[i] >> 16);\r
-                    int g = (FX_BYTE)(m_pPalette[i] >> 8);\r
-                    int b = (FX_BYTE)m_pPalette[i];\r
-                    gray[i] = (FX_BYTE)FXRGB2GRAY(r, g, b);\r
-                }\r
-                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;\r
-                if (!pMask) {\r
-                    return FALSE;\r
-                }\r
-                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {\r
-                    delete pMask;\r
-                    return FALSE;\r
-                }\r
-                FXSYS_memset8(pMask->GetBuffer(), gray[0], pMask->GetPitch() * m_Height);\r
-                for (int row = 0; row < m_Height; row ++) {\r
-                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;\r
-                    FX_LPBYTE dest_pos = (FX_LPBYTE)pMask->GetScanline(row);\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        if (src_pos[col / 8] & (1 << (7 - col % 8))) {\r
-                            *dest_pos = gray[1];\r
-                        }\r
-                        dest_pos ++;\r
-                    }\r
-                }\r
-                TakeOver(pMask);\r
-                delete pMask;\r
-                break;\r
-            }\r
-        case FXDIB_8bppRgb: {\r
-                if (m_pPalette == NULL) {\r
-                    return FALSE;\r
-                }\r
-                FX_BYTE gray[256];\r
-                for (int i = 0; i < 256; i ++) {\r
-                    int r = (FX_BYTE)(m_pPalette[i] >> 16);\r
-                    int g = (FX_BYTE)(m_pPalette[i] >> 8);\r
-                    int b = (FX_BYTE)m_pPalette[i];\r
-                    gray[i] = (FX_BYTE)FXRGB2GRAY(r, g, b);\r
-                }\r
-                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;\r
-                if (!pMask) {\r
-                    return FALSE;\r
-                }\r
-                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {\r
-                    delete pMask;\r
-                    return FALSE;\r
-                }\r
-                for (int row = 0; row < m_Height; row ++) {\r
-                    FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();\r
-                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        *dest_pos ++ = gray[*src_pos ++];\r
-                    }\r
-                }\r
-                TakeOver(pMask);\r
-                delete pMask;\r
-                break;\r
-            }\r
-        case FXDIB_Rgb: {\r
-                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;\r
-                if (!pMask) {\r
-                    return FALSE;\r
-                }\r
-                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {\r
-                    delete pMask;\r
-                    return FALSE;\r
-                }\r
-                for (int row = 0; row < m_Height; row ++) {\r
-                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;\r
-                    FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        *dest_pos ++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);\r
-                        src_pos += 3;\r
-                    }\r
-                }\r
-                TakeOver(pMask);\r
-                delete pMask;\r
-                break;\r
-            }\r
-        case FXDIB_Rgb32: {\r
-                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;\r
-                if (!pMask) {\r
-                    return FALSE;\r
-                }\r
-                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {\r
-                    delete pMask;\r
-                    return FALSE;\r
-                }\r
-                for (int row = 0; row < m_Height; row ++) {\r
-                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;\r
-                    FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        *dest_pos ++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);\r
-                        src_pos += 4;\r
-                    }\r
-                }\r
-                TakeOver(pMask);\r
-                delete pMask;\r
-                break;\r
-            }\r
-        default:\r
-            return FALSE;\r
-    }\r
-    return TRUE;\r
-}\r
-FX_BOOL CFX_DIBitmap::MultiplyAlpha(int alpha)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    switch (GetFormat()) {\r
-        case FXDIB_1bppMask:\r
-            if (!ConvertFormat(FXDIB_8bppMask)) {\r
-                return FALSE;\r
-            }\r
-            MultiplyAlpha(alpha);\r
-            break;\r
-        case FXDIB_8bppMask: {\r
-                for (int row = 0; row < m_Height; row ++) {\r
-                    FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch;\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        scan_line[col] = scan_line[col] * alpha / 255;\r
-                    }\r
-                }\r
-                break;\r
-            }\r
-        case FXDIB_Argb: {\r
-                for (int row = 0; row < m_Height; row ++) {\r
-                    FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch + 3;\r
-                    for (int col = 0; col < m_Width; col ++) {\r
-                        *scan_line = (*scan_line) * alpha / 255;\r
-                        scan_line += 4;\r
-                    }\r
-                }\r
-                break;\r
-            }\r
-        default:\r
-            if (HasAlpha()) {\r
-                m_pAlphaMask->MultiplyAlpha(alpha);\r
-            } else if (IsCmykImage()) {\r
-                if (!ConvertFormat((FXDIB_Format)(GetFormat() | 0x0200))) {\r
-                    return FALSE;\r
-                }\r
-                m_pAlphaMask->MultiplyAlpha(alpha);\r
-            } else {\r
-                if (!ConvertFormat(FXDIB_Argb)) {\r
-                    return FALSE;\r
-                }\r
-                MultiplyAlpha(alpha);\r
-            }\r
-            break;\r
-    }\r
-    return TRUE;\r
-}\r
-#if !defined(_FPDFAPI_MINI_) || defined(_FXCORE_FEATURE_ALL_)\r
-FX_DWORD CFX_DIBitmap::GetPixel(int x, int y) const\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return 0;\r
-    }\r
-    FX_LPBYTE pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;\r
-    switch (GetFormat()) {\r
-        case FXDIB_1bppMask: {\r
-                if ((*pos) & (1 << (7 - x % 8))) {\r
-                    return 0xff000000;\r
-                }\r
-                return 0;\r
-            }\r
-        case FXDIB_1bppRgb: {\r
-                if ((*pos) & (1 << (7 - x % 8))) {\r
-                    return m_pPalette ? m_pPalette[1] : 0xffffffff;\r
-                } else {\r
-                    return m_pPalette ? m_pPalette[0] : 0xff000000;\r
-                }\r
-                break;\r
-            }\r
-        case FXDIB_8bppMask:\r
-            return (*pos) << 24;\r
-        case FXDIB_8bppRgb:\r
-            return m_pPalette ? m_pPalette[*pos] : (0xff000000 | ((*pos) * 0x10101));\r
-        case FXDIB_Rgb:\r
-        case FXDIB_Rgba:\r
-        case FXDIB_Rgb32:\r
-            return FXARGB_GETDIB(pos) | 0xff000000;\r
-        case FXDIB_Argb:\r
-            return FXARGB_GETDIB(pos);\r
-        default:\r
-            break;\r
-    }\r
-    return 0;\r
-}\r
-#endif\r
-void CFX_DIBitmap::SetPixel(int x, int y, FX_DWORD color)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return;\r
-    }\r
-    if (x < 0 || x >= m_Width || y < 0 || y >= m_Height) {\r
-        return;\r
-    }\r
-    FX_LPBYTE pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;\r
-    switch (GetFormat()) {\r
-        case FXDIB_1bppMask:\r
-            if (color >> 24) {\r
-                *pos |= 1 << (7 - x % 8);\r
-            } else {\r
-                *pos &= ~(1 << (7 - x % 8));\r
-            }\r
-            break;\r
-        case FXDIB_1bppRgb:\r
-            if (m_pPalette) {\r
-                if (color == m_pPalette[1]) {\r
-                    *pos |= 1 << (7 - x % 8);\r
-                } else {\r
-                    *pos &= ~(1 << (7 - x % 8));\r
-                }\r
-            } else {\r
-                if (color == 0xffffffff) {\r
-                    *pos |= 1 << (7 - x % 8);\r
-                } else {\r
-                    *pos &= ~(1 << (7 - x % 8));\r
-                }\r
-            }\r
-            break;\r
-        case FXDIB_8bppMask:\r
-            *pos = (FX_BYTE)(color >> 24);\r
-            break;\r
-        case FXDIB_8bppRgb: {\r
-                if (m_pPalette) {\r
-                    for (int i = 0; i < 256; i ++) {\r
-                        if (m_pPalette[i] == color) {\r
-                            *pos = (FX_BYTE)i;\r
-                            return;\r
-                        }\r
-                    }\r
-                    *pos = 0;\r
-                } else {\r
-                    *pos = FXRGB2GRAY(FXARGB_R(color), FXARGB_G(color), FXARGB_B(color));\r
-                }\r
-                break;\r
-            }\r
-        case FXDIB_Rgb:\r
-        case FXDIB_Rgb32: {\r
-                int alpha = FXARGB_A(color);\r
-                pos[0] = (FXARGB_B(color) * alpha + pos[0] * (255 - alpha)) / 255;\r
-                pos[1] = (FXARGB_G(color) * alpha + pos[1] * (255 - alpha)) / 255;\r
-                pos[2] = (FXARGB_R(color) * alpha + pos[2] * (255 - alpha)) / 255;\r
-                break;\r
-            }\r
-        case FXDIB_Rgba: {\r
-                pos[0] = FXARGB_B(color);\r
-                pos[1] = FXARGB_G(color);\r
-                pos[2] = FXARGB_R(color);\r
-                break;\r
-            }\r
-        case FXDIB_Argb:\r
-            FXARGB_SETDIB(pos, color);\r
-            break;\r
-        default:\r
-            break;\r
-    }\r
-}\r
-void CFX_DIBitmap::DownSampleScanline(int line, FX_LPBYTE dest_scan, int dest_bpp,\r
-                                      int dest_width, FX_BOOL bFlipX, int clip_left, int clip_width) const\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return;\r
-    }\r
-    int src_Bpp = m_bpp / 8;\r
-    FX_LPBYTE scanline = m_pBuffer + line * m_Pitch;\r
-    if (src_Bpp == 0) {\r
-        for (int i = 0; i < clip_width; i ++) {\r
-            FX_DWORD dest_x = clip_left + i;\r
-            FX_DWORD src_x = dest_x * m_Width / dest_width;\r
-            if (bFlipX) {\r
-                src_x = m_Width - src_x - 1;\r
-            }\r
-#ifdef FOXIT_CHROME_BUILD\r
-            src_x %= m_Width;\r
-#endif\r
-            dest_scan[i] = (scanline[src_x / 8] & (1 << (7 - src_x % 8))) ? 255 : 0;\r
-        }\r
-    } else if (src_Bpp == 1) {\r
-        for (int i = 0; i < clip_width; i ++) {\r
-            FX_DWORD dest_x = clip_left + i;\r
-            FX_DWORD src_x = dest_x * m_Width / dest_width;\r
-            if (bFlipX) {\r
-                src_x = m_Width - src_x - 1;\r
-            }\r
-#ifdef FOXIT_CHROME_BUILD\r
-            src_x %= m_Width;\r
-#endif\r
-            int dest_pos = i;\r
-            if (m_pPalette) {\r
-                if (!IsCmykImage()) {\r
-                    dest_pos *= 3;\r
-                    FX_ARGB argb = m_pPalette[scanline[src_x]];\r
-                    dest_scan[dest_pos] = FXARGB_B(argb);\r
-                    dest_scan[dest_pos + 1] = FXARGB_G(argb);\r
-                    dest_scan[dest_pos + 2] = FXARGB_R(argb);\r
-                } else {\r
-                    dest_pos *= 4;\r
-                    FX_CMYK cmyk = m_pPalette[scanline[src_x]];\r
-                    dest_scan[dest_pos] = FXSYS_GetCValue(cmyk);\r
-                    dest_scan[dest_pos + 1] = FXSYS_GetMValue(cmyk);\r
-                    dest_scan[dest_pos + 2] = FXSYS_GetYValue(cmyk);\r
-                    dest_scan[dest_pos + 3] = FXSYS_GetKValue(cmyk);\r
-                }\r
-            } else {\r
-                dest_scan[dest_pos] = scanline[src_x];\r
-            }\r
-        }\r
-    } else {\r
-        for (int i = 0; i < clip_width; i ++) {\r
-            FX_DWORD dest_x = clip_left + i;\r
-            FX_DWORD src_x = bFlipX ? (m_Width - dest_x * m_Width / dest_width - 1) * src_Bpp : (dest_x * m_Width / dest_width) * src_Bpp;\r
-#ifdef FOXIT_CHROME_BUILD\r
-            src_x %= m_Width * src_Bpp;\r
-#endif\r
-            int dest_pos = i * src_Bpp;\r
-            for (int b = 0; b < src_Bpp; b ++) {\r
-                dest_scan[dest_pos + b] = scanline[src_x + b];\r
-            }\r
-        }\r
-    }\r
-}\r
-FX_BOOL CFX_DIBitmap::ConvertColorScale(FX_DWORD forecolor, FX_DWORD backcolor)\r
-{\r
-    ASSERT(!IsAlphaMask());\r
-    if (m_pBuffer == NULL || IsAlphaMask()) {\r
-        return FALSE;\r
-    }\r
-    int fc, fm, fy, fk, bc, bm, by, bk;\r
-    int fr, fg, fb, br, bg, bb;\r
-    FX_BOOL isCmykImage = IsCmykImage();\r
-    if (isCmykImage) {\r
-        fc = FXSYS_GetCValue(forecolor);\r
-        fm = FXSYS_GetMValue(forecolor);\r
-        fy = FXSYS_GetYValue(forecolor);\r
-        fk = FXSYS_GetKValue(forecolor);\r
-        bc = FXSYS_GetCValue(backcolor);\r
-        bm = FXSYS_GetMValue(backcolor);\r
-        by = FXSYS_GetYValue(backcolor);\r
-        bk = FXSYS_GetKValue(backcolor);\r
-    } else {\r
-        fr = FXSYS_GetRValue(forecolor);\r
-        fg = FXSYS_GetGValue(forecolor);\r
-        fb = FXSYS_GetBValue(forecolor);\r
-        br = FXSYS_GetRValue(backcolor);\r
-        bg = FXSYS_GetGValue(backcolor);\r
-        bb = FXSYS_GetBValue(backcolor);\r
-    }\r
-    if (m_bpp <= 8) {\r
-        if (isCmykImage) {\r
-            if (forecolor == 0xff && backcolor == 0 && m_pPalette == NULL) {\r
-                return TRUE;\r
-            }\r
-        } else if (forecolor == 0 && backcolor == 0xffffff && m_pPalette == NULL) {\r
-            return TRUE;\r
-        }\r
-        if (m_pPalette == NULL) {\r
-            BuildPalette();\r
-        }\r
-        int size = 1 << m_bpp;\r
-        if (isCmykImage) {\r
-            for (int i = 0; i < size; i ++) {\r
-                FX_BYTE b, g, r;\r
-                AdobeCMYK_to_sRGB1(FXSYS_GetCValue(m_pPalette[i]), FXSYS_GetMValue(m_pPalette[i]), FXSYS_GetYValue(m_pPalette[i]), FXSYS_GetKValue(m_pPalette[i]),\r
-                                   r, g, b);\r
-                int gray = 255 - FXRGB2GRAY(r, g, b);\r
-                m_pPalette[i] = CmykEncode(bc + (fc - bc) * gray / 255, bm + (fm - bm) * gray / 255,\r
-                                           by + (fy - by) * gray / 255, bk + (fk - bk) * gray / 255);\r
-            }\r
-        } else\r
-            for (int i = 0; i < size; i ++) {\r
-                int gray = FXRGB2GRAY(FXARGB_R(m_pPalette[i]), FXARGB_G(m_pPalette[i]), FXARGB_B(m_pPalette[i]));\r
-                m_pPalette[i] = FXARGB_MAKE(0xff, br + (fr - br) * gray / 255, bg + (fg - bg) * gray / 255,\r
-                                            bb + (fb - bb) * gray / 255);\r
-            }\r
-        return TRUE;\r
-    }\r
-    if (isCmykImage) {\r
-        if (forecolor == 0xff && backcolor == 0x00) {\r
-            for (int row = 0; row < m_Height; row ++) {\r
-                FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    FX_BYTE b, g, r;\r
-                    AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],\r
-                                       r, g, b);\r
-                    *scanline ++ = 0;\r
-                    *scanline ++ = 0;\r
-                    *scanline ++ = 0;\r
-                    *scanline ++ = 255 - FXRGB2GRAY(r, g, b);\r
-                }\r
-            }\r
-            return TRUE;\r
-        }\r
-    } else if (forecolor == 0 && backcolor == 0xffffff) {\r
-        for (int row = 0; row < m_Height; row ++) {\r
-            FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;\r
-            int gap = m_bpp / 8 - 2;\r
-            for (int col = 0; col < m_Width; col ++) {\r
-                int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);\r
-                *scanline ++ = gray;\r
-                *scanline ++ = gray;\r
-                *scanline    = gray;\r
-                scanline += gap;\r
-            }\r
-        }\r
-        return TRUE;\r
-    }\r
-    if (isCmykImage) {\r
-        for (int row = 0; row < m_Height; row ++) {\r
-            FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;\r
-            for (int col = 0; col < m_Width; col ++) {\r
-                FX_BYTE b, g, r;\r
-                AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],\r
-                                   r, g, b);\r
-                int gray = 255 - FXRGB2GRAY(r, g, b);\r
-                *scanline ++ = bc + (fc - bc) * gray / 255;\r
-                *scanline ++ = bm + (fm - bm) * gray / 255;\r
-                *scanline ++ = by + (fy - by) * gray / 255;\r
-                *scanline ++ = bk + (fk - bk) * gray / 255;\r
-            }\r
-        }\r
-    } else\r
-        for (int row = 0; row < m_Height; row ++) {\r
-            FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;\r
-            int gap = m_bpp / 8 - 2;\r
-            for (int col = 0; col < m_Width; col ++) {\r
-                int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);\r
-                *scanline ++ = bb + (fb - bb) * gray / 255;\r
-                *scanline ++ = bg + (fg - bg) * gray / 255;\r
-                *scanline    = br + (fr - br) * gray / 255;\r
-                scanline += gap;\r
-            }\r
-        }\r
-    return TRUE;\r
-}\r
-FX_BOOL CFX_DIBitmap::DitherFS(const FX_DWORD* pPalette, int pal_size, const FX_RECT* pRect)\r
-{\r
-    if (m_pBuffer == NULL) {\r
-        return FALSE;\r
-    }\r
-    if (m_bpp != 8 && m_pPalette != NULL && m_AlphaFlag != 0) {\r
-        return FALSE;\r
-    }\r
-    if (m_Width < 4 && m_Height < 4) {\r
-        return FALSE;\r
-    }\r
-    FX_RECT rect(0, 0, m_Width, m_Height);\r
-    if (pRect) {\r
-        rect.Intersect(*pRect);\r
-    }\r
-    FX_BYTE translate[256];\r
-    for (int i = 0; i < 256; i ++) {\r
-        int err2 = 65536;\r
-        for (int j = 0; j < pal_size; j ++) {\r
-            FX_BYTE entry = (FX_BYTE)pPalette[j];\r
-            int err = (int)entry - i;\r
-            if (err * err < err2) {\r
-                err2 = err * err;\r
-                translate[i] = entry;\r
-            }\r
-        }\r
-    }\r
-    for (int row = rect.top; row < rect.bottom; row ++) {\r
-        FX_LPBYTE scan = m_pBuffer + row * m_Pitch;\r
-        FX_LPBYTE next_scan = m_pBuffer + (row + 1) * m_Pitch;\r
-        for (int col = rect.left; col < rect.right; col ++) {\r
-            int src_pixel = scan[col];\r
-            int dest_pixel = translate[src_pixel];\r
-            scan[col] = (FX_BYTE)dest_pixel;\r
-            int error = -dest_pixel + src_pixel;\r
-            if (col < rect.right - 1) {\r
-                int src = scan[col + 1];\r
-                src += error * 7 / 16;\r
-                if (src > 255) {\r
-                    scan[col + 1] = 255;\r
-                } else if (src < 0) {\r
-                    scan[col + 1] = 0;\r
-                } else {\r
-                    scan[col + 1] = src;\r
-                }\r
-            }\r
-            if (col < rect.right - 1 && row < rect.bottom - 1) {\r
-                int src = next_scan[col + 1];\r
-                src += error * 1 / 16;\r
-                if (src > 255) {\r
-                    next_scan[col + 1] = 255;\r
-                } else if (src < 0) {\r
-                    next_scan[col + 1] = 0;\r
-                } else {\r
-                    next_scan[col + 1] = src;\r
-                }\r
-            }\r
-            if (row < rect.bottom - 1) {\r
-                int src = next_scan[col];\r
-                src += error * 5 / 16;\r
-                if (src > 255) {\r
-                    next_scan[col] = 255;\r
-                } else if (src < 0) {\r
-                    next_scan[col] = 0;\r
-                } else {\r
-                    next_scan[col] = src;\r
-                }\r
-            }\r
-            if (col > rect.left && row < rect.bottom - 1) {\r
-                int src = next_scan[col - 1];\r
-                src += error * 3 / 16;\r
-                if (src > 255) {\r
-                    next_scan[col - 1] = 255;\r
-                } else if (src < 0) {\r
-                    next_scan[col - 1] = 0;\r
-                } else {\r
-                    next_scan[col - 1] = src;\r
-                }\r
-            }\r
-        }\r
-    }\r
-    return TRUE;\r
-}\r
-CFX_DIBitmap* CFX_DIBSource::FlipImage(FX_BOOL bXFlip, FX_BOOL bYFlip) const\r
-{\r
-    CFX_DIBitmap* pFlipped = FX_NEW CFX_DIBitmap;\r
-    if (!pFlipped) {\r
-        return NULL;\r
-    }\r
-    if (!pFlipped->Create(m_Width, m_Height, GetFormat())) {\r
-        delete pFlipped;\r
-        return NULL;\r
-    }\r
-    pFlipped->CopyPalette(m_pPalette);\r
-    FX_LPBYTE pDestBuffer = pFlipped->GetBuffer();\r
-    int Bpp = m_bpp / 8;\r
-    for (int row = 0; row < m_Height; row ++) {\r
-        FX_LPCBYTE src_scan = GetScanline(row);\r
-        FX_LPBYTE dest_scan = pDestBuffer + m_Pitch * (bYFlip ? (m_Height - row - 1) : row);\r
-        if (!bXFlip) {\r
-            FXSYS_memcpy32(dest_scan, src_scan, m_Pitch);\r
-            continue;\r
-        }\r
-        if (m_bpp == 1) {\r
-            FXSYS_memset32(dest_scan, 0, m_Pitch);\r
-            for (int col = 0; col < m_Width; col ++)\r
-                if (src_scan[col / 8] & (1 << (7 - col % 8))) {\r
-                    int dest_col = m_Width - col - 1;\r
-                    dest_scan[dest_col / 8] |= (1 << (7 - dest_col % 8));\r
-                }\r
-        } else {\r
-            dest_scan += (m_Width - 1) * Bpp;\r
-            if (Bpp == 1) {\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    *dest_scan = *src_scan;\r
-                    dest_scan --;\r
-                    src_scan ++;\r
-                }\r
-            } else if (Bpp == 3) {\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    dest_scan[0] = src_scan[0];\r
-                    dest_scan[1] = src_scan[1];\r
-                    dest_scan[2] = src_scan[2];\r
-                    dest_scan -= 3;\r
-                    src_scan += 3;\r
-                }\r
-            } else {\r
-                ASSERT(Bpp == 4);\r
-                for (int col = 0; col < m_Width; col ++) {\r
-                    *(FX_DWORD*)dest_scan = *(FX_DWORD*)src_scan;\r
-                    dest_scan -= 4;\r
-                    src_scan += 4;\r
-                }\r
-            }\r
-        }\r
-    }\r
-    if (m_pAlphaMask) {\r
-        pDestBuffer = pFlipped->m_pAlphaMask->GetBuffer();\r
-        FX_DWORD dest_pitch = pFlipped->m_pAlphaMask->GetPitch();\r
-        for (int row = 0; row < m_Height; row ++) {\r
-            FX_LPCBYTE src_scan = m_pAlphaMask->GetScanline(row);\r
-            FX_LPBYTE dest_scan = pDestBuffer + dest_pitch * (bYFlip ? (m_Height - row - 1) : row);\r
-            if (!bXFlip) {\r
-                FXSYS_memcpy32(dest_scan, src_scan, dest_pitch);\r
-                continue;\r
-            }\r
-            dest_scan += (m_Width - 1);\r
-            for (int col = 0; col < m_Width; col ++) {\r
-                *dest_scan = *src_scan;\r
-                dest_scan --;\r
-                src_scan ++;\r
-            }\r
-        }\r
-    }\r
-    return pFlipped;\r
-}\r
-CFX_DIBExtractor::CFX_DIBExtractor(const CFX_DIBSource* pSrc)\r
-{\r
-    m_pBitmap = NULL;\r
-    if (pSrc->GetBuffer() == NULL) {\r
-        m_pBitmap = pSrc->Clone();\r
-    } else {\r
-        m_pBitmap = FX_NEW CFX_DIBitmap;\r
-        if (!m_pBitmap) {\r
-            return;\r
-        }\r
-        if (!m_pBitmap->Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat(), pSrc->GetBuffer())) {\r
-            delete m_pBitmap;\r
-            m_pBitmap = NULL;\r
-            return;\r
-        }\r
-        m_pBitmap->CopyPalette(pSrc->GetPalette());\r
-        m_pBitmap->CopyAlphaMask(pSrc->m_pAlphaMask);\r
-    }\r
-}\r
-CFX_DIBExtractor::~CFX_DIBExtractor()\r
-{\r
-    if (m_pBitmap) {\r
-        delete m_pBitmap;\r
-    }\r
-}\r
-CFX_FilteredDIB::CFX_FilteredDIB()\r
-{\r
-    m_pScanline = NULL;\r
-    m_pSrc = NULL;\r
-}\r
-CFX_FilteredDIB::~CFX_FilteredDIB()\r
-{\r
-    if (m_pSrc && m_bAutoDropSrc) {\r
-        delete m_pSrc;\r
-    }\r
-    if (m_pScanline) {\r
-        FX_Free(m_pScanline);\r
-    }\r
-}\r
-void CFX_FilteredDIB::LoadSrc(const CFX_DIBSource* pSrc, FX_BOOL bAutoDropSrc)\r
-{\r
-    m_pSrc = pSrc;\r
-    m_bAutoDropSrc = bAutoDropSrc;\r
-    m_Width = pSrc->GetWidth();\r
-    m_Height = pSrc->GetHeight();\r
-    FXDIB_Format format = GetDestFormat();\r
-    m_bpp = (FX_BYTE)format;\r
-    m_AlphaFlag = (FX_BYTE)(format >> 8);\r
-    m_Pitch = (m_Width * (format & 0xff) + 31) / 32 * 4;\r
-    m_pPalette = GetDestPalette();\r
-    m_pScanline = FX_Alloc(FX_BYTE, m_Pitch);\r
-}\r
-FX_LPCBYTE CFX_FilteredDIB::GetScanline(int line) const\r
-{\r
-    TranslateScanline(m_pScanline, m_pSrc->GetScanline(line));\r
-    return m_pScanline;\r
-}\r
-void CFX_FilteredDIB::DownSampleScanline(int line, FX_LPBYTE dest_scan, int dest_bpp,\r
-        int dest_width, FX_BOOL bFlipX, int clip_left, int clip_width) const\r
-{\r
-    m_pSrc->DownSampleScanline(line, dest_scan, dest_bpp, dest_width, bFlipX, clip_left, clip_width);\r
-    TranslateDownSamples(dest_scan, dest_scan, clip_width, dest_bpp);\r
-}\r
-CFX_ImageRenderer::CFX_ImageRenderer()\r
-{\r
-    m_Status = 0;\r
-    m_pTransformer = NULL;\r
-    m_bRgbByteOrder = FALSE;\r
-    m_BlendType = FXDIB_BLEND_NORMAL;\r
-}\r
-CFX_ImageRenderer::~CFX_ImageRenderer()\r
-{\r
-    if (m_pTransformer) {\r
-        delete m_pTransformer;\r
-    }\r
-}\r
-extern FX_RECT _FXDIB_SwapClipBox(FX_RECT& clip, int width, int height, FX_BOOL bFlipX, FX_BOOL bFlipY);\r
-FX_BOOL CFX_ImageRenderer::Start(CFX_DIBitmap* pDevice, const CFX_ClipRgn* pClipRgn,\r
-                                 const CFX_DIBSource* pSource, int bitmap_alpha,\r
-                                 FX_DWORD mask_color, const CFX_AffineMatrix* pMatrix,\r
-                                 FX_DWORD dib_flags, FX_BOOL bRgbByteOrder,\r
-                                 int alpha_flag, void* pIccTransform, int blend_type)\r
-{\r
-    m_Matrix = *pMatrix;\r
-    CFX_FloatRect image_rect_f = m_Matrix.GetUnitRect();\r
-    FX_RECT image_rect = image_rect_f.GetOutterRect();\r
-    m_ClipBox = pClipRgn ? pClipRgn->GetBox() : FX_RECT(0, 0, pDevice->GetWidth(), pDevice->GetHeight());\r
-    m_ClipBox.Intersect(image_rect);\r
-    if (m_ClipBox.IsEmpty()) {\r
-        return FALSE;\r
-    }\r
-    m_pDevice = pDevice;\r
-    m_pClipRgn = pClipRgn;\r
-    m_MaskColor = mask_color;\r
-    m_BitmapAlpha = bitmap_alpha;\r
-    m_Matrix = *pMatrix;\r
-    m_Flags = dib_flags;\r
-    m_AlphaFlag = alpha_flag;\r
-    m_pIccTransform = pIccTransform;\r
-    m_bRgbByteOrder = bRgbByteOrder;\r
-    m_BlendType = blend_type;\r
-    FX_BOOL ret = TRUE;\r
-    if ((FXSYS_fabs(m_Matrix.b) >= 0.5f || m_Matrix.a == 0) ||\r
-            (FXSYS_fabs(m_Matrix.c) >= 0.5f || m_Matrix.d == 0) ) {\r
-        if (FXSYS_fabs(m_Matrix.a) < FXSYS_fabs(m_Matrix.b) / 20 && FXSYS_fabs(m_Matrix.d) < FXSYS_fabs(m_Matrix.c) / 20 &&\r
-                FXSYS_fabs(m_Matrix.a) < 0.5f && FXSYS_fabs(m_Matrix.d) < 0.5f) {\r
-            int dest_width = image_rect.Width();\r
-            int dest_height = image_rect.Height();\r
-            FX_RECT bitmap_clip = m_ClipBox;\r
-            bitmap_clip.Offset(-image_rect.left, -image_rect.top);\r
-            bitmap_clip = _FXDIB_SwapClipBox(bitmap_clip, dest_width, dest_height, m_Matrix.c > 0, m_Matrix.b < 0);\r
-            m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color, m_ClipBox, TRUE,\r
-                               m_Matrix.c > 0, m_Matrix.b < 0, m_bRgbByteOrder, alpha_flag, pIccTransform, m_BlendType);\r
-            if (!m_Stretcher.Start(&m_Composer, pSource, dest_height, dest_width, bitmap_clip, dib_flags)) {\r
-                return FALSE;\r
-            }\r
-            m_Status = 1;\r
-            return TRUE;\r
-        }\r
-        m_Status = 2;\r
-        m_pTransformer = FX_NEW CFX_ImageTransformer;\r
-        if (!m_pTransformer) {\r
-            return FALSE;\r
-        }\r
-        m_pTransformer->Start(pSource, &m_Matrix, dib_flags, &m_ClipBox);\r
-        return TRUE;\r
-    }\r
-    int dest_width = image_rect.Width();\r
-    if (m_Matrix.a < 0) {\r
-        dest_width = -dest_width;\r
-    }\r
-    int dest_height = image_rect.Height();\r
-    if (m_Matrix.d > 0) {\r
-        dest_height = -dest_height;\r
-    }\r
-    if (dest_width == 0 || dest_height == 0) {\r
-        return FALSE;\r
-    }\r
-    FX_RECT bitmap_clip = m_ClipBox;\r
-    bitmap_clip.Offset(-image_rect.left, -image_rect.top);\r
-    m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color,\r
-                       m_ClipBox, FALSE, FALSE, FALSE, m_bRgbByteOrder, alpha_flag, pIccTransform, m_BlendType);\r
-    m_Status = 1;\r
-    ret = m_Stretcher.Start(&m_Composer, pSource, dest_width, dest_height, bitmap_clip, dib_flags);\r
-    return ret;\r
-}\r
-FX_BOOL CFX_ImageRenderer::Continue(IFX_Pause* pPause)\r
-{\r
-    if (m_Status == 1) {\r
-        return m_Stretcher.Continue(pPause);\r
-    } else if (m_Status == 2) {\r
-        if (m_pTransformer->Continue(pPause)) {\r
-            return TRUE;\r
-        }\r
-        CFX_DIBitmap* pBitmap = m_pTransformer->m_Storer.Detach();\r
-        if (pBitmap == NULL) {\r
-            return FALSE;\r
-        }\r
-        if (pBitmap->GetBuffer() == NULL) {\r
-            delete pBitmap;\r
-            return FALSE;\r
-        }\r
-        if (pBitmap->IsAlphaMask()) {\r
-            if (m_BitmapAlpha != 255) {\r
-                if (m_AlphaFlag >> 8) {\r
-                    m_AlphaFlag = (((FX_BYTE)((m_AlphaFlag & 0xff) * m_BitmapAlpha / 255)) | ((m_AlphaFlag >> 8) << 8));\r
-                } else {\r
-                    m_MaskColor = FXARGB_MUL_ALPHA(m_MaskColor, m_BitmapAlpha);\r
-                }\r
-            }\r
-            m_pDevice->CompositeMask(m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,\r
-                                     pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, m_MaskColor,\r
-                                     0, 0, m_BlendType, m_pClipRgn, m_bRgbByteOrder, m_AlphaFlag, m_pIccTransform);\r
-        } else {\r
-            if (m_BitmapAlpha != 255) {\r
-                pBitmap->MultiplyAlpha(m_BitmapAlpha);\r
-            }\r
-            m_pDevice->CompositeBitmap(m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,\r
-                                       pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, 0, 0, m_BlendType, m_pClipRgn, m_bRgbByteOrder, m_pIccTransform);\r
-        }\r
-        delete pBitmap;\r
-        return FALSE;\r
-    }\r
-    return FALSE;\r
-}\r
-CFX_BitmapStorer::CFX_BitmapStorer()\r
-{\r
-    m_pBitmap = NULL;\r
-}\r
-CFX_BitmapStorer::~CFX_BitmapStorer()\r
-{\r
-    if (m_pBitmap) {\r
-        delete m_pBitmap;\r
-    }\r
-}\r
-CFX_DIBitmap* CFX_BitmapStorer::Detach()\r
-{\r
-    CFX_DIBitmap* pBitmap = m_pBitmap;\r
-    m_pBitmap = NULL;\r
-    return pBitmap;\r
-}\r
-void CFX_BitmapStorer::Replace(CFX_DIBitmap* pBitmap)\r
-{\r
-    if (m_pBitmap) {\r
-        delete m_pBitmap;\r
-    }\r
-    m_pBitmap = pBitmap;\r
-}\r
-void CFX_BitmapStorer::ComposeScanline(int line, FX_LPCBYTE scanline, FX_LPCBYTE scan_extra_alpha)\r
-{\r
-    FX_LPBYTE dest_buf = (FX_LPBYTE)m_pBitmap->GetScanline(line);\r
-    FX_LPBYTE dest_alpha_buf = m_pBitmap->m_pAlphaMask ?\r
-                               (FX_LPBYTE)m_pBitmap->m_pAlphaMask->GetScanline(line) : NULL;\r
-    if (dest_buf) {\r
-        FXSYS_memcpy32(dest_buf, scanline, m_pBitmap->GetPitch());\r
-    }\r
-    if (dest_alpha_buf) {\r
-        FXSYS_memcpy32(dest_alpha_buf, scan_extra_alpha, m_pBitmap->m_pAlphaMask->GetPitch());\r
-    }\r
-}\r
-FX_BOOL CFX_BitmapStorer::SetInfo(int width, int height, FXDIB_Format src_format, FX_DWORD* pSrcPalette)\r
-{\r
-    m_pBitmap = FX_NEW CFX_DIBitmap;\r
-    if (!m_pBitmap) {\r
-        return FALSE;\r
-    }\r
-    if (!m_pBitmap->Create(width, height, src_format)) {\r
-        delete m_pBitmap;\r
-        m_pBitmap = NULL;\r
-        return FALSE;\r
-    }\r
-    if (pSrcPalette) {\r
-        m_pBitmap->CopyPalette(pSrcPalette);\r
-    }\r
-    return TRUE;\r
-}\r
+// Copyright 2014 PDFium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
+
+#include "../../../include/fxge/fx_dib.h"
+#include "../../../include/fxge/fx_ge.h"
+#include "../../../include/fxcodec/fx_codec.h"
+#include "dib_int.h"
+#include <limits.h>
+FX_BOOL ConvertBuffer(FXDIB_Format dest_format, FX_LPBYTE dest_buf, int dest_pitch, int width, int height,
+                      const CFX_DIBSource* pSrcBitmap, int src_left, int src_top, FX_DWORD*& pal, void* pIccTransform);
+void CmykDecode(FX_DWORD cmyk, int& c, int& m, int& y, int& k)
+{
+    c = FXSYS_GetCValue(cmyk);
+    m = FXSYS_GetMValue(cmyk);
+    y = FXSYS_GetYValue(cmyk);
+    k = FXSYS_GetKValue(cmyk);
+}
+void ArgbDecode(FX_DWORD argb, int& a, int& r, int& g, int& b)
+{
+    a = FXARGB_A(argb);
+    r = FXARGB_R(argb);
+    g = FXARGB_G(argb);
+    b = FXARGB_B(argb);
+}
+void ArgbDecode(FX_DWORD argb, int& a, FX_COLORREF& rgb)
+{
+    a = FXARGB_A(argb);
+    rgb = FXSYS_RGB(FXARGB_R(argb), FXARGB_G(argb), FXARGB_B(argb));
+}
+FX_DWORD ArgbEncode(int a, FX_COLORREF rgb)
+{
+    return FXARGB_MAKE(a, FXSYS_GetRValue(rgb), FXSYS_GetGValue(rgb), FXSYS_GetBValue(rgb));
+}
+CFX_DIBSource::CFX_DIBSource()
+{
+    m_bpp = 0;
+    m_AlphaFlag = 0;
+    m_Width = m_Height = 0;
+    m_Pitch = 0;
+    m_pPalette = NULL;
+    m_pAlphaMask = NULL;
+}
+CFX_DIBSource::~CFX_DIBSource()
+{
+    if (m_pPalette) {
+        FX_Free(m_pPalette);
+    }
+    if (m_pAlphaMask) {
+        delete m_pAlphaMask;
+    }
+}
+CFX_DIBitmap::CFX_DIBitmap()
+{
+    m_bExtBuf = FALSE;
+    m_pBuffer = NULL;
+    m_pPalette = NULL;
+}
+#define _MAX_OOM_LIMIT_        12000000
+FX_BOOL CFX_DIBitmap::Create(int width, int height, FXDIB_Format format, FX_LPBYTE pBuffer, int pitch)
+{
+    m_pBuffer = NULL;
+    m_bpp = (uint8_t)format;
+    m_AlphaFlag = (uint8_t)(format >> 8);
+    m_Width = m_Height = m_Pitch = 0;
+    if (width <= 0 || height <= 0 || pitch < 0) {
+        return FALSE;
+    }
+    if ((INT_MAX - 31) / width < (format & 0xff)) {
+        return FALSE;
+    }
+    if (!pitch) {
+        pitch = (width * (format & 0xff) + 31) / 32 * 4;
+    }
+    if ((1 << 30) / pitch < height) {
+        return FALSE;
+    }
+    if (pBuffer) {
+        m_pBuffer = pBuffer;
+        m_bExtBuf = TRUE;
+    } else {
+        int size = pitch * height + 4;
+        int oomlimit = _MAX_OOM_LIMIT_;
+        if (oomlimit >= 0 && size >= oomlimit) {
+            m_pBuffer = FX_TryAlloc(uint8_t, size);
+            if (m_pBuffer == NULL) {
+                return FALSE;
+            }
+        } else {
+            m_pBuffer = FX_Alloc(uint8_t, size);
+        }
+    }
+    m_Width = width;
+    m_Height = height;
+    m_Pitch = pitch;
+    if (HasAlpha() && format != FXDIB_Argb) {
+        FX_BOOL ret = TRUE;
+        ret = BuildAlphaMask();
+        if (!ret) {
+            if (!m_bExtBuf && m_pBuffer) {
+                FX_Free(m_pBuffer);
+                m_pBuffer = NULL;
+                m_Width = m_Height = m_Pitch = 0;
+                return FALSE;
+            }
+        }
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::Copy(const CFX_DIBSource* pSrc)
+{
+    if (m_pBuffer) {
+        return FALSE;
+    }
+    if (!Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat())) {
+        return FALSE;
+    }
+    CopyPalette(pSrc->GetPalette());
+    CopyAlphaMask(pSrc->m_pAlphaMask);
+    for (int row = 0; row < pSrc->GetHeight(); row ++) {
+        FXSYS_memcpy32(m_pBuffer + row * m_Pitch, pSrc->GetScanline(row), m_Pitch);
+    }
+    return TRUE;
+}
+CFX_DIBitmap::~CFX_DIBitmap()
+{
+    if (m_pBuffer && !m_bExtBuf) {
+        FX_Free(m_pBuffer);
+    }
+    m_pBuffer = NULL;
+}
+void CFX_DIBitmap::TakeOver(CFX_DIBitmap* pSrcBitmap)
+{
+    if (m_pBuffer && !m_bExtBuf) {
+        FX_Free(m_pBuffer);
+    }
+    if (m_pPalette) {
+        FX_Free(m_pPalette);
+    }
+    if (m_pAlphaMask) {
+        delete m_pAlphaMask;
+    }
+    m_pBuffer = pSrcBitmap->m_pBuffer;
+    m_pPalette = pSrcBitmap->m_pPalette;
+    m_pAlphaMask = pSrcBitmap->m_pAlphaMask;
+    pSrcBitmap->m_pBuffer = NULL;
+    pSrcBitmap->m_pPalette = NULL;
+    pSrcBitmap->m_pAlphaMask = NULL;
+    m_bpp = pSrcBitmap->m_bpp;
+    m_bExtBuf = pSrcBitmap->m_bExtBuf;
+    m_AlphaFlag = pSrcBitmap->m_AlphaFlag;
+    m_Width = pSrcBitmap->m_Width;
+    m_Height = pSrcBitmap->m_Height;
+    m_Pitch = pSrcBitmap->m_Pitch;
+}
+CFX_DIBitmap* CFX_DIBSource::Clone(const FX_RECT* pClip) const
+{
+    FX_RECT rect(0, 0, m_Width, m_Height);
+    if (pClip) {
+        rect.Intersect(*pClip);
+        if (rect.IsEmpty()) {
+            return NULL;
+        }
+    }
+    CFX_DIBitmap* pNewBitmap = FX_NEW CFX_DIBitmap;
+    if (!pNewBitmap) {
+        return NULL;
+    }
+    if (!pNewBitmap->Create(rect.Width(), rect.Height(), GetFormat())) {
+        delete pNewBitmap;
+        return NULL;
+    }
+    pNewBitmap->CopyPalette(m_pPalette);
+    pNewBitmap->CopyAlphaMask(m_pAlphaMask, pClip);
+    if (GetBPP() == 1 && rect.left % 8 != 0) {
+        int left_shift = rect.left % 32;
+        int right_shift = 32 - left_shift;
+        int dword_count = pNewBitmap->m_Pitch / 4;
+        for (int row = rect.top; row < rect.bottom; row ++) {
+            FX_DWORD* src_scan = (FX_DWORD*)GetScanline(row) + rect.left / 32;
+            FX_DWORD* dest_scan = (FX_DWORD*)pNewBitmap->GetScanline(row - rect.top);
+            for (int i = 0; i < dword_count; i ++) {
+                dest_scan[i] = (src_scan[i] << left_shift) | (src_scan[i + 1] >> right_shift);
+            }
+        }
+    } else {
+        int copy_len = (pNewBitmap->GetWidth() * pNewBitmap->GetBPP() + 7) / 8;
+        if (m_Pitch < (FX_DWORD)copy_len) {
+            copy_len = m_Pitch;
+        }
+        for (int row = rect.top; row < rect.bottom; row ++) {
+            FX_LPCBYTE src_scan = GetScanline(row) + rect.left * m_bpp / 8;
+            FX_LPBYTE dest_scan = (FX_LPBYTE)pNewBitmap->GetScanline(row - rect.top);
+            FXSYS_memcpy32(dest_scan, src_scan, copy_len);
+        }
+    }
+    return pNewBitmap;
+}
+void CFX_DIBSource::BuildPalette()
+{
+    if (m_pPalette) {
+        return;
+    }
+    if (GetBPP() == 1) {
+        m_pPalette = FX_Alloc(FX_DWORD, 2);
+        if(IsCmykImage()) {
+            m_pPalette[0] = 0xff;
+            m_pPalette[1] = 0;
+        } else {
+            m_pPalette[0] = 0xff000000;
+            m_pPalette[1] = 0xffffffff;
+        }
+    } else if (GetBPP() == 8) {
+        m_pPalette = FX_Alloc(FX_DWORD, 256);
+        if(IsCmykImage()) {
+            for (int i = 0; i < 256; i ++) {
+                m_pPalette[i] = 0xff - i;
+            }
+        } else {
+            for (int i = 0; i < 256; i ++) {
+                m_pPalette[i] = 0xff000000 | (i * 0x10101);
+            }
+        }
+    }
+}
+FX_BOOL CFX_DIBSource::BuildAlphaMask()
+{
+    if (m_pAlphaMask) {
+        return TRUE;
+    }
+    m_pAlphaMask = FX_NEW CFX_DIBitmap;
+    if (!m_pAlphaMask) {
+        return FALSE;
+    }
+    if (!m_pAlphaMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
+        delete m_pAlphaMask;
+        m_pAlphaMask = NULL;
+        return FALSE;
+    }
+    FXSYS_memset8(m_pAlphaMask->GetBuffer(), 0xff, m_pAlphaMask->GetHeight()*m_pAlphaMask->GetPitch());
+    return TRUE;
+}
+FX_DWORD CFX_DIBSource::GetPaletteEntry(int index) const
+{
+    ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
+    if (m_pPalette) {
+        return m_pPalette[index];
+    }
+    if (IsCmykImage()) {
+        if (GetBPP() == 1) {
+            return index ? 0 : 0xff;
+        }
+        return 0xff - index;
+    }
+    if (GetBPP() == 1) {
+        return index ? 0xffffffff : 0xff000000;
+    }
+    return index * 0x10101 | 0xff000000;
+}
+void CFX_DIBSource::SetPaletteEntry(int index, FX_DWORD color)
+{
+    ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
+    if (m_pPalette == NULL) {
+        BuildPalette();
+    }
+    m_pPalette[index] = color;
+}
+int CFX_DIBSource::FindPalette(FX_DWORD color) const
+{
+    ASSERT((GetBPP() == 1 || GetBPP() == 8) && !IsAlphaMask());
+    if (m_pPalette == NULL) {
+        if (IsCmykImage()) {
+            if (GetBPP() == 1) {
+                return ((uint8_t)color == 0xff) ? 0 : 1;
+            }
+            return 0xff - (uint8_t)color;
+        }
+        if (GetBPP() == 1) {
+            return ((uint8_t)color == 0xff) ? 1 : 0;
+        }
+        return (uint8_t)color;
+    }
+    int palsize = (1 << GetBPP());
+    for (int i = 0; i < palsize; i ++)
+        if (m_pPalette[i] == color) {
+            return i;
+        }
+    return -1;
+}
+void CFX_DIBitmap::Clear(FX_DWORD color)
+{
+    if (m_pBuffer == NULL) {
+        return;
+    }
+    switch (GetFormat()) {
+        case FXDIB_1bppMask:
+            FXSYS_memset8(m_pBuffer, (color & 0xff000000) ? 0xff : 0, m_Pitch * m_Height);
+            break;
+        case FXDIB_1bppRgb: {
+                int index = FindPalette(color);
+                FXSYS_memset8(m_pBuffer, index ? 0xff : 0, m_Pitch * m_Height);
+                break;
+            }
+        case FXDIB_8bppMask:
+            FXSYS_memset8(m_pBuffer, color >> 24, m_Pitch * m_Height);
+            break;
+        case FXDIB_8bppRgb: {
+                int index = FindPalette(color);
+                FXSYS_memset8(m_pBuffer, index, m_Pitch * m_Height);
+                break;
+            }
+        case FXDIB_Rgb:
+        case FXDIB_Rgba: {
+                int a, r, g, b;
+                ArgbDecode(color, a, r, g, b);
+                if (r == g && g == b) {
+                    FXSYS_memset8(m_pBuffer, r, m_Pitch * m_Height);
+                } else {
+                    int byte_pos = 0;
+                    for (int col = 0; col < m_Width; col ++) {
+                        m_pBuffer[byte_pos++] = b;
+                        m_pBuffer[byte_pos++] = g;
+                        m_pBuffer[byte_pos++] = r;
+                    }
+                    for (int row = 1; row < m_Height; row ++) {
+                        FXSYS_memcpy32(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);
+                    }
+                }
+                break;
+            }
+        case FXDIB_Rgb32:
+        case FXDIB_Argb: {
+                color = IsCmykImage() ? FXCMYK_TODIB(color) : FXARGB_TODIB(color);
+                for (int i = 0; i < m_Width; i ++) {
+                    ((FX_DWORD*)m_pBuffer)[i] = color;
+                }
+                for (int row = 1; row < m_Height; row ++) {
+                    FXSYS_memcpy32(m_pBuffer + row * m_Pitch, m_pBuffer, m_Pitch);
+                }
+                break;
+            }
+        default:
+            break;
+    }
+}
+void CFX_DIBSource::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)
+{
+    if (width == 0 || height == 0) {
+        return;
+    }
+    ASSERT(width > 0 && height > 0);
+    if (dest_left > m_Width || dest_top > m_Height) {
+        width = 0;
+        height = 0;
+        return;
+    }
+    int x_offset = dest_left - src_left;
+    int y_offset = dest_top - src_top;
+    FX_RECT src_rect(src_left, src_top, src_left + width, src_top + height);
+    FX_RECT src_bound(0, 0, src_width, src_height);
+    src_rect.Intersect(src_bound);
+    FX_RECT dest_rect(src_rect.left + x_offset, src_rect.top + y_offset,
+                      src_rect.right + x_offset, src_rect.bottom + y_offset);
+    FX_RECT dest_bound(0, 0, m_Width, m_Height);
+    dest_rect.Intersect(dest_bound);
+    if (pClipRgn) {
+        dest_rect.Intersect(pClipRgn->GetBox());
+    }
+    dest_left = dest_rect.left;
+    dest_top = dest_rect.top;
+    src_left = dest_left - x_offset;
+    src_top = dest_top - y_offset;
+    width = dest_rect.right - dest_rect.left;
+    height = dest_rect.bottom - dest_rect.top;
+}
+FX_BOOL CFX_DIBitmap::TransferBitmap(int dest_left, int dest_top, int width, int height,
+                                     const CFX_DIBSource* pSrcBitmap, int src_left, int src_top, void* pIccTransform)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    GetOverlapRect(dest_left, dest_top, width, height, pSrcBitmap->GetWidth(), pSrcBitmap->GetHeight(), src_left, src_top, NULL);
+    if (width == 0 || height == 0) {
+        return TRUE;
+    }
+    FXDIB_Format dest_format = GetFormat();
+    FXDIB_Format src_format = pSrcBitmap->GetFormat();
+    if (dest_format == src_format && pIccTransform == NULL) {
+        if (GetBPP() == 1) {
+            for (int row = 0; row < height; row ++) {
+                FX_LPBYTE dest_scan = m_pBuffer + (dest_top + row) * m_Pitch;
+                FX_LPCBYTE src_scan = pSrcBitmap->GetScanline(src_top + row);
+                for (int col = 0; col < width; col ++) {
+                    if (src_scan[(src_left + col) / 8] & (1 << (7 - (src_left + col) % 8))) {
+                        dest_scan[(dest_left + col) / 8] |= 1 << (7 - (dest_left + col) % 8);
+                    } else {
+                        dest_scan[(dest_left + col) / 8] &= ~(1 << (7 - (dest_left + col) % 8));
+                    }
+                }
+            }
+        } else {
+            int Bpp = GetBPP() / 8;
+            for (int row = 0; row < height; row ++) {
+                FX_LPBYTE dest_scan = m_pBuffer + (dest_top + row) * m_Pitch + dest_left * Bpp;
+                FX_LPCBYTE src_scan = pSrcBitmap->GetScanline(src_top + row) + src_left * Bpp;
+                FXSYS_memcpy32(dest_scan, src_scan, width * Bpp);
+            }
+        }
+    } else {
+        if (m_pPalette) {
+            return FALSE;
+        }
+        if (m_bpp == 8) {
+            dest_format = FXDIB_8bppMask;
+        }
+        FX_LPBYTE dest_buf = m_pBuffer + dest_top * m_Pitch + dest_left * GetBPP() / 8;
+        FX_DWORD* d_plt = NULL;
+        if(!ConvertBuffer(dest_format, dest_buf, m_Pitch, width, height, pSrcBitmap, src_left, src_top, d_plt, pIccTransform)) {
+            return FALSE;
+        }
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::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)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    ASSERT(HasAlpha() && (m_bpp >= 24));
+    ASSERT(pMask->IsAlphaMask());
+    if (!HasAlpha() || !pMask->IsAlphaMask() || m_bpp < 24) {
+        return FALSE;
+    }
+    GetOverlapRect(dest_left, dest_top, width, height, pMask->GetWidth(), pMask->GetHeight(), src_left, src_top, NULL);
+    if (width == 0 || height == 0) {
+        return TRUE;
+    }
+    int src_bpp = pMask->GetBPP();
+    int alpha;
+    FX_DWORD dst_color;
+    if (alpha_flag >> 8) {
+        alpha = alpha_flag & 0xff;
+        dst_color = FXCMYK_TODIB(color);
+    } else {
+        alpha = FXARGB_A(color);
+        dst_color = FXARGB_TODIB(color);
+    }
+    FX_LPBYTE color_p = (FX_LPBYTE)&dst_color;
+    if (pIccTransform && CFX_GEModule::Get()->GetCodecModule() && CFX_GEModule::Get()->GetCodecModule()->GetIccModule()) {
+        ICodec_IccModule* pIccModule = CFX_GEModule::Get()->GetCodecModule()->GetIccModule();
+        pIccModule->TranslateScanline(pIccTransform, color_p, color_p, 1);
+    } else {
+        if (alpha_flag >> 8 && !IsCmykImage())
+            AdobeCMYK_to_sRGB1(FXSYS_GetCValue(color), FXSYS_GetMValue(color), FXSYS_GetYValue(color), FXSYS_GetKValue(color),
+                               color_p[2], color_p[1], color_p[0]);
+        else if (!(alpha_flag >> 8) && IsCmykImage()) {
+            return FALSE;
+        }
+    }
+    if(!IsCmykImage()) {
+        color_p[3] = (uint8_t)alpha;
+    }
+    if (GetFormat() == FXDIB_Argb) {
+        for (int row = 0; row < height; row ++) {
+            FX_DWORD* dest_pos = (FX_DWORD*)(m_pBuffer + (dest_top + row) * m_Pitch + dest_left * 4);
+            FX_LPCBYTE src_scan = pMask->GetScanline(src_top + row);
+            if (src_bpp == 1) {
+                for (int col = 0; col < width; col ++) {
+                    int src_bitpos = src_left + col;
+                    if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {
+                        *dest_pos = dst_color;
+                    } else {
+                        *dest_pos = 0;
+                    }
+                    dest_pos ++;
+                }
+            } else {
+                src_scan += src_left;
+                dst_color = FXARGB_TODIB(dst_color);
+                dst_color &= 0xffffff;
+                for (int col = 0; col < width; col ++) {
+                    FXARGB_SETDIB(dest_pos++, dst_color | ((alpha * (*src_scan++) / 255) << 24));
+                }
+            }
+        }
+    } else {
+        int comps = m_bpp / 8;
+        for (int row = 0; row < height; row ++) {
+            FX_LPBYTE dest_color_pos = m_pBuffer + (dest_top + row) * m_Pitch + dest_left * comps;
+            FX_LPBYTE dest_alpha_pos = (FX_LPBYTE)m_pAlphaMask->GetScanline(dest_top + row) + dest_left;
+            FX_LPCBYTE src_scan = pMask->GetScanline(src_top + row);
+            if (src_bpp == 1) {
+                for (int col = 0; col < width; col ++) {
+                    int src_bitpos = src_left + col;
+                    if (src_scan[src_bitpos / 8] & (1 << (7 - src_bitpos % 8))) {
+                        FXSYS_memcpy32(dest_color_pos, color_p, comps);
+                        *dest_alpha_pos = 0xff;
+                    } else {
+                        FXSYS_memset32(dest_color_pos, 0, comps);
+                        *dest_alpha_pos = 0;
+                    }
+                    dest_color_pos += comps;
+                    dest_alpha_pos ++;
+                }
+            } else {
+                src_scan += src_left;
+                for (int col = 0; col < width; col ++) {
+                    FXSYS_memcpy32(dest_color_pos, color_p, comps);
+                    dest_color_pos += comps;
+                    *dest_alpha_pos++ = (alpha * (*src_scan++) / 255);
+                }
+            }
+        }
+    }
+    return TRUE;
+}
+void CFX_DIBSource::CopyPalette(const FX_DWORD* pSrc, FX_DWORD size)
+{
+    if (pSrc == NULL || GetBPP() > 8) {
+        if (m_pPalette) {
+            FX_Free(m_pPalette);
+        }
+        m_pPalette = NULL;
+    } else {
+        FX_DWORD pal_size = 1 << GetBPP();
+        if (m_pPalette == NULL) {
+            m_pPalette = FX_Alloc(FX_DWORD, pal_size);
+        }
+        if (pal_size > size) {
+            pal_size = size;
+        }
+        FXSYS_memcpy32(m_pPalette, pSrc, pal_size * sizeof(FX_DWORD));
+    }
+}
+void CFX_DIBSource::GetPalette(FX_DWORD* pal, int alpha) const
+{
+    ASSERT(GetBPP() <= 8 && !IsCmykImage());
+    if (GetBPP() == 1) {
+        pal[0] = ((m_pPalette ? m_pPalette[0] : 0xff000000) & 0xffffff) | (alpha << 24);
+        pal[1] = ((m_pPalette ? m_pPalette[1] : 0xffffffff) & 0xffffff) | (alpha << 24);
+        return;
+    }
+    if (m_pPalette) {
+        for (int i = 0; i < 256; i ++) {
+            pal[i] = (m_pPalette[i] & 0x00ffffff) | (alpha << 24);
+        }
+    } else {
+        for (int i = 0; i < 256; i ++) {
+            pal[i] = (i * 0x10101) | (alpha << 24);
+        }
+    }
+}
+CFX_DIBitmap* CFX_DIBSource::GetAlphaMask(const FX_RECT* pClip) const
+{
+    ASSERT(GetFormat() == FXDIB_Argb);
+    FX_RECT rect(0, 0, m_Width, m_Height);
+    if (pClip) {
+        rect.Intersect(*pClip);
+        if (rect.IsEmpty()) {
+            return NULL;
+        }
+    }
+    CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
+    if (!pMask) {
+        return NULL;
+    }
+    if (!pMask->Create(rect.Width(), rect.Height(), FXDIB_8bppMask)) {
+        delete pMask;
+        return NULL;
+    }
+    for (int row = rect.top; row < rect.bottom; row ++) {
+        FX_LPCBYTE src_scan = GetScanline(row) + rect.left * 4 + 3;
+        FX_LPBYTE dest_scan = (FX_LPBYTE)pMask->GetScanline(row - rect.top);
+        for (int col = rect.left; col < rect.right; col ++) {
+            *dest_scan ++ = *src_scan;
+            src_scan += 4;
+        }
+    }
+    return pMask;
+}
+FX_BOOL CFX_DIBSource::CopyAlphaMask(const CFX_DIBSource* pAlphaMask, const FX_RECT* pClip)
+{
+    if (!HasAlpha() || GetFormat() == FXDIB_Argb) {
+        return FALSE;
+    }
+    if (pAlphaMask) {
+        FX_RECT rect(0, 0, pAlphaMask->m_Width, pAlphaMask->m_Height);
+        if (pClip) {
+            rect.Intersect(*pClip);
+            if (rect.IsEmpty() || rect.Width() != m_Width || rect.Height() != m_Height) {
+                return FALSE;
+            }
+        } else {
+            if (pAlphaMask->m_Width != m_Width || pAlphaMask->m_Height != m_Height) {
+                return FALSE;
+            }
+        }
+        for (int row = 0; row < m_Height; row ++)
+            FXSYS_memcpy32((void*)m_pAlphaMask->GetScanline(row),
+                           pAlphaMask->GetScanline(row + rect.top) + rect.left, m_pAlphaMask->m_Pitch);
+    } else {
+        m_pAlphaMask->Clear(0xff000000);
+    }
+    return TRUE;
+}
+const int g_ChannelOffset[] = {0, 2, 1, 0, 0, 1, 2, 3, 3};
+FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, const CFX_DIBSource* pSrcBitmap, FXDIB_Channel srcChannel)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    CFX_DIBSource* pSrcClone = (CFX_DIBSource*)pSrcBitmap;
+    CFX_DIBitmap* pDst = this;
+    int destOffset, srcOffset;
+    if (srcChannel == FXDIB_Alpha) {
+        if (!pSrcBitmap->HasAlpha() && !pSrcBitmap->IsAlphaMask()) {
+            return FALSE;
+        }
+        if (pSrcBitmap->GetBPP() == 1) {
+            pSrcClone = pSrcBitmap->CloneConvert(FXDIB_8bppMask);
+            if (pSrcClone == NULL) {
+                return FALSE;
+            }
+        }
+        if(pSrcBitmap->GetFormat() == FXDIB_Argb) {
+            srcOffset = 3;
+        } else {
+            srcOffset = 0;
+        }
+    } else {
+        if (pSrcBitmap->IsAlphaMask()) {
+            return FALSE;
+        }
+        if (pSrcBitmap->GetBPP() < 24) {
+            if (pSrcBitmap->IsCmykImage()) {
+                pSrcClone = pSrcBitmap->CloneConvert((FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x20));
+            } else {
+                pSrcClone = pSrcBitmap->CloneConvert((FXDIB_Format)((pSrcBitmap->GetFormat() & 0xff00) | 0x18));
+            }
+            if (pSrcClone == NULL) {
+                return FALSE;
+            }
+        }
+        srcOffset = g_ChannelOffset[srcChannel];
+    }
+    if (destChannel == FXDIB_Alpha) {
+        if (IsAlphaMask()) {
+            if(!ConvertFormat(FXDIB_8bppMask)) {
+                if (pSrcClone != pSrcBitmap) {
+                    delete pSrcClone;
+                }
+                return FALSE;
+            }
+            destOffset = 0;
+        } else {
+            destOffset = 0;
+            if(!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
+                if (pSrcClone != pSrcBitmap) {
+                    delete pSrcClone;
+                }
+                return FALSE;
+            }
+            if (GetFormat() == FXDIB_Argb) {
+                destOffset = 3;
+            }
+        }
+    } else {
+        if (IsAlphaMask()) {
+            if (pSrcClone != pSrcBitmap) {
+                delete pSrcClone;
+            }
+            return FALSE;
+        }
+        if (GetBPP() < 24) {
+            if (HasAlpha()) {
+                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
+                    if (pSrcClone != pSrcBitmap) {
+                        delete pSrcClone;
+                    }
+                    return FALSE;
+                }
+            } else
+#if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_
+                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {
+#else
+                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {
+#endif
+                    if (pSrcClone != pSrcBitmap) {
+                        delete pSrcClone;
+                    }
+                    return FALSE;
+                }
+        }
+        destOffset = g_ChannelOffset[destChannel];
+    }
+    if (srcChannel == FXDIB_Alpha && pSrcClone->m_pAlphaMask) {
+        CFX_DIBitmap* pAlphaMask = pSrcClone->m_pAlphaMask;
+        if (pSrcClone->GetWidth() != m_Width || pSrcClone->GetHeight() != m_Height) {
+            if (pAlphaMask) {
+                pAlphaMask = pAlphaMask->StretchTo(m_Width, m_Height);
+                if (pAlphaMask == NULL) {
+                    if (pSrcClone != pSrcBitmap) {
+                        delete pSrcClone;
+                    }
+                    return FALSE;
+                }
+            }
+        }
+        if (pSrcClone != pSrcBitmap) {
+            pSrcClone->m_pAlphaMask = NULL;
+            delete pSrcClone;
+        }
+        pSrcClone = pAlphaMask;
+        srcOffset = 0;
+    } else if (pSrcClone->GetWidth() != m_Width || pSrcClone->GetHeight() != m_Height) {
+        CFX_DIBitmap* pSrcMatched = pSrcClone->StretchTo(m_Width, m_Height);
+        if (pSrcClone != pSrcBitmap) {
+            delete pSrcClone;
+        }
+        if (pSrcMatched == NULL) {
+            return FALSE;
+        }
+        pSrcClone = pSrcMatched;
+    }
+    if (destChannel == FXDIB_Alpha && m_pAlphaMask) {
+        pDst = m_pAlphaMask;
+        destOffset = 0;
+    }
+    int srcBytes = pSrcClone->GetBPP() / 8;
+    int destBytes = pDst->GetBPP() / 8;
+    for (int row = 0; row < m_Height; row ++) {
+        FX_LPBYTE dest_pos = (FX_LPBYTE)pDst->GetScanline(row) + destOffset;
+        FX_LPCBYTE src_pos = pSrcClone->GetScanline(row) + srcOffset;
+        for (int col = 0; col < m_Width; col ++) {
+            *dest_pos = *src_pos;
+            dest_pos += destBytes;
+            src_pos += srcBytes;
+        }
+    }
+    if (pSrcClone != pSrcBitmap && pSrcClone != pSrcBitmap->m_pAlphaMask) {
+        delete pSrcClone;
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::LoadChannel(FXDIB_Channel destChannel, int value)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    int destOffset;
+    if (destChannel == FXDIB_Alpha) {
+        if (IsAlphaMask()) {
+            if(!ConvertFormat(FXDIB_8bppMask)) {
+                return FALSE;
+            }
+            destOffset = 0;
+        } else {
+            destOffset = 0;
+            if(!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
+                return FALSE;
+            }
+            if (GetFormat() == FXDIB_Argb) {
+                destOffset = 3;
+            }
+        }
+    } else {
+        if (IsAlphaMask()) {
+            return FALSE;
+        }
+        if (GetBPP() < 24) {
+            if (HasAlpha()) {
+                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyka : FXDIB_Argb)) {
+                    return FALSE;
+                }
+            } else
+#if _FXM_PLATFORM_  == _FXM_PLATFORM_APPLE_
+                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb)) {
+                    return FALSE;
+                }
+#else
+                if (!ConvertFormat(IsCmykImage() ? FXDIB_Cmyk : FXDIB_Rgb32)) {
+                    return FALSE;
+                }
+#endif
+        }
+        destOffset = g_ChannelOffset[destChannel];
+    }
+    int Bpp = GetBPP() / 8;
+    if (Bpp == 1) {
+        FXSYS_memset8(m_pBuffer, value, m_Height * m_Pitch);
+        return TRUE;
+    }
+    if (destChannel == FXDIB_Alpha && m_pAlphaMask) {
+        FXSYS_memset8(m_pAlphaMask->GetBuffer(), value, m_pAlphaMask->GetHeight()*m_pAlphaMask->GetPitch());
+        return TRUE;
+    }
+    for (int row = 0; row < m_Height; row ++) {
+        FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch + destOffset;
+        for (int col = 0; col < m_Width; col ++) {
+            *scan_line = value;
+            scan_line += Bpp;
+        }
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::MultiplyAlpha(const CFX_DIBSource* pSrcBitmap)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    ASSERT(pSrcBitmap->IsAlphaMask());
+    if (!pSrcBitmap->IsAlphaMask()) {
+        return FALSE;
+    }
+    if (!IsAlphaMask() && !HasAlpha()) {
+        return LoadChannel(FXDIB_Alpha, pSrcBitmap, FXDIB_Alpha);
+    }
+    CFX_DIBitmap* pSrcClone = (CFX_DIBitmap*)pSrcBitmap;
+    if (pSrcBitmap->GetWidth() != m_Width || pSrcBitmap->GetHeight() != m_Height) {
+        pSrcClone = pSrcBitmap->StretchTo(m_Width, m_Height);
+        ASSERT(pSrcClone != NULL);
+        if (pSrcClone == NULL) {
+            return FALSE;
+        }
+    }
+    if (IsAlphaMask()) {
+        if(!ConvertFormat(FXDIB_8bppMask)) {
+            if (pSrcClone != pSrcBitmap) {
+                delete pSrcClone;
+            }
+            return FALSE;
+        }
+        for (int row = 0; row < m_Height; row ++) {
+            FX_LPBYTE dest_scan = m_pBuffer + m_Pitch * row;
+            FX_LPBYTE src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;
+            if (pSrcClone->GetBPP() == 1) {
+                for (int col = 0; col < m_Width; col ++) {
+                    if (!((1 << (7 - col % 8)) & src_scan[col / 8])) {
+                        dest_scan[col] = 0;
+                    }
+                }
+            } else {
+                for (int col = 0; col < m_Width; col ++) {
+                    *dest_scan = (*dest_scan) * src_scan[col] / 255;
+                    dest_scan ++;
+                }
+            }
+        }
+    } else {
+        if(GetFormat() == FXDIB_Argb) {
+            if (pSrcClone->GetBPP() == 1) {
+                if (pSrcClone != pSrcBitmap) {
+                    delete pSrcClone;
+                }
+                return FALSE;
+            }
+            for (int row = 0; row < m_Height; row ++) {
+                FX_LPBYTE dest_scan = m_pBuffer + m_Pitch * row + 3;
+                FX_LPBYTE src_scan = pSrcClone->m_pBuffer + pSrcClone->m_Pitch * row;
+                for (int col = 0; col < m_Width; col ++) {
+                    *dest_scan = (*dest_scan) * src_scan[col] / 255;
+                    dest_scan += 4;
+                }
+            }
+        } else {
+            m_pAlphaMask->MultiplyAlpha(pSrcClone);
+        }
+    }
+    if (pSrcClone != pSrcBitmap) {
+        delete pSrcClone;
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::GetGrayData(void* pIccTransform)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    switch (GetFormat()) {
+        case FXDIB_1bppRgb: {
+                if (m_pPalette == NULL) {
+                    return FALSE;
+                }
+                uint8_t gray[2];
+                for (int i = 0; i < 2; i ++) {
+                    int r = (uint8_t)(m_pPalette[i] >> 16);
+                    int g = (uint8_t)(m_pPalette[i] >> 8);
+                    int b = (uint8_t)m_pPalette[i];
+                    gray[i] = (uint8_t)FXRGB2GRAY(r, g, b);
+                }
+                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
+                if (!pMask) {
+                    return FALSE;
+                }
+                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
+                    delete pMask;
+                    return FALSE;
+                }
+                FXSYS_memset8(pMask->GetBuffer(), gray[0], pMask->GetPitch() * m_Height);
+                for (int row = 0; row < m_Height; row ++) {
+                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
+                    FX_LPBYTE dest_pos = (FX_LPBYTE)pMask->GetScanline(row);
+                    for (int col = 0; col < m_Width; col ++) {
+                        if (src_pos[col / 8] & (1 << (7 - col % 8))) {
+                            *dest_pos = gray[1];
+                        }
+                        dest_pos ++;
+                    }
+                }
+                TakeOver(pMask);
+                delete pMask;
+                break;
+            }
+        case FXDIB_8bppRgb: {
+                if (m_pPalette == NULL) {
+                    return FALSE;
+                }
+                uint8_t gray[256];
+                for (int i = 0; i < 256; i ++) {
+                    int r = (uint8_t)(m_pPalette[i] >> 16);
+                    int g = (uint8_t)(m_pPalette[i] >> 8);
+                    int b = (uint8_t)m_pPalette[i];
+                    gray[i] = (uint8_t)FXRGB2GRAY(r, g, b);
+                }
+                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
+                if (!pMask) {
+                    return FALSE;
+                }
+                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
+                    delete pMask;
+                    return FALSE;
+                }
+                for (int row = 0; row < m_Height; row ++) {
+                    FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
+                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
+                    for (int col = 0; col < m_Width; col ++) {
+                        *dest_pos ++ = gray[*src_pos ++];
+                    }
+                }
+                TakeOver(pMask);
+                delete pMask;
+                break;
+            }
+        case FXDIB_Rgb: {
+                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
+                if (!pMask) {
+                    return FALSE;
+                }
+                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
+                    delete pMask;
+                    return FALSE;
+                }
+                for (int row = 0; row < m_Height; row ++) {
+                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
+                    FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
+                    for (int col = 0; col < m_Width; col ++) {
+                        *dest_pos ++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);
+                        src_pos += 3;
+                    }
+                }
+                TakeOver(pMask);
+                delete pMask;
+                break;
+            }
+        case FXDIB_Rgb32: {
+                CFX_DIBitmap* pMask = FX_NEW CFX_DIBitmap;
+                if (!pMask) {
+                    return FALSE;
+                }
+                if (!pMask->Create(m_Width, m_Height, FXDIB_8bppMask)) {
+                    delete pMask;
+                    return FALSE;
+                }
+                for (int row = 0; row < m_Height; row ++) {
+                    FX_LPBYTE src_pos = m_pBuffer + row * m_Pitch;
+                    FX_LPBYTE dest_pos = pMask->GetBuffer() + row * pMask->GetPitch();
+                    for (int col = 0; col < m_Width; col ++) {
+                        *dest_pos ++ = FXRGB2GRAY(src_pos[2], src_pos[1], *src_pos);
+                        src_pos += 4;
+                    }
+                }
+                TakeOver(pMask);
+                delete pMask;
+                break;
+            }
+        default:
+            return FALSE;
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::MultiplyAlpha(int alpha)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    switch (GetFormat()) {
+        case FXDIB_1bppMask:
+            if (!ConvertFormat(FXDIB_8bppMask)) {
+                return FALSE;
+            }
+            MultiplyAlpha(alpha);
+            break;
+        case FXDIB_8bppMask: {
+                for (int row = 0; row < m_Height; row ++) {
+                    FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch;
+                    for (int col = 0; col < m_Width; col ++) {
+                        scan_line[col] = scan_line[col] * alpha / 255;
+                    }
+                }
+                break;
+            }
+        case FXDIB_Argb: {
+                for (int row = 0; row < m_Height; row ++) {
+                    FX_LPBYTE scan_line = m_pBuffer + row * m_Pitch + 3;
+                    for (int col = 0; col < m_Width; col ++) {
+                        *scan_line = (*scan_line) * alpha / 255;
+                        scan_line += 4;
+                    }
+                }
+                break;
+            }
+        default:
+            if (HasAlpha()) {
+                m_pAlphaMask->MultiplyAlpha(alpha);
+            } else if (IsCmykImage()) {
+                if (!ConvertFormat((FXDIB_Format)(GetFormat() | 0x0200))) {
+                    return FALSE;
+                }
+                m_pAlphaMask->MultiplyAlpha(alpha);
+            } else {
+                if (!ConvertFormat(FXDIB_Argb)) {
+                    return FALSE;
+                }
+                MultiplyAlpha(alpha);
+            }
+            break;
+    }
+    return TRUE;
+}
+FX_DWORD CFX_DIBitmap::GetPixel(int x, int y) const
+{
+    if (m_pBuffer == NULL) {
+        return 0;
+    }
+    FX_LPBYTE pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;
+    switch (GetFormat()) {
+        case FXDIB_1bppMask: {
+                if ((*pos) & (1 << (7 - x % 8))) {
+                    return 0xff000000;
+                }
+                return 0;
+            }
+        case FXDIB_1bppRgb: {
+                if ((*pos) & (1 << (7 - x % 8))) {
+                    return m_pPalette ? m_pPalette[1] : 0xffffffff;
+                } else {
+                    return m_pPalette ? m_pPalette[0] : 0xff000000;
+                }
+                break;
+            }
+        case FXDIB_8bppMask:
+            return (*pos) << 24;
+        case FXDIB_8bppRgb:
+            return m_pPalette ? m_pPalette[*pos] : (0xff000000 | ((*pos) * 0x10101));
+        case FXDIB_Rgb:
+        case FXDIB_Rgba:
+        case FXDIB_Rgb32:
+            return FXARGB_GETDIB(pos) | 0xff000000;
+        case FXDIB_Argb:
+            return FXARGB_GETDIB(pos);
+        default:
+            break;
+    }
+    return 0;
+}
+void CFX_DIBitmap::SetPixel(int x, int y, FX_DWORD color)
+{
+    if (m_pBuffer == NULL) {
+        return;
+    }
+    if (x < 0 || x >= m_Width || y < 0 || y >= m_Height) {
+        return;
+    }
+    FX_LPBYTE pos = m_pBuffer + y * m_Pitch + x * GetBPP() / 8;
+    switch (GetFormat()) {
+        case FXDIB_1bppMask:
+            if (color >> 24) {
+                *pos |= 1 << (7 - x % 8);
+            } else {
+                *pos &= ~(1 << (7 - x % 8));
+            }
+            break;
+        case FXDIB_1bppRgb:
+            if (m_pPalette) {
+                if (color == m_pPalette[1]) {
+                    *pos |= 1 << (7 - x % 8);
+                } else {
+                    *pos &= ~(1 << (7 - x % 8));
+                }
+            } else {
+                if (color == 0xffffffff) {
+                    *pos |= 1 << (7 - x % 8);
+                } else {
+                    *pos &= ~(1 << (7 - x % 8));
+                }
+            }
+            break;
+        case FXDIB_8bppMask:
+            *pos = (uint8_t)(color >> 24);
+            break;
+        case FXDIB_8bppRgb: {
+                if (m_pPalette) {
+                    for (int i = 0; i < 256; i ++) {
+                        if (m_pPalette[i] == color) {
+                            *pos = (uint8_t)i;
+                            return;
+                        }
+                    }
+                    *pos = 0;
+                } else {
+                    *pos = FXRGB2GRAY(FXARGB_R(color), FXARGB_G(color), FXARGB_B(color));
+                }
+                break;
+            }
+        case FXDIB_Rgb:
+        case FXDIB_Rgb32: {
+                int alpha = FXARGB_A(color);
+                pos[0] = (FXARGB_B(color) * alpha + pos[0] * (255 - alpha)) / 255;
+                pos[1] = (FXARGB_G(color) * alpha + pos[1] * (255 - alpha)) / 255;
+                pos[2] = (FXARGB_R(color) * alpha + pos[2] * (255 - alpha)) / 255;
+                break;
+            }
+        case FXDIB_Rgba: {
+                pos[0] = FXARGB_B(color);
+                pos[1] = FXARGB_G(color);
+                pos[2] = FXARGB_R(color);
+                break;
+            }
+        case FXDIB_Argb:
+            FXARGB_SETDIB(pos, color);
+            break;
+        default:
+            break;
+    }
+}
+void CFX_DIBitmap::DownSampleScanline(int line, FX_LPBYTE dest_scan, int dest_bpp,
+                                      int dest_width, FX_BOOL bFlipX, int clip_left, int clip_width) const
+{
+    if (m_pBuffer == NULL) {
+        return;
+    }
+    int src_Bpp = m_bpp / 8;
+    FX_LPBYTE scanline = m_pBuffer + line * m_Pitch;
+    if (src_Bpp == 0) {
+        for (int i = 0; i < clip_width; i ++) {
+            FX_DWORD dest_x = clip_left + i;
+            FX_DWORD src_x = dest_x * m_Width / dest_width;
+            if (bFlipX) {
+                src_x = m_Width - src_x - 1;
+            }
+            src_x %= m_Width;
+            dest_scan[i] = (scanline[src_x / 8] & (1 << (7 - src_x % 8))) ? 255 : 0;
+        }
+    } else if (src_Bpp == 1) {
+        for (int i = 0; i < clip_width; i ++) {
+            FX_DWORD dest_x = clip_left + i;
+            FX_DWORD src_x = dest_x * m_Width / dest_width;
+            if (bFlipX) {
+                src_x = m_Width - src_x - 1;
+            }
+            src_x %= m_Width;
+            int dest_pos = i;
+            if (m_pPalette) {
+                if (!IsCmykImage()) {
+                    dest_pos *= 3;
+                    FX_ARGB argb = m_pPalette[scanline[src_x]];
+                    dest_scan[dest_pos] = FXARGB_B(argb);
+                    dest_scan[dest_pos + 1] = FXARGB_G(argb);
+                    dest_scan[dest_pos + 2] = FXARGB_R(argb);
+                } else {
+                    dest_pos *= 4;
+                    FX_CMYK cmyk = m_pPalette[scanline[src_x]];
+                    dest_scan[dest_pos] = FXSYS_GetCValue(cmyk);
+                    dest_scan[dest_pos + 1] = FXSYS_GetMValue(cmyk);
+                    dest_scan[dest_pos + 2] = FXSYS_GetYValue(cmyk);
+                    dest_scan[dest_pos + 3] = FXSYS_GetKValue(cmyk);
+                }
+            } else {
+                dest_scan[dest_pos] = scanline[src_x];
+            }
+        }
+    } else {
+        for (int i = 0; i < clip_width; i ++) {
+            FX_DWORD dest_x = clip_left + i;
+            FX_DWORD src_x = bFlipX ? (m_Width - dest_x * m_Width / dest_width - 1) * src_Bpp : (dest_x * m_Width / dest_width) * src_Bpp;
+            src_x %= m_Width * src_Bpp;
+            int dest_pos = i * src_Bpp;
+            for (int b = 0; b < src_Bpp; b ++) {
+                dest_scan[dest_pos + b] = scanline[src_x + b];
+            }
+        }
+    }
+}
+FX_BOOL CFX_DIBitmap::ConvertColorScale(FX_DWORD forecolor, FX_DWORD backcolor)
+{
+    ASSERT(!IsAlphaMask());
+    if (m_pBuffer == NULL || IsAlphaMask()) {
+        return FALSE;
+    }
+    int fc, fm, fy, fk, bc, bm, by, bk;
+    int fr, fg, fb, br, bg, bb;
+    FX_BOOL isCmykImage = IsCmykImage();
+    if (isCmykImage) {
+        fc = FXSYS_GetCValue(forecolor);
+        fm = FXSYS_GetMValue(forecolor);
+        fy = FXSYS_GetYValue(forecolor);
+        fk = FXSYS_GetKValue(forecolor);
+        bc = FXSYS_GetCValue(backcolor);
+        bm = FXSYS_GetMValue(backcolor);
+        by = FXSYS_GetYValue(backcolor);
+        bk = FXSYS_GetKValue(backcolor);
+    } else {
+        fr = FXSYS_GetRValue(forecolor);
+        fg = FXSYS_GetGValue(forecolor);
+        fb = FXSYS_GetBValue(forecolor);
+        br = FXSYS_GetRValue(backcolor);
+        bg = FXSYS_GetGValue(backcolor);
+        bb = FXSYS_GetBValue(backcolor);
+    }
+    if (m_bpp <= 8) {
+        if (isCmykImage) {
+            if (forecolor == 0xff && backcolor == 0 && m_pPalette == NULL) {
+                return TRUE;
+            }
+        } else if (forecolor == 0 && backcolor == 0xffffff && m_pPalette == NULL) {
+            return TRUE;
+        }
+        if (m_pPalette == NULL) {
+            BuildPalette();
+        }
+        int size = 1 << m_bpp;
+        if (isCmykImage) {
+            for (int i = 0; i < size; i ++) {
+                uint8_t b, g, r;
+                AdobeCMYK_to_sRGB1(FXSYS_GetCValue(m_pPalette[i]), FXSYS_GetMValue(m_pPalette[i]), FXSYS_GetYValue(m_pPalette[i]), FXSYS_GetKValue(m_pPalette[i]),
+                                   r, g, b);
+                int gray = 255 - FXRGB2GRAY(r, g, b);
+                m_pPalette[i] = CmykEncode(bc + (fc - bc) * gray / 255, bm + (fm - bm) * gray / 255,
+                                           by + (fy - by) * gray / 255, bk + (fk - bk) * gray / 255);
+            }
+        } else
+            for (int i = 0; i < size; i ++) {
+                int gray = FXRGB2GRAY(FXARGB_R(m_pPalette[i]), FXARGB_G(m_pPalette[i]), FXARGB_B(m_pPalette[i]));
+                m_pPalette[i] = FXARGB_MAKE(0xff, br + (fr - br) * gray / 255, bg + (fg - bg) * gray / 255,
+                                            bb + (fb - bb) * gray / 255);
+            }
+        return TRUE;
+    }
+    if (isCmykImage) {
+        if (forecolor == 0xff && backcolor == 0x00) {
+            for (int row = 0; row < m_Height; row ++) {
+                FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
+                for (int col = 0; col < m_Width; col ++) {
+                    uint8_t b, g, r;
+                    AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],
+                                       r, g, b);
+                    *scanline ++ = 0;
+                    *scanline ++ = 0;
+                    *scanline ++ = 0;
+                    *scanline ++ = 255 - FXRGB2GRAY(r, g, b);
+                }
+            }
+            return TRUE;
+        }
+    } else if (forecolor == 0 && backcolor == 0xffffff) {
+        for (int row = 0; row < m_Height; row ++) {
+            FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
+            int gap = m_bpp / 8 - 2;
+            for (int col = 0; col < m_Width; col ++) {
+                int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);
+                *scanline ++ = gray;
+                *scanline ++ = gray;
+                *scanline    = gray;
+                scanline += gap;
+            }
+        }
+        return TRUE;
+    }
+    if (isCmykImage) {
+        for (int row = 0; row < m_Height; row ++) {
+            FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
+            for (int col = 0; col < m_Width; col ++) {
+                uint8_t b, g, r;
+                AdobeCMYK_to_sRGB1(scanline[0], scanline[1], scanline[2], scanline[3],
+                                   r, g, b);
+                int gray = 255 - FXRGB2GRAY(r, g, b);
+                *scanline ++ = bc + (fc - bc) * gray / 255;
+                *scanline ++ = bm + (fm - bm) * gray / 255;
+                *scanline ++ = by + (fy - by) * gray / 255;
+                *scanline ++ = bk + (fk - bk) * gray / 255;
+            }
+        }
+    } else {
+        for (int row = 0; row < m_Height; row ++) {
+            FX_LPBYTE scanline = m_pBuffer + row * m_Pitch;
+            int gap = m_bpp / 8 - 2;
+            for (int col = 0; col < m_Width; col ++) {
+                int gray = FXRGB2GRAY(scanline[2], scanline[1], scanline[0]);
+                *scanline ++ = bb + (fb - bb) * gray / 255;
+                *scanline ++ = bg + (fg - bg) * gray / 255;
+                *scanline    = br + (fr - br) * gray / 255;
+                scanline += gap;
+            }
+        }
+    }
+    return TRUE;
+}
+FX_BOOL CFX_DIBitmap::DitherFS(const FX_DWORD* pPalette, int pal_size, const FX_RECT* pRect)
+{
+    if (m_pBuffer == NULL) {
+        return FALSE;
+    }
+    if (m_bpp != 8 && m_pPalette != NULL && m_AlphaFlag != 0) {
+        return FALSE;
+    }
+    if (m_Width < 4 && m_Height < 4) {
+        return FALSE;
+    }
+    FX_RECT rect(0, 0, m_Width, m_Height);
+    if (pRect) {
+        rect.Intersect(*pRect);
+    }
+    uint8_t translate[256];
+    for (int i = 0; i < 256; i ++) {
+        int err2 = 65536;
+        for (int j = 0; j < pal_size; j ++) {
+            uint8_t entry = (uint8_t)pPalette[j];
+            int err = (int)entry - i;
+            if (err * err < err2) {
+                err2 = err * err;
+                translate[i] = entry;
+            }
+        }
+    }
+    for (int row = rect.top; row < rect.bottom; row ++) {
+        FX_LPBYTE scan = m_pBuffer + row * m_Pitch;
+        FX_LPBYTE next_scan = m_pBuffer + (row + 1) * m_Pitch;
+        for (int col = rect.left; col < rect.right; col ++) {
+            int src_pixel = scan[col];
+            int dest_pixel = translate[src_pixel];
+            scan[col] = (uint8_t)dest_pixel;
+            int error = -dest_pixel + src_pixel;
+            if (col < rect.right - 1) {
+                int src = scan[col + 1];
+                src += error * 7 / 16;
+                if (src > 255) {
+                    scan[col + 1] = 255;
+                } else if (src < 0) {
+                    scan[col + 1] = 0;
+                } else {
+                    scan[col + 1] = src;
+                }
+            }
+            if (col < rect.right - 1 && row < rect.bottom - 1) {
+                int src = next_scan[col + 1];
+                src += error * 1 / 16;
+                if (src > 255) {
+                    next_scan[col + 1] = 255;
+                } else if (src < 0) {
+                    next_scan[col + 1] = 0;
+                } else {
+                    next_scan[col + 1] = src;
+                }
+            }
+            if (row < rect.bottom - 1) {
+                int src = next_scan[col];
+                src += error * 5 / 16;
+                if (src > 255) {
+                    next_scan[col] = 255;
+                } else if (src < 0) {
+                    next_scan[col] = 0;
+                } else {
+                    next_scan[col] = src;
+                }
+            }
+            if (col > rect.left && row < rect.bottom - 1) {
+                int src = next_scan[col - 1];
+                src += error * 3 / 16;
+                if (src > 255) {
+                    next_scan[col - 1] = 255;
+                } else if (src < 0) {
+                    next_scan[col - 1] = 0;
+                } else {
+                    next_scan[col - 1] = src;
+                }
+            }
+        }
+    }
+    return TRUE;
+}
+CFX_DIBitmap* CFX_DIBSource::FlipImage(FX_BOOL bXFlip, FX_BOOL bYFlip) const
+{
+    CFX_DIBitmap* pFlipped = FX_NEW CFX_DIBitmap;
+    if (!pFlipped) {
+        return NULL;
+    }
+    if (!pFlipped->Create(m_Width, m_Height, GetFormat())) {
+        delete pFlipped;
+        return NULL;
+    }
+    pFlipped->CopyPalette(m_pPalette);
+    FX_LPBYTE pDestBuffer = pFlipped->GetBuffer();
+    int Bpp = m_bpp / 8;
+    for (int row = 0; row < m_Height; row ++) {
+        FX_LPCBYTE src_scan = GetScanline(row);
+        FX_LPBYTE dest_scan = pDestBuffer + m_Pitch * (bYFlip ? (m_Height - row - 1) : row);
+        if (!bXFlip) {
+            FXSYS_memcpy32(dest_scan, src_scan, m_Pitch);
+            continue;
+        }
+        if (m_bpp == 1) {
+            FXSYS_memset32(dest_scan, 0, m_Pitch);
+            for (int col = 0; col < m_Width; col ++)
+                if (src_scan[col / 8] & (1 << (7 - col % 8))) {
+                    int dest_col = m_Width - col - 1;
+                    dest_scan[dest_col / 8] |= (1 << (7 - dest_col % 8));
+                }
+        } else {
+            dest_scan += (m_Width - 1) * Bpp;
+            if (Bpp == 1) {
+                for (int col = 0; col < m_Width; col ++) {
+                    *dest_scan = *src_scan;
+                    dest_scan --;
+                    src_scan ++;
+                }
+            } else if (Bpp == 3) {
+                for (int col = 0; col < m_Width; col ++) {
+                    dest_scan[0] = src_scan[0];
+                    dest_scan[1] = src_scan[1];
+                    dest_scan[2] = src_scan[2];
+                    dest_scan -= 3;
+                    src_scan += 3;
+                }
+            } else {
+                ASSERT(Bpp == 4);
+                for (int col = 0; col < m_Width; col ++) {
+                    *(FX_DWORD*)dest_scan = *(FX_DWORD*)src_scan;
+                    dest_scan -= 4;
+                    src_scan += 4;
+                }
+            }
+        }
+    }
+    if (m_pAlphaMask) {
+        pDestBuffer = pFlipped->m_pAlphaMask->GetBuffer();
+        FX_DWORD dest_pitch = pFlipped->m_pAlphaMask->GetPitch();
+        for (int row = 0; row < m_Height; row ++) {
+            FX_LPCBYTE src_scan = m_pAlphaMask->GetScanline(row);
+            FX_LPBYTE dest_scan = pDestBuffer + dest_pitch * (bYFlip ? (m_Height - row - 1) : row);
+            if (!bXFlip) {
+                FXSYS_memcpy32(dest_scan, src_scan, dest_pitch);
+                continue;
+            }
+            dest_scan += (m_Width - 1);
+            for (int col = 0; col < m_Width; col ++) {
+                *dest_scan = *src_scan;
+                dest_scan --;
+                src_scan ++;
+            }
+        }
+    }
+    return pFlipped;
+}
+CFX_DIBExtractor::CFX_DIBExtractor(const CFX_DIBSource* pSrc)
+{
+    m_pBitmap = NULL;
+    if (pSrc->GetBuffer() == NULL) {
+        m_pBitmap = pSrc->Clone();
+    } else {
+        m_pBitmap = FX_NEW CFX_DIBitmap;
+        if (!m_pBitmap) {
+            return;
+        }
+        if (!m_pBitmap->Create(pSrc->GetWidth(), pSrc->GetHeight(), pSrc->GetFormat(), pSrc->GetBuffer())) {
+            delete m_pBitmap;
+            m_pBitmap = NULL;
+            return;
+        }
+        m_pBitmap->CopyPalette(pSrc->GetPalette());
+        m_pBitmap->CopyAlphaMask(pSrc->m_pAlphaMask);
+    }
+}
+CFX_DIBExtractor::~CFX_DIBExtractor()
+{
+    if (m_pBitmap) {
+        delete m_pBitmap;
+    }
+}
+CFX_FilteredDIB::CFX_FilteredDIB()
+{
+    m_pScanline = NULL;
+    m_pSrc = NULL;
+}
+CFX_FilteredDIB::~CFX_FilteredDIB()
+{
+    if (m_pSrc && m_bAutoDropSrc) {
+        delete m_pSrc;
+    }
+    if (m_pScanline) {
+        FX_Free(m_pScanline);
+    }
+}
+void CFX_FilteredDIB::LoadSrc(const CFX_DIBSource* pSrc, FX_BOOL bAutoDropSrc)
+{
+    m_pSrc = pSrc;
+    m_bAutoDropSrc = bAutoDropSrc;
+    m_Width = pSrc->GetWidth();
+    m_Height = pSrc->GetHeight();
+    FXDIB_Format format = GetDestFormat();
+    m_bpp = (uint8_t)format;
+    m_AlphaFlag = (uint8_t)(format >> 8);
+    m_Pitch = (m_Width * (format & 0xff) + 31) / 32 * 4;
+    m_pPalette = GetDestPalette();
+    m_pScanline = FX_Alloc(uint8_t, m_Pitch);
+}
+FX_LPCBYTE CFX_FilteredDIB::GetScanline(int line) const
+{
+    TranslateScanline(m_pScanline, m_pSrc->GetScanline(line));
+    return m_pScanline;
+}
+void CFX_FilteredDIB::DownSampleScanline(int line, FX_LPBYTE dest_scan, int dest_bpp,
+        int dest_width, FX_BOOL bFlipX, int clip_left, int clip_width) const
+{
+    m_pSrc->DownSampleScanline(line, dest_scan, dest_bpp, dest_width, bFlipX, clip_left, clip_width);
+    TranslateDownSamples(dest_scan, dest_scan, clip_width, dest_bpp);
+}
+CFX_ImageRenderer::CFX_ImageRenderer()
+{
+    m_Status = 0;
+    m_pTransformer = NULL;
+    m_bRgbByteOrder = FALSE;
+    m_BlendType = FXDIB_BLEND_NORMAL;
+}
+CFX_ImageRenderer::~CFX_ImageRenderer()
+{
+    if (m_pTransformer) {
+        delete m_pTransformer;
+    }
+}
+extern FX_RECT _FXDIB_SwapClipBox(FX_RECT& clip, int width, int height, FX_BOOL bFlipX, FX_BOOL bFlipY);
+FX_BOOL CFX_ImageRenderer::Start(CFX_DIBitmap* pDevice, const CFX_ClipRgn* pClipRgn,
+                                 const CFX_DIBSource* pSource, int bitmap_alpha,
+                                 FX_DWORD mask_color, const CFX_AffineMatrix* pMatrix,
+                                 FX_DWORD dib_flags, FX_BOOL bRgbByteOrder,
+                                 int alpha_flag, void* pIccTransform, int blend_type)
+{
+    m_Matrix = *pMatrix;
+    CFX_FloatRect image_rect_f = m_Matrix.GetUnitRect();
+    FX_RECT image_rect = image_rect_f.GetOutterRect();
+    m_ClipBox = pClipRgn ? pClipRgn->GetBox() : FX_RECT(0, 0, pDevice->GetWidth(), pDevice->GetHeight());
+    m_ClipBox.Intersect(image_rect);
+    if (m_ClipBox.IsEmpty()) {
+        return FALSE;
+    }
+    m_pDevice = pDevice;
+    m_pClipRgn = pClipRgn;
+    m_MaskColor = mask_color;
+    m_BitmapAlpha = bitmap_alpha;
+    m_Matrix = *pMatrix;
+    m_Flags = dib_flags;
+    m_AlphaFlag = alpha_flag;
+    m_pIccTransform = pIccTransform;
+    m_bRgbByteOrder = bRgbByteOrder;
+    m_BlendType = blend_type;
+    FX_BOOL ret = TRUE;
+    if ((FXSYS_fabs(m_Matrix.b) >= 0.5f || m_Matrix.a == 0) ||
+            (FXSYS_fabs(m_Matrix.c) >= 0.5f || m_Matrix.d == 0) ) {
+        if (FXSYS_fabs(m_Matrix.a) < FXSYS_fabs(m_Matrix.b) / 20 && FXSYS_fabs(m_Matrix.d) < FXSYS_fabs(m_Matrix.c) / 20 &&
+                FXSYS_fabs(m_Matrix.a) < 0.5f && FXSYS_fabs(m_Matrix.d) < 0.5f) {
+            int dest_width = image_rect.Width();
+            int dest_height = image_rect.Height();
+            FX_RECT bitmap_clip = m_ClipBox;
+            bitmap_clip.Offset(-image_rect.left, -image_rect.top);
+            bitmap_clip = _FXDIB_SwapClipBox(bitmap_clip, dest_width, dest_height, m_Matrix.c > 0, m_Matrix.b < 0);
+            m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color, m_ClipBox, TRUE,
+                               m_Matrix.c > 0, m_Matrix.b < 0, m_bRgbByteOrder, alpha_flag, pIccTransform, m_BlendType);
+            if (!m_Stretcher.Start(&m_Composer, pSource, dest_height, dest_width, bitmap_clip, dib_flags)) {
+                return FALSE;
+            }
+            m_Status = 1;
+            return TRUE;
+        }
+        m_Status = 2;
+        m_pTransformer = FX_NEW CFX_ImageTransformer;
+        if (!m_pTransformer) {
+            return FALSE;
+        }
+        m_pTransformer->Start(pSource, &m_Matrix, dib_flags, &m_ClipBox);
+        return TRUE;
+    }
+    int dest_width = image_rect.Width();
+    if (m_Matrix.a < 0) {
+        dest_width = -dest_width;
+    }
+    int dest_height = image_rect.Height();
+    if (m_Matrix.d > 0) {
+        dest_height = -dest_height;
+    }
+    if (dest_width == 0 || dest_height == 0) {
+        return FALSE;
+    }
+    FX_RECT bitmap_clip = m_ClipBox;
+    bitmap_clip.Offset(-image_rect.left, -image_rect.top);
+    m_Composer.Compose(pDevice, pClipRgn, bitmap_alpha, mask_color,
+                       m_ClipBox, FALSE, FALSE, FALSE, m_bRgbByteOrder, alpha_flag, pIccTransform, m_BlendType);
+    m_Status = 1;
+    ret = m_Stretcher.Start(&m_Composer, pSource, dest_width, dest_height, bitmap_clip, dib_flags);
+    return ret;
+}
+FX_BOOL CFX_ImageRenderer::Continue(IFX_Pause* pPause)
+{
+    if (m_Status == 1) {
+        return m_Stretcher.Continue(pPause);
+    } else if (m_Status == 2) {
+        if (m_pTransformer->Continue(pPause)) {
+            return TRUE;
+        }
+        CFX_DIBitmap* pBitmap = m_pTransformer->m_Storer.Detach();
+        if (pBitmap == NULL) {
+            return FALSE;
+        }
+        if (pBitmap->GetBuffer() == NULL) {
+            delete pBitmap;
+            return FALSE;
+        }
+        if (pBitmap->IsAlphaMask()) {
+            if (m_BitmapAlpha != 255) {
+                if (m_AlphaFlag >> 8) {
+                    m_AlphaFlag = (((uint8_t)((m_AlphaFlag & 0xff) * m_BitmapAlpha / 255)) | ((m_AlphaFlag >> 8) << 8));
+                } else {
+                    m_MaskColor = FXARGB_MUL_ALPHA(m_MaskColor, m_BitmapAlpha);
+                }
+            }
+            m_pDevice->CompositeMask(m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,
+                                     pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, m_MaskColor,
+                                     0, 0, m_BlendType, m_pClipRgn, m_bRgbByteOrder, m_AlphaFlag, m_pIccTransform);
+        } else {
+            if (m_BitmapAlpha != 255) {
+                pBitmap->MultiplyAlpha(m_BitmapAlpha);
+            }
+            m_pDevice->CompositeBitmap(m_pTransformer->m_ResultLeft, m_pTransformer->m_ResultTop,
+                                       pBitmap->GetWidth(), pBitmap->GetHeight(), pBitmap, 0, 0, m_BlendType, m_pClipRgn, m_bRgbByteOrder, m_pIccTransform);
+        }
+        delete pBitmap;
+        return FALSE;
+    }
+    return FALSE;
+}
+CFX_BitmapStorer::CFX_BitmapStorer()
+{
+    m_pBitmap = NULL;
+}
+CFX_BitmapStorer::~CFX_BitmapStorer()
+{
+    if (m_pBitmap) {
+        delete m_pBitmap;
+    }
+}
+CFX_DIBitmap* CFX_BitmapStorer::Detach()
+{
+    CFX_DIBitmap* pBitmap = m_pBitmap;
+    m_pBitmap = NULL;
+    return pBitmap;
+}
+void CFX_BitmapStorer::Replace(CFX_DIBitmap* pBitmap)
+{
+    if (m_pBitmap) {
+        delete m_pBitmap;
+    }
+    m_pBitmap = pBitmap;
+}
+void CFX_BitmapStorer::ComposeScanline(int line, FX_LPCBYTE scanline, FX_LPCBYTE scan_extra_alpha)
+{
+    FX_LPBYTE dest_buf = (FX_LPBYTE)m_pBitmap->GetScanline(line);
+    FX_LPBYTE dest_alpha_buf = m_pBitmap->m_pAlphaMask ?
+                               (FX_LPBYTE)m_pBitmap->m_pAlphaMask->GetScanline(line) : NULL;
+    if (dest_buf) {
+        FXSYS_memcpy32(dest_buf, scanline, m_pBitmap->GetPitch());
+    }
+    if (dest_alpha_buf) {
+        FXSYS_memcpy32(dest_alpha_buf, scan_extra_alpha, m_pBitmap->m_pAlphaMask->GetPitch());
+    }
+}
+FX_BOOL CFX_BitmapStorer::SetInfo(int width, int height, FXDIB_Format src_format, FX_DWORD* pSrcPalette)
+{
+    m_pBitmap = FX_NEW CFX_DIBitmap;
+    if (!m_pBitmap) {
+        return FALSE;
+    }
+    if (!m_pBitmap->Create(width, height, src_format)) {
+        delete m_pBitmap;
+        m_pBitmap = NULL;
+        return FALSE;
+    }
+    if (pSrcPalette) {
+        m_pBitmap->CopyPalette(pSrcPalette);
+    }
+    return TRUE;
+}