os/graphics/graphicsdeviceinterface/directgdiadaptation/swsrc/swdirectgdibitblt.cpp
Update contrib.
1 // Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
16 #include "swdirectgdiengine.h"
17 #include "swdirectgdidriverimpl.h"
18 #include "directgdiadapter.h"
19 #include <bitdrawinterfaceid.h>
20 #include <bmalphablend.h>
21 #include <graphics/bitmap.inl>
22 #include <graphics/gdi/gdiinline.inl>
25 @see MDirectGdiEngine::BitBlt()
26 @panic DGDIAdapter 7, aBitmap is invalid (debug only).
28 void CSwDirectGdiEngine::BitBlt(const TPoint& aDestPos,
29 const CFbsBitmap& aBitmap,
30 const TRect& aSrceRect)
32 if (aBitmap.ExtendedBitmapType() != KNullUid)
34 iDriver->SetError(KErrNotSupported); // Not supported for extended bitmaps
38 TRect srceRect(aSrceRect);
39 const TPoint destPoint(aDestPos + iOrigin + srceRect.iTl - aSrceRect.iTl);
40 const TPoint offset(srceRect.iTl - destPoint);
42 TRect targetRect(destPoint,srceRect.Size());
43 aBitmap.BeginDataAccess();
45 CBitwiseBitmap* srce = static_cast<const CSwDirectGdiBitmap&>(aBitmap).Address();
46 GRAPHICS_ASSERT_DEBUG(srce,EDirectGdiPanicInvalidBitmap);
48 const TInt limit = iDefaultRegionPtr->Count();
50 TBool opaqueSource = (!IsAlphaChannel(aBitmap.DisplayMode())) && (iDrawMode == DirectGdi::EDrawModePEN);
53 for (TInt count = 0; count < limit; count++)
55 clipRect = (*iDefaultRegionPtr)[count];
56 if(!clipRect.Intersects(targetRect))
61 clipRect.Intersection(targetRect);
63 TRect clippedSrceRect(clipRect);
64 clippedSrceRect.Move(offset);
68 iDrawMode = DirectGdi::EDrawModeWriteAlpha; // write rather than blend.
71 DoBitBlt(clipRect.iTl, srce, aBitmap.DataAddress(), aBitmap.DataStride(), clippedSrceRect);
75 iDrawMode = DirectGdi::EDrawModePEN; // set it back to how it was.
78 iDrawDevice->UpdateRegion(clipRect);
81 aBitmap.EndDataAccess(ETrue);
85 @see MDirectGdiEngine::BitBltMasked(const TPoint&, const CFbsBitmap&, const TRect&, const CFbsBitmap&, TBool)
86 @panic DGDIAdapter 7, if either aMaskBitmap or aBitmap are invalid (debug only).
88 void CSwDirectGdiEngine::BitBltMasked(const TPoint& aDestPos,
89 const CFbsBitmap& aBitmap,
90 const TRect& aSrcRect,
91 const CFbsBitmap& aMaskBitmap,
94 if ((aBitmap.ExtendedBitmapType() != KNullUid) || (aMaskBitmap.ExtendedBitmapType() != KNullUid))
96 iDriver->SetError(KErrNotSupported); // Not supported for extended bitmaps
100 TRect localSrcRect(aSrcRect);
101 const TPoint destPoint(aDestPos + iOrigin + localSrcRect.iTl - aSrcRect.iTl);
102 const TRect destRect(destPoint, localSrcRect.Size());
103 const TPoint offset(localSrcRect.iTl - destPoint);
105 TRect targetRect(destRect);
106 aBitmap.BeginDataAccess();
107 aMaskBitmap.BeginDataAccess();
109 CBitwiseBitmap* srcebmp = static_cast<const CSwDirectGdiBitmap&>(aBitmap).Address();
110 CBitwiseBitmap* maskbmp = static_cast<const CSwDirectGdiBitmap&>(aMaskBitmap).Address();
112 GRAPHICS_ASSERT_DEBUG(srcebmp,EDirectGdiPanicInvalidBitmap);
113 GRAPHICS_ASSERT_DEBUG(maskbmp,EDirectGdiPanicInvalidBitmap);
115 const TDisplayMode maskMode = maskbmp->DisplayMode();
116 const TInt limit = iDefaultRegionPtr->Count();
117 TBool opaqueSource = (!IsAlphaChannel(aBitmap.DisplayMode())) && (iDrawMode == DirectGdi::EDrawModePEN);
119 for (TInt count = 0; count < limit; count++)
121 clipRect = (*iDefaultRegionPtr)[count];
122 if (!clipRect.Intersects(targetRect))
127 clipRect.Intersection(targetRect);
128 TRect clippedSrceRect(clipRect);
129 clippedSrceRect.Move(offset);
133 iDrawMode = DirectGdi::EDrawModeWriteAlpha; // ie write rather than blend
136 DoBitBltMasked(clipRect.iTl, srcebmp, aBitmap.DataAddress(), clippedSrceRect, maskbmp,
137 aMaskBitmap.DataAddress(), aInvertMask);
141 iDrawMode = DirectGdi::EDrawModePEN; // set to default
143 iDrawDevice->UpdateRegion(clipRect);
146 aBitmap.EndDataAccess(ETrue);
147 aMaskBitmap.EndDataAccess(ETrue);
152 @see MDirectGdiEngine::BitBltMasked(const TPoint&, const CFbsBitmap&, const TRect&, const CFbsBitmap&, const TPoint&)
153 @panic DGDIAdapter 7, if either aBitmap or aMaskBitmap are invalid.
154 @panic DGDIAdapter 1022, if the top-left corner of aSrcRect is out of bounds (debug only).
156 void CSwDirectGdiEngine::BitBltMasked(const TPoint& aDestPt,
157 const CFbsBitmap& aBitmap, const TRect& aSrcRect,
158 const CFbsBitmap& aMaskBitmap, const TPoint& aAlphaPt)
160 if ((aBitmap.ExtendedBitmapType() != KNullUid) || (aMaskBitmap.ExtendedBitmapType() != KNullUid))
162 iDriver->SetError(KErrNotSupported); // Not supported for extended bitmaps
166 TRect srcRect(aSrcRect);
167 //Calculate the destination rect
168 TPoint destPt(aDestPt + iOrigin);
169 TRect destRect(destPt, srcRect.Size());
170 TPoint offset(srcRect.iTl - destPt);
171 TRect targetRect(destRect);
173 aBitmap.BeginDataAccess();
174 aMaskBitmap.BeginDataAccess();
175 CBitwiseBitmap* srcBmp = static_cast<const CSwDirectGdiBitmap&>(aBitmap).Address();
176 CBitwiseBitmap* alphaBmp = static_cast<const CSwDirectGdiBitmap&>(aMaskBitmap).Address();
177 GRAPHICS_ASSERT_DEBUG(srcBmp, EDirectGdiPanicInvalidBitmap);
178 GRAPHICS_ASSERT_DEBUG(alphaBmp, EDirectGdiPanicInvalidBitmap);
179 TUint32* srcDataAddr = aBitmap.DataAddress();
180 TUint32* alphaDataAddr = aMaskBitmap.DataAddress();
182 //For each region - find the clipping rect and draw
183 TInt limit = iDefaultRegionPtr->Count ();
185 for (TInt count=0; count<limit;count++)
187 clipRect=(*iDefaultRegionPtr)[count];
188 if ( !clipRect.Intersects(targetRect))
192 //targetRect was constructed from destRect. destRect was constructed from srcRect.
193 clipRect.Intersection (targetRect);
194 TRect clippedSrcRect(clipRect);
195 clippedSrcRect.Move(offset);//clippedSrcRect - maps to a part of srcRect now.
196 TPoint shift(clippedSrcRect.iTl - srcRect.iTl);
197 GRAPHICS_ASSERT_DEBUG(shift.iX >= 0, EDirectGdiPanicNegativeShift);
198 GRAPHICS_ASSERT_DEBUG(shift.iY >= 0, EDirectGdiPanicNegativeShift);
199 DoBitBltAlpha (clipRect.iTl, srcBmp, srcDataAddr, clippedSrcRect,
200 alphaBmp, alphaDataAddr, aAlphaPt + shift, EFalse);
201 iDrawDevice->UpdateRegion (clipRect);
204 aBitmap.EndDataAccess(ETrue);
205 aMaskBitmap.EndDataAccess(ETrue);
210 Calculates the position into the scanline for the given x coordinate.
212 @param aX The given x-coordinate.
213 @param aDisplayMode The applied display mode.
214 @return The memory offset, or 0 if the mode is not supported.
215 @panic DGDIAdapter 1009, if aDisplayMode is not supported (debug only).
217 TUint CSwDirectGdiEngine::MemoryOffsetForPixelPitch(TUint aX, TDisplayMode aDisplayMode) const
219 switch (aDisplayMode)
227 GRAPHICS_PANIC_DEBUG(EDirectGdiPanicInvalidDisplayMode);
232 TUint32* CSwDirectGdiEngine::GetScanLineOffsetPtr(CBitwiseBitmap* aSrce, TUint32*& aSlptr,
233 TInt aLength, TPoint aPixel, TUint32* aBase,
234 TLineScanningPosition& aLineScanningPosition, TUint aXOffset)
236 aSrce->GetScanLinePtr(aSlptr, aLength, aPixel, aBase, aLineScanningPosition);
237 return (TUint32*)((TUint8*)aSlptr + aXOffset);
241 Performs the actual bitblt to the device.
242 This function may also be called by DrawBitmap(), and DrawRect() when using a patterned brush,
243 so any changes to this function may impact on them also.
245 @pre aSrce A bitmap with non-zero dimensions. aSrceRect has been clipped against the target.
247 @param aDest The target position on the device which will contain the top left
248 corner of the source bitmap.
249 @param aSrce The bitmap object that contains the pixels to draw.
250 @param aBase The address of the bitmap pixels.
251 @param aStride The length in bytes between scanlines in memory.
252 @param aSrceRect The area of the bitmap to draw from.
253 @panic DGDIAdapter 1013, if aDest is fully outside of the bounds of the target, or aSrceRect.iTl is not
254 within the drawing area (debug only).
256 void CSwDirectGdiEngine::DoBitBlt(const TPoint& aDest,
257 CBitwiseBitmap* aSrce,
260 const TRect& aSrceRect)
262 // Does multiple bitmap widths for painting rects only
263 const TInt width = aSrceRect.Width ();
266 TRect deviceDestRect;
267 iDrawDevice->GetDrawRect(deviceDestRect);
268 GRAPHICS_ASSERT_DEBUG(aDest.iX >= deviceDestRect.iTl.iX, EDirectGdiPanicOutOfBounds);
269 GRAPHICS_ASSERT_DEBUG(aDest.iY >= deviceDestRect.iTl.iY, EDirectGdiPanicOutOfBounds);
270 GRAPHICS_ASSERT_DEBUG((aDest.iX + aSrceRect.Width()) <= deviceDestRect.iBr.iX, EDirectGdiPanicOutOfBounds);
271 GRAPHICS_ASSERT_DEBUG((aDest.iY + aSrceRect.Height()) <= deviceDestRect.iBr.iY, EDirectGdiPanicOutOfBounds);
272 GRAPHICS_ASSERT_DEBUG(aDest.iX >= 0 && aDest.iY >= 0, EDirectGdiPanicOutOfBounds);
273 GRAPHICS_ASSERT_DEBUG(aSrceRect.iTl.iX >= 0 && aSrceRect.iTl.iY >= 0, EDirectGdiPanicOutOfBounds);
276 TSize srcSize = aSrce->SizeInPixels ();
278 const TPoint KZeroPoint(0,0);
279 TAny* interface = NULL;
280 if (iDrawMode == DirectGdi::EDrawModeWriteAlpha &&
281 aSrceRect.iBr.iX <= srcSize.iWidth &&
282 aSrceRect.iBr.iY <= srcSize.iHeight &&
283 !aSrce->IsCompressed() &&
284 aSrce->DisplayMode() == iDrawDevice->DisplayMode() &&
285 iDrawDevice->GetInterface(KFastBlit2InterfaceID, interface) == KErrNone)
287 // Conditions in CFbsBitGc allow for optimised blitting.
288 // The draw device supports the optimised blitting function.
289 // Operation may fail regardless due to unacceptable conditions in the draw device.
290 MFastBlit2* fastBlit = reinterpret_cast<MFastBlit2*>(interface);
291 if (fastBlit && (fastBlit->WriteBitmapBlock(aDest, aBase, aStride, srcSize, aSrceRect) == KErrNone))
297 MFastBlend* fastBlend=NULL;
298 if (FastBlendInterface(aSrce,NULL,fastBlend)==KErrNone)
300 if (fastBlend->FastBlendBitmap(aDest, aBase, aStride, srcSize, aSrceRect, aSrce->DisplayMode(), GcDrawMode(iDrawMode), CFbsDrawDevice::ENoShadow)== KErrNone)
306 const TInt scanLineBytes = iDrawDevice->ScanLineBytes();
307 TUint32* scanLineBuffer = iDrawDevice->ScanLineBuffer();
308 TPtr8 scanLineDes((TUint8*)scanLineBuffer, scanLineBytes, scanLineBytes);
310 const TDisplayMode dispMode = ScanLineBufferDisplayMode(iDrawDevice);
312 const TBool useScanLinePtr = (dispMode == aSrce->DisplayMode()) &&
313 (TDisplayModeUtils::NumDisplayModeBitsPerPixel(dispMode) >= 8);
315 TUint32* slptr = NULL;
317 TUint32* lastScanLine = NULL;
320 lastScanLine = aSrce->ScanLineAddress(aBase, aSrceRect.iBr.iY-1);
323 TInt srceWidth = srcSize.iWidth;
324 TInt partlinestart = aSrceRect.iTl.iX % srceWidth;
326 if (partlinestart < 0)
328 partlinestart += srceWidth;
331 const TInt partlinelength = Min(srceWidth - partlinestart, width);
332 TInt destX = aDest.iX;
333 const TInt destXlimit = destX+width;
334 const CGraphicsContext::TDrawMode drawMode = GcDrawMode(iDrawMode);
337 if (partlinestart > 0 && partlinelength > 0)
339 TPoint srcecoord1(partlinestart, aSrceRect.iTl.iY);
340 TInt desty = aDest.iY;
342 TLineScanningPosition lineScanPos(aBase);
346 offset = MemoryOffsetForPixelPitch(partlinestart, dispMode);
347 if (aSrce->IsCompressed ())
350 while (srcecoord1.iY < aSrceRect.iBr.iY)
352 scanLineBuffer = GetScanLineOffsetPtr (aSrce, slptr,
353 partlinelength, srcecoord1, aBase, lineScanPos, offset);
354 if (srcecoord1.iY == aSrceRect.iTl.iY)
356 aSrce->SetCompressionBookmark(lineScanPos, aBase, NULL);
358 iDrawDevice->WriteLine(aDest.iX, desty, partlinelength,
359 scanLineBuffer, drawMode);
366 while (srcecoord1.iY < aSrceRect.iBr.iY)
368 scanLineBuffer = GetScanLineOffsetPtr (aSrce, slptr,
369 partlinelength, srcecoord1, aBase, lineScanPos,
373 iDrawDevice->WriteLine (aDest.iX, desty,
374 partlinelength, scanLineBuffer, drawMode);
375 scanLineBuffer = (TUint32*)((TUint8*)scanLineBuffer + aStride);
379 while ((srcecoord1.iY < aSrceRect.iBr.iY) && (scanLineBuffer < lastScanLine)) ;
385 for ( ; srcecoord1.iY < aSrceRect.iBr.iY; srcecoord1.iY++, desty++)
387 aSrce->GetScanLine (scanLineDes, srcecoord1, partlinelength,
388 EFalse, KZeroPoint, dispMode, aBase, lineScanPos);
389 if ( srcecoord1.iY==aSrceRect.iTl.iY)
391 aSrce->SetCompressionBookmark (lineScanPos, aBase, NULL);
393 iDrawDevice->WriteLine (aDest.iX, desty, partlinelength,
394 scanLineBuffer, drawMode);
398 destX += partlinelength;
401 // multiple complete lines - columns
402 TInt numcolumns = (destXlimit - destX) / srceWidth;
406 TPoint srcecoord2(0, aSrceRect.iTl.iY);
407 TInt desty = aDest.iY;
409 TLineScanningPosition lineScanPos(aBase);
413 if (aSrce->IsCompressed())
415 while (srcecoord2.iY < aSrceRect.iBr.iY)
417 TPoint coord(srcecoord2);
418 aSrce->GetScanLinePtr (slptr, srceWidth, coord, aBase, lineScanPos);
419 if (srcecoord2.iY == aSrceRect.iTl.iY)
421 aSrce->SetCompressionBookmark(lineScanPos, aBase, NULL);
423 TInt tempdestX = destX;
424 for (TInt count = 0; count < numcolumns; count++, tempdestX += srceWidth)
426 iDrawDevice->WriteLine(tempdestX, desty, srceWidth, slptr, drawMode);
434 while (srcecoord2.iY < aSrceRect.iBr.iY)
436 TPoint coord(srcecoord2);
437 aSrce->GetScanLinePtr (slptr, srceWidth, coord, aBase, lineScanPos);
440 TInt tempdestX = destX;
441 for (TInt count = 0; count < numcolumns; count++, tempdestX += srceWidth)
443 iDrawDevice->WriteLine (tempdestX, desty, srceWidth, slptr, drawMode);
445 slptr = (TUint32*)((TUint8*)slptr + aStride);
449 while ((srcecoord2.iY < aSrceRect.iBr.iY) && (slptr < lastScanLine));
455 for (; srcecoord2.iY < aSrceRect.iBr.iY; srcecoord2.iY++, desty++)
457 TInt tempdestX = destX;
458 TPoint coord(srcecoord2);
459 aSrce->GetScanLinePtr (slptr, srceWidth, coord, aBase, lineScanPos);
460 if (srcecoord2.iY == aSrceRect.iTl.iY)
462 aSrce->SetCompressionBookmark (lineScanPos, aBase, NULL);
464 for (TInt count = 0; count < numcolumns; count++, tempdestX += srceWidth)
466 aSrce->GetScanLine(slptr, scanLineDes, coord, srceWidth,
467 EFalse, KZeroPoint, dispMode);
468 iDrawDevice->WriteLine(tempdestX, desty, srceWidth, scanLineBuffer, drawMode);
473 destX += numcolumns * srceWidth;
477 if (destX < destXlimit)
479 const TInt restofline = destXlimit - destX;
480 TPoint srcecoord3(0, aSrceRect.iTl.iY);
481 TInt desty = aDest.iY;
483 TLineScanningPosition lineScanPos(aBase);
488 if (aSrce->IsCompressed())
490 while (srcecoord3.iY < aSrceRect.iBr.iY)
492 scanLineBuffer = GetScanLineOffsetPtr (aSrce, slptr,
493 srceWidth, srcecoord3, aBase, lineScanPos, offset);
494 if (srcecoord3.iY == aSrceRect.iTl.iY)
496 aSrce->SetCompressionBookmark (lineScanPos, aBase, NULL);
498 iDrawDevice->WriteLine(destX, desty, restofline, scanLineBuffer, drawMode);
505 while (srcecoord3.iY < aSrceRect.iBr.iY)
507 scanLineBuffer = GetScanLineOffsetPtr (aSrce, slptr,
508 srceWidth, srcecoord3, aBase, lineScanPos, offset);
511 iDrawDevice->WriteLine(destX, desty, restofline, scanLineBuffer, drawMode);
512 scanLineBuffer = (TUint32*)((TUint8*)scanLineBuffer + aStride);
516 while ((srcecoord3.iY < aSrceRect.iBr.iY) && (scanLineBuffer < lastScanLine));
522 for (; srcecoord3.iY < aSrceRect.iBr.iY; srcecoord3.iY++, desty++)
524 aSrce->GetScanLine (scanLineDes, srcecoord3, srceWidth, EFalse,
525 KZeroPoint, dispMode, aBase, lineScanPos);
526 if (srcecoord3.iY == aSrceRect.iTl.iY)
528 aSrce->SetCompressionBookmark (lineScanPos, aBase, NULL);
530 iDrawDevice->WriteLine(destX, desty, restofline, scanLineBuffer, drawMode);
536 Performs the masked bitblt to the device.
538 @param aDest The target position on the device which will contain the top left
539 corner of the source bitmap.
540 @param aSourceBitmap The bitmap object that contains the pixels to draw.
541 @param aSourceBase The address of the source bitmap pixels.
542 @param aSourceRect The area of the bitmap to draw from.
543 @param aMaskBitmap The bitmap object that contains the mask.
544 @param aMaskBase The address of the mask pixels.
545 @param aInvertMask Inverts the mask if ETrue.
546 @panic DGDIAdapter 1013, if aDest is outside of the device bounds (debug only).
548 void CSwDirectGdiEngine::DoBitBltMasked(const TPoint& aDest,
549 CBitwiseBitmap* aSourceBitmap,
550 TUint32* aSourceBase,
551 const TRect& aSourceRect,
552 CBitwiseBitmap* aMaskBitmap,
557 TRect deviceDestRect;
558 iDrawDevice->GetDrawRect (deviceDestRect);
561 GRAPHICS_ASSERT_DEBUG (aDest.iX >= deviceDestRect.iTl.iX, EDirectGdiPanicOutOfBounds);
562 GRAPHICS_ASSERT_DEBUG (aDest.iY >= deviceDestRect.iTl.iY, EDirectGdiPanicOutOfBounds);
563 GRAPHICS_ASSERT_DEBUG ((aDest.iX + aSourceRect.Width()) <= deviceDestRect.iBr.iX, EDirectGdiPanicOutOfBounds);
564 GRAPHICS_ASSERT_DEBUG ((aDest.iY + aSourceRect.Height()) <= deviceDestRect.iBr.iY, EDirectGdiPanicOutOfBounds);
565 const TPoint KZeroPoint(0,0);
567 MFastBlend* fastBlend=NULL;
568 if (FastBlendInterface(aSourceBitmap,aMaskBitmap,fastBlend)==KErrNone)
570 if (fastBlend->FastBlendBitmapMasked(aDest, aSourceBase, aSourceBitmap->DataStride(), aSourceBitmap->SizeInPixels(), aSourceRect, aSourceBitmap->DisplayMode(),
571 aMaskBase, aMaskBitmap->DataStride(), aMaskBitmap->DisplayMode(), aMaskBitmap->SizeInPixels(), aSourceRect.iTl, aInvertMask,
572 GcDrawMode(iDrawMode), CFbsDrawDevice::ENoShadow)==KErrNone)
578 if (aMaskBitmap->DisplayMode() == EGray256)
580 DoBitBltAlpha (aDest, aSourceBitmap, aSourceBase, aSourceRect,
581 aMaskBitmap, aMaskBase, aSourceRect.iTl, EFalse);
583 // if screen driver is 16MAP we avoid logical operator pen modes by using DoBitBltAlpha() for blitting
584 else if(iDrawDevice->ScanLineDisplayMode() == EColor16MAP)
586 DoBitBltAlpha (aDest, aSourceBitmap, aSourceBase, aSourceRect,
587 aMaskBitmap, aMaskBase, aSourceRect.iTl, aInvertMask);
589 else if (aSourceBitmap == aMaskBitmap)
591 const TInt width = aSourceRect.Width();
592 const TDisplayMode dispMode = ScanLineBufferDisplayMode(iDrawDevice);
593 const CGraphicsContext::TDrawMode drawMode = aInvertMask ? CGraphicsContext::EDrawModeAND : CGraphicsContext::EDrawModeOR;
594 TPoint srcePoint(aSourceRect.iTl);
595 TInt destY = aDest.iY;
597 TLineScanningPosition lineScanPos(aSourceBase);
599 const TBool useScanLinePtr = (dispMode == aSourceBitmap->DisplayMode() &&
600 (TDisplayModeUtils::NumDisplayModeBitsPerPixel(dispMode)>=8));
604 TUint32* scanLineBuffer = NULL;
605 TUint32* slptr = NULL;
606 TUint offset = MemoryOffsetForPixelPitch (srcePoint.iX, dispMode);
608 if (aSourceBitmap->IsCompressed())
610 for ( ; srcePoint.iY < aSourceRect.iBr.iY; destY++,
613 scanLineBuffer = GetScanLineOffsetPtr (
614 aSourceBitmap, slptr, width, srcePoint,
615 aSourceBase, lineScanPos, offset);
617 iDrawDevice->WriteLine (aDest.iX, destY, width,
618 scanLineBuffer, drawMode);
623 TUint stride = aSourceBitmap->DataStride ();
624 TUint32* lastScanLine = aSourceBitmap->ScanLineAddress(aSourceBase, aSourceRect.iBr.iY-1);
626 while (srcePoint.iY < aSourceRect.iBr.iY)
628 scanLineBuffer = GetScanLineOffsetPtr(aSourceBitmap, slptr, width, srcePoint,
629 aSourceBase, lineScanPos, offset);
632 iDrawDevice->WriteLine (aDest.iX, destY, width, scanLineBuffer, drawMode);
633 scanLineBuffer = (TUint32*)((TUint8*)scanLineBuffer + stride);
637 while ((srcePoint.iY < aSourceRect.iBr.iY) && (scanLineBuffer < lastScanLine));
643 const TInt scanLineBytes = iDrawDevice->ScanLineBytes();
644 TUint32* scanLineBuffer = iDrawDevice->ScanLineBuffer();
645 TPtr8 scanLineDes((TUint8*)scanLineBuffer, scanLineBytes,
647 for (; srcePoint.iY < aSourceRect.iBr.iY; destY++,
650 aSourceBitmap->GetScanLine (scanLineDes, srcePoint,
651 width, EFalse, KZeroPoint, dispMode,
652 aSourceBase, lineScanPos);
654 iDrawDevice->WriteLine (aDest.iX, destY, width,
655 scanLineBuffer, drawMode);
661 DoBitBltMaskedFlicker(aDest, aSourceBitmap, aSourceBase,
662 aSourceRect, aMaskBitmap, aMaskBase, aInvertMask);
667 @see DoBitBltMasked()
669 void CSwDirectGdiEngine::DoBitBltMaskedFlicker(const TPoint& aDest,
670 CBitwiseBitmap* aSourceBitmap,
671 TUint32* aSourceBase,
672 const TRect& aSourceRect,
673 CBitwiseBitmap* aMaskBitmap,
677 const TInt width = aSourceRect.Width();
678 TInt destY = aDest.iY;
679 TPoint srcePoint(aSourceRect.iTl);
681 TLineScanningPosition lineScanPos(aSourceBase);
682 TLineScanningPosition lineScanPosMask(aMaskBase);
684 const TDisplayMode srcFormat = aSourceBitmap->DisplayMode();
685 const TDisplayMode maskFormat = aMaskBitmap->DisplayMode();
687 if (aMaskBitmap->IsCompressed())
689 HBufC8* hBuf = CFbsBitmap::GetDecompressionBuffer(aMaskBitmap->DataStride() + 4);
692 iDriver->SetError(KErrNoMemory);
693 return; // Out of memory so do not draw anything
695 lineScanPosMask.iScanLineBuffer = hBuf;
698 TAny* interface = NULL;
699 if ( (srcFormat == EColor16MU || srcFormat == EColor64K ) &&
700 maskFormat == EGray2 &&
701 aSourceBitmap->SizeInPixels().iWidth <= aMaskBitmap->SizeInPixels().iWidth &&
702 aSourceBitmap->SizeInPixels().iHeight <= aMaskBitmap->SizeInPixels().iHeight &&
703 iDrawDevice->GetInterface(KFastBlitInterfaceID, interface) == KErrNone )
705 // Parameters allow optimised code path
707 TUint32* srcPtr=NULL;
708 TUint32* maskPtr=NULL;
709 MFastBlit* fastBlit = reinterpret_cast<MFastBlit*>(interface);
710 while (srcePoint.iY < aSourceRect.iBr.iY)
712 aSourceBitmap->GetScanLinePtr(srcPtr, length, srcePoint, aSourceBase, lineScanPos);
713 aMaskBitmap->GetScanLinePtr(maskPtr, length, srcePoint, aMaskBase, lineScanPosMask);
715 fastBlit->WriteMaskLineEx(aDest.iX,destY,length,srcePoint.iX,srcPtr,srcFormat,srcePoint.iX,maskPtr,aInvertMask);
723 const TDisplayMode dispMode = ScanLineBufferDisplayMode(iDrawDevice);
724 const CGraphicsContext::TDrawMode drawMode = aInvertMask ? CGraphicsContext::EDrawModeAND : CGraphicsContext::EDrawModeANDNOT;
726 TUint32* scanLineBuffer = iDrawDevice->ScanLineBuffer();
727 const TInt scanLineBytes = iDrawDevice->ScanLineBytes();
728 TPtr8 scanLineDes((TUint8*)scanLineBuffer,scanLineBytes,scanLineBytes);
729 TLineScanningPosition lineScanPos2(aSourceBase);
730 const TPoint KZeroPoint(0,0);
732 //scanline modifications required if using different modes, bits per pixel less than 8
733 if ( (dispMode == aSourceBitmap->DisplayMode()) &&
734 (TDisplayModeUtils::NumDisplayModeBitsPerPixel(dispMode)>=8))
736 TUint offset = MemoryOffsetForPixelPitch(srcePoint.iX, dispMode);
738 //mask scanline modifications required for EInvertPen, different screen modes
739 if ((drawMode != CGraphicsContext::EDrawModeANDNOT) && (dispMode == aMaskBitmap->DisplayMode()))
741 TUint32* scanLineBufferMask = NULL;
742 //stride jumping not possible with compressed bitmaps
743 if (aSourceBitmap->IsCompressed() || aMaskBitmap->IsCompressed())
745 for (; srcePoint.iY < aSourceRect.iBr.iY; destY++,srcePoint.iY++)
747 scanLineBuffer = GetScanLineOffsetPtr(aSourceBitmap, slptr, width, srcePoint, aSourceBase, lineScanPos, offset);
748 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer, CGraphicsContext::EDrawModeXOR);
749 scanLineBufferMask = GetScanLineOffsetPtr(aMaskBitmap, slptr, width, srcePoint, aMaskBase, lineScanPosMask, offset);
750 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBufferMask,drawMode);
751 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer, CGraphicsContext::EDrawModeXOR);
756 TUint strideSrc = aSourceBitmap->DataStride();
757 TUint strideMask = aMaskBitmap->DataStride();
758 TUint32* lastScanLineSrc = aSourceBitmap->ScanLineAddress(aSourceBase,aSourceRect.iBr.iY-1);
759 TUint32* lastScanLineMask = aMaskBitmap->ScanLineAddress(aMaskBase,aSourceRect.iBr.iY-1);
761 while (srcePoint.iY < aSourceRect.iBr.iY)
763 scanLineBuffer = GetScanLineOffsetPtr(aSourceBitmap, slptr, width, srcePoint, aSourceBase, lineScanPos, offset);
764 scanLineBufferMask = GetScanLineOffsetPtr(aMaskBitmap, slptr, width, srcePoint, aMaskBase, lineScanPosMask, offset);
767 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,CGraphicsContext::EDrawModeXOR);
768 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBufferMask,drawMode);
769 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,CGraphicsContext::EDrawModeXOR);
773 while ((srcePoint.iY < aSourceRect.iBr.iY) &&
774 (scanLineBuffer < lastScanLineSrc) &&
775 (scanLineBufferMask < lastScanLineMask) &&
776 ((scanLineBuffer = (TUint32*)((TUint8*)scanLineBuffer + strideSrc))>(TUint32*)0) &&
777 ((scanLineBufferMask = (TUint32*)((TUint8*)scanLineBufferMask + strideMask))>(TUint32*)0) );
783 TUint32* scanLineBufferPtr = NULL;
784 //stride jumping not possible with compressed bitmaps
785 if (aSourceBitmap->IsCompressed())
787 for (; srcePoint.iY < aSourceRect.iBr.iY; destY++,srcePoint.iY++)
789 scanLineBufferPtr = GetScanLineOffsetPtr(aSourceBitmap, slptr, width, srcePoint, aSourceBase, lineScanPos, offset);
790 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBufferPtr,CGraphicsContext::EDrawModeXOR);
791 aMaskBitmap->GetScanLine(scanLineDes,srcePoint,width,EFalse , KZeroPoint, dispMode, aMaskBase, lineScanPosMask);
792 TileScanLine(scanLineDes, width, srcePoint, aMaskBitmap, lineScanPosMask, aMaskBase, dispMode);
793 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,drawMode);
794 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBufferPtr,CGraphicsContext::EDrawModeXOR);
799 TUint stride = aSourceBitmap->DataStride();
800 TUint32* lastScanLine = aSourceBitmap->ScanLineAddress(aSourceBase,aSourceRect.iBr.iY-1);
801 while (srcePoint.iY < aSourceRect.iBr.iY)
803 scanLineBufferPtr = GetScanLineOffsetPtr(aSourceBitmap, slptr, width, srcePoint, aSourceBase, lineScanPos, offset);
806 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBufferPtr,CGraphicsContext::EDrawModeXOR);
807 aMaskBitmap->GetScanLine(scanLineDes,srcePoint,width,EFalse, KZeroPoint, dispMode,aMaskBase, lineScanPosMask);
808 TileScanLine(scanLineDes, width, srcePoint, aMaskBitmap, lineScanPosMask, aMaskBase, dispMode);
809 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,drawMode);
810 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBufferPtr,CGraphicsContext::EDrawModeXOR);
814 while ((srcePoint.iY < aSourceRect.iBr.iY) && (scanLineBufferPtr < lastScanLine) &&
815 ((scanLineBufferPtr = (TUint32*)((TUint8*)scanLineBufferPtr + stride))>(TUint32*)0));
822 for (; srcePoint.iY < aSourceRect.iBr.iY; destY++,srcePoint.iY++)
824 aSourceBitmap->GetScanLine(scanLineDes,srcePoint,width,EFalse, KZeroPoint,
825 dispMode,aSourceBase,lineScanPos);
827 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,CGraphicsContext::EDrawModeXOR);
828 aMaskBitmap->GetScanLine(scanLineDes,srcePoint,width,EFalse , KZeroPoint, dispMode,
829 aMaskBase, lineScanPosMask);
830 TileScanLine(scanLineDes, width, srcePoint, aMaskBitmap, lineScanPosMask, aMaskBase, dispMode);
831 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,drawMode);
832 aSourceBitmap->GetScanLine(scanLineDes,srcePoint,width,EFalse , KZeroPoint ,dispMode,
833 aSourceBase,lineScanPos2);
835 iDrawDevice->WriteLine(aDest.iX,destY,width,scanLineBuffer,CGraphicsContext::EDrawModeXOR);
841 @see DoBitBltMasked()
843 void CSwDirectGdiEngine::DoBitBltAlpha(const TPoint& aDest ,CBitwiseBitmap* aSourceBitmap,
844 TUint32* aSourceBase, const TRect& aSourceRect,
845 CBitwiseBitmap* aMaskBitmap, TUint32* aMaskBase,
846 const TPoint& aAlphaPoint, TBool aInvertMask)
848 MFastBlend* fastBlend=NULL;
849 if (FastBlendInterface(aSourceBitmap,aMaskBitmap,fastBlend)==KErrNone)
851 if (fastBlend->FastBlendBitmapMasked(aDest, aSourceBase, aSourceBitmap->DataStride(), aSourceBitmap->SizeInPixels(), aSourceRect, aSourceBitmap->DisplayMode(),
852 aMaskBase, aMaskBitmap->DataStride(), aMaskBitmap->DisplayMode(), aMaskBitmap->SizeInPixels(), aAlphaPoint, aInvertMask,
853 GcDrawMode(iDrawMode), CFbsDrawDevice::ENoShadow)==KErrNone)
859 const TPoint KZeroPoint(0,0);
860 const TInt KScanLineLength = 256;
861 const TInt KRgbSize = 4;
863 TUint8 srceRgbBuffer[KScanLineLength * KRgbSize];
864 TUint8 maskBuffer[KScanLineLength];
865 TUint8* srceRgbBufferPtr(srceRgbBuffer);
867 TPtr8 srceRgbDes(srceRgbBuffer, KScanLineLength * KRgbSize, KScanLineLength * KRgbSize);
868 TPtr8 maskDes(maskBuffer, KScanLineLength, KScanLineLength);
870 TInt srceY = aSourceRect.iTl.iY;
871 TInt destY = aDest.iY;
872 TInt alphaY = aAlphaPoint.iY;
874 TLineScanningPosition lineScanPosSrc(aSourceBase);
875 TLineScanningPosition lineScanPosMask(aMaskBase);
876 TDisplayMode sourceMode = aSourceBitmap->DisplayMode();
878 if (aMaskBitmap->IsCompressed())
880 HBufC8* hBuf= CFbsBitmap::GetDecompressionBuffer(aMaskBitmap->DataStride());
883 iDriver->SetError(KErrNoMemory); // Out of memory so do not draw anything
886 lineScanPosMask.iScanLineBuffer = hBuf;
889 TAny* interface = NULL;
890 if ( (sourceMode == EColor16MU || sourceMode == EColor64K) &&
891 aMaskBitmap->DisplayMode() == EGray256 && // ensure a monochrome mask isn't passed in as an alpha channel
892 aSourceBitmap->SizeInPixels().iWidth <= aMaskBitmap->SizeInPixels().iWidth &&
893 aSourceBitmap->SizeInPixels().iHeight <= aMaskBitmap->SizeInPixels().iHeight &&
894 iDrawDevice->GetInterface(KFastBlitInterfaceID, interface) == KErrNone )
896 TInt length = aSourceRect.Width();
897 const TInt srceX = aSourceRect.iTl.iX;
898 const TInt alphaX = aAlphaPoint.iX;
899 const TInt destX = aDest.iX;
900 MFastBlit* fastBlit = reinterpret_cast<MFastBlit*>(interface);
902 while (srceY < aSourceRect.iBr.iY)
906 TPoint srcPoint(srceX, srceY);
907 TPoint maskPoint(alphaX, alphaY);
909 aSourceBitmap->GetScanLinePtr(srcPtr, length, srcPoint, aSourceBase, lineScanPosSrc);
910 aMaskBitmap->GetScanLinePtr(maskPtr, length, maskPoint, aMaskBase, lineScanPosMask);
912 fastBlit->WriteAlphaLineEx(destX, destY, length, srceX, srcPtr,
913 sourceMode, alphaX, maskPtr, MAlphaBlend::EShdwBefore);
922 const TBool useScanLinePtr = ( (EColor16MA == aSourceBitmap->DisplayMode()));
923 TUint32* slptr = NULL;
926 while (srceY < aSourceRect.iBr.iY)
928 TInt srceX = aSourceRect.iTl.iX;
929 TInt destX = aDest.iX;
930 TInt alphaX = aAlphaPoint.iX;
932 while (srceX < aSourceRect.iBr.iX)
934 TPoint srcePoint(srceX,srceY);
935 TPoint alphaPoint(alphaX,alphaY);
936 const TInt width = Min(KScanLineLength, aSourceRect.iBr.iX - srceX);
940 offset = MemoryOffsetForPixelPitch(srceX, EColor16MU);
941 srceRgbBufferPtr = (TUint8*)GetScanLineOffsetPtr(aSourceBitmap, slptr, width,
942 srcePoint, aSourceBase, lineScanPosSrc, offset);
946 aSourceBitmap->GetScanLine(srceRgbDes,srcePoint,width,EFalse,KZeroPoint,
947 ERgb,aSourceBase,lineScanPosSrc);
950 aMaskBitmap->GetScanLine(maskDes, alphaPoint, width, EFalse, KZeroPoint,
951 EGray256, aMaskBase, lineScanPosMask);
952 TileScanLine(maskDes, width, alphaPoint, aMaskBitmap, lineScanPosMask, aMaskBase, EGray256);
954 // aInvertMask is not used for alpha channels (EGray256 mask)
955 if (aInvertMask && aMaskBitmap->DisplayMode() != EGray256)
957 for (TInt i = 0; i < width; ++i)
959 maskBuffer[i] = ~maskBuffer[i];
963 iDrawDevice->WriteRgbAlphaLine(destX, destY, width, srceRgbBufferPtr, maskBuffer, GcDrawMode(iDrawMode));
965 srceX += KScanLineLength;
966 destX += KScanLineLength;
967 alphaX += KScanLineLength;
977 Tiles the scan line if its length in pixels is less than aLengthInPixels.
979 @param aScanLine A pointer to the scan line buffer.
980 @param aLengthInPixels The scan line size in pixels.
981 @param aSrcPt Position of the first pixel in aMaskBitmap that should be used as a source
982 for the pixels in scan line buffer.
983 @param aMaskBitmap Any additional pixels for the scan line buffer will be taken from here.
984 @param aScanLinePos This argument is used for some internal optimisations. It should not be
985 modified by the caller.
986 @param aMaskBase The base address of aMaskBitmap data.
987 @param aDisplayMode Any additional pixels should be taken from aMaskBitmap using aDisplayMode
988 as an argument for GetScanLine() call.
989 @panic DGDIAdapter 1021, if the memory required for the scanline is greater than the size of aScanLine (debug only).
991 void CSwDirectGdiEngine::TileScanLine(TPtr8& aScanLine,
992 TInt aLengthInPixels,
993 const TPoint& aSrcPt,
994 const CBitwiseBitmap* aMaskBitmap,
995 TLineScanningPosition& aScanLinePos,
997 TDisplayMode aDisplayMode
1000 TInt lengthInBytes = CFbsBitmap::ScanLineLength(aLengthInPixels, aDisplayMode);
1001 GRAPHICS_ASSERT_DEBUG(lengthInBytes <= aScanLine.MaxLength(), EDirectGdiPanicInvalidArg);
1002 TInt scanLineLength = aScanLine.Length();
1003 if(scanLineLength < lengthInBytes && aSrcPt.iX > 0)
1005 //If, for example, src bmp is 100 pixels width, mask bmp is 20 pixels width and src
1006 //rect is (10, 0, 100, 1) -> We will read only pixels 10..19 from the first scan line
1007 //of the mask bmp. We have to have 90 mask bmp pixels.
1008 //So we have to make a second mask bmp read startig from pixel 0 - 10 pixels length.
1009 TInt maxLen = Min(aScanLine.MaxLength() - scanLineLength, aSrcPt.iX);
1010 TPtr8 maskDes2(const_cast <TUint8*> (aScanLine.Ptr()) + scanLineLength, maxLen, maxLen);
1011 TPoint srcPt(0, aSrcPt.iY);
1012 TPoint zeroPt(0, 0);
1013 aMaskBitmap->GetScanLine(maskDes2, srcPt, maxLen, EFalse, zeroPt, aDisplayMode, aMaskBase, aScanLinePos);
1014 aScanLine.SetLength(scanLineLength + maskDes2.Length());
1015 scanLineLength = aScanLine.Length();
1017 if(scanLineLength >= lengthInBytes || scanLineLength == 0)
1021 //If we still don't have enough mask bmp pixels - we have to tile the scan line
1022 TInt repeatCnt = lengthInBytes / scanLineLength - 1;
1023 TInt bytesLeft = lengthInBytes % scanLineLength;
1024 const TUint8* src = aScanLine.Ptr();
1025 TUint8* dest = const_cast <TUint8*> (src) + scanLineLength;
1026 for(;repeatCnt>0;dest+=scanLineLength,repeatCnt--)
1028 Mem::Copy(dest, src, scanLineLength);
1032 Mem::Copy(dest, src, bytesLeft);
1034 aScanLine.SetLength(lengthInBytes);
1038 Draws a masked rectangular section of the source bitmap and does a compress/stretch to
1039 fit a given destination rectangle. It uses DoBitBltMasked() if no stretching is involved.
1041 @see DrawBitmapMasked()
1043 @param aDestRect The target position on the device containing the top left corner of the source bitmap.
1044 @param aSourceBitmap The bitmap object that contains the pixels to draw.
1045 @param aSourceBase The address of the source bitmap pixels.
1046 @param aSourceRect The area of the bitmap to draw from.
1047 @param aMaskBitmap The bitmap object that contains the mask.
1048 @param aMaskBase The address of the mask pixels.
1049 @param aInvertMask Inverts the mask if ETrue.
1050 @param aClipRect A clipping rectangle.
1051 @panic DGDIAdapter 1013, if the clipping rectangle is fully outside of the device bounds (debug only).
1053 void CSwDirectGdiEngine::DoDrawBitmapMasked(const TRect& aDestRect,
1054 CBitwiseBitmap* aSourceBitmap,
1055 TUint32* aSourceBase,
1056 const TRect& aSourceRect,
1057 CBitwiseBitmap* aMaskBitmap,
1060 const TRect& aClipRect)
1062 CFbsDrawDevice* drawDevice = iDrawDevice;
1064 TRect deviceDestRect;
1065 drawDevice->GetDrawRect(deviceDestRect);
1066 GRAPHICS_ASSERT_DEBUG(aClipRect.iTl.iX >= deviceDestRect.iTl.iX, EDirectGdiPanicOutOfBounds);
1067 GRAPHICS_ASSERT_DEBUG(aClipRect.iTl.iY >= deviceDestRect.iTl.iY, EDirectGdiPanicOutOfBounds);
1068 GRAPHICS_ASSERT_DEBUG(aClipRect.iBr.iX <= deviceDestRect.iBr.iX, EDirectGdiPanicOutOfBounds);
1069 GRAPHICS_ASSERT_DEBUG(aClipRect.iBr.iY <= deviceDestRect.iBr.iY, EDirectGdiPanicOutOfBounds);
1072 // The clipped version of the destination rectangle
1073 TRect clippedDestRect(aDestRect);
1074 clippedDestRect.Intersection(aClipRect);
1076 // If the source rectangle and the destination rectangle are same,
1077 // no stretch/compress operation required, just do BitBltMasked
1078 if (aDestRect.Size() == aSourceRect.Size())
1080 if (!clippedDestRect.IsEmpty())
1082 const TPoint destPoint(clippedDestRect.iTl);
1083 clippedDestRect.Move(aSourceRect.iTl - aDestRect.iTl);
1084 DoBitBltMasked(destPoint,
1095 MFastBlend* fastBlend=NULL;
1096 if (FastBlendInterface(aSourceBitmap,aMaskBitmap,fastBlend)==KErrNone)
1098 if (fastBlend->FastBlendBitmapMaskedScaled(aClipRect, aDestRect, aSourceRect, aSourceBase, aSourceBitmap->DataStride(),
1099 aSourceBitmap->DisplayMode(),aSourceBitmap->SizeInPixels(),
1100 aMaskBase, aMaskBitmap->DataStride(), aMaskBitmap->DisplayMode(), aMaskBitmap->SizeInPixels(), aInvertMask,
1101 GcDrawMode(iDrawMode), CFbsDrawDevice::ENoShadow)==KErrNone)
1107 TUint32* scanLineBuffer = drawDevice->ScanLineBuffer();
1108 const TInt scanLineBytes = drawDevice->ScanLineBytes();
1109 TPtr8 scanLineDes(reinterpret_cast<TUint8*>(scanLineBuffer),scanLineBytes,scanLineBytes);
1111 const TInt KScanLineLength = 256;
1112 const TInt KRgbSize = 4;
1113 TUint8 maskBuffer[KScanLineLength];
1115 TUint8 sourceBuffer[KScanLineLength*KRgbSize];
1116 TPtr8 sourceDes(sourceBuffer,KScanLineLength*KRgbSize,KScanLineLength*KRgbSize);
1118 const TDisplayMode dispMode = ScanLineBufferDisplayMode(drawDevice);
1119 CGraphicsContext::TDrawMode drawMode = aInvertMask ? CGraphicsContext::EDrawModeAND : CGraphicsContext::EDrawModeANDNOT;
1120 // If the source bitmap and the mask bitmap are same, draw the source bitmap either
1121 // with EDrawModeAND or EDrawModeOR based on aInvertMask parameter.
1122 if (aSourceBitmap == aMaskBitmap)
1124 drawMode = aInvertMask ? CGraphicsContext::EDrawModeAND : CGraphicsContext::EDrawModeOR;
1128 TInt bitmapXStart = 0;
1129 xLine.Construct(TPoint(aSourceRect.iTl.iX,aDestRect.iTl.iX),
1130 TPoint(aSourceRect.iBr.iX,aDestRect.iBr.iX),TLinearDDA::ELeft);
1131 xLine.JumpToYCoord2(bitmapXStart,clippedDestRect.iTl.iX);
1134 TPoint yCoord(aSourceRect.iTl.iY,aDestRect.iTl.iY);
1135 yLine.Construct(yCoord,TPoint(aSourceRect.iBr.iY,aDestRect.iBr.iY),TLinearDDA::ELeft);
1137 yLine.JumpToYCoord2(dummy,clippedDestRect.iTl.iY);
1138 yCoord.SetXY(dummy,clippedDestRect.iTl.iY);
1140 const TInt srceWidth = aSourceRect.Width();
1141 const TInt destWidth = aDestRect.Width();
1142 const TInt clipWidth = clippedDestRect.Width();
1143 const TInt clipStrch = clippedDestRect.iTl.iX - aDestRect.iTl.iX;
1144 const TInt sourceBmpWidth = aSourceBitmap->SizeInPixels().iWidth;
1145 const TInt maskWidth = aMaskBitmap->SizeInPixels().iWidth;
1146 const TInt maskHeight = aMaskBitmap->SizeInPixels().iHeight;
1148 TLineScanningPosition lineScanPos(aSourceBase);
1149 TLineScanningPosition lineScanPos2(aSourceBase);
1150 TLineScanningPosition lineScanPosMask(aMaskBase);
1152 HBufC8* alphaBuffer = NULL;
1153 TPtr8 alphaBufferDes(NULL, 0);
1154 const TDisplayMode maskDisplayMode = aMaskBitmap->DisplayMode();
1156 // Mask inversion is not supported if the original source mask format is EGray256.
1157 // Note that this is only used for pre-multiplied alpha targets.
1158 TUint8 maskInverter = (aInvertMask && maskDisplayMode != EGray256) ? 0xFF : 0x00;
1160 if (aSourceBitmap != aMaskBitmap)
1162 // Get buffer to be used to convert non-EGray256 masks to EGray256 when display mode is EColor16MAP
1163 // or to tile the mask when the mask width is smaller than the source bitmap width.
1164 if (maskDisplayMode != EGray256 && (dispMode == EColor16MAP || maskWidth < sourceBmpWidth))
1166 alphaBuffer = CFbsBitmap::GetExtraBuffer(CFbsBitmap::ScanLineLength(maskWidth, EGray256));
1169 return; // Out of memory so do not draw anything
1171 alphaBufferDes.Set(alphaBuffer->Des());
1174 // Get buffer to be used for decompressing compressed masks when mask is EGray256
1175 if (maskDisplayMode == EGray256 && aMaskBitmap->IsCompressed())
1177 HBufC8* hBuf= CFbsBitmap::GetDecompressionBuffer(aMaskBitmap->DataStride());
1180 return; // Out of memory so do not draw anything
1182 lineScanPosMask.iScanLineBuffer = hBuf;
1185 const TPoint KZeroPoint(0,0);
1187 while (yCoord.iY < clippedDestRect.iBr.iY)
1189 // Draw only the source bitmap, if the source bitmap and the mask bitmap are same.
1190 // else draw both the bitmaps
1191 if (aSourceBitmap == aMaskBitmap)
1193 aSourceBitmap->StretchScanLine(scanLineDes,TPoint(bitmapXStart,yCoord.iX),
1194 clipStrch,clipWidth,destWidth,aSourceRect.iTl.iX,
1195 srceWidth, KZeroPoint,dispMode,aSourceBase,lineScanPos);
1196 if (yCoord.iY==clippedDestRect.iTl.iY)
1197 aSourceBitmap->SetCompressionBookmark(lineScanPos,aSourceBase,NULL);
1198 drawDevice->WriteLine(clippedDestRect.iTl.iX,yCoord.iY,clipWidth,scanLineBuffer,drawMode);
1200 else if ((maskDisplayMode == EGray256) || (dispMode == EColor16MAP))
1202 // Stretch the source bitmap and the mask bitmap for KScanLineLength as stretch length
1203 // then do alpha blending for this length. If the length is more then KScanLineLength
1204 // repeat it till you stretch complete destination length.
1205 const TPoint startPt(bitmapXStart,yCoord.iX);
1206 TInt clipWidthPart = clippedDestRect.Width();
1207 TBool loopLast = ETrue;
1208 if(clipWidthPart > KScanLineLength)
1210 clipWidthPart = KScanLineLength;
1213 TInt clipIncStrch = clippedDestRect.iTl.iX - aDestRect.iTl.iX;
1214 TInt startClip=clippedDestRect.iTl.iX;
1215 TPoint sourceDestXCoords(bitmapXStart,clippedDestRect.iTl.iX);
1216 xLine.Construct(TPoint(aSourceRect.iTl.iX,aDestRect.iTl.iX),
1217 TPoint(aSourceRect.iBr.iX,aDestRect.iBr.iX),TLinearDDA::ELeft);
1218 xLine.JumpToYCoord2(sourceDestXCoords.iX,sourceDestXCoords.iY);
1219 TPoint srcPixel(bitmapXStart % maskWidth,yCoord.iX % maskHeight);
1222 TUint32* maskScanLinePtr32 = NULL;
1223 TPoint currentYValue(0,srcPixel.iY);
1224 aMaskBitmap->GetScanLinePtr(maskScanLinePtr32, currentYValue, maskWidth, aMaskBase, lineScanPosMask);
1225 // To implement non EGray256 mask support with EColor16MAP display mode, we convert
1226 // the mask to EGray256.
1227 if (maskDisplayMode != EGray256) // Convert scan-line to EGray256 and set maskScanLinePtr32 to the conversion.
1229 aMaskBitmap->GetScanLine(maskScanLinePtr32, alphaBufferDes, currentYValue, maskWidth, EFalse, TPoint(0, 0), EGray256);
1230 maskScanLinePtr32 = (TUint32*)alphaBuffer->Ptr();
1232 TUint8* maskScanLinePtr = reinterpret_cast<TUint8*>(maskScanLinePtr32);
1234 // Outer loop over all KScanLineLengths
1237 aSourceBitmap->StretchScanLine(sourceDes,startPt,clipIncStrch,clipWidthPart,destWidth,
1238 aSourceRect.iTl.iX,srceWidth, KZeroPoint ,EColor16MU,aSourceBase,lineScanPos);
1239 // Inner loop to tile the mask if necessary
1240 spaceLeft = clipWidthPart;
1242 srcPixel.iX = sourceDestXCoords.iX % maskWidth;
1244 // Invert the mask using the inversion mask.
1245 maskBuffer[(sourceDestXCoords.iY-clippedDestRect.iTl.iX)%KScanLineLength]=
1246 maskInverter^maskScanLinePtr[srcPixel.iX];
1247 xLine.NextStep(sourceDestXCoords);
1248 } while (--spaceLeft>0);
1250 if (yCoord.iY == clippedDestRect.iTl.iY && startClip == clippedDestRect.iTl.iX)
1252 aSourceBitmap->SetCompressionBookmark(lineScanPos,aSourceBase,NULL);
1253 aMaskBitmap->SetCompressionBookmark(lineScanPosMask,aMaskBase,NULL);
1255 drawDevice->WriteRgbAlphaLine(startClip,yCoord.iY,clipWidthPart,sourceBuffer,maskBuffer, GcDrawMode(iDrawMode));
1260 startClip+=KScanLineLength;
1261 if (clippedDestRect.iBr.iX - startClip <= KScanLineLength)
1264 clipWidthPart = clippedDestRect.iBr.iX - startClip;
1266 clipIncStrch += KScanLineLength;
1271 aSourceBitmap->StretchScanLine(scanLineDes,TPoint(bitmapXStart,yCoord.iX),
1272 clipStrch,clipWidth,destWidth,aSourceRect.iTl.iX,
1273 srceWidth, KZeroPoint ,dispMode,aSourceBase,lineScanPos2);
1274 drawDevice->WriteLine(clippedDestRect.iTl.iX,yCoord.iY,clipWidth,scanLineBuffer,CGraphicsContext::EDrawModeXOR);
1276 TInt maskXStart = bitmapXStart % maskWidth;
1277 if(maskWidth < sourceBmpWidth)
1279 TPoint sourceDestXCoords(bitmapXStart,clippedDestRect.iTl.iX);
1280 xLine.Construct(TPoint(aSourceRect.iTl.iX,aDestRect.iTl.iX),
1281 TPoint(aSourceRect.iBr.iX,aDestRect.iBr.iX),TLinearDDA::ELeft);
1282 xLine.JumpToYCoord2(sourceDestXCoords.iX,sourceDestXCoords.iY);
1283 TPoint srcPixel(maskXStart,yCoord.iX % maskHeight);
1284 TInt spaceLeft = clipWidth;
1285 TPoint prevSourceDestXCoords(-1,-1);
1287 aMaskBitmap->GetScanLine(alphaBufferDes, TPoint(0,srcPixel.iY), maskWidth, EFalse, TPoint(0, 0), EGray256, aMaskBase, lineScanPosMask);
1289 // Loop to tile the mask
1291 if (sourceDestXCoords.iY != prevSourceDestXCoords.iY)
1293 if (sourceDestXCoords.iX != prevSourceDestXCoords.iX)
1295 srcPixel.iX = sourceDestXCoords.iX % maskWidth;
1296 if (srcPixel.iX < 0)
1297 srcPixel.iX += maskWidth;
1298 maskRgbValue = TRgb::Gray256((*alphaBuffer)[srcPixel.iX]);
1300 drawDevice->WriteRgb(sourceDestXCoords.iY,yCoord.iY,maskRgbValue,drawMode);
1303 prevSourceDestXCoords = sourceDestXCoords;
1304 xLine.SingleStep(sourceDestXCoords);
1305 } while (spaceLeft > 0);
1309 // No need to tile the mask
1310 aMaskBitmap->StretchScanLine(scanLineDes,TPoint(maskXStart,yCoord.iX % maskHeight),
1311 clipStrch,clipWidth,destWidth,aSourceRect.iTl.iX,
1312 srceWidth, KZeroPoint ,dispMode,aMaskBase,lineScanPosMask);
1313 drawDevice->WriteLine(clippedDestRect.iTl.iX,yCoord.iY,clipWidth,scanLineBuffer,drawMode);
1314 // Redo stretching of the aSourceBitmap scanline
1315 aSourceBitmap->StretchScanLine(scanLineDes,TPoint(bitmapXStart,yCoord.iX),
1316 clipStrch,clipWidth,destWidth,aSourceRect.iTl.iX,
1317 srceWidth, KZeroPoint ,dispMode,aSourceBase,lineScanPos2);
1320 if (yCoord.iY==clippedDestRect.iTl.iY)
1322 aSourceBitmap->SetCompressionBookmark(lineScanPos2,aSourceBase,NULL);
1323 aMaskBitmap->SetCompressionBookmark(lineScanPosMask,aMaskBase,NULL);
1326 drawDevice->WriteLine(clippedDestRect.iTl.iX,yCoord.iY,clipWidth,scanLineBuffer,CGraphicsContext::EDrawModeXOR);
1328 yLine.NextStep(yCoord);
1332 TInt CSwDirectGdiEngine::FastBlendInterface(const CBitwiseBitmap* aSource, const CBitwiseBitmap* aMask, MFastBlend*& aFastBlend) const
1334 #if defined(__ALLOW_FAST_BLEND_DISABLE__)
1335 if (iFastBlendDisabled)
1336 return(KErrNotSupported);
1338 if ((aSource && aSource->IsCompressed()) || (aMask && aMask->IsCompressed()))
1339 return(KErrNotSupported);
1340 TAny* interface=NULL;
1341 TInt ret= iDrawDevice->GetInterface(KFastBlendInterfaceID, interface);
1342 aFastBlend=(MFastBlend*)interface;
1347 Returns the pixel-format to be used when extracting a scan-line through CBitwiseBitmap::GetScanLine(), CBitwiseBitmap::GetVerticalScanLine(), and CBitwiseBitmap::StretchScanLine() for consumption by CFbsDrawDevice::WriteLine() and associated methods.
1349 @see CBitwiseBitmap::GetScanLine()
1350 @see CBitwiseBitmap::GetVerticalScanLine()
1351 @see CBitwiseBitmap::StretchScanLine()
1352 @see CFbsDrawDevice::WriteLine()
1355 TDisplayMode CSwDirectGdiEngine::ScanLineBufferDisplayMode(CFbsDrawDevice* aDrawDevice)
1357 return iDrawMode == DirectGdi::EDrawModeWriteAlpha ? aDrawDevice->DisplayMode() : aDrawDevice->ScanLineDisplayMode();