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.
589 lines
17 KiB
589 lines
17 KiB
//+-------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 2000.
|
|
//
|
|
// File: bitmap.cxx
|
|
//
|
|
// Contents: Support for Windows/OLE data types for oleprx32.dll.
|
|
// Used to be transmit_as routines, now user_marshal routines.
|
|
//
|
|
// This file contains support for HBITMAP.
|
|
//
|
|
// Functions:
|
|
// HBITMAP_UserSize
|
|
// HBITMAP_UserMarshal
|
|
// HBITMAP_UserUnmarshal
|
|
// HBITMAP_UserFree
|
|
// HBITMAP_UserSize64
|
|
// HBITMAP_UserMarshal64
|
|
// HBITMAP_UserUnmarshal64
|
|
// HBITMAP_UserFree64
|
|
//
|
|
// History: 13-Dec-00 JohnDoty Migrated from transmit.cxx
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
#include "stdrpc.hxx"
|
|
#pragma hdrstop
|
|
|
|
#include <oleauto.h>
|
|
#include <objbase.h>
|
|
#include "transmit.hxx"
|
|
#include <rpcwdt.h>
|
|
#include <storext.h>
|
|
#include <valid.h>
|
|
#include <obase.h>
|
|
#include <stream.hxx>
|
|
|
|
#include "carefulreader.hxx"
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserSize
|
|
//
|
|
// Synopsis: Get the wire size the HBITMAP handle and data.
|
|
//
|
|
// Derivation: Union of a long and the bitmap handle and then struct.
|
|
//
|
|
// history: May-95 Ryszardk Created.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned long __RPC_USER
|
|
HBITMAP_UserSize (
|
|
unsigned long * pFlags,
|
|
unsigned long Offset,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
if ( !pHBitmap )
|
|
return Offset;
|
|
|
|
BITMAP bm;
|
|
HBITMAP hBitmap = *pHBitmap;
|
|
|
|
LENGTH_ALIGN( Offset, 3 );
|
|
|
|
// The encapsulated union.
|
|
// Discriminant and then handle or pointer from the union arm.
|
|
// Union discriminant is 4 bytes + handle is represented by a long.
|
|
Offset += sizeof(long) + sizeof(long);
|
|
|
|
if ( ! *pHBitmap )
|
|
return Offset;
|
|
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
{
|
|
// Pointee of the union arm for the remote case.
|
|
// Conformat size, 6 fields, size, conf array.
|
|
|
|
Offset += 4 + 4 * sizeof(LONG) + 2 * sizeof(WORD) + 4;
|
|
|
|
// Get information about the bitmap
|
|
if (FALSE == GetObjectA(hBitmap, sizeof(BITMAP), &bm))
|
|
{
|
|
RAISE_RPC_EXCEPTION(HRESULT_FROM_WIN32(GetLastError()));
|
|
}
|
|
|
|
ULONG ulDataSize = bm.bmPlanes * bm.bmHeight * bm.bmWidthBytes;
|
|
|
|
Offset += ulDataSize;
|
|
}
|
|
|
|
return( Offset ) ;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserMarshall
|
|
//
|
|
// Synopsis: Marshalls an HBITMAP object into the RPC buffer.
|
|
//
|
|
// history: May-95 Ryszardk Created.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned char __RPC_FAR * __RPC_USER
|
|
HBITMAP_UserMarshal (
|
|
unsigned long * pFlags,
|
|
unsigned char * pBuffer,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
if ( !pHBitmap )
|
|
return pBuffer;
|
|
|
|
UserNdrDebugOut((UNDR_OUT4, "HBITMAP_UserMarshal\n"));
|
|
|
|
ALIGN( pBuffer, 3 );
|
|
|
|
// Discriminant of the encapsulated union and union arm.
|
|
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
{
|
|
// userHBITMAP
|
|
|
|
*( PULONG_LV_CAST pBuffer)++ = WDT_DATA_MARKER;
|
|
*( PULONG_LV_CAST pBuffer)++ = PtrToUlong(*pHBitmap);
|
|
|
|
if ( ! *pHBitmap )
|
|
return pBuffer;
|
|
|
|
// Get information about the bitmap
|
|
BITMAP bm;
|
|
HBITMAP hBitmap = *pHBitmap;
|
|
|
|
if (FALSE == GetObject(hBitmap, sizeof(BITMAP), &bm))
|
|
{
|
|
RpcRaiseException(HRESULT_FROM_WIN32(GetLastError()));
|
|
}
|
|
|
|
DWORD dwCount = bm.bmPlanes * bm.bmHeight * bm.bmWidthBytes;
|
|
|
|
*( PULONG_LV_CAST pBuffer)++ = dwCount;
|
|
|
|
// Get the bm structure fields.
|
|
ulong ulBmSize = 4 * sizeof(LONG) + 2 * sizeof( WORD );
|
|
|
|
memcpy( pBuffer, (void *) &bm, ulBmSize );
|
|
pBuffer += ulBmSize;
|
|
|
|
// Get the raw bits.
|
|
if (0 == GetBitmapBits( hBitmap, dwCount, pBuffer ) )
|
|
{
|
|
RpcRaiseException(HRESULT_FROM_WIN32(GetLastError()));
|
|
}
|
|
pBuffer += dwCount;
|
|
}
|
|
else
|
|
{
|
|
// Sending a handle.
|
|
*( PULONG_LV_CAST pBuffer)++ = WDT_HANDLE_MARKER;
|
|
*( PULONG_LV_CAST pBuffer)++ = PtrToUlong(*(HANDLE *)pHBitmap);
|
|
}
|
|
|
|
return( pBuffer );
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserUnmarshallWorker
|
|
//
|
|
// Synopsis: Unmarshalls an HBITMAP object from the RPC buffer.
|
|
//
|
|
// history: Aug-99 JohnStra Created.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned char __RPC_FAR * __RPC_USER
|
|
HBITMAP_UserUnmarshalWorker (
|
|
unsigned long * pFlags,
|
|
unsigned char * pBuffer,
|
|
HBITMAP * pHBitmap,
|
|
ULONG_PTR BufferSize )
|
|
{
|
|
HBITMAP hBitmap;
|
|
|
|
// Align the buffer and save the fixup size.
|
|
UCHAR* pBufferStart = pBuffer;
|
|
ALIGN( pBuffer, 3 );
|
|
ULONG_PTR cbFixup = (ULONG_PTR)(pBuffer - pBufferStart);
|
|
|
|
// Check for EOB before accessing discriminant and handle.
|
|
CHECK_BUFFER_SIZE( BufferSize, cbFixup + (2 * sizeof( ULONG )) );
|
|
|
|
// Get Discriminant and handle. Caller checked for EOB.
|
|
unsigned long UnionDisc = *( PULONG_LV_CAST pBuffer)++;
|
|
hBitmap = (HBITMAP)LongToHandle( *( PLONG_LV_CAST pBuffer)++ );
|
|
|
|
if ( IS_DATA_MARKER(UnionDisc) )
|
|
{
|
|
if ( GDI_HANDLE_PASSING(*pFlags) )
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
|
|
if ( hBitmap )
|
|
{
|
|
ulong ulBmSize = 4 * sizeof(LONG) + 2 * sizeof( WORD );
|
|
|
|
// Check for EOB before accessing metadata.
|
|
CHECK_BUFFER_SIZE(
|
|
BufferSize,
|
|
cbFixup + (3 * sizeof( ULONG )) + ulBmSize );
|
|
|
|
DWORD dwCount = *( PULONG_LV_CAST pBuffer)++;
|
|
BITMAP * pBm = (BITMAP *) pBuffer;
|
|
|
|
// verify dwCount matches the bitmap.
|
|
if ( dwCount != (DWORD) pBm->bmPlanes * pBm->bmHeight * pBm->bmWidthBytes )
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
|
|
pBuffer += ulBmSize;
|
|
|
|
// Check for EOB before accessing data.
|
|
CHECK_BUFFER_SIZE(
|
|
BufferSize,
|
|
cbFixup + (3 * sizeof(ULONG)) + ulBmSize + dwCount);
|
|
|
|
// Create a bitmap based on the BITMAP structure and the raw bits in
|
|
// the transmission buffer
|
|
|
|
hBitmap = CreateBitmap( pBm->bmWidth,
|
|
pBm->bmHeight,
|
|
pBm->bmPlanes,
|
|
pBm->bmBitsPixel,
|
|
pBuffer );
|
|
|
|
pBuffer += dwCount;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( !IS_HANDLE_MARKER( UnionDisc ) )
|
|
{
|
|
RAISE_RPC_EXCEPTION( RPC_S_INVALID_TAG );
|
|
}
|
|
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
}
|
|
|
|
// A new bitmap handle is ready, destroy the old one, if needed.
|
|
|
|
if ( *pHBitmap )
|
|
DeleteObject( *pHBitmap );
|
|
|
|
*pHBitmap = hBitmap;
|
|
|
|
return( pBuffer );
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserUnmarshall
|
|
//
|
|
// Synopsis: Unmarshalls an HBITMAP object from the RPC buffer.
|
|
//
|
|
// history: May-95 Ryszardk Created.
|
|
// Aug-99 JohnStra Factored bulk of work into a worker
|
|
// routine in order to add consistency
|
|
// checks.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned char __RPC_FAR * __RPC_USER
|
|
HBITMAP_UserUnmarshal (
|
|
unsigned long * pFlags,
|
|
unsigned char * pBuffer,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
UserNdrDebugOut((UNDR_OUT4, "HBITMAP_UserUnmarshal\n"));
|
|
|
|
// Get the buffer size and ptr to buffer.
|
|
CUserMarshalInfo MarshalInfo( pFlags, pBuffer );
|
|
ULONG_PTR BufferSize = MarshalInfo.GetBufferSize();
|
|
UCHAR* pBufferStart = MarshalInfo.GetBuffer();
|
|
|
|
pBuffer = HBITMAP_UserUnmarshalWorker( pFlags,
|
|
pBufferStart,
|
|
pHBitmap,
|
|
BufferSize );
|
|
return( pBuffer );
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserFree
|
|
//
|
|
// Synopsis: Free an HBITMAP.
|
|
//
|
|
// history: May-95 Ryszardk Created.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void __RPC_USER
|
|
HBITMAP_UserFree(
|
|
unsigned long * pFlags,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
UserNdrDebugOut((UNDR_OUT4, "HBITMAP_UserFree\n"));
|
|
|
|
if( pHBitmap && *pHBitmap )
|
|
{
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
{
|
|
DeleteObject( *pHBitmap );
|
|
}
|
|
}
|
|
}
|
|
|
|
#if defined(_WIN64)
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserSize64
|
|
//
|
|
// Synopsis: Get the wire size the HBITMAP handle and data.
|
|
//
|
|
// Derivation: Union of a long and the bitmap handle and then struct.
|
|
//
|
|
// history: Dec-00 JohnDoty Created from 32bit functions.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned long __RPC_USER
|
|
HBITMAP_UserSize64 (
|
|
unsigned long * pFlags,
|
|
unsigned long Offset,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
if ( !pHBitmap )
|
|
return Offset;
|
|
|
|
BITMAP bm;
|
|
HBITMAP hBitmap = *pHBitmap;
|
|
|
|
LENGTH_ALIGN( Offset, 7 );
|
|
|
|
// The encapsulated union.
|
|
// (aligned on 8)
|
|
// discriminant 4
|
|
// (align on 8) 4
|
|
// handle or ptr 8
|
|
Offset += 16;
|
|
|
|
if ( ! *pHBitmap )
|
|
return Offset;
|
|
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
{
|
|
// Pointee of the union arm for the remote case.
|
|
// (aligned on 8)
|
|
// conformance 8
|
|
// 4xlong 16
|
|
// 2xword 4
|
|
// size 4
|
|
// data ulDataSize;
|
|
if (FALSE == GetObject(hBitmap, sizeof(BITMAP), &bm))
|
|
RAISE_RPC_EXCEPTION(HRESULT_FROM_WIN32(GetLastError()));
|
|
|
|
ULONG ulDataSize = bm.bmPlanes * bm.bmHeight * bm.bmWidthBytes;
|
|
Offset += 32 + ulDataSize;
|
|
}
|
|
|
|
return( Offset ) ;
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserMarshal64
|
|
//
|
|
// Synopsis: Marshalls an HBITMAP object into the RPC buffer.
|
|
//
|
|
// history: Dec-00 JohnDoty Created from 32bit functions.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned char __RPC_FAR * __RPC_USER
|
|
HBITMAP_UserMarshal64 (
|
|
unsigned long * pFlags,
|
|
unsigned char * pBuffer,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
if ( !pHBitmap )
|
|
return pBuffer;
|
|
|
|
UserNdrDebugOut((UNDR_OUT4, "HBITMAP_UserMarshal64\n"));
|
|
|
|
ALIGN( pBuffer, 7 );
|
|
|
|
// Discriminant of the encapsulated union and union arm.
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
{
|
|
// userHBITMAP
|
|
*( PULONG_LV_CAST pBuffer)++ = WDT_DATA_MARKER;
|
|
ALIGN( pBuffer, 7 );
|
|
*( PHYPER_LV_CAST pBuffer)++ = (hyper)(*pHBitmap);
|
|
|
|
if ( ! *pHBitmap )
|
|
return pBuffer;
|
|
|
|
// Get information about the bitmap
|
|
BITMAP bm;
|
|
HBITMAP hBitmap = *pHBitmap;
|
|
|
|
if (FALSE == GetObject(hBitmap, sizeof(BITMAP), &bm))
|
|
RpcRaiseException(HRESULT_FROM_WIN32(GetLastError()));
|
|
|
|
ULONG ulCount = bm.bmPlanes * bm.bmHeight * bm.bmWidthBytes;
|
|
|
|
// Conformance...
|
|
*(PHYPER_LV_CAST pBuffer)++ = ulCount;
|
|
|
|
// Get the bm structure fields.
|
|
ulong ulBmSize = 4 * sizeof(LONG) + 2 * sizeof( WORD );
|
|
memcpy( pBuffer, &bm, ulBmSize );
|
|
pBuffer += ulBmSize;
|
|
|
|
// Get the raw bits.
|
|
*(PULONG_LV_CAST pBuffer)++ = ulCount;
|
|
|
|
if (0 == GetBitmapBits( hBitmap, ulCount, pBuffer ) )
|
|
RpcRaiseException(HRESULT_FROM_WIN32(GetLastError()));
|
|
|
|
pBuffer += ulCount;
|
|
}
|
|
else
|
|
{
|
|
// Sending a handle.
|
|
*( PULONG_LV_CAST pBuffer)++ = WDT_HANDLE64_MARKER;
|
|
ALIGN( pBuffer, 7 );
|
|
*( PHYPER_LV_CAST pBuffer)++ = (hyper)(*(HANDLE *)pHBitmap);
|
|
}
|
|
|
|
return( pBuffer );
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserUnmarshallWorker64
|
|
//
|
|
// Synopsis: Unmarshalls an HBITMAP object from the RPC buffer.
|
|
//
|
|
// history: Dec-00 JohnDoty Created from 32bit functions.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
unsigned char __RPC_FAR * __RPC_USER
|
|
HBITMAP_UserUnmarshalWorker64 (
|
|
unsigned long * pFlags,
|
|
unsigned char * pBuffer,
|
|
HBITMAP * pHBitmap,
|
|
ULONG_PTR BufferSize )
|
|
{
|
|
CarefulBufferReader stream(pBuffer, BufferSize);
|
|
|
|
stream.Align(8);
|
|
|
|
// Get Discriminant and handle.
|
|
unsigned long UnionDisc = stream.ReadULONGNA();
|
|
HBITMAP hBitmap = (HBITMAP)stream.ReadHYPER();
|
|
|
|
if ( IS_DATA_MARKER( UnionDisc) )
|
|
{
|
|
if ( GDI_HANDLE_PASSING(*pFlags) )
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
|
|
if ( hBitmap )
|
|
{
|
|
DWORD dwCount = (DWORD)stream.ReadHYPERNA();
|
|
|
|
// Check for EOB before accessing metadata.
|
|
ulong ulBmSize = 4 * sizeof(LONG) + 2 * sizeof(WORD);
|
|
stream.CheckSize(ulBmSize);
|
|
|
|
BITMAP * pBm = (BITMAP *)stream.GetBuffer();
|
|
// verify dwCount matches the bitmap.
|
|
if ( dwCount != (DWORD) pBm->bmPlanes * pBm->bmHeight * pBm->bmWidthBytes )
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
|
|
stream.Advance(ulBmSize);
|
|
|
|
if (stream.ReadULONGNA() != dwCount)
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
|
|
// Check for EOB before accessing data.
|
|
stream.CheckSize(dwCount);
|
|
|
|
// Create a bitmap based on the BITMAP structure and the raw bits in
|
|
// the transmission buffer
|
|
hBitmap = CreateBitmap( pBm->bmWidth,
|
|
pBm->bmHeight,
|
|
pBm->bmPlanes,
|
|
pBm->bmBitsPixel,
|
|
stream.GetBuffer() );
|
|
if (hBitmap == NULL)
|
|
RpcRaiseException(HRESULT_FROM_WIN32(GetLastError()));
|
|
|
|
stream.Advance(dwCount);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( !IS_HANDLE64_MARKER( UnionDisc ) )
|
|
{
|
|
RAISE_RPC_EXCEPTION( RPC_S_INVALID_TAG );
|
|
}
|
|
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
RAISE_RPC_EXCEPTION( RPC_X_BAD_STUB_DATA );
|
|
}
|
|
|
|
// A new bitmap handle is ready, destroy the old one, if needed.
|
|
if ( *pHBitmap )
|
|
DeleteObject( *pHBitmap );
|
|
|
|
*pHBitmap = hBitmap;
|
|
|
|
return( stream.GetBuffer() );
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserUnmarshal64
|
|
//
|
|
// Synopsis: Unmarshalls an HBITMAP object from the RPC buffer.
|
|
//
|
|
// history: Dec-00 JohnDoty Created from 32bit functions.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
unsigned char __RPC_FAR * __RPC_USER
|
|
HBITMAP_UserUnmarshal64 (
|
|
unsigned long * pFlags,
|
|
unsigned char * pBuffer,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
UserNdrDebugOut((UNDR_OUT4, "HBITMAP_UserUnmarshal\n"));
|
|
|
|
// Get the buffer size and ptr to buffer.
|
|
CUserMarshalInfo MarshalInfo( pFlags, pBuffer );
|
|
ULONG_PTR BufferSize = MarshalInfo.GetBufferSize();
|
|
UCHAR* pBufferStart = MarshalInfo.GetBuffer();
|
|
|
|
pBuffer = HBITMAP_UserUnmarshalWorker64( pFlags,
|
|
pBufferStart,
|
|
pHBitmap,
|
|
BufferSize );
|
|
return( pBuffer );
|
|
}
|
|
|
|
//+-------------------------------------------------------------------------
|
|
//
|
|
// Function: HBITMAP_UserFree64
|
|
//
|
|
// Synopsis: Free an HBITMAP.
|
|
//
|
|
// history: Dec-00 JohnDoty Created from 32bit functions.
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
void __RPC_USER
|
|
HBITMAP_UserFree64 (
|
|
unsigned long * pFlags,
|
|
HBITMAP * pHBitmap )
|
|
{
|
|
UserNdrDebugOut((UNDR_OUT4, "HBITMAP_UserFree\n"));
|
|
|
|
if( pHBitmap && *pHBitmap )
|
|
{
|
|
if ( GDI_DATA_PASSING(*pFlags) )
|
|
{
|
|
DeleteObject( *pHBitmap );
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|