2020-09-30 17:12:32 +02:00

724 lines
16 KiB
C

/****************************** Module Header ******************************\
* Module Name: hdata.c
* Copyright (c) 1985 - 1999, Microsoft Corporation
* DDE Manager data handle functions
* Created: 11/12/91 Sanford Staab
*/
#define DDEMLDB
#include "precomp.h"
#pragma hdrstop
/*
* DdeCreateDataHandle (DDEML API)
* Description
* History:
* 11-1-91 sanfords Created.
*/
HDDEDATA DdeCreateDataHandle(
DWORD idInst,
LPBYTE pSrc,
DWORD cb,
DWORD cbOff,
HSZ hszItem,
UINT wFmt,
UINT afCmd)
{
PCL_INSTANCE_INFO pcii;
HDDEDATA hRet = 0;
if (cb == -1) {
RIPMSG0(RIP_ERROR, "DdeCreateDataHandle called with cb == -1\n");
return NULL;
}
EnterDDECrit;
pcii = ValidateInstance((HANDLE)LongToHandle( idInst ));
if (pcii == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
goto Exit;
}
if (afCmd & ~HDATA_APPOWNED) {
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
if (cb + cbOff < sizeof(DWORD) && pSrc == NULL &&
(wFmt == CF_METAFILEPICT ||
wFmt == CF_DSPMETAFILEPICT ||
wFmt == CF_DIB ||
wFmt == CF_BITMAP ||
wFmt == CF_DSPBITMAP ||
wFmt == CF_PALETTE ||
wFmt == CF_ENHMETAFILE ||
wFmt == CF_DSPENHMETAFILE)) {
/*
* We have the nasty possibility of blowing up in FreeDDEData if we
* don't initialize the data for formats with indirect data to 0.
* This is because GlobalLock/GlobalSize do not adequately validate
* random numbers given to them.
*/
cb += 4;
}
hRet = InternalCreateDataHandle(pcii, pSrc, cb, cbOff,
hszItem ? afCmd : (afCmd | HDATA_EXECUTE),
(WORD)((afCmd & HDATA_APPOWNED) ? 0 : DDE_FRELEASE), (WORD)wFmt);
if (!hRet) {
SetLastDDEMLError(pcii, DMLERR_MEMORY_ERROR);
}
Exit:
LeaveDDECrit;
return (hRet);
}
/*
* InternalCreateDataHandle
* Description:
* Worker function for creating a data handle. If cb is -1, pSrc is
* a GMEM_DDESHARE data handle. 0 is return ed on error.
* History:
* 11-19-91 sanfords Created.
*/
HDDEDATA InternalCreateDataHandle(
PCL_INSTANCE_INFO pcii,
LPBYTE pSrc,
DWORD cb, // cb of actual data to initialize with
DWORD cbOff, // offset from start of data
DWORD flags,
WORD wStatus,
WORD wFmt)
{
PDDEMLDATA pdd;
HDDEDATA hRet;
LPBYTE p;
DWORD cbOff2;
CheckDDECritIn;
pdd = (PDDEMLDATA)DDEMLAlloc(sizeof(DDEMLDATA));
if (pdd == NULL) {
return (0);
}
if (cb == -1) {
pdd->hDDE = (HANDLE)pSrc;
} else {
if (flags & HDATA_EXECUTE) {
cbOff2 = 0;
} else {
cbOff2 = sizeof(WORD) + sizeof(WORD); // skip wStatus, wFmt
}
pdd->hDDE = UserGlobalAlloc(GMEM_DDESHARE | GMEM_MOVEABLE | GMEM_ZEROINIT,
cb + cbOff + cbOff2);
if (pdd->hDDE == NULL) {
DDEMLFree(pdd);
return (0);
}
if (!(flags & HDATA_EXECUTE)) {
PDDE_DATA pdde;
USERGLOBALLOCK(pdd->hDDE, pdde);
UserAssert(pdde);
pdde->wStatus = wStatus;
pdde->wFmt = wFmt;
USERGLOBALUNLOCK(pdd->hDDE);
}
}
pdd->flags = (WORD)flags;
hRet = (HDDEDATA)CreateHandle((ULONG_PTR)pdd, HTYPE_DATA_HANDLE,
InstFromHandle(pcii->hInstClient));
if (!hRet) {
WOWGLOBALFREE(pdd->hDDE);
DDEMLFree(pdd);
return (0);
}
if (cb != -1 && pSrc != NULL) {
USERGLOBALLOCK(pdd->hDDE, p);
UserAssert(p);
RtlCopyMemory(p + cbOff + cbOff2, pSrc, cb);
USERGLOBALUNLOCK(pdd->hDDE);
pdd->flags |= HDATA_INITIALIZED;
}
return (hRet);
}
/*
* DdeAddData (DDEML API)
* Description:
* Copys data from a user buffer to a data handles. Reallocates if needed.
* History:
* 11-1-91 sanfords Created.
*/
HDDEDATA DdeAddData(
HDDEDATA hData,
LPBYTE pSrc,
DWORD cb,
DWORD cbOff)
{
LPSTR pMem;
PDDEMLDATA pdd;
PCL_INSTANCE_INFO pcii;
HDDEDATA hRet = 0;
EnterDDECrit;
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)hData, HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
goto Exit;
}
pcii = PciiFromHandle((HANDLE)hData);
if (pcii == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
goto Exit;
}
if (!(pdd->flags & HDATA_EXECUTE)) {
cbOff += 4;
}
if (cb + cbOff > UserGlobalSize(pdd->hDDE)) {
pdd->hDDE = UserGlobalReAlloc(pdd->hDDE, cb + cbOff,
GMEM_MOVEABLE | GMEM_ZEROINIT);
}
USERGLOBALLOCK(pdd->hDDE, pMem);
if (pMem == NULL) {
SetLastDDEMLError(pcii, DMLERR_MEMORY_ERROR);
goto Exit;
}
hRet = hData;
if (pSrc != NULL) {
try {
RtlCopyMemory(pMem + cbOff, pSrc, cb);
pdd->flags |= HDATA_INITIALIZED;
} except(W32ExceptionHandler(FALSE, RIP_WARNING)) {
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
hRet = 0;
}
}
USERGLOBALUNLOCK(pdd->hDDE);
Exit:
LeaveDDECrit;
return (hRet);
}
/*
* DdeGetData (DDEML API)
* Description:
* Copys data from a data handle into a user buffer.
* History:
* 11-1-91 sanfords Created.
*/
DWORD DdeGetData(
HDDEDATA hData,
LPBYTE pDst,
DWORD cbMax,
DWORD cbOff)
{
DWORD cbCopied = 0;
DWORD cbSize;
PDDEMLDATA pdd;
PCL_INSTANCE_INFO pcii;
EnterDDECrit;
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)hData,
HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
goto Exit;
}
pcii = PciiFromHandle((HANDLE)hData);
if (pcii == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
goto Exit;
}
if (!(pdd->flags & HDATA_EXECUTE)) {
cbOff += 4;
}
cbSize = (DWORD)UserGlobalSize(pdd->hDDE);
if (cbOff >= cbSize) {
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
if (pDst == NULL) {
cbCopied = cbSize - cbOff;
goto Exit;
} else {
LPSTR pMem;
cbCopied = min(cbMax, cbSize - cbOff);
USERGLOBALLOCK(pdd->hDDE, pMem);
UserAssert(pMem);
try {
RtlCopyMemory(pDst, pMem + cbOff, cbCopied);
} except(W32ExceptionHandler(FALSE, RIP_WARNING)) {
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
cbCopied = 0;
}
if (pMem != NULL) {
USERGLOBALUNLOCK(pdd->hDDE);
}
}
Exit:
LeaveDDECrit;
return (cbCopied);
}
/*
* DdeAccessData (DDEML API)
* Description:
* Locks a data handle for access.
* History:
* 11-1-91 sanfords Created.
*/
LPBYTE DdeAccessData(
HDDEDATA hData,
LPDWORD pcbDataSize)
{
PCL_INSTANCE_INFO pcii;
PDDEMLDATA pdd;
LPBYTE pRet = NULL;
DWORD cbOff;
EnterDDECrit;
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)hData,
HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
goto Exit;
}
pcii = PciiFromHandle((HANDLE)hData);
cbOff = pdd->flags & HDATA_EXECUTE ? 0 : 4;
if (pcbDataSize != NULL) {
*pcbDataSize = (DWORD)UserGlobalSize(pdd->hDDE) - cbOff;
}
USERGLOBALLOCK(pdd->hDDE, pRet);
UserAssert(pRet);
pRet = (LPBYTE)pRet + cbOff;
Exit:
LeaveDDECrit;
return (pRet);
}
/*
* DdeUnaccessData (DDEML API)
* Description:
* Unlocks a data handle
* History:
* 11-1-91 sanfords Created.
*/
BOOL DdeUnaccessData(
HDDEDATA hData)
{
PDDEMLDATA pdd;
BOOL fSuccess = FALSE;
EnterDDECrit;
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)hData,
HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
goto Exit;
}
USERGLOBALUNLOCK(pdd->hDDE);
fSuccess = TRUE;
Exit:
LeaveDDECrit;
return (fSuccess);
}
/*
* DdeFreeDataHandle (DDEML API)
* Description:
* Releases application interest in a data handle.
* History:
* 11-1-91 sanfords Created.
*/
BOOL DdeFreeDataHandle(
HDDEDATA hData)
{
PDDEMLDATA pdd;
BOOL fSuccess = FALSE;
EnterDDECrit;
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)hData,
HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
goto Exit;
}
if (pdd->flags & HDATA_NOAPPFREE) {
fSuccess = TRUE;
goto Exit;
}
fSuccess = InternalFreeDataHandle(hData, TRUE);
Exit:
LeaveDDECrit;
return (fSuccess);
}
/*
* InternalFreeDataHandle
* Description:
* Frees a data handle and its contents. The contents are NOT freed for
* APPOWNED data handles unless fIgnorefRelease is set.
* History:
* 11-19-91 sanfords Created.
*/
BOOL InternalFreeDataHandle(
HDDEDATA hData,
BOOL fIgnorefRelease)
{
PDDEMLDATA pdd;
CheckDDECritIn;
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)hData,
HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
return (FALSE);
}
if (pdd->flags & HDATA_EXECUTE) {
if (!(pdd->flags & HDATA_APPOWNED) || fIgnorefRelease) {
WOWGLOBALFREE(pdd->hDDE);
}
} else {
FreeDDEData(pdd->hDDE, fIgnorefRelease, TRUE);
}
DDEMLFree(pdd);
DestroyHandle((HANDLE)hData);
return (TRUE);
}
/*
* ApplyFreeDataHandle
* Description:
* Used during data handle cleanup.
* History:
* 11-19-91 sanfords Created.
*/
BOOL ApplyFreeDataHandle(
HANDLE hData)
{
BOOL fRet;
CheckDDECritOut;
EnterDDECrit;
fRet = InternalFreeDataHandle((HDDEDATA)hData, FALSE);
LeaveDDECrit;
return(fRet);
}
/*
* FreeDDEData
* Description:
* Used for freeing DDE data including any special indirect objects
* associated with the data depending on the format. This function
* SHOULD NOT BE USED TO FREE EXECUTE DATA!
* The data is not freed if the fRelease bit is clear and fIgnoreRelease
* is FALSE.
* The fFreeTruelyGlobalObjects parameter is used to distinguish tracking
* layer frees from DDEML frees. Data in certain formats (CF_BITMAP,
* CF_PALETTE) is maintained on the gdi CSR server side. When this is
* passed between processes, gdi is not able to maintain multiple process
* ownership on these objects so the objects must be made global. Thus
* the tracking layer should NOT free these objects on behalf of another
* process because they are truely global- however, DDEML can do this
* because it is following the protocol which delclares who is in charge
* of freeing global data. (YUCK!)
* History:
* 11-19-91 sanfords Created.
*/
/*
* WARNING: This is exported for NetDDE use - DO NOT CHANGE THE PARAMETERS!
*/
VOID FreeDDEData(
HANDLE hDDE,
BOOL fIgnorefRelease,
BOOL fFreeTruelyGlobalObjects)
{
PDDE_DATA pdde;
LPMETAFILEPICT pmfPict;
DWORD cb;
USERGLOBALLOCK(hDDE, pdde);
if (pdde == NULL) {
return ;
}
if ((pdde->wStatus & DDE_FRELEASE) || fIgnorefRelease) {
cb = (DWORD)GlobalSize(hDDE);
/*
* Because there is the possibility that the data never got
* initialized we need to do this in a try-except so we
* behave nicely.
*/
switch (pdde->wFmt) {
case CF_BITMAP:
case CF_DSPBITMAP:
case CF_PALETTE:
if (cb >= sizeof(HANDLE)) {
if (fFreeTruelyGlobalObjects) {
if (pdde->Data != 0) {
DeleteObject((HANDLE)pdde->Data);
}
} else {
/*
* !fFreeTruelyGlobalObject implies we are only freeing
* the Gdi proxy. (another process may still have this
* object in use.)
* ChrisWil: removed this call. No longer
* applicable in KMode.
* GdiDeleteLocalObject((ULONG)pdde->Data);
*/
}
}
break;
case CF_DIB:
if (cb >= sizeof(HANDLE)) {
if (pdde->Data != 0) {
WOWGLOBALFREE((HANDLE)pdde->Data);
}
}
break;
case CF_METAFILEPICT:
case CF_DSPMETAFILEPICT:
if (cb >= sizeof(HANDLE)) {
if (pdde->Data != 0) {
USERGLOBALLOCK(pdde->Data, pmfPict);
if (pmfPict != NULL) {
if (GlobalSize((HANDLE)pdde->Data) >= sizeof(METAFILEPICT)) {
DeleteMetaFile(pmfPict->hMF);
}
USERGLOBALUNLOCK((HANDLE)pdde->Data);
WOWGLOBALFREE((HANDLE)pdde->Data);
}
}
}
break;
case CF_ENHMETAFILE:
case CF_DSPENHMETAFILE:
if (cb >= sizeof(HANDLE)) {
if (pdde->Data != 0) {
DeleteEnhMetaFile((HANDLE)pdde->Data);
}
}
break;
}
USERGLOBALUNLOCK(hDDE);
WOWGLOBALFREE(hDDE);
} else {
USERGLOBALUNLOCK(hDDE);
}
}
HBITMAP CopyBitmap(
HBITMAP hbm)
{
BITMAP bm;
HBITMAP hbm2, hbmOld1, hbmOld2;
HDC hdc, hdcMem1, hdcMem2;
if (!GetObject(hbm, sizeof(BITMAP), &bm)) {
return(0);
}
hdc = NtUserGetDC(NULL); // screen DC
if (!hdc) {
return(0);
}
hdcMem1 = CreateCompatibleDC(hdc);
if (!hdcMem1) {
goto Cleanup3;
}
hdcMem2 = CreateCompatibleDC(hdc);
if (!hdcMem2) {
goto Cleanup2;
}
hbmOld1 = SelectObject(hdcMem1, hbm);
hbm2 = CreateCompatibleBitmap(hdcMem1, bm.bmWidth, bm.bmHeight);
if (!hbm2) {
goto Cleanup1;
}
hbmOld2 = SelectObject(hdcMem2, hbm2);
BitBlt(hdcMem2, 0, 0, bm.bmWidth, bm.bmHeight, hdcMem1, 0, 0, SRCCOPY);
SelectObject(hdcMem1, hbmOld1);
SelectObject(hdcMem2, hbmOld2);
Cleanup1:
DeleteDC(hdcMem2);
Cleanup2:
DeleteDC(hdcMem1);
Cleanup3:
NtUserReleaseDC(NULL, hdc);
return(hbm2);
}
HPALETTE CopyPalette(
HPALETTE hpal)
{
int cPalEntries;
LOGPALETTE *plp;
if (!GetObject(hpal, sizeof(int), &cPalEntries)) {
return(0);
}
plp = (LOGPALETTE *)DDEMLAlloc(sizeof(LOGPALETTE) +
(cPalEntries - 1) * sizeof(PALETTEENTRY));
if (!plp) {
return(0);
}
if (!GetPaletteEntries(hpal, 0, cPalEntries, plp->palPalEntry)) {
DDEMLFree(plp);
return(0);
}
plp->palVersion = 0x300;
plp->palNumEntries = (WORD)cPalEntries;
hpal = CreatePalette(plp);
if (hpal &&
!SetPaletteEntries(hpal, 0, cPalEntries, plp->palPalEntry)) {
hpal = 0;
}
DDEMLFree(plp);
return(hpal);
}
/*
* CopyDDEData
* Description:
* Used to copy DDE data apropriately.
* History:
* 11-19-91 sanfords Created.
*/
HANDLE CopyDDEData(
HANDLE hDDE,
BOOL fIsExecute)
{
HANDLE hDDENew;
PDDE_DATA pdde, pddeNew;
LPMETAFILEPICT pmfPict;
HANDLE hmfPict;
hDDENew = UserGlobalAlloc(GMEM_DDESHARE | GMEM_MOVEABLE,
UserGlobalSize(hDDE));
if (!hDDENew) {
return (0);
}
USERGLOBALLOCK(hDDE, pdde);
if (pdde == NULL) {
UserGlobalFree(hDDENew);
return (0);
}
USERGLOBALLOCK(hDDENew, pddeNew);
UserAssert(pddeNew);
RtlCopyMemory(pddeNew, pdde, UserGlobalSize(hDDE));
if (!fIsExecute) {
switch (pdde->wFmt) {
case CF_BITMAP:
case CF_DSPBITMAP:
pddeNew->Data = (ULONG_PTR)CopyBitmap((HBITMAP)pdde->Data);
break;
case CF_PALETTE:
pddeNew->Data = (ULONG_PTR)CopyPalette((HPALETTE)pdde->Data);
break;
case CF_DIB:
pddeNew->Data = (ULONG_PTR)CopyDDEData((HANDLE)pdde->Data, TRUE);
break;
case CF_METAFILEPICT:
case CF_DSPMETAFILEPICT:
hmfPict = CopyDDEData((HANDLE)pdde->Data, TRUE);
USERGLOBALLOCK(hmfPict, pmfPict);
if (pmfPict == NULL) {
WOWGLOBALFREE(hmfPict);
USERGLOBALUNLOCK(hDDENew);
WOWGLOBALFREE(hDDENew);
USERGLOBALUNLOCK(hDDE);
return (FALSE);
}
pmfPict->hMF = CopyMetaFile(pmfPict->hMF, NULL);
USERGLOBALUNLOCK(hmfPict);
pddeNew->Data = (ULONG_PTR)hmfPict;
break;
case CF_ENHMETAFILE:
case CF_DSPENHMETAFILE:
pddeNew->Data = (ULONG_PTR)CopyEnhMetaFile((HANDLE)pdde->Data, NULL);
break;
}
}
USERGLOBALUNLOCK(hDDENew);
USERGLOBALUNLOCK(hDDE);
return (hDDENew);
}