429 lines
12 KiB
C++
429 lines
12 KiB
C++
#include "precomp.h"
|
|
|
|
|
|
//
|
|
// SSI.CPP
|
|
// Save Screenbits Interceptor
|
|
//
|
|
// Copyright(c) Microsoft 1997-
|
|
//
|
|
|
|
#define MLZ_FILE_ZONE ZONE_CORE
|
|
|
|
|
|
|
|
|
|
//
|
|
// SSI_HostStarting()
|
|
//
|
|
// Called when we start to host, figures out the max save bitmap bits size
|
|
// etc.
|
|
//
|
|
BOOL ASHost::SSI_HostStarting(void)
|
|
{
|
|
DebugEntry(ASHost::SSI_HostStarting);
|
|
|
|
m_pShare->SSI_RecalcCaps(TRUE);
|
|
|
|
DebugExitBOOL(ASHost::SSI_HostStarting, TRUE);
|
|
return(TRUE);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// SSI_ViewStarted()
|
|
//
|
|
// Called when someone we are viewing has started to host. Creates save bits
|
|
// bitmap for them.
|
|
//
|
|
BOOL ASShare::SSI_ViewStarting(ASPerson * pasPerson)
|
|
{
|
|
BOOL rc = FALSE;
|
|
HDC hdcScreen = NULL;
|
|
|
|
DebugEntry(ASShare::SSI_ViewStarting);
|
|
|
|
ValidateView(pasPerson);
|
|
|
|
//
|
|
// ASSERT that this persons' variables are clear.
|
|
//
|
|
ASSERT(pasPerson->m_pView->m_ssiBitmapHeight == 0);
|
|
ASSERT(pasPerson->m_pView->m_ssiBitmap == NULL);
|
|
ASSERT(pasPerson->m_pView->m_ssiOldBitmap == NULL);
|
|
|
|
//
|
|
// Does this person support savebits?
|
|
//
|
|
if (!pasPerson->cpcCaps.orders.capsSaveBitmapSize)
|
|
{
|
|
// No receive SSI capability, bail out now.
|
|
rc = TRUE;
|
|
DC_QUIT;
|
|
}
|
|
|
|
//
|
|
// Store the height of this host's bitmap.
|
|
//
|
|
pasPerson->m_pView->m_ssiBitmapHeight = (int)
|
|
(pasPerson->cpcCaps.orders.capsSaveBitmapSize / TSHR_SSI_BITMAP_WIDTH);
|
|
|
|
//
|
|
// If the calculated bitmap size is not exactly divisible by the bitmap
|
|
// width increase the bitmap height to fit in the partial row.
|
|
//
|
|
if (pasPerson->cpcCaps.orders.capsSaveBitmapSize % TSHR_SSI_BITMAP_WIDTH)
|
|
{
|
|
pasPerson->m_pView->m_ssiBitmapHeight += pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity;
|
|
}
|
|
|
|
TRACE_OUT(("Person [%d] SSI Bitmap height %d",
|
|
pasPerson->mcsID,
|
|
pasPerson->m_pView->m_ssiBitmapHeight));
|
|
|
|
//
|
|
// Create this host's save screen bitmap.
|
|
//
|
|
hdcScreen = GetDC(NULL);
|
|
if (hdcScreen == NULL)
|
|
{
|
|
ERROR_OUT(( "Failed to get screen surface"));
|
|
DC_QUIT;
|
|
}
|
|
|
|
//
|
|
// Create the save screen bitmap DC.
|
|
//
|
|
ASSERT(pasPerson->m_pView->m_ssiDC == NULL);
|
|
pasPerson->m_pView->m_ssiDC = CreateCompatibleDC(hdcScreen);
|
|
if (!pasPerson->m_pView->m_ssiDC)
|
|
{
|
|
ERROR_OUT(("Failed to create SSI DC"));
|
|
DC_QUIT;
|
|
}
|
|
|
|
//
|
|
// Create the save screen bitmap.
|
|
//
|
|
ASSERT(pasPerson->m_pView->m_ssiBitmap == NULL);
|
|
pasPerson->m_pView->m_ssiBitmap = CreateCompatibleBitmap(hdcScreen,
|
|
TSHR_SSI_BITMAP_WIDTH, pasPerson->m_pView->m_ssiBitmapHeight);
|
|
if (!pasPerson->m_pView->m_ssiBitmap)
|
|
{
|
|
ERROR_OUT(("SSI_ViewStarting: can't create bitmap for person %x",
|
|
pasPerson->mcsID));
|
|
DC_QUIT;
|
|
}
|
|
|
|
//
|
|
// Select the save screen bitmap into the DC
|
|
//
|
|
ASSERT(pasPerson->m_pView->m_ssiOldBitmap == NULL);
|
|
pasPerson->m_pView->m_ssiOldBitmap = SelectBitmap(pasPerson->m_pView->m_ssiDC,
|
|
pasPerson->m_pView->m_ssiBitmap);
|
|
|
|
rc = TRUE;
|
|
|
|
DC_EXIT_POINT:
|
|
|
|
if (hdcScreen != NULL)
|
|
{
|
|
ReleaseDC(NULL, hdcScreen);
|
|
}
|
|
|
|
DebugExitBOOL(ASShare::SSI_ViewStarting, rc);
|
|
return(rc);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// SSI_ViewEnded()
|
|
//
|
|
// Called when someone we are viewing has stopped hosting, so we can clean
|
|
// up our view data for them.
|
|
//
|
|
void ASShare::SSI_ViewEnded(ASPerson * pasPerson)
|
|
{
|
|
DebugEntry(ASShare::SSI_ViewEnded);
|
|
|
|
ValidateView(pasPerson);
|
|
|
|
//
|
|
// Deselect the save screen bitmap if there is one
|
|
//
|
|
if (pasPerson->m_pView->m_ssiOldBitmap != NULL)
|
|
{
|
|
SelectBitmap(pasPerson->m_pView->m_ssiDC, pasPerson->m_pView->m_ssiOldBitmap);
|
|
pasPerson->m_pView->m_ssiOldBitmap = NULL;
|
|
}
|
|
|
|
//
|
|
// Delete the save screen bitmap
|
|
//
|
|
if (pasPerson->m_pView->m_ssiBitmap != NULL)
|
|
{
|
|
DeleteBitmap(pasPerson->m_pView->m_ssiBitmap);
|
|
pasPerson->m_pView->m_ssiBitmap = NULL;
|
|
}
|
|
|
|
//
|
|
// Delete the save screen DC
|
|
//
|
|
if (pasPerson->m_pView->m_ssiDC != NULL)
|
|
{
|
|
DeleteDC(pasPerson->m_pView->m_ssiDC);
|
|
pasPerson->m_pView->m_ssiDC = NULL;
|
|
}
|
|
|
|
DebugExitVOID(ASShare::SSI_ViewEnded);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// SSI_SyncOutgoing()
|
|
// Called when NEW (3.0) dude starts to share, a share is created, or
|
|
// someone new joins the share.
|
|
// Resets save state for OUTGOING save/restore orders.
|
|
//
|
|
void ASHost::SSI_SyncOutgoing(void)
|
|
{
|
|
OSI_ESCAPE_HEADER request;
|
|
|
|
DebugEntry(ASHost::SSI_SyncOutgoing);
|
|
|
|
//
|
|
// Discard any saved bitmaps. This ensures that the subsequent
|
|
// datastream will not refer to any previously sent data.
|
|
//
|
|
//
|
|
// Make sure the display driver resets the save level. Note we don't
|
|
// really care what happens in the display driver, so don't bother with
|
|
// a special request block - use a standard request header.
|
|
//
|
|
OSI_FunctionRequest(SSI_ESC_RESET_LEVEL, &request, sizeof(request));
|
|
|
|
DebugExitVOID(ASHost::SSI_SyncOutgoing);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// FUNCTION: SSI_SaveBitmap
|
|
//
|
|
// DESCRIPTION:
|
|
// Replays a SaveBitmap order by saving or restoring a specified area of
|
|
// the user's desktop bitmap.
|
|
//
|
|
//
|
|
void ASShare::SSI_SaveBitmap
|
|
(
|
|
ASPerson * pasPerson,
|
|
LPSAVEBITMAP_ORDER pSaveBitmap
|
|
)
|
|
{
|
|
RECT screenBitmapRect;
|
|
RECT saveBitmapRect;
|
|
int xSaveBitmap;
|
|
int ySaveBitmap;
|
|
int xScreenBitmap;
|
|
int yScreenBitmap;
|
|
int cxTile;
|
|
int cyTile;
|
|
|
|
DebugEntry(ASShare::SSI_SaveBitmap);
|
|
|
|
ValidateView(pasPerson);
|
|
|
|
if ((pSaveBitmap->Operation != SV_SAVEBITS) &&
|
|
(pSaveBitmap->Operation != SV_RESTOREBITS))
|
|
{
|
|
ERROR_OUT(("SSI_SaveBitmap: unrecognized SV_ value %d",
|
|
pSaveBitmap->Operation));
|
|
DC_QUIT;
|
|
}
|
|
|
|
//
|
|
// Calculate the (x,y) start position from the pel start position
|
|
// given in the order.
|
|
//
|
|
ySaveBitmap = (pSaveBitmap->SavedBitmapPosition /
|
|
(TSHR_SSI_BITMAP_WIDTH *
|
|
(UINT)pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity)) *
|
|
pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity;
|
|
|
|
xSaveBitmap = (pSaveBitmap->SavedBitmapPosition -
|
|
(ySaveBitmap *
|
|
(UINT)TSHR_SSI_BITMAP_WIDTH)) /
|
|
pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity;
|
|
|
|
|
|
screenBitmapRect.left = pSaveBitmap->nLeftRect
|
|
- pasPerson->m_pView->m_dsScreenOrigin.x;
|
|
screenBitmapRect.top = pSaveBitmap->nTopRect
|
|
- pasPerson->m_pView->m_dsScreenOrigin.y;
|
|
screenBitmapRect.right = pSaveBitmap->nRightRect + 1
|
|
- pasPerson->m_pView->m_dsScreenOrigin.x;
|
|
screenBitmapRect.bottom = pSaveBitmap->nBottomRect + 1
|
|
- pasPerson->m_pView->m_dsScreenOrigin.y;
|
|
saveBitmapRect.left = 0;
|
|
saveBitmapRect.top = 0;
|
|
saveBitmapRect.right = TSHR_SSI_BITMAP_WIDTH;
|
|
saveBitmapRect.bottom = pasPerson->m_pView->m_ssiBitmapHeight;
|
|
|
|
//
|
|
// Start tiling in the top left corner of the Screen Bitmap rectangle.
|
|
//
|
|
xScreenBitmap = screenBitmapRect.left;
|
|
yScreenBitmap = screenBitmapRect.top;
|
|
|
|
//
|
|
// The height of the tile is the vertical granularity (or less - if
|
|
// the Screen Bitmap rect is thinner than the granularity).
|
|
//
|
|
cyTile = min(screenBitmapRect.bottom - yScreenBitmap,
|
|
(int)pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity );
|
|
|
|
//
|
|
// Repeat while there are more tiles in the Screen Bitmap rect to
|
|
// process.
|
|
//
|
|
while (yScreenBitmap < screenBitmapRect.bottom)
|
|
{
|
|
//
|
|
// The width of the tile is the minimum of:
|
|
//
|
|
// - the width of the remaining rectangle in the current strip of
|
|
// the Screen Bitmap rectangle
|
|
//
|
|
// - the width of the remaining empty space in the current strip of
|
|
// the Save Bitmap
|
|
//
|
|
//
|
|
cxTile = min( saveBitmapRect.right - xSaveBitmap,
|
|
screenBitmapRect.right - xScreenBitmap );
|
|
|
|
TRACE_OUT(( "screen(%d,%d) save(%d,%d) cx(%d) cy(%d)",
|
|
xScreenBitmap,
|
|
yScreenBitmap,
|
|
xSaveBitmap,
|
|
ySaveBitmap,
|
|
cxTile,
|
|
cyTile ));
|
|
|
|
//
|
|
// Save or Restore this tile
|
|
//
|
|
if (pSaveBitmap->Operation == SV_SAVEBITS)
|
|
{
|
|
//
|
|
// Save user's desktop area to SSI bitmap
|
|
//
|
|
BitBlt(pasPerson->m_pView->m_ssiDC,
|
|
xSaveBitmap, ySaveBitmap, cxTile, cyTile,
|
|
pasPerson->m_pView->m_usrDC,
|
|
xScreenBitmap, yScreenBitmap, SRCCOPY);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Restore user's desktop area from SSI bitmap
|
|
//
|
|
BitBlt(pasPerson->m_pView->m_usrDC,
|
|
xScreenBitmap, yScreenBitmap, cxTile, cyTile,
|
|
pasPerson->m_pView->m_ssiDC,
|
|
xSaveBitmap, ySaveBitmap, SRCCOPY);
|
|
}
|
|
|
|
//
|
|
// Move to the next tile in the Screen Bitmap rectangle.
|
|
//
|
|
xScreenBitmap += cxTile;
|
|
if (xScreenBitmap >= screenBitmapRect.right)
|
|
{
|
|
xScreenBitmap = screenBitmapRect.left;
|
|
yScreenBitmap += cyTile;
|
|
cyTile = min( screenBitmapRect.bottom - yScreenBitmap,
|
|
(int)pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity );
|
|
}
|
|
|
|
//
|
|
// Move to the next free space in the Save Bitmap.
|
|
//
|
|
xSaveBitmap += ROUNDUP(cxTile, pasPerson->cpcCaps.orders.capsSaveBitmapXGranularity);
|
|
if (xSaveBitmap >= saveBitmapRect.right)
|
|
{
|
|
xSaveBitmap = saveBitmapRect.left;
|
|
ySaveBitmap += ROUNDUP(cyTile, pasPerson->cpcCaps.orders.capsSaveBitmapYGranularity);
|
|
}
|
|
}
|
|
|
|
DC_EXIT_POINT:
|
|
DebugExitVOID(ASShare::SSI_SaveBitmap);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// SSI_RecalcCaps()
|
|
//
|
|
// Called when we are hosting and someone joins/leaves the share.
|
|
//
|
|
// When 2.x COMPAT IS GONE, THIS IS OBSOLETE
|
|
//
|
|
void ASShare::SSI_RecalcCaps(BOOL fJoiner)
|
|
{
|
|
ASPerson * pasT;
|
|
SSI_NEW_CAPABILITIES newCapabilities;
|
|
|
|
DebugEntry(ASShare::SSI_RecalcCaps);
|
|
|
|
if (!m_pHost)
|
|
{
|
|
//
|
|
// Nothing to do. Note that we recalc when someone joins AND
|
|
// when someone leaves, like SBC.
|
|
//
|
|
DC_QUIT;
|
|
}
|
|
|
|
ValidatePerson(m_pasLocal);
|
|
|
|
//
|
|
// Enumerate all the save screen bitmap receive capabilities of the
|
|
// parties in the share. The usable size of the send save screen
|
|
// bitmap is then the minimum of all the remote receive sizes and the
|
|
// local send size.
|
|
//
|
|
|
|
//
|
|
// Copy the locally registered send save screen bitmap size capability
|
|
// to our global variable used to communicate with the enumeration
|
|
// function SSIEnumBitmapCacheCaps().
|
|
//
|
|
m_pHost->m_ssiSaveBitmapSize = m_pasLocal->cpcCaps.orders.capsSaveBitmapSize;
|
|
|
|
//
|
|
// Set up the new capabilities structure...
|
|
//
|
|
newCapabilities.sendSaveBitmapSize = m_pHost->m_ssiSaveBitmapSize;
|
|
|
|
newCapabilities.xGranularity = TSHR_SSI_BITMAP_X_GRANULARITY;
|
|
|
|
newCapabilities.yGranularity = TSHR_SSI_BITMAP_Y_GRANULARITY;
|
|
|
|
//
|
|
// ... and pass it through to the driver.
|
|
//
|
|
if (!OSI_FunctionRequest(SSI_ESC_NEW_CAPABILITIES, (LPOSI_ESCAPE_HEADER)&newCapabilities,
|
|
sizeof(newCapabilities)))
|
|
{
|
|
ERROR_OUT(("SSI_ESC_NEW_CAPABILITIES failed"));
|
|
}
|
|
|
|
DC_EXIT_POINT:
|
|
DebugExitVOID(ASHost::SSI_RecalcCaps);
|
|
}
|