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

721 lines
20 KiB
C

/****************************** Module Header ******************************\
* Module Name: ddetrack.c
* Copyright (c) 1985 - 1999, Microsoft Corporation
* client sied DDE tracking routines
* 10-22-91 sanfords created
*/
#include "precomp.h"
#pragma hdrstop
DWORD _ClientCopyDDEIn1(
HANDLE hClient, // client handle to dde data or ddepack data
PINTDDEINFO pi) // info for transfer
{
PBYTE pData;
DWORD flags;
// zero out everything but the flags
flags = pi->flags;
RtlZeroMemory(pi, sizeof(INTDDEINFO));
pi->flags = flags;
USERGLOBALLOCK(hClient, pData);
if (pData == NULL) { // bad hClient
RIPMSG0(RIP_WARNING, "_ClientCopyDDEIn1:GlobalLock failed.");
return (FAIL_POST);
}
if (flags & XS_PACKED) {
if (UserGlobalSize(hClient) < sizeof(DDEPACK)) {
/*
* must be a low memory condition. fail.
*/
return(FAIL_POST);
}
pi->DdePack = *(PDDEPACK)pData;
USERGLOBALUNLOCK(hClient);
UserGlobalFree(hClient); // packed data handles are not WOW matched.
hClient = NULL;
if (!(flags & (XS_LOHANDLE | XS_HIHANDLE))) {
if (flags & XS_EXECUTE && flags & XS_FREESRC) {
/*
* free execute ACK data
*/
WOWGLOBALFREE((HANDLE)pi->DdePack.uiHi);
}
return (DO_POST); // no direct data
}
if (flags & XS_LOHANDLE) {
pi->hDirect = (HANDLE)pi->DdePack.uiLo;
} else {
pi->hDirect = (HANDLE)pi->DdePack.uiHi;
}
if (pi->hDirect == 0) {
return (DO_POST); // must be warm link
}
USERGLOBALLOCK(pi->hDirect, pi->pDirect);
if (pi->pDirect == NULL) {
RIPMSG1(RIP_ERROR, "_ClientCopyDDEIn1:GlobalLock failed for hDirect %lx.",pi->hDirect);
return FAILNOFREE_POST;
}
pData = pi->pDirect;
pi->cbDirect = (UINT)UserGlobalSize(pi->hDirect);
} else { // not packed - must be execute data or we wouldn't be called
UserAssert(flags & XS_EXECUTE);
pi->cbDirect = (UINT)UserGlobalSize(hClient);
pi->hDirect = hClient;
pi->pDirect = pData;
hClient = NULL;
}
if (flags & XS_DATA) {
PDDE_DATA pDdeData = (PDDE_DATA)pData;
/*
* Assert that the hClient has been freed. If not this code will return
* the wrong thing on failure
*/
UserAssert(flags & XS_PACKED);
// check here for indirect data
switch (pDdeData->wFmt) {
case CF_BITMAP:
case CF_DSPBITMAP:
// Imediately following the dde data header is a bitmap handle.
UserAssert(pi->cbDirect >= sizeof(DDE_DATA));
pi->hIndirect = (HANDLE)pDdeData->Data;
if (pi->hIndirect == 0) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEIn1:GdiConvertBitmap failed");
return(FAILNOFREE_POST);
}
// pi->cbIndirect = 0; // zero init.
// pi->pIndirect = NULL; // zero init.
pi->flags |= XS_BITMAP;
break;
case CF_DIB:
// Imediately following the dde data header is a global data handle
// to the DIB bits.
UserAssert(pi->cbDirect >= sizeof(DDE_DATA));
pi->flags |= XS_DIB;
pi->hIndirect = (HANDLE)pDdeData->Data;
USERGLOBALLOCK(pi->hIndirect, pi->pIndirect);
if (pi->pIndirect == NULL) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEIn1:CF_DIB GlobalLock failed.");
return (FAILNOFREE_POST);
}
pi->cbIndirect = (UINT)UserGlobalSize(pi->hIndirect);
break;
case CF_PALETTE:
UserAssert(pi->cbDirect >= sizeof(DDE_DATA));
pi->hIndirect = (HANDLE) pDdeData->Data;
if (pi->hIndirect == 0) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEIn1:GdiConvertPalette failed.");
return(FAILNOFREE_POST);
}
// pi->cbIndirect = 0; // zero init.
// pi->pIndirect = NULL; // zero init.
pi->flags |= XS_PALETTE;
break;
case CF_DSPMETAFILEPICT:
case CF_METAFILEPICT:
// This format holds a global data handle which contains
// a METAFILEPICT structure that in turn contains
// a GDI metafile.
UserAssert(pi->cbDirect >= sizeof(DDE_DATA));
pi->hIndirect = GdiConvertMetaFilePict((HANDLE)pDdeData->Data);
if (pi->hIndirect == 0) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEIn1:GdiConvertMetaFilePict failed");
return(FAILNOFREE_POST);
}
// pi->cbIndirect = 0; // zero init.
// pi->pIndirect = NULL; // zero init.
pi->flags |= XS_METAFILEPICT;
break;
case CF_ENHMETAFILE:
case CF_DSPENHMETAFILE:
UserAssert(pi->cbDirect >= sizeof(DDE_DATA));
pi->hIndirect = GdiConvertEnhMetaFile((HENHMETAFILE)pDdeData->Data);
if (pi->hIndirect == 0) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEIn1:GdiConvertEnhMetaFile failed");
return(FAILNOFREE_POST);
}
// pi->cbIndirect = 0; // zero init.
// pi->pIndirect = NULL; // zero init.
pi->flags |= XS_ENHMETAFILE;
break;
}
}
return (DO_POST);
}
/*
* unlocks and frees DDE data pointers as appropriate
*/
VOID _ClientCopyDDEIn2(
PINTDDEINFO pi)
{
if (pi->cbDirect) {
USERGLOBALUNLOCK(pi->hDirect);
if (pi->flags & XS_FREESRC) {
WOWGLOBALFREE(pi->hDirect);
}
}
if (pi->cbIndirect) {
USERGLOBALUNLOCK(pi->hIndirect);
if (pi->flags & XS_FREESRC) {
WOWGLOBALFREE(pi->hIndirect);
}
}
}
/*
* returns fHandleValueChanged.
*/
BOOL FixupDdeExecuteIfNecessary(
HGLOBAL *phCommands,
BOOL fNeedUnicode)
{
UINT cbLen;
UINT cbSrc = (UINT)GlobalSize(*phCommands);
LPVOID pstr;
HGLOBAL hTemp;
BOOL fHandleValueChanged = FALSE;
USERGLOBALLOCK(*phCommands, pstr);
if (cbSrc && pstr != NULL) {
BOOL fIsUnicodeText;
#ifdef ISTEXTUNICODE_WORKS
int flags;
flags = (IS_TEXT_UNICODE_UNICODE_MASK |
IS_TEXT_UNICODE_REVERSE_MASK |
(IS_TEXT_UNICODE_NOT_UNICODE_MASK &
(~IS_TEXT_UNICODE_ILLEGAL_CHARS)) |
IS_TEXT_UNICODE_NOT_ASCII_MASK);
fIsUnicodeText = RtlIsTextUnicode(pstr, cbSrc - 2, &flags);
#else
fIsUnicodeText = ((cbSrc >= sizeof(WCHAR)) && (((LPSTR)pstr)[1] == '\0'));
#endif
if (!fIsUnicodeText && fNeedUnicode) {
LPWSTR pwsz;
/*
* Contents needs to be UNICODE.
*/
cbLen = strlen(pstr) + 1;
cbSrc = min(cbSrc, cbLen);
pwsz = UserLocalAlloc(HEAP_ZERO_MEMORY, cbSrc * sizeof(WCHAR));
if (pwsz != NULL) {
if (NT_SUCCESS(RtlMultiByteToUnicodeN(
pwsz,
cbSrc * sizeof(WCHAR),
NULL,
(PCHAR)pstr,
cbSrc))) {
USERGLOBALUNLOCK(*phCommands);
if ((hTemp = GlobalReAlloc(
*phCommands,
cbSrc * sizeof(WCHAR),
GMEM_MOVEABLE)) != NULL) {
fHandleValueChanged = (hTemp != *phCommands);
*phCommands = hTemp;
USERGLOBALLOCK(*phCommands, pstr);
pwsz[cbSrc - 1] = L'\0';
wcscpy(pstr, pwsz);
}
}
UserLocalFree(pwsz);
}
} else if (fIsUnicodeText && !fNeedUnicode) {
LPSTR psz;
/*
* Contents needs to be ANSI.
*/
cbLen = (wcslen(pstr) + 1) * sizeof(WCHAR);
cbSrc = min(cbSrc, cbLen);
psz = UserLocalAlloc(HEAP_ZERO_MEMORY, cbSrc);
if (psz != NULL) {
if (NT_SUCCESS(RtlUnicodeToMultiByteN(
psz,
cbSrc,
NULL,
(PWSTR)pstr,
cbSrc))) {
USERGLOBALUNLOCK(*phCommands);
if ((hTemp = GlobalReAlloc(
*phCommands,
cbSrc / sizeof(WCHAR),
GMEM_MOVEABLE)) != NULL) {
fHandleValueChanged = (hTemp != *phCommands);
*phCommands = hTemp;
USERGLOBALLOCK(*phCommands, pstr);
UserAssert(pstr);
psz[cbSrc - 1] = '\0';
strcpy(pstr, psz);
}
}
UserLocalFree(psz);
}
}
USERGLOBALUNLOCK(*phCommands);
}
return(fHandleValueChanged);
}
/*
* Allocates and locks global handles as appropriate in preperation
* for thunk copying.
*/
HANDLE _ClientCopyDDEOut1(
PINTDDEINFO pi)
{
HANDLE hDdePack = NULL;
PDDEPACK pDdePack = NULL;
if (pi->flags & XS_PACKED) {
/*
* make a wrapper for the data
*/
hDdePack = UserGlobalAlloc(GMEM_DDESHARE | GMEM_FIXED,
sizeof(DDEPACK));
pDdePack = (PDDEPACK)hDdePack;
if (pDdePack == NULL) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEOut1:Couldn't allocate DDEPACK");
return (NULL);
}
*pDdePack = pi->DdePack;
}
if (pi->cbDirect) {
pi->hDirect = UserGlobalAlloc(GMEM_DDESHARE | GMEM_MOVEABLE, pi->cbDirect);
if (pi->hDirect == NULL) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEOut1:Couldn't allocate hDirect");
if (hDdePack) {
UserGlobalFree(hDdePack);
}
return (NULL);
}
USERGLOBALLOCK(pi->hDirect, pi->pDirect);
UserAssert(pi->pDirect);
// fixup packed data reference to direct data
if (pDdePack != NULL) {
if (pi->flags & XS_LOHANDLE) {
pDdePack->uiLo = HandleToUlong(pi->hDirect);
UserAssert((ULONG_PTR)pDdePack->uiLo == (ULONG_PTR)pi->hDirect);
} else if (pi->flags & XS_HIHANDLE) {
pDdePack->uiHi = HandleToUlong(pi->hDirect);
UserAssert((ULONG_PTR)pDdePack->uiHi == (ULONG_PTR)pi->hDirect);
}
}
if (pi->cbIndirect) {
pi->hIndirect = UserGlobalAlloc(GMEM_DDESHARE | GMEM_MOVEABLE,
pi->cbIndirect);
if (pi->hIndirect == NULL) {
RIPMSG0(RIP_WARNING, "_ClientCopyDDEOut1:Couldn't allocate hIndirect");
USERGLOBALUNLOCK(pi->hDirect);
UserGlobalFree(pi->hDirect);
if (hDdePack) {
UserGlobalFree(hDdePack);
}
return (NULL);
}
USERGLOBALLOCK(pi->hIndirect, pi->pIndirect);
UserAssert(pi->pIndirect);
}
}
if (hDdePack) {
return (hDdePack);
} else {
return (pi->hDirect);
}
}
/*
* Fixes up internal poniters after thunk copy and unlocks handles.
*/
BOOL _ClientCopyDDEOut2(
PINTDDEINFO pi)
{
BOOL fSuccess = TRUE;
/*
* done with copies - now fixup indirect references
*/
if (pi->hIndirect) {
PDDE_DATA pDdeData = (PDDE_DATA)pi->pDirect;
switch (pDdeData->wFmt) {
case CF_BITMAP:
case CF_DSPBITMAP:
case CF_PALETTE:
pDdeData->Data = (ULONG_PTR)pi->hIndirect;
fSuccess = (BOOL)pDdeData->Data;
break;
case CF_METAFILEPICT:
case CF_DSPMETAFILEPICT:
pDdeData->Data = (ULONG_PTR)GdiCreateLocalMetaFilePict(pi->hIndirect);
fSuccess = (BOOL)pDdeData->Data;
break;
case CF_DIB:
pDdeData->Data = (ULONG_PTR)pi->hIndirect;
fSuccess = (BOOL)pDdeData->Data;
USERGLOBALUNLOCK(pi->hIndirect);
break;
case CF_ENHMETAFILE:
case CF_DSPENHMETAFILE:
pDdeData->Data = (ULONG_PTR)GdiCreateLocalEnhMetaFile(pi->hIndirect);
fSuccess = (BOOL)pDdeData->Data;
break;
default:
RIPMSG0(RIP_WARNING, "_ClientCopyDDEOut2:Unknown format w/indirect data.");
fSuccess = FALSE;
USERGLOBALUNLOCK(pi->hIndirect);
}
}
UserAssert(pi->hDirect); // if its null, we didn't need to call this function.
USERGLOBALUNLOCK(pi->hDirect);
if (pi->flags & XS_EXECUTE) {
/*
* Its possible that in RAW DDE cases where the app allocated the
* execute data as non-moveable, we have a different hDirect
* than we started with. This needs to be noted and passed
* back to the server. (Very RARE case)
*/
FixupDdeExecuteIfNecessary(&pi->hDirect,
pi->flags & XS_UNICODE);
}
return fSuccess;
}
/*
* This routine is called by the tracking layer when it frees DDE objects
* on behalf of a client. This cleans up the LOCAL objects associated
* with the DDE objects. It should NOT remove truely global objects such
* as bitmaps or palettes except in the XS_DUMPMSG case which is for
* faked Posts.
*/
#if DBG
/*
* Help track down a bug where I suspect the xxxFreeListFree is
* freeing a handle already freed by some other means which has
* since been reallocated and is trashing the client heap. (SAS)
*/
HANDLE DDEHandleLastFreed = 0;
#endif
BOOL _ClientFreeDDEHandle(
HANDLE hDDE,
DWORD flags)
{
PDDEPACK pDdePack;
HANDLE hNew;
if (flags & XS_PACKED) {
pDdePack = (PDDEPACK)hDDE;
if (pDdePack == NULL) {
return (FALSE);
}
if (flags & XS_LOHANDLE) {
hNew = (HANDLE)pDdePack->uiLo;
} else {
hNew = (HANDLE)pDdePack->uiHi;
}
WOWGLOBALFREE(hDDE);
hDDE = hNew;
}
/*
* Do a range check and call GlobalFlags to validate, just to prevent heap checking
* from complaining during the GlobalSize call.
* Is this leaking atoms??
*/
if ((hDDE <= (HANDLE)0xFFFF)
|| (GlobalFlags(hDDE) == GMEM_INVALID_HANDLE)
|| !GlobalSize(hDDE)) {
/*
* There may be cases where apps improperly freed stuff
* when they shouldn't have so make sure this handle
* is valid by the time it gets here.
* See SvSpontAdvise; it posts a message with an atom in uiHi. Then from _PostMessage
* in the kernel side, we might end up here. So it's not only for apps...
*/
return(FALSE);
}
if (flags & XS_DUMPMSG) {
if (flags & XS_PACKED) {
if (!IS_PTR(hNew)) {
GlobalDeleteAtom(LOWORD((ULONG_PTR)hNew));
if (!(flags & XS_DATA)) {
return(TRUE); // ACK
}
}
} else {
if (!(flags & XS_EXECUTE)) {
GlobalDeleteAtom(LOWORD((ULONG_PTR)hDDE)); // REQUEST, UNADVISE
return(TRUE);
}
}
}
if (flags & XS_DATA) {
// POKE, DATA
#if DBG
DDEHandleLastFreed = hDDE;
#endif
FreeDDEData(hDDE,
(flags & XS_DUMPMSG) ? FALSE : TRUE, // fIgnorefRelease
(flags & XS_DUMPMSG) ? TRUE : FALSE); // fDestroyTruelyGlobalObjects
} else {
// ADVISE, EXECUTE
#if DBG
DDEHandleLastFreed = hDDE;
#endif
WOWGLOBALFREE(hDDE); // covers ADVISE case (fmt but no data)
}
return (TRUE);
}
DWORD _ClientGetDDEFlags(
HANDLE hDDE,
DWORD flags)
{
PDDEPACK pDdePack;
PWORD pw;
HANDLE hData;
DWORD retval = 0;
pDdePack = (PDDEPACK)hDDE;
if (pDdePack == NULL) {
return (0);
}
if (flags & XS_DATA) {
if (pDdePack->uiLo) {
hData = (HANDLE)pDdePack->uiLo;
USERGLOBALLOCK(hData, pw);
if (pw != NULL) {
retval = (DWORD)*pw; // first word is hData is wStatus
USERGLOBALUNLOCK(hData);
}
}
} else {
retval = (DWORD)pDdePack->uiLo;
}
return (retval);
}
LPARAM APIENTRY PackDDElParam(
UINT msg,
UINT_PTR uiLo,
UINT_PTR uiHi)
{
PDDEPACK pDdePack;
HANDLE h;
switch (msg) {
case WM_DDE_EXECUTE:
return((LPARAM)uiHi);
case WM_DDE_ACK:
case WM_DDE_ADVISE:
case WM_DDE_DATA:
case WM_DDE_POKE:
h = UserGlobalAlloc(GMEM_DDESHARE | GMEM_FIXED, sizeof(DDEPACK));
pDdePack = (PDDEPACK)h;
if (pDdePack == NULL) {
return(0);
}
pDdePack->uiLo = uiLo;
pDdePack->uiHi = uiHi;
return((LPARAM)h);
default:
return(MAKELONG((WORD)uiLo, (WORD)uiHi));
}
}
BOOL APIENTRY UnpackDDElParam(
UINT msg,
LPARAM lParam,
PUINT_PTR puiLo,
PUINT_PTR puiHi)
{
PDDEPACK pDdePack;
switch (msg) {
case WM_DDE_EXECUTE:
if (puiLo != NULL) {
*puiLo = 0L;
}
if (puiHi != NULL) {
*puiHi = (UINT_PTR)lParam;
}
return(TRUE);
case WM_DDE_ACK:
case WM_DDE_ADVISE:
case WM_DDE_DATA:
case WM_DDE_POKE:
pDdePack = (PDDEPACK)lParam;
if (pDdePack == NULL || !GlobalHandle(pDdePack)) {
return(FALSE);
}
if (puiLo != NULL) {
*puiLo = pDdePack->uiLo;
}
if (puiHi != NULL) {
*puiHi = pDdePack->uiHi;
}
return(TRUE);
default:
if (puiLo != NULL) {
*puiLo = (UINT)LOWORD(lParam);
}
if (puiHi != NULL) {
*puiHi = (UINT)HIWORD(lParam);
}
return(TRUE);
}
}
BOOL APIENTRY FreeDDElParam(
UINT msg,
LPARAM lParam)
{
switch (msg) {
case WM_DDE_ACK:
case WM_DDE_ADVISE:
case WM_DDE_DATA:
case WM_DDE_POKE:
/*
* Do a range check and call GlobalFlags to validate,
* just to prevent heap checking from complaining
*/
if ((lParam > (LPARAM)0xFFFF) && GlobalFlags((HANDLE)lParam) != GMEM_INVALID_HANDLE) {
if (GlobalHandle((HANDLE)lParam))
return(UserGlobalFree((HANDLE)lParam) == NULL);
}
default:
return(TRUE);
}
}
LPARAM APIENTRY ReuseDDElParam(
LPARAM lParam,
UINT msgIn,
UINT msgOut,
UINT_PTR uiLo,
UINT_PTR uiHi)
{
PDDEPACK pDdePack;
switch (msgIn) {
case WM_DDE_ACK:
case WM_DDE_DATA:
case WM_DDE_POKE:
case WM_DDE_ADVISE:
// Incoming message was packed...
switch (msgOut) {
case WM_DDE_EXECUTE:
FreeDDElParam(msgIn, lParam);
return((LPARAM)uiHi);
case WM_DDE_ACK:
case WM_DDE_ADVISE:
case WM_DDE_DATA:
case WM_DDE_POKE:
/*
* This must be a valid handle
*/
UserAssert(GlobalFlags((HANDLE)lParam) != GMEM_INVALID_HANDLE);
UserAssert(GlobalSize((HANDLE)lParam) == sizeof(DDEPACK));
// Actual cases where lParam can be reused.
pDdePack = (PDDEPACK)lParam;
if (pDdePack == NULL) {
return(0); // the only error case
}
pDdePack->uiLo = uiLo;
pDdePack->uiHi = uiHi;
return(lParam);
default:
FreeDDElParam(msgIn, lParam);
return(MAKELONG((WORD)uiLo, (WORD)uiHi));
}
default:
// Incoming message was not packed ==> PackDDElParam()
return(PackDDElParam(msgOut, uiLo, uiHi));
}
}