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

945 lines
26 KiB
C

/****************************** Module Header ******************************\
* Module Name: xact.c
* Copyright (c) 1985 - 1999, Microsoft Corporation
* DDE Manager transaction processing module
* Created: 11/3/91 Sanford Staab
*/
#include "precomp.h"
#pragma hdrstop
/*
* DdeClientTransaction (DDEML API)
* Description:
* Initiates all DDE transactions.
* History:
* 11-1-91 sanfords Created.
*/
HDDEDATA DdeClientTransaction(
LPBYTE pData,
DWORD cbData,
HCONV hConv,
HSZ hszItem,
UINT wFmt,
UINT wType,
DWORD ulTimeout,
LPDWORD pulResult)
{
MSG msg;
PCL_INSTANCE_INFO pcii = NULL;
HDDEDATA hRet = 0;
PCL_CONV_INFO pci;
PDDEMLDATA pdd = NULL;
PXACT_INFO pxi;
BOOL fStarted;
PDDE_DATA pdde;
EnterDDECrit;
pci = (PCL_CONV_INFO)ValidateCHandle((HANDLE)hConv,
HTYPE_CLIENT_CONVERSATION, HINST_ANY);
if (pci == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
goto Exit;
}
pcii = pci->ci.pcii;
if (ulTimeout != TIMEOUT_ASYNC && GetClientInfo()->CI_flags & CI_IN_SYNC_TRANSACTION) {
SetLastDDEMLError(pcii, DMLERR_REENTRANCY);
goto Exit;
}
if (!(pci->ci.state & ST_CONNECTED)) {
SetLastDDEMLError(pcii, DMLERR_NO_CONV_ESTABLISHED);
goto Exit;
}
switch (wType) {
case XTYP_POKE:
case XTYP_ADVSTART:
case XTYP_ADVSTART | XTYPF_NODATA:
case XTYP_ADVSTART | XTYPF_ACKREQ:
case XTYP_ADVSTART | XTYPF_NODATA | XTYPF_ACKREQ:
case XTYP_REQUEST:
case XTYP_ADVSTOP:
if (hszItem == 0) {
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
break;
case XTYP_EXECUTE: // just ignore wFmt & hszItem
break;
default:
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
pxi = DDEMLAlloc(sizeof(XACT_INFO));
if (pxi == NULL) {
SetLastDDEMLError(pcii, DMLERR_MEMORY_ERROR);
goto Exit;
}
switch (wType) {
case XTYP_EXECUTE:
case XTYP_POKE:
if ((LONG)cbData == -1L) {
// We are accepting an existing data handle for export to another
// app.
pdd = (PDDEMLDATA)ValidateCHandle((HANDLE)pData,
HTYPE_DATA_HANDLE, HINST_ANY);
if (pdd == NULL) {
InvParam:
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
DDEMLFree(pxi);
goto Exit;
}
// make sure data handle holds apropriate data for this transaction
if ((pdd->flags & HDATA_EXECUTE && wType != XTYP_EXECUTE) ||
(!(pdd->flags & HDATA_EXECUTE) && wType == XTYP_EXECUTE)) {
goto InvParam;
}
// To simplify life, use a copy if this handle is potentially
// a relay or APPOWNED handle.
if (pdd->flags & (HDATA_APPOWNED | HDATA_NOAPPFREE)) {
pxi->hDDESent = CopyDDEData(pdd->hDDE, wType == XTYP_EXECUTE);
if (!pxi->hDDESent) {
MemErr:
DDEMLFree(pxi);
SetLastDDEMLError(pcii, DMLERR_MEMORY_ERROR);
goto Exit;
}
USERGLOBALLOCK(pxi->hDDESent, pdde);
if (pdde == NULL) {
FreeDDEData(pxi->hDDESent, TRUE, TRUE);
goto MemErr;
}
pdde->wStatus = DDE_FRELEASE;
USERGLOBALUNLOCK(pxi->hDDESent);
} else {
pxi->hDDESent = pdd->hDDE;
}
// make sure handle has proper format
if (wType == XTYP_POKE) {
USERGLOBALLOCK(pxi->hDDESent, pdde);
if (pdde == NULL) {
goto InvParam;
}
pdde->wFmt = (WORD)wFmt;
USERGLOBALUNLOCK(pxi->hDDESent);
}
} else { // Convert data in buffer into an apropriate hDDE
if (wType == XTYP_POKE) {
pxi->hDDESent = AllocAndSetDDEData(pData, cbData,
DDE_FRELEASE, (WORD)wFmt);
} else {
pxi->hDDESent = AllocAndSetDDEData(pData, cbData, 0, 0);
}
if (!pxi->hDDESent) {
goto MemErr;
}
}
}
// FINALLY - start the transaction
pxi->pcoi = (PCONV_INFO)pci;
pxi->gaItem = LocalToGlobalAtom(LATOM_FROM_HSZ(hszItem)); // pxi copy
pxi->wFmt = (WORD)wFmt;
pxi->wType = (WORD)wType;
switch (wType) {
case XTYP_ADVSTART:
case XTYP_ADVSTART | XTYPF_NODATA:
case XTYP_ADVSTART | XTYPF_ACKREQ:
case XTYP_ADVSTART | XTYPF_NODATA | XTYPF_ACKREQ:
fStarted = ClStartAdvise(pxi);
break;
case XTYP_ADVSTOP:
fStarted = ClStartUnadvise(pxi);
break;
case XTYP_EXECUTE:
fStarted = ClStartExecute(pxi);
break;
case XTYP_POKE:
fStarted = ClStartPoke(pxi);
break;
case XTYP_REQUEST:
fStarted = ClStartRequest(pxi);
}
if (!fStarted) {
// if we copied or allocated data - free it.
if (pxi->hDDESent && (pdd == NULL || pxi->hDDESent != pdd->hDDE)) {
FreeDDEData(pxi->hDDESent, FALSE, TRUE); // free data copy
}
GlobalDeleteAtom(pxi->gaItem); // pxi copy
DDEMLFree(pxi);
goto Exit;
}
if (pdd != NULL && !(pdd->flags & (HDATA_NOAPPFREE | HDATA_APPOWNED))) {
// invalidate given handle on success - unless we copied it because
// the app will either be return ing it from a callback or potentially
// using it again.
DDEMLFree(pdd);
DestroyHandle((HANDLE)pData);
}
if (ulTimeout == TIMEOUT_ASYNC) {
// asynchronous transaction
if (pulResult != NULL) {
pxi->hXact = CreateHandle((ULONG_PTR)pxi, HTYPE_TRANSACTION,
InstFromHandle(pcii->hInstClient));
*pulResult = HandleToUlong(pxi->hXact);
}
hRet = (HDDEDATA)TRUE;
} else {
// synchronous transaction
GetClientInfo()->CI_flags |= CI_IN_SYNC_TRANSACTION;
pcii->flags |= IIF_IN_SYNC_XACT;
pxi->flags |= XIF_SYNCHRONOUS;
NtUserSetTimer(pci->ci.hwndConv, TID_TIMEOUT, ulTimeout, NULL);
LeaveDDECrit;
CheckDDECritOut;
GetMessage(&msg, (HWND)NULL, 0, 0);
/*
* stay in modal loop until a timeout happens.
*/
while (msg.hwnd != pci->ci.hwndConv || msg.message != WM_TIMER ||
(msg.wParam != TID_TIMEOUT)) {
if (!CallMsgFilter(&msg, MSGF_DDEMGR))
DispatchMessage(&msg);
GetMessage(&msg, (HWND)NULL, 0, 0);
}
EnterDDECrit;
NtUserKillTimer(pci->ci.hwndConv, TID_TIMEOUT);
GetClientInfo()->CI_flags &= ~CI_IN_SYNC_TRANSACTION;
pcii->flags &= ~IIF_IN_SYNC_XACT;
if (pxi->flags & XIF_COMPLETE) {
if (pulResult != NULL) {
*pulResult = pxi->wStatus; // NACK status bits
}
switch (wType) {
case XTYP_ADVSTART:
case XTYP_ADVSTART | XTYPF_NODATA:
case XTYP_ADVSTART | XTYPF_ACKREQ:
case XTYP_ADVSTART | XTYPF_NODATA | XTYPF_ACKREQ:
case XTYP_ADVSTOP:
case XTYP_EXECUTE:
case XTYP_POKE:
hRet = (HDDEDATA)((pxi->wStatus & DDE_FACK) ? TRUE : FALSE);
if (!hRet) {
if (pxi->wStatus & DDE_FBUSY) {
SetLastDDEMLError(pcii, DMLERR_BUSY);
} else {
SetLastDDEMLError(pcii, DMLERR_NOTPROCESSED);
}
}
break;
case XTYP_REQUEST:
if (pxi->hDDEResult == 0) {
hRet = (HDDEDATA)((pxi->wStatus & DDE_FACK) ? TRUE : FALSE);
if (!hRet) {
if (pxi->wStatus & DDE_FBUSY) {
SetLastDDEMLError(pcii, DMLERR_BUSY);
} else {
SetLastDDEMLError(pcii, DMLERR_NOTPROCESSED);
}
}
break;
}
// Note that if the incoming data didn't have the DDE_FRELEASE
// bit set, the transaction code would have made a copy so
// the app is free to keep is as long as he likes.
hRet = InternalCreateDataHandle(pcii, (LPBYTE)pxi->hDDEResult, (DWORD)-1, 0,
HDATA_READONLY, 0, 0);
pxi->hDDEResult = 0; // so cleanup doesn't free it.
}
(pxi->pfnResponse)((struct tagXACT_INFO *)pxi, 0, 0); // cleanup transaction
} else { // Timed out
// abandon the transaction and make it asyncronous so it will
// clean itself up when the response finally comes in.
pxi->flags &= ~XIF_SYNCHRONOUS;
pxi->flags |= XIF_ABANDONED;
switch (wType) {
case XTYP_ADVSTART:
case XTYP_ADVSTART | XTYPF_NODATA:
case XTYP_ADVSTART | XTYPF_ACKREQ:
case XTYP_ADVSTART | XTYPF_NODATA | XTYPF_ACKREQ:
SetLastDDEMLError(pcii, DMLERR_ADVACKTIMEOUT);
break;
case XTYP_ADVSTOP:
SetLastDDEMLError(pcii, DMLERR_UNADVACKTIMEOUT);
break;
case XTYP_EXECUTE:
SetLastDDEMLError(pcii, DMLERR_EXECACKTIMEOUT);
break;
case XTYP_POKE:
SetLastDDEMLError(pcii, DMLERR_POKEACKTIMEOUT);
break;
case XTYP_REQUEST:
SetLastDDEMLError(pcii, DMLERR_DATAACKTIMEOUT);
break;
}
// cleanup of pxi happens when transaction actually completes.
}
}
if (pci->ci.state & ST_FREE_CONV_RES_NOW) {
/*
* The conversation was terminated during the synchronous transaction
* so we need to clean up now that we are out of the loop.
*/
FreeConversationResources((PCONV_INFO)pci);
}
Exit:
/*
* Because this API is capable of blocking DdeUninitialize(), we check
* before exit to see if it needs to be called.
*/
if (pcii != NULL &&
(pcii->afCmd & APPCMD_UNINIT_ASAP) &&
// !(pcii->flags & IIF_IN_SYNC_XACT) &&
!pcii->cInDDEMLCallback) {
DdeUninitialize(HandleToUlong(pcii->hInstClient));
hRet = 0;
}
LeaveDDECrit;
return (hRet);
}
/*
* GetConvContext
* Description:
* Retrieves conversation context information from the DDEML client window
* given. pl points to a CONVCONTEXT structure.
* History:
* 11-12-91 sanfords Created.
*/
VOID GetConvContext(
HWND hwnd,
LONG *pl)
{
int i;
for (i = 0; i < sizeof(CONVCONTEXT); i += 4) {
*pl++ = GetWindowLong(hwnd, GWL_CONVCONTEXT + i);
}
}
/*
* SetConvContext
* Description:
* History:
* 11-19-92 sanfords Created.
*/
VOID SetConvContext(
HWND hwnd,
LONG *pl)
{
int i;
for (i = 0; i < sizeof(CONVCONTEXT); i += 4) {
SetWindowLong(hwnd, GWL_CONVCONTEXT + i, *pl++);
}
}
/*
* DdeQueryConvInfo (DDEML API)
* Description:
* Retrieves detailed conversation information on a per conversation/
* transaction basis.
* History:
* 11-12-91 sanfords Created.
*/
UINT DdeQueryConvInfo(
HCONV hConv,
DWORD idTransaction,
PCONVINFO pConvInfo)
{
PCONV_INFO pcoi;
PXACT_INFO pxi;
CONVINFO ci;
UINT uiRet = 0;
EnterDDECrit;
if (!ValidateTransaction(hConv, (HANDLE)LongToHandle( idTransaction ), &pcoi, &pxi)) {
goto Exit;
}
try {
if (pConvInfo->cb > sizeof(CONVINFO)) {
SetLastDDEMLError(pcoi->pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
ci.cb = pConvInfo->cb;
ci.hConvPartner = 0; // no longer supported.
ci.hszSvcPartner = NORMAL_HSZ_FROM_LATOM(pcoi->laService);
ci.hszServiceReq = NORMAL_HSZ_FROM_LATOM(pcoi->laServiceRequested);
ci.hszTopic = NORMAL_HSZ_FROM_LATOM(pcoi->laTopic);
ci.wStatus = pcoi->state;
ci.wLastError = (WORD)pcoi->pcii->LastError;
if (pcoi->state & ST_CLIENT) {
ci.hConvList = ((PCL_CONV_INFO)pcoi)->hConvList;
GetConvContext(pcoi->hwndConv, (LONG *)&ci.ConvCtxt);
} else {
ci.hConvList = 0;
if (pcoi->state & ST_ISLOCAL) {
GetConvContext(pcoi->hwndPartner, (LONG *)&ci.ConvCtxt);
} else {
ci.ConvCtxt = DefConvContext;
}
}
if (pxi == NULL) {
ci.hUser = pcoi->hUser;
ci.hszItem = 0;
ci.wFmt = 0;
ci.wType = 0;
ci.wConvst = XST_CONNECTED;
} else {
ci.hUser = pxi->hUser;
// BUG - not fixable - This will result in extra local atoms
// since we can never know when he is done with them.
ci.hszItem = NORMAL_HSZ_FROM_LATOM(GlobalToLocalAtom(pxi->gaItem));
ci.wFmt = pxi->wFmt;
ci.wType = pxi->wType;
ci.wConvst = pxi->state;
}
ci.hwnd = pcoi->hwndConv;
ci.hwndPartner = pcoi->hwndPartner;
RtlCopyMemory((LPSTR)pConvInfo, (LPSTR)&ci, pConvInfo->cb);
} except(W32ExceptionHandler(FALSE, RIP_WARNING)) {
SetLastDDEMLError(pcoi->pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
uiRet = TRUE;
Exit:
LeaveDDECrit;
return (uiRet);
}
/*
* DdeSetUserHandle (DDEML API)
* Description:
* Sets a user DWORD on a per conversation/transaction basis.
* History:
* 11-12-91 sanfords Created.
*/
BOOL DdeSetUserHandle(
HCONV hConv,
DWORD id,
DWORD_PTR hUser)
{
PCONV_INFO pcoi;
PXACT_INFO pxi;
BOOL fRet = FALSE;
EnterDDECrit;
if (!ValidateTransaction(hConv, (HANDLE)LongToHandle( id ), &pcoi, &pxi)) {
goto Exit;
}
if (pxi == NULL) {
pcoi->hUser = hUser;
} else {
pxi->hUser = hUser;
}
fRet = TRUE;
Exit:
LeaveDDECrit;
return (fRet);
}
VOID AbandonTransaction(
PCONV_INFO pcoi,
PXACT_INFO pxi)
{
if (pxi != NULL) {
pxi->flags |= XIF_ABANDONED;
} else {
for (pxi = pcoi->pxiIn; pxi != NULL; pxi = pxi->next) {
pxi->flags |= XIF_ABANDONED;
}
}
}
BOOL AbandonEnumerateProc(
HWND hwnd,
LPARAM idTransaction)
{
PCONV_INFO pcoi;
pcoi = (PCONV_INFO)GetWindowLongPtr(hwnd, GWLP_PCI);
if (!pcoi || !(pcoi->state & ST_CLIENT)) {
return(TRUE);
}
while (pcoi) {
AbandonTransaction(pcoi, (PXACT_INFO)idTransaction);
pcoi = pcoi->next;
}
return(TRUE);
}
/*
* DdeAbandonTransaction (DDEML API)
* Description:
* Cancels application interest in completing an asynchronous transaction.
* History:
* 11-12-91 sanfords Created.
*/
BOOL DdeAbandonTransaction(
DWORD idInst,
HCONV hConv,
DWORD idTransaction)
{
PCONV_INFO pcoi;
PXACT_INFO pxi;
PCL_INSTANCE_INFO pcii;
BOOL fRet = FALSE;
EnterDDECrit;
pcii = ValidateInstance((HANDLE)LongToHandle( idInst ));
if (hConv == 0 && idTransaction == 0) {
EnumChildWindows(pcii->hwndMother, AbandonEnumerateProc, 0);
goto Exit;
}
if (idTransaction == 0) {
idTransaction = QID_SYNC;
}
if (!ValidateTransaction(hConv, (HANDLE)LongToHandle( idTransaction ), &pcoi, &pxi)) {
goto Exit;
}
if (pcii == NULL || pcoi->pcii != pcii) {
SetLastDDEMLError(pcoi->pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
AbandonTransaction(pcoi, pxi);
fRet = TRUE;
Exit:
LeaveDDECrit;
return (fRet);
}
/*
* UpdateLinkIfChanged
* Description:
* Helper function for updating a link
* Returns: TRUE if pxi was used - ie fMustReallocPxi
* History:
* 3-11-92 sanfords Created.
* 8-24-92 sanfords added cLinksToGo
*/
BOOL UpdateLinkIfChanged(
PADVISE_LINK paLink,
PXACT_INFO pxi,
PCONV_INFO pcoi,
PADVISE_LINK paLinkLast,
PBOOL pfSwapped,
DWORD cLinksToGo)
{
ADVISE_LINK aLinkT;
CheckDDECritIn;
*pfSwapped = FALSE;
if (paLink->state & ADVST_CHANGED && !(paLink->state & ADVST_WAITING)) {
pxi->pfnResponse = SvRespAdviseDataAck;
pxi->pcoi = pcoi;
pxi->gaItem = LocalToGlobalAtom(paLink->laItem); // pxi copy
pxi->wFmt = paLink->wFmt;
pxi->wType = paLink->wType;
paLink->state &= ~ADVST_CHANGED;
if (SvStartAdviseUpdate(pxi, cLinksToGo)) {
if (pxi->wType & DDE_FACKREQ) {
paLink->state |= ADVST_WAITING;
/*
* swap paLink with the last non-moved link to make ack search find
* oldest updated format.
*/
if (paLink != paLinkLast) {
aLinkT = *paLink;
RtlMoveMemory(paLink, paLink + 1,
(PBYTE)paLinkLast - (PBYTE)paLink);
*paLinkLast = aLinkT;
*pfSwapped = TRUE;
}
}
return(TRUE);
} else {
GlobalDeleteAtom(pxi->gaItem); // pxi copy
return(FALSE);
}
}
return(FALSE);
}
/*
* DdePostAdvise (DDEML API)
* Description:
* Updates outstanding server advise links as needed.
* History:
* 11-12-91 sanfords Created.
*/
BOOL DdePostAdvise(
DWORD idInst,
HSZ hszTopic,
HSZ hszItem)
{
PCL_INSTANCE_INFO pcii;
PSVR_CONV_INFO psi;
PXACT_INFO pxi;
PADVISE_LINK paLink;
BOOL fRet = FALSE, fSwapped, fFound;
int iServer, iLink;
PLINK_COUNT pLinkCount;
#if DBG
int cLinks;
#endif
EnterDDECrit;
pcii = ValidateInstance((HANDLE)LongToHandle( idInst ));
if (pcii == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
goto Exit;
}
if ((ValidateHSZ(hszTopic) == HSZT_INVALID) ||
(ValidateHSZ(hszItem) == HSZT_INVALID)) {
SetLastDDEMLError(pcii, DMLERR_INVALIDPARAMETER);
goto Exit;
}
/*
* Initialize all link counters and check if any links qualify
*/
fFound = FALSE;
for (pLinkCount = pcii->pLinkCount;
pLinkCount; pLinkCount = pLinkCount->next) {
pLinkCount->Count = pLinkCount->Total;
fFound |= pLinkCount->laTopic == LATOM_FROM_HSZ(hszTopic) &&
pLinkCount->laItem == LATOM_FROM_HSZ(hszItem);
}
if (!fFound && hszTopic && hszItem) {
fRet = TRUE;
goto Exit;
}
/*
* preallocate incase we are low on memory.
*/
pxi = DDEMLAlloc(sizeof(XACT_INFO));
if (pxi == NULL) {
SetLastDDEMLError(pcii, DMLERR_MEMORY_ERROR);
fRet = FALSE;
goto Exit;
}
/*
* For each server window on the specified topic
*/
for (iServer = 0; iServer < pcii->cServerLookupAlloc; iServer++) {
if (hszTopic == 0 ||
pcii->aServerLookup[iServer].laTopic == LATOM_FROM_HSZ(hszTopic)) {
/*
* For each conversation within that window
*/
psi = (PSVR_CONV_INFO)GetWindowLongPtr(
pcii->aServerLookup[iServer].hwndServer, GWLP_PSI);
UserAssert(psi != NULL && psi->ci.pcii == pcii); // sanity check
while (psi != NULL) {
/*
* UpdateLinkIfChanged might leave the critical section so lock this conversation
*/
psi->ci.cLocks++;
#if DBG
/*
* Rememeber the number of links so we can assert if they change during the loop below
*/
cLinks = psi->ci.cLinks;
#endif
/*
* For each active link on the given item...
*/
for (paLink = psi->ci.aLinks, iLink = 0;
iLink < psi->ci.cLinks; paLink++, iLink++) {
if (hszItem == 0 ||
paLink->laItem == LATOM_FROM_HSZ(hszItem)) {
// Bit of a hack here. For FACKREQ links, we don't want the server to
// outrun the client so we set the ADVST_WAITING bit till the ack is
// received. When the ack comes in, the protocol code has to search
// the aLinks array again to locate the apropriate link state flags and
// clear the ADVST_WAITING flag. At that time, if the ADVST_CHANGED flag
// is set, it is cleared and another SvStartAdviseUpdate transaction
// is started to get the link up to date. To complicate matters,
// the ACK contains no format information. Thus we need to move
// the Link info to the end of the list so that the right format
// is updated when the ack comes in.
paLink->state |= ADVST_CHANGED;
if (UpdateLinkIfChanged(paLink, pxi, &psi->ci,
&psi->ci.aLinks[psi->ci.cLinks - 1],
&fSwapped, --paLink->pLinkCount->Count)) {
if (fSwapped) {
paLink--;
}
/*
* preallocate for next advise
*/
pxi = DDEMLAlloc(sizeof(XACT_INFO));
if (pxi == NULL) {
SetLastDDEMLError(pcii, DMLERR_MEMORY_ERROR);
/*
* Unlock the conversation
*/
psi->ci.cLocks--;
if ((psi->ci.cLocks == 0) && (psi->ci.state & ST_FREE_CONV_RES_NOW)) {
RIPMSG1(RIP_ERROR, "DdePostAdvise: Conversation terminated. psi:%#p", psi);
FreeConversationResources((PCONV_INFO)psi);
}
goto Exit;
}
}
/*
* We might have left the crit sect...
*/
UserAssert(pcii == ValidateInstance((HANDLE)LongToHandle( idInst )));
}
}
#if DBG
if (cLinks != psi->ci.cLinks) {
RIPMSG1(RIP_ERROR, "DdePostAdvise: cLinks changed. psi:%#p", psi);
}
#endif
/*
* If the converstaion got nuked, stop working on this conversation chain.
*/
psi->ci.cLocks--;
if ((psi->ci.cLocks == 0) && (psi->ci.state & ST_FREE_CONV_RES_NOW)) {
RIPMSG1(RIP_ERROR, "DdePostAdvise: Conversation terminated. psi:%#p", psi);
FreeConversationResources((PCONV_INFO)psi);
break;
}
psi = (PSVR_CONV_INFO)psi->ci.next; // next conversation
}
}
}
DDEMLFree(pxi);
fRet = TRUE;
Exit:
/*
* Because callbacks are capable of blocking DdeUninitialize(), we check
* before exit to see if it needs to be called.
*/
UserAssert(pcii == ValidateInstance((HANDLE)LongToHandle( idInst )));
if (pcii != NULL &&
pcii->afCmd & APPCMD_UNINIT_ASAP &&
!(pcii->flags & IIF_IN_SYNC_XACT) &&
!pcii->cInDDEMLCallback) {
DdeUninitialize(HandleToUlong(pcii->hInstClient));
fRet = TRUE;
}
LeaveDDECrit;
return (fRet);
}
/*
* LinkTransaction
* Description:
* Adds a transaction structure to the associated conversation's transaction
* queue.
* History:
* 11-12-91 sanfords Created.
*/
VOID LinkTransaction(
PXACT_INFO pxi)
{
CheckDDECritIn;
pxi->next = NULL;
if (pxi->pcoi->pxiOut == NULL) {
pxi->pcoi->pxiIn = pxi->pcoi->pxiOut = pxi;
} else {
pxi->pcoi->pxiIn->next = pxi;
pxi->pcoi->pxiIn = pxi;
}
#if DBG
/*
* Temporary check to find stress bug - make sure pxi list is not
* looped on itself. If it is, this loop will never exit and things
* will get investigated. (sanfords)
*/
{
PXACT_INFO pxiT;
for (pxiT = pxi->pcoi->pxiOut; pxiT != NULL; pxiT = pxiT->next) {
;
}
}
#endif // DBG
}
/*
* UnlinkTransaction
* Description:
* Removes a transaction structure from the associated conversation's transaction
* queue.
* History:
* 11-12-91 sanfords Created.
*/
VOID UnlinkTransaction(
PXACT_INFO pxi)
{
CheckDDECritIn;
if (pxi == pxi->pcoi->pxiOut) {
pxi->pcoi->pxiOut = pxi->next;
if (pxi->next == NULL) {
pxi->pcoi->pxiIn = NULL;
}
}
}
/*
* ValidateTransaction
* Description:
* Common validation code for DDEML APIs that take a conversation handle
* and a transaction ID. *ppxi may be null on return if hXact was 0.
* Returns fSuccess.
* History:
* 11-12-91 sanfords Created.
*/
BOOL ValidateTransaction(
HCONV hConv,
HANDLE hXact,
PCONV_INFO *ppcoi,
PXACT_INFO *ppxi)
{
PCL_INSTANCE_INFO pcii;
*ppcoi = (PCONV_INFO)ValidateCHandle((HANDLE)hConv,
HTYPE_CLIENT_CONVERSATION, HINST_ANY);
if (*ppcoi == NULL) {
*ppcoi = (PCONV_INFO)ValidateCHandle((HANDLE)hConv,
HTYPE_SERVER_CONVERSATION, HINST_ANY);
}
if (*ppcoi == NULL) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
return (FALSE);
}
pcii = ValidateInstance((*ppcoi)->pcii->hInstClient);
if (pcii != (*ppcoi)->pcii) {
BestSetLastDDEMLError(DMLERR_INVALIDPARAMETER);
return (FALSE);
}
if (hXact == (HANDLE)IntToPtr( QID_SYNC )) {
*ppxi = NULL;
} else {
*ppxi = (PXACT_INFO)ValidateCHandle(hXact, HTYPE_TRANSACTION,
InstFromHandle((*ppcoi)->pcii->hInstClient));
if (*ppxi == NULL) {
SetLastDDEMLError((*ppcoi)->pcii, DMLERR_INVALIDPARAMETER);
return (FALSE);
}
}
return (TRUE);
}