|
|
/*****************************************************************************\
* Smart 256 Colour Bank Manager * * Copyright (c) 1992 Microsoft Corporation \*****************************************************************************/
#include "driver.h"
/*****************************************************************************\
* pcoBankStart - Start the bank enumeration using the clip object. * * Used when the destination is the screen and we can't do the clipping * ourselves (as we can for blt's). \*****************************************************************************/
CLIPOBJ* pcoBankStart( PPDEV ppdev, RECTL* prclScans, SURFOBJ* pso, CLIPOBJ* pco) { LONG iTopScan = max(0, prclScans->top);
if (pco) { iTopScan = max(prclScans->top, pco->rclBounds.top); } // Adjust for those weird cases where we're asked to start enumerating
// below the bottom of the screen:
iTopScan = min(iTopScan, (LONG) ppdev->cyScreen - 1);
// Map in the bank:
if (iTopScan < ppdev->rcl1WindowClip.top || iTopScan >= ppdev->rcl1WindowClip.bottom) { ppdev->pfnBankControl(ppdev, iTopScan, JustifyTop); }
// Remember what the last scan is that we're going to, and
// make sure we only try to go as far as we need to. It could
// happen that when get a prclScans bigger than the screen:
ppdev->iLastScan = min(prclScans->bottom, (LONG) ppdev->cyScreen);
if (pco) { ppdev->iLastScan = min(ppdev->iLastScan, pco->rclBounds.bottom); }
pso->pvScan0 = ppdev->pvBitmapStart;
if (pco == NULL) { // The call may have come down to us as having no clipping, but
// we have to clip to the banks, so use our own clip object:
pco = ppdev->pcoNull; pco->rclBounds = ppdev->rcl1WindowClip; } else { // Save the engine's clip object data that we'll be tromping on:
ppdev->rclSaveBounds = pco->rclBounds; ppdev->iSaveDComplexity = pco->iDComplexity; ppdev->fjSaveOptions = pco->fjOptions;
// Let engine know it has to pay attention to the rclBounds of the
// clip object:
pco->fjOptions |= OC_BANK_CLIP;
if (pco->iDComplexity == DC_TRIVIAL) pco->iDComplexity = DC_RECT;
// Use the bank bounds if they are tighter than the existing
// bounds. We don't have to check the left case here because we
// know that ppdev->rcl1WindowClip.left == 0.
if (pco->rclBounds.top <= ppdev->rcl1WindowClip.top) pco->rclBounds.top = ppdev->rcl1WindowClip.top;
if (pco->rclBounds.right >= ppdev->rcl1WindowClip.right) pco->rclBounds.right = ppdev->rcl1WindowClip.right;
if (pco->rclBounds.bottom >= ppdev->rcl1WindowClip.bottom) pco->rclBounds.bottom = ppdev->rcl1WindowClip.bottom;
if ((pco->rclBounds.top >= pco->rclBounds.bottom) || (pco->rclBounds.left >= pco->rclBounds.right)) { // It's conceivable that we could get a situation where our
// draw rectangle is completely disjoint from the specified
// rectangle's rclBounds. Make sure we don't puke on our
// shoes:
pco->rclBounds.left = 0; pco->rclBounds.top = 0; pco->rclBounds.right = 0; pco->rclBounds.bottom = 0; ppdev->iLastScan = 0; } }
return(pco); }
/*****************************************************************************\
* bBankEnum - Continue the bank enumeration. \*****************************************************************************/
BOOL bBankEnum(PPDEV ppdev, SURFOBJ* pso, CLIPOBJ* pco) { // If we're on the first portion of a broken raster, get the next:
LONG yNewTop = ppdev->rcl1WindowClip.bottom;
DISPDBG((4, "bBankEnum: Start.\n"));
if (ppdev->flBank & BANK_BROKEN_RASTER1) ppdev->pfnBankNext(ppdev);
else if (ppdev->rcl1WindowClip.bottom < ppdev->iLastScan) ppdev->pfnBankControl(ppdev, yNewTop, JustifyTop);
else { // Okay, that was the last bank, so restore our structures:
if (pco != ppdev->pcoNull) { pco->rclBounds = ppdev->rclSaveBounds; pco->iDComplexity = ppdev->iSaveDComplexity; pco->fjOptions = ppdev->fjSaveOptions; }
return(FALSE); }
// Adjust the pvScan0 because we've moved the window to view
// a different area:
pso->pvScan0 = ppdev->pvBitmapStart;
if (pco == ppdev->pcoNull) { // If were given a NULL clip object originally, we don't have
// to worry about clipping to ppdev->rclSaveBounds:
pco->rclBounds.top = yNewTop; pco->rclBounds.left = ppdev->rcl1WindowClip.left; pco->rclBounds.bottom = ppdev->rcl1WindowClip.bottom; pco->rclBounds.right = ppdev->rcl1WindowClip.right; } else { // Use the bank bounds if they are tighter than the bounds
// we were originally given:
pco->rclBounds = ppdev->rclSaveBounds;
if (pco->rclBounds.top <= yNewTop) pco->rclBounds.top = yNewTop;
if (pco->rclBounds.left <= ppdev->rcl1WindowClip.left) pco->rclBounds.left = ppdev->rcl1WindowClip.left;
if (pco->rclBounds.right >= ppdev->rcl1WindowClip.right) pco->rclBounds.right = ppdev->rcl1WindowClip.right;
if (pco->rclBounds.bottom >= ppdev->rcl1WindowClip.bottom) pco->rclBounds.bottom = ppdev->rcl1WindowClip.bottom; }
DISPDBG((4, "bBankEnum: Leaving.\n"));
return(TRUE); }
/***************************************************************************\
* vBankStartBltSrc - Start the bank enumeration for when the screen is * the source. \***************************************************************************/
VOID vBankStartBltSrc( PPDEV ppdev, SURFOBJ* pso, POINTL* pptlSrc, RECTL* prclDest, POINTL* pptlNewSrc, RECTL* prclNewDest) { LONG xRightSrc; LONG yBottomSrc; LONG iTopScan = max(0, pptlSrc->y);
DISPDBG((4, "vBankStartBltSrc: Entering.\n"));
if (iTopScan >= (LONG) ppdev->cyScreen) { // In some instances we may be asked to start on a scan below the screen.
// Since we obviously won't be drawing anything, don't bother mapping in
// a bank:
return; }
// Map in the bank:
if (iTopScan < ppdev->rcl1WindowClip.top || iTopScan >= ppdev->rcl1WindowClip.bottom) { ppdev->pfnBankControl(ppdev, iTopScan, JustifyTop); }
if (ppdev->rcl1WindowClip.right <= pptlSrc->x) { // We have to watch out for those rare cases where we're starting
// on a broken raster and we won't be drawing on the first part:
ASSERTVGA(ppdev->flBank & BANK_BROKEN_RASTER1, "Weird start bounds");
ppdev->pfnBankNext(ppdev); }
pso->pvScan0 = ppdev->pvBitmapStart;
// Adjust the source:
pptlNewSrc->x = pptlSrc->x; pptlNewSrc->y = pptlSrc->y;
// Adjust the destination:
prclNewDest->left = prclDest->left; prclNewDest->top = prclDest->top;
yBottomSrc = pptlSrc->y + prclDest->bottom - prclDest->top; prclNewDest->bottom = min(ppdev->rcl1WindowClip.bottom, yBottomSrc); prclNewDest->bottom += prclDest->top - pptlSrc->y;
xRightSrc = pptlSrc->x + prclDest->right - prclDest->left; prclNewDest->right = min(ppdev->rcl1WindowClip.right, xRightSrc); prclNewDest->right += prclDest->left - pptlSrc->x;
DISPDBG((4, "vBankStartBltSrc: Leaving.\n")); }
/***************************************************************************\
* bBankEnumBltSrc - Continue the bank enumeration for when the screen is * the source. \***************************************************************************/
BOOL bBankEnumBltSrc( PPDEV ppdev, SURFOBJ* pso, POINTL* pptlSrc, RECTL* prclDest, POINTL* pptlNewSrc, RECTL* prclNewDest) { LONG xLeftSrc; LONG xRightSrc; LONG yBottomSrc;
LONG cx = prclDest->right - prclDest->left; LONG cy = prclDest->bottom - prclDest->top;
LONG dx; LONG dy;
LONG yBottom = min(pptlSrc->y + cy, (LONG) ppdev->cyScreen); LONG yNewTop = ppdev->rcl1WindowClip.bottom;
DISPDBG((4, "bBankEnumBltSrc: Entering.\n"));
if (ppdev->flBank & BANK_BROKEN_RASTER1) { ppdev->pfnBankNext(ppdev); if (ppdev->rcl1WindowClip.left >= pptlSrc->x + cx) { if (ppdev->rcl1WindowClip.bottom < yBottom) ppdev->pfnBankNext(ppdev); else { // We're done:
return(FALSE); } } } else if (yNewTop < yBottom) { ppdev->pfnBankControl(ppdev, yNewTop, JustifyTop); if (ppdev->rcl1WindowClip.right <= pptlSrc->x) { ASSERTVGA(ppdev->flBank & BANK_BROKEN_RASTER1, "Weird bounds"); ppdev->pfnBankNext(ppdev); } } else { // We're done:
return(FALSE); }
// Adjust the source:
pso->pvScan0 = ppdev->pvBitmapStart;
pptlNewSrc->x = max(ppdev->rcl1WindowClip.left, pptlSrc->x); pptlNewSrc->y = yNewTop;
// Adjust the destination:
dy = prclDest->top - pptlSrc->y; // y delta from source to dest
prclNewDest->top = yNewTop + dy;
yBottomSrc = pptlSrc->y + cy; prclNewDest->bottom = min(ppdev->rcl1WindowClip.bottom, yBottomSrc) + dy;
dx = prclDest->left - pptlSrc->x; // x delta from source to dest
xLeftSrc = pptlSrc->x; prclNewDest->left = pptlNewSrc->x + dx;
xRightSrc = pptlSrc->x + cx; prclNewDest->right = min(ppdev->rcl1WindowClip.right, xRightSrc) + dx;
DISPDBG((4, "bBankEnumBltSrc: Leaving.\n"));
return(TRUE); }
/***************************************************************************\
* vBankStartBltDest - Start the bank enumeration for when the screen is * the destination. \***************************************************************************/
VOID vBankStartBltDest( PPDEV ppdev, SURFOBJ* pso, POINTL* pptlSrc, RECTL* prclDest, POINTL* pptlNewSrc, RECTL* prclNewDest) { LONG iTopScan = max(0, prclDest->top);
DISPDBG((4, "vBankSTartBltDest: Entering.\n"));
if (iTopScan >= (LONG) ppdev->cyScreen) { // In some instances we may be asked to start on a scan below the screen.
// Since we obviously won't be drawing anything, don't bother mapping in
// a bank:
return; }
// Map in the bank:
if (iTopScan < ppdev->rcl1WindowClip.top || iTopScan >= ppdev->rcl1WindowClip.bottom) { ppdev->pfnBankControl(ppdev, iTopScan, JustifyTop); }
if (ppdev->rcl1WindowClip.right <= prclDest->left) { // We have to watch out for those rare cases where we're starting
// on a broken raster and we won't be drawing on the first part:
ASSERTVGA(ppdev->flBank & BANK_BROKEN_RASTER1, "Weird start bounds"); ppdev->pfnBankNext(ppdev); }
pso->pvScan0 = ppdev->pvBitmapStart;
// Adjust the destination:
prclNewDest->left = prclDest->left; prclNewDest->top = prclDest->top; prclNewDest->bottom = min(ppdev->rcl1WindowClip.bottom, prclDest->bottom); prclNewDest->right = min(ppdev->rcl1WindowClip.right, prclDest->right);
// Adjust the source if there is one:
if (pptlSrc != NULL) *pptlNewSrc = *pptlSrc;
DISPDBG((4, "vBankStartBltDest: Leaving.\n")); }
/***************************************************************************\
* bBankEnumBltDest - Continue the bank enumeration for when the screen is * the destination. \***************************************************************************/
BOOL bBankEnumBltDest( PPDEV ppdev, SURFOBJ* pso, POINTL* pptlSrc, RECTL* prclDest, POINTL* pptlNewSrc, RECTL* prclNewDest) { LONG yBottom = min(prclDest->bottom, (LONG) ppdev->cyScreen); LONG yNewTop = ppdev->rcl1WindowClip.bottom;
DISPDBG((4, "bBankEnumBltDest: Entering.\n"));
if (ppdev->flBank & BANK_BROKEN_RASTER1) { ppdev->pfnBankNext(ppdev); if (ppdev->rcl1WindowClip.left >= prclDest->right) { if (ppdev->rcl1WindowClip.bottom < yBottom) ppdev->pfnBankNext(ppdev); else { // We're done:
return(FALSE); } } } else if (yNewTop < yBottom) { ppdev->pfnBankControl(ppdev, yNewTop, JustifyTop); if (ppdev->rcl1WindowClip.right <= prclDest->left) { ASSERTVGA(ppdev->flBank & BANK_BROKEN_RASTER1, "Weird bounds"); ppdev->pfnBankNext(ppdev); } } else { // We're done:
return(FALSE); }
pso->pvScan0 = ppdev->pvBitmapStart;
// Adjust the destination:
prclNewDest->top = yNewTop; prclNewDest->left = max(ppdev->rcl1WindowClip.left, prclDest->left); prclNewDest->bottom = min(ppdev->rcl1WindowClip.bottom, prclDest->bottom); prclNewDest->right = min(ppdev->rcl1WindowClip.right, prclDest->right);
// Adjust the source if there is one:
if (pptlSrc != NULL) { pptlNewSrc->x = pptlSrc->x + (prclNewDest->left - prclDest->left); pptlNewSrc->y = pptlSrc->y + (prclNewDest->top - prclDest->top); }
DISPDBG((4, "bBankEnumBltDest: Leaving.\n"));
return(TRUE); }
|