Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1319 lines
43 KiB

/******************************Module*Header*******************************\
* Module Name: bitblt.c
*
* Banked Frame Buffer bitblit
*
* Copyright (c) 1992 Microsoft Corporation
*
\**************************************************************************/
#include "driver.h"
/************************************************************************\
* bIntersectRect
*
* Calculates the intersection between *prcSrc1 and *prcSrc2,
* returning the resulting rect in *prcDst. Returns TRUE if
* *prcSrc1 intersects *prcSrc2, FALSE otherwise. If there is no
* intersection, an empty rect is returned in *prcDst.
\************************************************************************/
static const RECTL rclEmpty = { 0, 0, 0, 0 };
BOOL bIntersectRect(
PRECTL prcDst,
PRECTL prcSrc1,
PRECTL prcSrc2)
{
prcDst->left = max(prcSrc1->left, prcSrc2->left);
prcDst->right = min(prcSrc1->right, prcSrc2->right);
// check for empty rect
if (prcDst->left < prcDst->right)
{
prcDst->top = max(prcSrc1->top, prcSrc2->top);
prcDst->bottom = min(prcSrc1->bottom, prcSrc2->bottom);
// check for empty rect
if (prcDst->top < prcDst->bottom)
return(TRUE); // not empty
}
// empty rect
*prcDst = rclEmpty;
return(FALSE);
}
/******************************Public*Routine******************************\
* BOOL bPuntScreenToScreenCopyBits(ppdev, pco, pxlo, prclDest, pptlSrc)
*
* Performs a screen-to-screen CopyBits entirely using an intermediate
* temporary buffer and GDI.
*
* We found that on most machines it was faster to have the engine copy
* the source to a buffer, then blit the buffer to the destination, than
* to have optimized ASM code that copies a word at a time. The reason?
* The engine does d-word moves, which are faster than word moves even
* going over the bus to a 16 bit display device.
*
* We could also write optimized ASM code that does d-word moves, but the
* win will be marginal, we're time constrained, we also need a routine
* like this to handle complex clip objects and palette translates, and
* most of the other times we can use planar copies for important things
* like scrolls, anyways.
*
\**************************************************************************/
BOOL bPuntScreenToScreenCopyBits(
PPDEV ppdev,
CLIPOBJ* pco,
XLATEOBJ* pxlo,
RECTL* prclDest,
POINTL* pptlSrc)
{
RECTL rclDest;
POINTL ptlSrc;
BOOL b = TRUE;
ULONG ulWidth;
LONG ulBurstSize;
LONG xLeft;
LONG xRight;
SURFOBJ* pso = ppdev->pSurfObj;
SURFOBJ* psoTmp = ppdev->psoTmp;
ULONG ulAlign;
PVOID savedpvScan0 = psoTmp->pvScan0;
LONG savedlDelta = psoTmp->lDelta;
USHORT savedfjBitmap = psoTmp->fjBitmap;
xLeft = prclDest->left;
xRight = prclDest->right;
if (pco && (pco->iDComplexity != DC_TRIVIAL))
{
xLeft = max(xLeft,pco->rclBounds.left);
xRight = min(xRight,pco->rclBounds.right);
}
ulAlign = xLeft & 3;
ulWidth = (((xRight + 3) & ~3) - (xLeft & ~3));
ulBurstSize = min((GLOBAL_BUFFER_SIZE/ulWidth),(ULONG)(prclDest->bottom-prclDest->top));
ASSERTVGA (ulBurstSize > 0, "VGA256: bPuntScreenToScreenBitBlt ulBurstSize <= 0\n");
// set up pso to use stack memory
psoTmp->pvScan0 = ppdev->pvTmpBuf;
psoTmp->lDelta = ulWidth;
psoTmp->fjBitmap |= BMF_TOPDOWN;
if (prclDest->top < pptlSrc->y)
{
////////////////////////////////////////////////////////////////
// Do a top-to-bottom copy:
////////////////////////////////////////////////////////////////
LONG ySrcBottom;
LONG yDestBottom;
LONG yDestTop = prclDest->top;
LONG ySrcTop = pptlSrc->y;
LONG ySrcLast = ySrcTop + (prclDest->bottom - prclDest->top);
if (ySrcTop < ppdev->rcl1WindowClip.top ||
ySrcTop >= ppdev->rcl1WindowClip.bottom)
{
ppdev->pfnBankControl(ppdev, ySrcTop, JustifyTop);
}
pso->pvScan0 = ppdev->pvBitmapStart;
while (TRUE)
{
// Copy some scans into the temporary buffer:
ySrcBottom = min(ySrcLast, ppdev->rcl1WindowClip.bottom);
ySrcBottom = min(ySrcBottom,ySrcTop+ulBurstSize);
ptlSrc.x = pptlSrc->x;
ptlSrc.y = ySrcTop;
rclDest.left = ulAlign; // make sure buffer is aligned to dst
rclDest.top = 0;
rclDest.right = xRight - xLeft + ulAlign;
rclDest.bottom = ySrcBottom - ySrcTop;
b &= EngCopyBits(psoTmp, pso, NULL, NULL, &rclDest, &ptlSrc);
yDestBottom = yDestTop + rclDest.bottom;
if (ppdev->rcl1WindowClip.top >= yDestBottom)
{
ppdev->pfnBankControl(ppdev, yDestBottom - 1, JustifyBottom);
pso->pvScan0 = ppdev->pvBitmapStart;
}
while (TRUE)
{
// Copy the temporary buffer into one or more destination
// banks:
LONG yThisTop;
LONG yThisBottom;
LONG yOffset;
yThisBottom = min(yDestBottom, ppdev->rcl1WindowClip.bottom);
yThisTop = max(yDestTop, ppdev->rcl1WindowClip.top);
yOffset = yThisTop - yDestTop;
ptlSrc.x = ulAlign;
ptlSrc.y = yOffset;
rclDest.left = xLeft;
rclDest.top = yThisTop;
rclDest.right = xRight;
rclDest.bottom = yThisBottom;
b &= EngCopyBits(pso, psoTmp, pco, pxlo, &rclDest, &ptlSrc);
if (yOffset == 0)
break;
if (ppdev->rcl1WindowClip.top >= yThisTop)
{
ppdev->pfnBankControl(ppdev, yThisTop - 1, JustifyBottom);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
if (ySrcBottom >= ySrcLast)
break;
yDestTop = yDestBottom;
ySrcTop = ySrcBottom;
if (ySrcTop >= ppdev->rcl1WindowClip.bottom)
{
ppdev->pfnBankControl(ppdev, ySrcTop, JustifyTop);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
}
else
{
////////////////////////////////////////////////////////////////
// Do a bottom-to-top copy:
////////////////////////////////////////////////////////////////
LONG ySrcTop;
LONG yDestTop;
LONG yDestBottom = prclDest->bottom;
LONG ySrcFirst = pptlSrc->y;
LONG ySrcBottom = ySrcFirst + (prclDest->bottom - prclDest->top);
if (ySrcBottom <= ppdev->rcl1WindowClip.top ||
ySrcBottom > ppdev->rcl1WindowClip.bottom)
{
ppdev->pfnBankControl(ppdev, ySrcBottom - 1, JustifyBottom);
}
pso->pvScan0 = ppdev->pvBitmapStart;
while (TRUE)
{
// Copy some scans into the temporary buffer:
ySrcTop = max(ySrcFirst, ppdev->rcl1WindowClip.top);
ySrcTop = max(ySrcTop,ySrcBottom-ulBurstSize);
ptlSrc.x = pptlSrc->x;
ptlSrc.y = ySrcTop;
rclDest.left = ulAlign;
rclDest.top = 0;
rclDest.right = xRight - xLeft + ulAlign;
rclDest.bottom = ySrcBottom - ySrcTop;
b &= EngCopyBits(psoTmp, pso, NULL, NULL, &rclDest, &ptlSrc);
yDestTop = yDestBottom - rclDest.bottom;
if (ppdev->rcl1WindowClip.bottom <= yDestTop)
{
ppdev->pfnBankControl(ppdev, yDestTop, JustifyTop);
pso->pvScan0 = ppdev->pvBitmapStart;
}
while (TRUE)
{
// Copy the temporary buffer into one or more destination
// banks:
LONG yThisTop;
LONG yThisBottom;
LONG yOffset;
yThisTop = max(yDestTop, ppdev->rcl1WindowClip.top);
yThisBottom = min(yDestBottom, ppdev->rcl1WindowClip.bottom);
yOffset = yThisTop - yDestTop;
ptlSrc.x = ulAlign;
ptlSrc.y = yOffset;
rclDest.left = xLeft;
rclDest.top = yThisTop;
rclDest.right = xRight;
rclDest.bottom = yThisBottom;
b &= EngCopyBits(pso, psoTmp, pco, pxlo, &rclDest, &ptlSrc);
if (yThisBottom == yDestBottom)
break;
if (ppdev->rcl1WindowClip.bottom <= yThisBottom)
{
ppdev->pfnBankControl(ppdev, yThisBottom, JustifyTop);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
if (ySrcTop <= ySrcFirst)
break;
yDestBottom = yDestTop;
ySrcBottom = ySrcTop;
if (ppdev->rcl1WindowClip.top >= ySrcBottom)
{
ppdev->pfnBankControl(ppdev, ySrcBottom - 1, JustifyBottom);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
}
// restore initial values to pso
ppdev->psoTmp->pvScan0 = savedpvScan0;
ppdev->psoTmp->lDelta = savedlDelta;
ppdev->psoTmp->fjBitmap = savedfjBitmap;
return(b);
}
/******************************Public*Routine******************************\
* BOOL bPuntScreenToScreenBitBlt(...)
*
* Performs a screen-to-screen BitBlt entirely using an intermediate temporary
* buffer and GDI.
*
* This function is basically a clone of bPuntScreenToScreenCopyBits,
* except that it can handle funky ROPs and stuff.
\**************************************************************************/
BOOL bPuntScreenToScreenBitBlt(
PPDEV ppdev,
SURFOBJ* psoMask,
CLIPOBJ* pco,
XLATEOBJ* pxlo,
RECTL* prclDest,
POINTL* pptlSrc,
POINTL* pptlMask,
BRUSHOBJ* pbo,
POINTL* pptlBrush,
ROP4 rop4)
{
RECTL rclDest; // Temporary destination rectangle
POINTL ptlSrc; // Temporary source point
POINTL ptlMask; // Temporary mask offset
POINTL ptlMaskAdjust; // Adjustment for mask offset
BOOL b = TRUE;
ULONG ulWidth;
LONG ulBurstSize;
LONG xLeft;
LONG xRight;
SURFOBJ* pso = ppdev->pSurfObj;
SURFOBJ* psoTmp = ppdev->psoTmp;
ULONG ulAlign;
PVOID savedpvScan0 = psoTmp->pvScan0;
LONG savedlDelta = psoTmp->lDelta;
USHORT savedfjBitmap = psoTmp->fjBitmap;
xLeft = prclDest->left;
xRight = prclDest->right;
if (pco && (pco->iDComplexity != DC_TRIVIAL))
{
xLeft = max(xLeft,pco->rclBounds.left);
xRight = min(xRight,pco->rclBounds.right);
}
if (psoMask != NULL)
{
ptlMaskAdjust.x = xLeft - pptlMask->x;
ptlMaskAdjust.y = prclDest->top - pptlMask->y;
}
ulAlign = xLeft & 3;
ulWidth = (((xRight + 3) & ~3) - (xLeft & ~3));
ulBurstSize = min((GLOBAL_BUFFER_SIZE/ulWidth),(ULONG)(prclDest->bottom-prclDest->top));
ASSERTVGA (ulBurstSize > 0, "bPuntScreenToScreenBitBlt ulBurstSize <= 0\n");
// set up pso to use stack memory
psoTmp->pvScan0 = ppdev->pvTmpBuf;
psoTmp->lDelta = ulWidth;
psoTmp->fjBitmap |= BMF_TOPDOWN;
if (prclDest->top < pptlSrc->y)
{
////////////////////////////////////////////////////////////////
// Do a top-to-bottom copy:
////////////////////////////////////////////////////////////////
LONG ySrcBottom;
LONG yDestBottom;
LONG yDestTop = prclDest->top;
LONG ySrcTop = pptlSrc->y;
LONG ySrcLast = ySrcTop + (prclDest->bottom - prclDest->top);
if (ySrcTop < ppdev->rcl1WindowClip.top ||
ySrcTop >= ppdev->rcl1WindowClip.bottom)
{
ppdev->pfnBankControl(ppdev, ySrcTop, JustifyTop);
}
pso->pvScan0 = ppdev->pvBitmapStart;
while (TRUE)
{
// Copy some scans into the temporary buffer:
ySrcBottom = min(ySrcLast, ppdev->rcl1WindowClip.bottom);
ySrcBottom = min(ySrcBottom,ySrcTop+ulBurstSize);
ptlSrc.x = pptlSrc->x;
ptlSrc.y = ySrcTop;
rclDest.left = ulAlign;
rclDest.top = 0;
rclDest.right = xRight - xLeft + ulAlign;
rclDest.bottom = ySrcBottom - ySrcTop;
b &= EngCopyBits(psoTmp, pso, NULL, NULL, &rclDest, &ptlSrc);
yDestBottom = yDestTop + rclDest.bottom;
if (ppdev->rcl1WindowClip.top >= yDestBottom)
{
ppdev->pfnBankControl(ppdev, yDestBottom - 1, JustifyBottom);
pso->pvScan0 = ppdev->pvBitmapStart;
}
while (TRUE)
{
// Copy the temporary buffer into one or more destination
// banks:
LONG yThisTop;
LONG yThisBottom;
LONG yOffset;
yThisBottom = min(yDestBottom, ppdev->rcl1WindowClip.bottom);
yThisTop = max(yDestTop, ppdev->rcl1WindowClip.top);
yOffset = yThisTop - yDestTop;
ptlSrc.x = ulAlign;
ptlSrc.y = yOffset;
rclDest.left = xLeft;
rclDest.top = yThisTop;
rclDest.right = xRight;
rclDest.bottom = yThisBottom;
ptlMask.x = rclDest.left - ptlMaskAdjust.x;
ptlMask.y = rclDest.top - ptlMaskAdjust.y;
b &= EngBitBlt(pso, psoTmp, psoMask, pco, pxlo, &rclDest,
&ptlSrc, &ptlMask, pbo, pptlBrush, rop4);
if (yOffset == 0)
break;
if (ppdev->rcl1WindowClip.top >= yThisTop)
{
ppdev->pfnBankControl(ppdev, yThisTop - 1, JustifyBottom);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
if (ySrcBottom >= ySrcLast)
break;
yDestTop = yDestBottom;
ySrcTop = ySrcBottom;
if (ySrcTop >= ppdev->rcl1WindowClip.bottom)
{
ppdev->pfnBankControl(ppdev, ySrcTop, JustifyTop);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
}
else
{
////////////////////////////////////////////////////////////////
// Do a bottom-to-top copy:
////////////////////////////////////////////////////////////////
LONG ySrcTop;
LONG yDestTop;
LONG yDestBottom = prclDest->bottom;
LONG ySrcFirst = pptlSrc->y;
LONG ySrcBottom = ySrcFirst + (prclDest->bottom - prclDest->top);
if (ySrcBottom <= ppdev->rcl1WindowClip.top ||
ySrcBottom > ppdev->rcl1WindowClip.bottom)
{
ppdev->pfnBankControl(ppdev, ySrcBottom - 1, JustifyBottom);
}
pso->pvScan0 = ppdev->pvBitmapStart;
while (TRUE)
{
// Copy some scans into the temporary buffer:
ySrcTop = max(ySrcFirst, ppdev->rcl1WindowClip.top);
ySrcTop = max(ySrcTop,ySrcBottom-ulBurstSize);
ptlSrc.x = pptlSrc->x;
ptlSrc.y = ySrcTop;
rclDest.left = ulAlign;
rclDest.top = 0;
rclDest.right = xRight - xLeft + ulAlign;
rclDest.bottom = ySrcBottom - ySrcTop;
b &= EngCopyBits(psoTmp, pso, NULL, NULL, &rclDest, &ptlSrc);
yDestTop = yDestBottom - rclDest.bottom;
if (ppdev->rcl1WindowClip.bottom <= yDestTop)
{
ppdev->pfnBankControl(ppdev, yDestTop, JustifyTop);
pso->pvScan0 = ppdev->pvBitmapStart;
}
while (TRUE)
{
// Copy the temporary buffer into one or more destination
// banks:
LONG yThisTop;
LONG yThisBottom;
LONG yOffset;
yThisTop = max(yDestTop, ppdev->rcl1WindowClip.top);
yThisBottom = min(yDestBottom, ppdev->rcl1WindowClip.bottom);
yOffset = yThisTop - yDestTop;
ptlSrc.x = ulAlign;
ptlSrc.y = yOffset;
rclDest.left = xLeft;
rclDest.top = yThisTop;
rclDest.right = xRight;
rclDest.bottom = yThisBottom;
ptlMask.x = rclDest.left - ptlMaskAdjust.x;
ptlMask.y = rclDest.top - ptlMaskAdjust.y;
b &= EngBitBlt(pso, psoTmp, psoMask, pco, pxlo, &rclDest,
&ptlSrc, &ptlMask, pbo, pptlBrush, rop4);
if (yThisBottom == yDestBottom)
break;
if (ppdev->rcl1WindowClip.bottom <= yThisBottom)
{
ppdev->pfnBankControl(ppdev, yThisBottom, JustifyTop);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
if (ySrcTop <= ySrcFirst)
break;
yDestBottom = yDestTop;
ySrcBottom = ySrcTop;
if (ppdev->rcl1WindowClip.top >= ySrcBottom)
{
ppdev->pfnBankControl(ppdev, ySrcBottom - 1, JustifyBottom);
pso->pvScan0 = ppdev->pvBitmapStart;
}
}
}
// restore initial values to pso
ppdev->psoTmp->pvScan0 = savedpvScan0;
ppdev->psoTmp->lDelta = savedlDelta;
ppdev->psoTmp->fjBitmap = savedfjBitmap;
return(b);
}
/******************************Public*Data*********************************\
* ROP to mix translation table
*
* Table to translate ternary raster ops to mixes (binary raster ops). Ternary
* raster ops that can't be translated to mixes are translated to 0 (0 is not
* a valid mix).
*
\**************************************************************************/
UCHAR jRop3ToMix[256] = {
R2_BLACK, 0, 0, 0, 0, R2_NOTMERGEPEN, 0, 0,
0, 0, R2_MASKNOTPEN, 0, 0, 0, 0, R2_NOTCOPYPEN,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
R2_MASKPENNOT, 0, 0, 0, 0, R2_NOT, 0, 0,
0, 0, R2_XORPEN, 0, 0, 0, 0, R2_NOTMASKPEN,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
R2_MASKPEN, 0, 0, 0, 0, R2_NOTXORPEN, 0, 0,
0, 0, R2_NOP, 0, 0, 0, 0, R2_MERGENOTPEN,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0,
R2_COPYPEN, 0, 0, 0, 0, R2_MERGEPENNOT, 0, 0,
0, 0, R2_MERGEPEN, 0, 0, 0, 0, R2_WHITE
};
/******************************Public*Routine******************************\
* BOOL DrvBitBlt(psoDest, psoSrc, psoMask, pco, pxlo, prclDest, pptlSrc,
* pptlMask, pbo, pptlBrush, rop4)
*
* This routine will handle any blit. Perhaps glacially, but it will be
* handled.
\**************************************************************************/
BOOL DrvBitBlt(
SURFOBJ* psoDest,
SURFOBJ* psoSrc,
SURFOBJ* psoMask,
CLIPOBJ* pco,
XLATEOBJ* pxlo,
RECTL* prclDest,
POINTL* pptlSrc,
POINTL* pptlMask,
BRUSHOBJ* pbo,
POINTL* pptlBrush,
ROP4 rop4)
{
BOOL b;
POINTL ptlSrc;
RECTL rclDest;
PPDEV ppdev;
SURFOBJ* pso;
MIX mix; // Mix, when solid fill performed
BYTE jClipping;
RECTL rclTmp;
POINTL ptlTmp;
BBENUM bben; // Clip enumerator
BOOL bMore; // Clip continuation flag
POINTL ptlMask; // Temporary mask for engine call-backs
POINTL ptlMaskAdjust; // Adjustment for mask
INT iCopyDir;
RBRUSH_COLOR rbc; // Pointer to RBRUSH or iSolidColor value
PFNFILL pfnFill = vTrgBlt; // Pointer to appropriate fill routine
// (solid color by default)
DISPDBG((3, "DrvBitBlt: Entering."));
// Set up the clipping type
if (pco == (CLIPOBJ *) NULL) {
// No CLIPOBJ provided, so we don't have to worry about clipping
jClipping = DC_TRIVIAL;
} else {
// Use the CLIPOBJ-provided clipping
jClipping = pco->iDComplexity;
}
// Handle solid fills to the VGA surface with special-case code if planar
// mode is supported.
// LATER handle non-planar also
if (psoDest->iType == STYPE_DEVICE) {
// Destination is the VGA surface
// Masked cases must be handled differently
if ((((PPDEV) psoDest->dhsurf)->fl & DRIVER_PLANAR_CAPABLE) &&
((rop4 & 0xFF) == ((rop4 >> 8) & 0xFF))) {
// Special-case static code for no-mask cases
// Calculate mix from ROP if possible (not possible if it's truly a
// ternary rop or a real rop4, but we can treat all pure binary
// rops as mixes rather than rop4s)
mix = jRop3ToMix[rop4 & 0xFF];
switch (mix) {
case R2_MASKNOTPEN:
case R2_NOTCOPYPEN:
case R2_XORPEN:
case R2_MASKPEN:
case R2_NOTXORPEN:
case R2_MERGENOTPEN:
case R2_COPYPEN:
case R2_MERGEPEN:
case R2_NOTMERGEPEN:
case R2_MASKPENNOT:
case R2_NOTMASKPEN:
case R2_MERGEPENNOT:
// vTrgBlt can only handle solid color fills
if (pbo->iSolidColor != 0xffffffff)
{
rbc.iSolidColor = pbo->iSolidColor;
}
else
{
rbc.prb = (RBRUSH*) pbo->pvRbrush;
if (rbc.prb == NULL)
{
rbc.prb = (RBRUSH*) BRUSHOBJ_pvGetRbrush(pbo);
if (rbc.prb == NULL)
{
// If we haven't realized the brush, punt the call
// to the engine:
break;
}
}
if (!(rbc.prb->fl & RBRUSH_BLACKWHITE) &&
(mix != R2_COPYPEN))
{
// Only black/white brushes can handle ROPs other
// than COPYPEN:
break;
}
if (rbc.prb->fl & RBRUSH_NCOLOR)
pfnFill = vColorPat;
else
pfnFill = vMonoPat;
}
// Rops that are implicit solid colors
case R2_NOT:
case R2_WHITE:
case R2_BLACK:
// We can do a special-case solid fill
switch(jClipping) {
case DC_TRIVIAL:
// Just fill the rectangle:
(*pfnFill)((PPDEV) psoDest->dhsurf, 1,
prclDest, mix, rbc, pptlBrush);
break;
case DC_RECT:
// Clip the solid fill to the clip rectangle
if (!bIntersectRect(&rclTmp, prclDest,
&pco->rclBounds))
return(TRUE);
// Fill the clipped rectangle
(*pfnFill)((PPDEV) psoDest->dhsurf, 1,
&rclTmp, mix, rbc, pptlBrush);
break;
case DC_COMPLEX:
ppdev = (PPDEV) psoDest->dhsurf;
CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES,
CD_ANY, BB_RECT_LIMIT);
do {
bMore = CLIPOBJ_bEnum(pco,(ULONG) sizeof(bben),
(PVOID) &bben);
if (bben.c > 0)
{
RECTL* prclEnd = &bben.arcl[bben.c];
RECTL* prcl = &bben.arcl[0];
do {
bIntersectRect(prcl, prcl, prclDest);
prcl++;
} while (prcl < prclEnd);
(*pfnFill)(ppdev, bben.c, bben.arcl,
mix, rbc, pptlBrush);
}
} while(bMore);
}
case R2_NOP:
return(TRUE);
default:
break;
}
}
}
// Get the correct surface object for the target and the source
if (psoDest->iType == STYPE_DEVICE) {
if ((psoSrc != NULL) && (psoSrc->iType == STYPE_DEVICE)) {
////////////////////////////////////////////////////////////////
// BitBlt screen-to-screen:
////////////////////////////////////////////////////////////////
ppdev = (PPDEV) psoDest->dhsurf;
// See if we can do a simple CopyBits:
if (rop4 == 0x0000CCCC)
{
ppdev = (PPDEV) psoDest->dhsurf;
// We can handle quadpixel-aligned screen-to-screen blts with
// no translation:
if ((((pptlSrc->x ^ prclDest->left) & 3) == 0) &&
(ppdev->fl & DRIVER_PLANAR_CAPABLE) &&
((pxlo == NULL) || (pxlo->flXlate & XO_TRIVIAL)))
{
switch(jClipping)
{
case DC_TRIVIAL:
vPlanarCopyBits(ppdev, prclDest, pptlSrc);
return(TRUE);
case DC_RECT:
// Clip the target rectangle to the clip rectangle:
if (!bIntersectRect(&rclTmp, prclDest, &pco->rclBounds))
{
DISPDBG((0, "DrvBitBlt: Nothing to draw."));
return(TRUE);
}
ptlTmp.x = pptlSrc->x + rclTmp.left - prclDest->left;
ptlTmp.y = pptlSrc->y + rclTmp.top - prclDest->top;
vPlanarCopyBits(ppdev, &rclTmp, &ptlTmp);
return(TRUE);
case DC_COMPLEX:
if (pptlSrc->y >= prclDest->top)
{
if (pptlSrc->x >= prclDest->left)
iCopyDir = CD_RIGHTDOWN;
else
iCopyDir = CD_LEFTDOWN;
}
else
{
if (pptlSrc->x >= prclDest->left)
iCopyDir = CD_RIGHTUP;
else
iCopyDir = CD_LEFTUP;
}
CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES, iCopyDir, 0);
do {
RECTL* prcl;
RECTL* prclEnd;
bMore = CLIPOBJ_bEnum(pco, (ULONG) sizeof(bben),
(PVOID) &bben);
prclEnd = &bben.arcl[bben.c];
for (prcl = bben.arcl; prcl < prclEnd; prcl++)
{
if (bIntersectRect(prcl, prclDest, prcl))
{
ptlTmp.x = pptlSrc->x + prcl->left - prclDest->left;
ptlTmp.y = pptlSrc->y + prcl->top - prclDest->top;
vPlanarCopyBits(ppdev, prcl, &ptlTmp);
}
}
} while (bMore);
return(TRUE);
}
}
// Can't handle in hardware, so punt:
return(bPuntScreenToScreenCopyBits(ppdev,
pco,
pxlo,
prclDest,
pptlSrc));
}
// It's more complicated than a CopyBits, so punt it:
return(bPuntScreenToScreenBitBlt(ppdev,
psoMask,
pco,
pxlo,
prclDest,
pptlSrc,
pptlMask,
pbo,
pptlBrush,
rop4));
}
////////////////////////////////////////////////////////////////
// BitBlt to screen:
////////////////////////////////////////////////////////////////
ppdev = (PPDEV) psoDest->dhsurf;
if ((rop4 == 0x0000CCCC) &&
(psoSrc->iBitmapFormat == BMF_8BPP) &&
((pxlo == NULL) || (pxlo->flXlate & XO_TRIVIAL)))
{
// We have special code for the common 8bpp from memory to screen
// with no ROPs:
switch(jClipping)
{
case DC_TRIVIAL:
vSrcCopy8bpp(ppdev, prclDest, pptlSrc,
psoSrc->lDelta, psoSrc->pvScan0);
return(TRUE);
case DC_RECT:
// Clip the blt to the clip rectangle
if(bIntersectRect(&rclTmp, prclDest, &pco->rclBounds)){
ptlTmp.x = pptlSrc->x + rclTmp.left - prclDest->left;
ptlTmp.y = pptlSrc->y + rclTmp.top - prclDest->top;
vSrcCopy8bpp(ppdev, &rclTmp, &ptlTmp,
psoSrc->lDelta, psoSrc->pvScan0);
}
return(TRUE);
case DC_COMPLEX:
CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES,
CD_ANY, BB_RECT_LIMIT);
do {
bMore = CLIPOBJ_bEnum(pco,(ULONG) sizeof(bben),
(PVOID) &bben);
if (bben.c > 0)
{
RECTL* prclEnd = &bben.arcl[bben.c];
RECTL* prcl = &bben.arcl[0];
do {
if (bIntersectRect(prcl, prcl, prclDest)){
ptlTmp.x = pptlSrc->x + prcl->left
- prclDest->left;
ptlTmp.y = pptlSrc->y + prcl->top
- prclDest->top;
vSrcCopy8bpp(ppdev, prcl, &ptlTmp,
psoSrc->lDelta, psoSrc->pvScan0);
}
prcl++;
} while (prcl < prclEnd);
}
} while(bMore);
return(TRUE);
}
}
// Punt the memory-to-screen call back to the engine:
if (psoMask != NULL)
{
ptlMaskAdjust.x = prclDest->left - pptlMask->x;
ptlMaskAdjust.y = prclDest->top - pptlMask->y;
}
pso = ppdev->pSurfObj;
vBankStartBltDest(ppdev, pso, pptlSrc, prclDest, &ptlSrc, &rclDest);
do {
ptlMask.x = rclDest.left - ptlMaskAdjust.x;
ptlMask.y = rclDest.top - ptlMaskAdjust.y;
b = EngBitBlt(pso,
psoSrc,
psoMask,
pco,
pxlo,
&rclDest,
&ptlSrc,
&ptlMask,
pbo,
pptlBrush,
rop4);
} while (b && bBankEnumBltDest(ppdev, pso, pptlSrc, prclDest,
&ptlSrc, &rclDest));
return(b);
}
else if ((psoSrc != NULL) && (psoSrc->iType == STYPE_DEVICE))
{
////////////////////////////////////////////////////////////////
// BitBlt from screen:
////////////////////////////////////////////////////////////////
if (psoMask != NULL)
{
ptlMaskAdjust.x = prclDest->left - pptlMask->x;
ptlMaskAdjust.y = prclDest->top - pptlMask->y;
}
ppdev = (PPDEV) psoSrc->dhsurf;
pso = ppdev->pSurfObj;
vBankStartBltSrc(ppdev, pso, pptlSrc, prclDest, &ptlSrc, &rclDest);
do {
ptlMask.x = rclDest.left - ptlMaskAdjust.x;
ptlMask.y = rclDest.top - ptlMaskAdjust.y;
b = EngBitBlt(psoDest,
pso,
psoMask,
pco,
pxlo,
&rclDest,
&ptlSrc,
&ptlMask,
pbo,
pptlBrush,
rop4);
} while (b && bBankEnumBltSrc(ppdev, pso, pptlSrc, prclDest,
&ptlSrc, &rclDest));
return(b);
}
RIP("Got a funky format?");
return(FALSE);
}
/***************************************************************************\
* DrvCopyBits
\***************************************************************************/
BOOL DrvCopyBits(
SURFOBJ* psoDest,
SURFOBJ* psoSrc,
CLIPOBJ* pco,
XLATEOBJ* pxlo,
RECTL* prclDest,
POINTL* pptlSrc)
{
BOOL b;
POINTL ptlSrc;
RECTL rclDest;
PPDEV ppdev;
SURFOBJ* pso;
BBENUM bben;
BOOL bMore;
BYTE jClipping;
POINTL ptlTmp;
RECTL rclTmp;
INT iCopyDir;
// Get the correct surface object for the target and the source
if (psoDest->iType == STYPE_DEVICE)
{
// We have to special case screen-to-screen operations:
if ((psoSrc != NULL) && (psoSrc->iType == STYPE_DEVICE))
{
////////////////////////////////////////////////////////////////
// CopyBits screen-to-screen:
////////////////////////////////////////////////////////////////
ppdev = (PPDEV) psoDest->dhsurf;
// We check to see if we can do a planar copy, because usually
// it will be faster. But the hardware has to be capable of
// doing it, and the source and destination must be 4-pel
// aligned.
if ((((pptlSrc->x ^ prclDest->left) & 3) == 0) &&
(ppdev->fl & DRIVER_PLANAR_CAPABLE) &&
((pxlo == NULL) || (pxlo->flXlate & XO_TRIVIAL)))
{
jClipping = (pco != NULL) ? pco->iDComplexity : DC_TRIVIAL;
switch(jClipping)
{
case DC_TRIVIAL:
vPlanarCopyBits(ppdev, prclDest, pptlSrc);
return(TRUE);
case DC_RECT:
// Clip the target rectangle to the clip rectangle:
if (!bIntersectRect(&rclTmp, prclDest, &pco->rclBounds))
{
DISPDBG((0, "DrvCopyBits: Nothing to draw."));
return(TRUE);
}
ptlTmp.x = pptlSrc->x + rclTmp.left - prclDest->left;
ptlTmp.y = pptlSrc->y + rclTmp.top - prclDest->top;
vPlanarCopyBits(ppdev, &rclTmp, &ptlTmp);
return(TRUE);
case DC_COMPLEX:
if (pptlSrc->y >= prclDest->top)
{
if (pptlSrc->x >= prclDest->left)
iCopyDir = CD_RIGHTDOWN;
else
iCopyDir = CD_LEFTDOWN;
}
else
{
if (pptlSrc->x >= prclDest->left)
iCopyDir = CD_RIGHTUP;
else
iCopyDir = CD_LEFTUP;
}
CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES, iCopyDir, 0);
do {
RECTL* prcl;
RECTL* prclEnd;
bMore = CLIPOBJ_bEnum(pco, (ULONG) sizeof(bben),
(PVOID) &bben);
prclEnd = &bben.arcl[bben.c];
for (prcl = bben.arcl; prcl < prclEnd; prcl++)
{
if (bIntersectRect(prcl, prclDest, prcl))
{
ptlTmp.x = pptlSrc->x + prcl->left - prclDest->left;
ptlTmp.y = pptlSrc->y + prcl->top - prclDest->top;
vPlanarCopyBits(ppdev, prcl, &ptlTmp);
}
}
} while (bMore);
return(TRUE);
}
}
return(bPuntScreenToScreenCopyBits(ppdev,
pco,
pxlo,
prclDest,
pptlSrc));
}
////////////////////////////////////////////////////////////////
// CopyBits to screen:
////////////////////////////////////////////////////////////////
ppdev = (PPDEV) psoDest->dhsurf;
if ((psoSrc->iBitmapFormat == BMF_8BPP) &&
((pxlo == NULL) || (pxlo->flXlate & XO_TRIVIAL)))
{
// We have special code for the common 8bpp from memory to screen
// with no ROPs:
jClipping = (pco != NULL) ? pco->iDComplexity : DC_TRIVIAL;
switch(jClipping)
{
case DC_TRIVIAL:
vSrcCopy8bpp(ppdev, prclDest, pptlSrc,
psoSrc->lDelta, psoSrc->pvScan0);
return(TRUE);
case DC_RECT:
// Clip the blt to the clip rectangle
if( bIntersectRect(&rclTmp, prclDest, &pco->rclBounds)){
ptlTmp.x = pptlSrc->x + rclTmp.left - prclDest->left;
ptlTmp.y = pptlSrc->y + rclTmp.top - prclDest->top;
vSrcCopy8bpp(ppdev, &rclTmp, &ptlTmp,
psoSrc->lDelta, psoSrc->pvScan0);
}
return(TRUE);
case DC_COMPLEX:
CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES,
CD_ANY, BB_RECT_LIMIT);
do {
bMore = CLIPOBJ_bEnum(pco,(ULONG) sizeof(bben),
(PVOID) &bben);
if (bben.c > 0)
{
RECTL* prclEnd = &bben.arcl[bben.c];
RECTL* prcl = &bben.arcl[0];
do {
ASSERTVGA((prcl->bottom - prcl->top) > 0,
"DrvCopyBits: enum rect height <= 0\n");
ASSERTVGA((prcl->right - prcl->left) > 0,
"DrvCopyBits: enum rect width <= 0\n");
if(bIntersectRect(prcl, prcl, prclDest)){
ptlTmp.x = pptlSrc->x + prcl->left
- prclDest->left;
ptlTmp.y = pptlSrc->y + prcl->top
- prclDest->top;
vSrcCopy8bpp(ppdev, prcl, &ptlTmp,
psoSrc->lDelta, psoSrc->pvScan0);
}
prcl++;
} while (prcl < prclEnd);
}
} while(bMore);
return(TRUE);
}
}
// Fall back to the engine:
pso = ppdev->pSurfObj;
vBankStartBltDest(ppdev, pso, pptlSrc, prclDest, &ptlSrc, &rclDest);
do {
b = EngCopyBits(pso,
psoSrc,
pco,
pxlo,
&rclDest,
&ptlSrc);
} while (b && bBankEnumBltDest(ppdev, pso, pptlSrc, prclDest,
&ptlSrc, &rclDest));
return(b);
}
else if ((psoSrc != NULL) && (psoSrc->iType == STYPE_DEVICE))
{
////////////////////////////////////////////////////////////////
// CopyBits from screen:
////////////////////////////////////////////////////////////////
ppdev = (PPDEV) psoSrc->dhsurf;
pso = ppdev->pSurfObj;
vBankStartBltSrc(ppdev, pso, pptlSrc, prclDest, &ptlSrc, &rclDest);
do {
b = EngCopyBits(psoDest,
pso,
pco,
pxlo,
&rclDest,
&ptlSrc);
} while (b && bBankEnumBltSrc(ppdev, pso, pptlSrc, prclDest,
&ptlSrc, &rclDest));
return(b);
}
/* we should never be here */
return FALSE;
}