/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: bmpfast.cxx,v $ * * $Revision: 1.7 $ * * last change: $Author: obo $ $Date: 2006-09-17 11:59:00 $ * * The Contents of this file are made available subject to * the terms of GNU Lesser General Public License Version 2.1. * * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2005 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_vcl.hxx" #include #ifndef NO_OPTIMIZED_BITMAP_ACCESS #include #define _SOLAR__PRIVATE 1 #include #define FAST_ARGB_BGRA #include static bool bDisableFastBitops = (getenv( "SAL_DISABLE_BITMAPS_OPTS" ) != NULL); typedef unsigned char PIXBYTE; class BasePixelPtr { public: BasePixelPtr( PIXBYTE* p = NULL ) : mpPixel( p ) {} void SetRawPtr( PIXBYTE* pRawPtr ) { mpPixel = pRawPtr; } PIXBYTE* GetRawPtr( void ) const { return mpPixel; } void AddByteOffset( int nByteOffset ) { mpPixel += nByteOffset; } bool operator<( const BasePixelPtr& rCmp ) const { return (mpPixel < rCmp.mpPixel); } protected: PIXBYTE* mpPixel; }; template class TrueColorPixelPtr : public BasePixelPtr { public: PIXBYTE GetRed() const; PIXBYTE GetGreen() const; PIXBYTE GetBlue() const; PIXBYTE GetAlpha() const; void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const; void SetAlpha( PIXBYTE a ) const; void operator++(int); }; // ======================================================================= // template specializations for truecolor pixel formats template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 3; } PIXBYTE GetRed() const { return mpPixel[0]; } PIXBYTE GetGreen() const { return mpPixel[1]; } PIXBYTE GetBlue() const { return mpPixel[2]; } PIXBYTE GetAlpha() const { return 0; } void SetAlpha( PIXBYTE ) const {} void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[0] = r; mpPixel[1] = g; mpPixel[2] = b; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 3; } PIXBYTE GetRed() const { return mpPixel[2]; } PIXBYTE GetGreen() const { return mpPixel[1]; } PIXBYTE GetBlue() const { return mpPixel[0]; } PIXBYTE GetAlpha() const { return 0; } void SetAlpha( PIXBYTE ) const {} void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[0] = b; mpPixel[1] = g; mpPixel[2] = r; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 4; } PIXBYTE GetRed() const { return mpPixel[1]; } PIXBYTE GetGreen() const { return mpPixel[2]; } PIXBYTE GetBlue() const { return mpPixel[3]; } PIXBYTE GetAlpha() const { return mpPixel[0]; } void SetAlpha( PIXBYTE a ) const { mpPixel[0] = a; } void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[1] = r; mpPixel[2] = g; mpPixel[3] = b; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 4; } PIXBYTE GetRed() const { return mpPixel[3]; } PIXBYTE GetGreen() const { return mpPixel[2]; } PIXBYTE GetBlue() const { return mpPixel[1]; } PIXBYTE GetAlpha() const { return mpPixel[0]; } void SetAlpha( PIXBYTE a ) const { mpPixel[0] = a; } void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[1] = b; mpPixel[2] = g; mpPixel[3] = r; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 4; } PIXBYTE GetRed() const { return mpPixel[0]; } PIXBYTE GetGreen() const { return mpPixel[1]; } PIXBYTE GetBlue() const { return mpPixel[2]; } PIXBYTE GetAlpha() const { return mpPixel[3]; } void SetAlpha( PIXBYTE a ) const{ mpPixel[3] = a; } void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[0] = r; mpPixel[1] = g; mpPixel[2] = b; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 4; } PIXBYTE GetRed() const { return mpPixel[2]; } PIXBYTE GetGreen() const { return mpPixel[1]; } PIXBYTE GetBlue() const { return mpPixel[0]; } PIXBYTE GetAlpha() const { return mpPixel[3]; } void SetAlpha( PIXBYTE a ) const{ mpPixel[3] = a; } void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[0] = b; mpPixel[1] = g; mpPixel[2] = r; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 2; } // TODO: non565-RGB PIXBYTE GetRed() const { return (mpPixel[0] & 0xF8U); } PIXBYTE GetGreen() const { return (mpPixel[0]<<5U) | ((mpPixel[1]>>3U)&28U); } PIXBYTE GetBlue() const { return (mpPixel[1]<<3U); } PIXBYTE GetAlpha() const { return 0; } void SetAlpha( PIXBYTE ) const {} void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[0] = ((g >> 5U) & 7U) | (r & 0xF8U); mpPixel[1] = ((g & 28U)<< 3U) | (b >> 3U); } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 2; } // TODO: non565-RGB PIXBYTE GetRed() const { return (mpPixel[1] & 0xF8U); } PIXBYTE GetGreen() const { return (mpPixel[1]<<5U) | ((mpPixel[0]>>3U)&28U); } PIXBYTE GetBlue() const { return (mpPixel[0]<<3U); } PIXBYTE GetAlpha() const { return 0; } void SetAlpha( PIXBYTE ) const {} void SetColor( PIXBYTE r, PIXBYTE g, PIXBYTE b ) const { mpPixel[0] = ((g & 28U)<< 3U) | (b >> 3U); mpPixel[1] = ((g >> 5U) & 7U) | (r & 0xF8U); } }; // ----------------------------------------------------------------------- template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 1; } PIXBYTE GetAlpha() const { return mpPixel[0]; } void SetAlpha( PIXBYTE a ) const { mpPixel[0] = a; } void SetColor( PIXBYTE, PIXBYTE, PIXBYTE ) const {} }; // TODO: for some reason many Alpha maps are BMP_FORMAT_8BIT_PAL // they should be BMP_FORMAT_8BIT_TC_MASK template <> class TrueColorPixelPtr : public TrueColorPixelPtr {}; #if 0 template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 3; } unsigned GetAlpha() const { unsigned nAlpha = mpPixel[0]; nAlpha |= mpPixel[1] << 8U; nAlpha |= mpPixel[2] << 16U; return nAlpha; } void SetAlpha( unsigned nAlpha ) const { mpPixel[0] = nAlpha; mpPixel[1] = nAlpha >> 8U; mpPixel[2] = nAlpha >> 16U; } }; template <> class TrueColorPixelPtr : public BasePixelPtr { public: void operator++() { mpPixel += 4; } unsigned GetAlpha() const { #ifdef OSL_BIGENDIAN unsigned nAlpha = *reinterpret_cast( mpPixel ); #else unsigned nAlpha = mpPixel[0]; nAlpha |= mpPixel[1] << 8U; nAlpha |= mpPixel[2] << 16U; nAlpha |= mpPixel[3] << 24U; #endif return nAlpha; } void SetAlpha( unsigned nAlpha ) const { #ifdef OSL_BIGENDIAN *reinterpret_cast( mpPixel ) = nAlpha; #else mpPixel[0] = nAlpha; mpPixel[1] = nAlpha >> 8U; mpPixel[2] = nAlpha >> 16U; mpPixel[3] = nAlpha >> 24U; #endif } }; #endif // ======================================================================= // converting truecolor formats template inline void ImplConvertPixel( const TrueColorPixelPtr& rDst, const TrueColorPixelPtr& rSrc ) { rDst.SetColor( rSrc.GetRed(), rSrc.GetGreen(), rSrc.GetBlue() ); rDst.SetAlpha( rSrc.GetAlpha() ); } // ----------------------------------------------------------------------- template <> inline void ImplConvertPixel ( const TrueColorPixelPtr& rDst, const TrueColorPixelPtr& rSrc ) { // byte swapping const PIXBYTE* pSrc = rSrc.GetRawPtr(); PIXBYTE* pDst = rDst.GetRawPtr(); pDst[1] = pSrc[0]; pDst[0] = pSrc[1]; } // ----------------------------------------------------------------------- template inline void ImplConvertLine( const TrueColorPixelPtr& rDst, const TrueColorPixelPtr& rSrc, int nPixelCount ) { TrueColorPixelPtr aDst( rDst ); TrueColorPixelPtr aSrc( rSrc ); while( --nPixelCount >= 0 ) { ImplConvertPixel( aDst, aSrc ); ++aSrc; ++aDst; } } // ======================================================================= // alpha blending truecolor pixels template inline void ImplBlendPixels( const TrueColorPixelPtr& rDst, const TrueColorPixelPtr& rSrc, unsigned nAlphaVal ) { if( !nAlphaVal ) ImplConvertPixel( rDst, rSrc ); else if( nAlphaVal != ~(~0 << ALPHABITS) ) { static const unsigned nAlphaShift = (ALPHABITS > 8) ? 8 : ALPHABITS; if( ALPHABITS > nAlphaShift ) nAlphaVal >>= ALPHABITS - nAlphaShift; int nR = rDst.GetRed(); int nS = rSrc.GetRed(); nR = nS + (((nR - nS) * nAlphaVal) >> nAlphaShift); int nG = rDst.GetGreen(); nS = rSrc.GetGreen(); nG = nS + (((nG - nS) * nAlphaVal) >> nAlphaShift); int nB = rDst.GetBlue(); nS = rSrc.GetBlue(); nB = nS + (((nB - nS) * nAlphaVal) >> nAlphaShift); rDst.SetColor( sal::static_int_cast(nR), sal::static_int_cast(nG), sal::static_int_cast(nB) ); } } // ----------------------------------------------------------------------- template inline void ImplBlendLines( const TrueColorPixelPtr& rDst, const TrueColorPixelPtr& rSrc, const TrueColorPixelPtr& rMsk, int nPixelCount ) { TrueColorPixelPtr aMsk( rMsk ); TrueColorPixelPtr aDst( rDst ); TrueColorPixelPtr aSrc( rSrc ); while( --nPixelCount >= 0 ) { ImplBlendPixels( aDst, aSrc, aMsk.GetAlpha() ); ++aDst; ++aSrc; ++aMsk; } } // ----------------------------------------------------------------------- template inline void ImplBlendLines( const TrueColorPixelPtr& rDst, const TrueColorPixelPtr& rSrc, unsigned nAlphaVal, int nPixelCount ) { if( nAlphaVal == ~(~0 << ALPHABITS) ) ImplConvertLine( rDst, rSrc, nPixelCount ); else if( nAlphaVal ) { TrueColorPixelPtr aSrc( rSrc ); TrueColorPixelPtr aDst( rDst ); while( --nPixelCount >= 0 ) { ImplBlendPixels( aDst, aSrc, nAlphaVal ); ++aDst; ++aSrc; } } } // ======================================================================= static bool ImplCopyImage( BitmapBuffer& rDstBuffer, const BitmapBuffer& rSrcBuffer ) { const int nSrcLinestep = rSrcBuffer.mnScanlineSize; int nDstLinestep = rDstBuffer.mnScanlineSize; const PIXBYTE* pRawSrc = rSrcBuffer.mpBits; PIXBYTE* pRawDst = rDstBuffer.mpBits; // source and destination don't match upside down if( BMP_FORMAT_TOP_DOWN & (rSrcBuffer.mnFormat ^ rDstBuffer.mnFormat) ) { pRawDst += (rSrcBuffer.mnHeight - 1) * nDstLinestep; nDstLinestep = -rDstBuffer.mnScanlineSize; } else if( nSrcLinestep == nDstLinestep ) { memcpy( pRawDst, pRawSrc, rSrcBuffer.mnHeight * nDstLinestep ); return true; } int nByteWidth = nSrcLinestep; if( nByteWidth > rDstBuffer.mnScanlineSize ) nByteWidth = rDstBuffer.mnScanlineSize; for( int y = rSrcBuffer.mnHeight; --y >= 0; ) { memcpy( pRawDst, pRawSrc, nByteWidth ); pRawSrc += nSrcLinestep; pRawDst += nDstLinestep; } return true; } // ----------------------------------------------------------------------- template bool ImplConvertToBitmap( TrueColorPixelPtr& rSrcLine, BitmapBuffer& rDstBuffer, const BitmapBuffer& rSrcBuffer ) { // help the compiler to avoid instantiations of unneeded conversions DBG_ASSERT( SRCFMT != DSTFMT, "ImplConvertToBitmap into same format"); if( SRCFMT == DSTFMT ) return false; const int nSrcLinestep = rSrcBuffer.mnScanlineSize; int nDstLinestep = rDstBuffer.mnScanlineSize; TrueColorPixelPtr aDstLine; aDstLine.SetRawPtr( rDstBuffer.mpBits ); // source and destination don't match upside down if( BMP_FORMAT_TOP_DOWN & (rSrcBuffer.mnFormat ^ rDstBuffer.mnFormat) ) { aDstLine.AddByteOffset( (rSrcBuffer.mnHeight - 1) * nDstLinestep ); nDstLinestep = -nDstLinestep; } for( int y = rSrcBuffer.mnHeight; --y >= 0; ) { ImplConvertLine( aDstLine, rSrcLine, rSrcBuffer.mnWidth ); rSrcLine.AddByteOffset( nSrcLinestep ); aDstLine.AddByteOffset( nDstLinestep ); } return true; } // ----------------------------------------------------------------------- template inline bool ImplConvertFromBitmap( BitmapBuffer& rDst, const BitmapBuffer& rSrc ) { TrueColorPixelPtr aSrcType; aSrcType.SetRawPtr( rSrc.mpBits ); // select the matching instantiation for the destination's bitmap format switch( rDst.mnFormat & ~BMP_FORMAT_TOP_DOWN ) { case BMP_FORMAT_1BIT_MSB_PAL: case BMP_FORMAT_1BIT_LSB_PAL: case BMP_FORMAT_4BIT_MSN_PAL: case BMP_FORMAT_4BIT_LSN_PAL: case BMP_FORMAT_8BIT_PAL: break; case BMP_FORMAT_8BIT_TC_MASK: // return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_24BIT_TC_MASK: // return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_32BIT_TC_MASK: // return ImplConvertToBitmap( aSrcType, rDst, rSrc ); break; case BMP_FORMAT_16BIT_TC_MSB_MASK: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_16BIT_TC_LSB_MASK: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_24BIT_TC_BGR: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_24BIT_TC_RGB: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_32BIT_TC_ABGR: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); #ifdef FAST_ARGB_BGRA case BMP_FORMAT_32BIT_TC_ARGB: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); case BMP_FORMAT_32BIT_TC_BGRA: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); #endif case BMP_FORMAT_32BIT_TC_RGBA: return ImplConvertToBitmap( aSrcType, rDst, rSrc ); } #ifdef DEBUG static int nNotAccelerated = 0; if( rSrc.mnWidth * rSrc.mnHeight >= 4000 ) if( ++nNotAccelerated == 100 ) { int foo = 0; (void)foo; // so no warning is created when building on pro with debug DBG_WARNING2( "ImplConvertFromBitmap for not accelerated case (0x%04X->0x%04X)", rSrc.mnFormat, rDst.mnFormat ); } #endif return false; } // ======================================================================= // an universal stretching conversion is overkill in most common situations // => performance benefits for speeding up the non-stretching cases bool ImplFastBitmapConversion( BitmapBuffer& rDst, const BitmapBuffer& rSrc, const SalTwoRect& rTR ) { if( bDisableFastBitops ) return false; // horizontal mirroring not implemented yet if( rTR.mnDestWidth < 0 ) return false; // vertical mirroring if( rTR.mnDestHeight < 0 ) // TODO: rDst.mnFormat ^= BMP_FORMAT_TOP_DOWN; return false; // offseted conversion is not implemented yet if( rTR.mnSrcX || rTR.mnSrcY ) return false; if( rTR.mnDestX || rTR.mnDestY ) return false; // stretched conversion is not implemented yet if( rTR.mnDestWidth != rTR.mnSrcWidth ) return false; if( rTR.mnDestHeight!= rTR.mnSrcHeight ) return false; // check source image size if( rSrc.mnWidth < rTR.mnSrcX + rTR.mnSrcWidth ) return false; if( rSrc.mnHeight < rTR.mnSrcY + rTR.mnSrcHeight ) return false; // check dest image size if( rDst.mnWidth < rTR.mnDestX + rTR.mnDestWidth ) return false; if( rDst.mnHeight < rTR.mnDestY + rTR.mnDestHeight ) return false; const ULONG nSrcFormat = rSrc.mnFormat & ~BMP_FORMAT_TOP_DOWN; const ULONG nDstFormat = rDst.mnFormat & ~BMP_FORMAT_TOP_DOWN; // TODO: also implement conversions for 16bit colormasks with non-565 format if( nSrcFormat & (BMP_FORMAT_16BIT_TC_LSB_MASK | BMP_FORMAT_16BIT_TC_MSB_MASK) ) if( rSrc.maColorMask.GetRedMask() != 0xF800 || rSrc.maColorMask.GetGreenMask()!= 0x07E0 || rSrc.maColorMask.GetBlueMask() != 0x001F ) return false; if( nDstFormat & (BMP_FORMAT_16BIT_TC_LSB_MASK | BMP_FORMAT_16BIT_TC_MSB_MASK) ) if( rDst.maColorMask.GetRedMask() != 0xF800 || rDst.maColorMask.GetGreenMask()!= 0x07E0 || rDst.maColorMask.GetBlueMask() != 0x001F ) return false; // special handling of trivial cases if( nSrcFormat == nDstFormat ) { // accelerated palette conversions not yet implemented if( rSrc.maPalette != rDst.maPalette ) return false; return ImplCopyImage( rDst, rSrc ); } // select the matching instantiation for the source's bitmap format switch( nSrcFormat ) { case BMP_FORMAT_1BIT_MSB_PAL: case BMP_FORMAT_1BIT_LSB_PAL: case BMP_FORMAT_4BIT_MSN_PAL: case BMP_FORMAT_4BIT_LSN_PAL: case BMP_FORMAT_8BIT_PAL: break; case BMP_FORMAT_8BIT_TC_MASK: // return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_24BIT_TC_MASK: // return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_32BIT_TC_MASK: // return ImplConvertFromBitmap( rDst, rSrc ); break; case BMP_FORMAT_16BIT_TC_MSB_MASK: return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_16BIT_TC_LSB_MASK: return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_24BIT_TC_BGR: return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_24BIT_TC_RGB: return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_32BIT_TC_ABGR: return ImplConvertFromBitmap( rDst, rSrc ); #ifdef FAST_ARGB_BGRA case BMP_FORMAT_32BIT_TC_ARGB: return ImplConvertFromBitmap( rDst, rSrc ); case BMP_FORMAT_32BIT_TC_BGRA: return ImplConvertFromBitmap( rDst, rSrc ); #endif case BMP_FORMAT_32BIT_TC_RGBA: return ImplConvertFromBitmap( rDst, rSrc ); } #ifdef DEBUG static int nNotAccelerated = 0; if( rSrc.mnWidth * rSrc.mnHeight >= 4000 ) { if( ++nNotAccelerated == 100 ) { int foo = 0; (void)foo; // so no warning is created when building on pro with debug DBG_WARNING2( "ImplFastBitmapConversion for not accelerated case (0x%04X->0x%04X)", rSrc.mnFormat, rDst.mnFormat ); } } #endif return false; } // ======================================================================= template //,ULONG MSKFMT> bool ImplBlendToBitmap( TrueColorPixelPtr& rSrcLine, BitmapBuffer& rDstBuffer, const BitmapBuffer& rSrcBuffer, const BitmapBuffer& rMskBuffer ) { //DBG_ASSERT( rMskBuffer.mnFormat == MSKFMT, "FastBmp BlendImage: wrong MSKFMT" ); DBG_ASSERT( rMskBuffer.mnFormat == BMP_FORMAT_8BIT_PAL, "FastBmp BlendImage: unusual MSKFMT" ); const int nSrcLinestep = rSrcBuffer.mnScanlineSize; int nMskLinestep = rMskBuffer.mnScanlineSize; int nDstLinestep = rDstBuffer.mnScanlineSize; TrueColorPixelPtr aMskLine; aMskLine.SetRawPtr( rMskBuffer.mpBits ); TrueColorPixelPtr aDstLine; aDstLine.SetRawPtr( rDstBuffer.mpBits ); // special case for single line masks if( rMskBuffer.mnHeight == 1 ) nMskLinestep = 0; // source and mask don't match: upside down if( (rSrcBuffer.mnFormat ^ rMskBuffer.mnFormat) & BMP_FORMAT_TOP_DOWN ) { aMskLine.AddByteOffset( (rSrcBuffer.mnHeight - 1) * nMskLinestep ); nMskLinestep = -nMskLinestep; } // source and destination don't match: upside down if( (rSrcBuffer.mnFormat ^ rDstBuffer.mnFormat) & BMP_FORMAT_TOP_DOWN ) { aDstLine.AddByteOffset( (rSrcBuffer.mnHeight - 1) * nDstLinestep ); nDstLinestep = -nDstLinestep; } for( int y = rSrcBuffer.mnHeight; --y >= 0; ) { ImplBlendLines<8>( aDstLine, rSrcLine, aMskLine, rDstBuffer.mnWidth ); aDstLine.AddByteOffset( nDstLinestep ); rSrcLine.AddByteOffset( nSrcLinestep ); aMskLine.AddByteOffset( nMskLinestep ); } return true; } // some specializations to reduce the code size template <> inline bool ImplBlendToBitmap( TrueColorPixelPtr&, BitmapBuffer& rDstBuffer, const BitmapBuffer& rSrcBuffer, const BitmapBuffer& rMskBuffer ) { TrueColorPixelPtr aSrcType; aSrcType.SetRawPtr( rSrcBuffer.mpBits ); return ImplBlendToBitmap( aSrcType, rDstBuffer, rSrcBuffer, rMskBuffer ); } template <> inline bool ImplBlendToBitmap( TrueColorPixelPtr&, BitmapBuffer& rDstBuffer, const BitmapBuffer& rSrcBuffer, const BitmapBuffer& rMskBuffer ) { TrueColorPixelPtr aSrcType; aSrcType.SetRawPtr( rSrcBuffer.mpBits ); return ImplBlendToBitmap( aSrcType, rDstBuffer, rSrcBuffer, rMskBuffer ); } template <> inline bool ImplBlendToBitmap( TrueColorPixelPtr&, BitmapBuffer& rDstBuffer, const BitmapBuffer& rSrcBuffer, const BitmapBuffer& rMskBuffer ) { TrueColorPixelPtr aSrcType; aSrcType.SetRawPtr( rSrcBuffer.mpBits ); return ImplBlendToBitmap( aSrcType, rDstBuffer, rSrcBuffer, rMskBuffer ); } // ----------------------------------------------------------------------- template bool ImplBlendFromBitmap( BitmapBuffer& rDst, const BitmapBuffer& rSrc, const BitmapBuffer& rMsk ) { TrueColorPixelPtr aSrcType; aSrcType.SetRawPtr( rSrc.mpBits ); // select the matching instantiation for the destination's bitmap format switch( rDst.mnFormat & ~BMP_FORMAT_TOP_DOWN ) { case BMP_FORMAT_1BIT_MSB_PAL: case BMP_FORMAT_1BIT_LSB_PAL: case BMP_FORMAT_4BIT_MSN_PAL: case BMP_FORMAT_4BIT_LSN_PAL: case BMP_FORMAT_8BIT_PAL: break; case BMP_FORMAT_8BIT_TC_MASK: // return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_24BIT_TC_MASK: // return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_32BIT_TC_MASK: // return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); break; case BMP_FORMAT_16BIT_TC_MSB_MASK: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_16BIT_TC_LSB_MASK: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_24BIT_TC_BGR: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_24BIT_TC_RGB: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_32BIT_TC_ABGR: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); #ifdef FAST_ARGB_BGRA case BMP_FORMAT_32BIT_TC_ARGB: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); case BMP_FORMAT_32BIT_TC_BGRA: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); #endif case BMP_FORMAT_32BIT_TC_RGBA: return ImplBlendToBitmap( aSrcType, rDst, rSrc, rMsk ); } #ifdef DEBUG static int nNotAccelerated = 0; if( rSrc.mnWidth * rSrc.mnHeight >= 4000 ) if( ++nNotAccelerated == 100 ) { int foo = 0; (void)foo; // so no warning is created when building on pro with debug DBG_WARNING3( "ImplBlendFromBitmap for not accelerated case (0x%04X*0x%04X->0x%04X)", rSrc.mnFormat, rMsk.mnFormat, rDst.mnFormat ); } #endif return false; } // ----------------------------------------------------------------------- bool ImplFastBitmapBlending( BitmapWriteAccess& rDstWA, const BitmapReadAccess& rSrcRA, const BitmapReadAccess& rMskRA, const SalTwoRect& rTR ) { if( bDisableFastBitops ) return false; // accelerated blending of paletted bitmaps not implemented yet if( rSrcRA.HasPalette() ) return false; if( rDstWA.HasPalette() ) return false; // TODO: either get rid of mask's use of 8BIT_PAL or check the palette // horizontal mirroring not implemented yet if( rTR.mnDestWidth < 0 ) return false; // vertical mirroring if( rTR.mnDestHeight < 0 ) // TODO: rDst.mnFormat ^= BMP_FORMAT_TOP_DOWN; return false; // offseted blending is not implemented yet if( rTR.mnSrcX || rTR.mnSrcY ) return false; if( rTR.mnDestX || rTR.mnDestY ) return false; // stretched blending is not implemented yet if( rTR.mnDestWidth != rTR.mnSrcWidth ) return false; if( rTR.mnDestHeight!= rTR.mnSrcHeight ) return false; // check source image size if( rSrcRA.Width() < rTR.mnSrcX + rTR.mnSrcWidth ) return false; if( rSrcRA.Height() < rTR.mnSrcY + rTR.mnSrcHeight ) return false; // check mask image size if( rMskRA.Width() < rTR.mnSrcX + rTR.mnSrcWidth ) return false; if( rMskRA.Height() < rTR.mnSrcY + rTR.mnSrcHeight ) if( rMskRA.Height() != 1 ) return false; // check dest image size if( rDstWA.Width() < rTR.mnDestX + rTR.mnDestWidth ) return false; if( rDstWA.Height() < rTR.mnDestY + rTR.mnDestHeight ) return false; BitmapBuffer& rDst = *rDstWA.ImplGetBitmapBuffer(); const BitmapBuffer& rSrc = *rSrcRA.ImplGetBitmapBuffer(); const BitmapBuffer& rMsk = *rMskRA.ImplGetBitmapBuffer(); const ULONG nSrcFormat = rSrc.mnFormat & ~BMP_FORMAT_TOP_DOWN; const ULONG nDstFormat = rDst.mnFormat & ~BMP_FORMAT_TOP_DOWN; // accelerated conversions for 16bit colormasks with non-565 format are not yet implemented if( nSrcFormat & (BMP_FORMAT_16BIT_TC_LSB_MASK | BMP_FORMAT_16BIT_TC_MSB_MASK) ) if( rSrc.maColorMask.GetRedMask() != 0xF800 || rSrc.maColorMask.GetGreenMask()!= 0x07E0 || rSrc.maColorMask.GetBlueMask() != 0x001F) return false; if( nDstFormat & (BMP_FORMAT_16BIT_TC_LSB_MASK | BMP_FORMAT_16BIT_TC_MSB_MASK) ) if( rDst.maColorMask.GetRedMask() != 0xF800 || rDst.maColorMask.GetGreenMask()!= 0x07E0 || rDst.maColorMask.GetBlueMask() != 0x001F) return false; // select the matching instantiation for the source's bitmap format switch( nSrcFormat ) { case BMP_FORMAT_1BIT_MSB_PAL: case BMP_FORMAT_1BIT_LSB_PAL: case BMP_FORMAT_4BIT_MSN_PAL: case BMP_FORMAT_4BIT_LSN_PAL: case BMP_FORMAT_8BIT_PAL: break; case BMP_FORMAT_8BIT_TC_MASK: // return ImplBlendFromBitmap( rDst, rSrc ); case BMP_FORMAT_24BIT_TC_MASK: // return ImplBlendFromBitmap( rDst, rSrc ); case BMP_FORMAT_32BIT_TC_MASK: // return ImplBlendFromBitmap( rDst, rSrc ); break; case BMP_FORMAT_16BIT_TC_MSB_MASK: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); case BMP_FORMAT_16BIT_TC_LSB_MASK: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); case BMP_FORMAT_24BIT_TC_BGR: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); case BMP_FORMAT_24BIT_TC_RGB: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); case BMP_FORMAT_32BIT_TC_ABGR: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); #ifdef FAST_ARGB_BGRA case BMP_FORMAT_32BIT_TC_ARGB: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); case BMP_FORMAT_32BIT_TC_BGRA: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); #endif case BMP_FORMAT_32BIT_TC_RGBA: return ImplBlendFromBitmap( rDst, rSrc, rMsk ); } #ifdef DEBUG static int nNotAccelerated = 0; if( rSrc.mnWidth * rSrc.mnHeight >= 4000 ) if( ++nNotAccelerated == 100 ) { int foo = 0; (void)foo; // so no warning is created when building on pro with debug DBG_WARNING3( "ImplFastBlend for not accelerated case (0x%04X*0x%04X->0x%04X)", rSrc.mnFormat, rMsk.mnFormat, rDst.mnFormat ); } #endif return false; } bool ImplFastEraseBitmap( BitmapBuffer& rDst, const BitmapColor& rColor ) { if( bDisableFastBitops ) return false; const ULONG nDstFormat = rDst.mnFormat & ~BMP_FORMAT_TOP_DOWN; // erasing a bitmap is often just a byte-wise memory fill bool bByteFill = true; BYTE nFillByte; switch( nDstFormat ) { case BMP_FORMAT_1BIT_MSB_PAL: case BMP_FORMAT_1BIT_LSB_PAL: nFillByte = rColor.GetIndex(); nFillByte = static_cast( -(nFillByte & 1) ); // 0x00 or 0xFF break; case BMP_FORMAT_4BIT_MSN_PAL: case BMP_FORMAT_4BIT_LSN_PAL: nFillByte = rColor.GetIndex(); nFillByte &= 0x0F; nFillByte |= (nFillByte << 4); break; case BMP_FORMAT_8BIT_PAL: case BMP_FORMAT_8BIT_TC_MASK: nFillByte = rColor.GetIndex(); break; case BMP_FORMAT_24BIT_TC_MASK: case BMP_FORMAT_24BIT_TC_BGR: case BMP_FORMAT_24BIT_TC_RGB: nFillByte = rColor.GetRed(); if( (nFillByte != rColor.GetGreen()) || (nFillByte != rColor.GetBlue()) ) bByteFill = false; break; default: bByteFill = false; nFillByte = 0x00; break; } if( bByteFill ) { long nByteCount = rDst.mnHeight * rDst.mnScanlineSize; rtl_fillMemory( rDst.mpBits, nByteCount, nFillByte ); return true; } // TODO: handle other bitmap formats switch( nDstFormat ) { case BMP_FORMAT_32BIT_TC_MASK: case BMP_FORMAT_16BIT_TC_MSB_MASK: case BMP_FORMAT_16BIT_TC_LSB_MASK: case BMP_FORMAT_24BIT_TC_BGR: case BMP_FORMAT_24BIT_TC_RGB: case BMP_FORMAT_32BIT_TC_ABGR: #ifdef FAST_ARGB_BGRA case BMP_FORMAT_32BIT_TC_ARGB: case BMP_FORMAT_32BIT_TC_BGRA: #endif case BMP_FORMAT_32BIT_TC_RGBA: break; default: break; } return false; } // ======================================================================= #else // NO_OPTIMIZED_BITMAP_ACCESS bool ImplFastBitmapConversion( BitmapBuffer&, const BitmapBuffer& ) { return false; } bool ImplFastBitmapBlending( BitmapWriteAccess&, const BitmapReadAccess&, const BitmapReadAccess&, const Size&, const Point& ) { return false; } bool ImplFastEraseBitmap( BitmapBuffer&, const BitmapColor& ) { return false; } #endif