/****************************** Module Header ******************************\ * Module Name: ddemlwp.c * Copyright (c) 1985 - 1999, Microsoft Corporation * DDE Manager client side window procedures * Created: 11/3/91 Sanford Staab */ #include "precomp.h" #pragma hdrstop VOID ProcessDDEMLInitiate(PCL_INSTANCE_INFO pcii, HWND hwndClient, GATOM aServer, GATOM aTopic); LRESULT DDEMLMotherWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) /* * Description: * Handles WM_DDE_INITIATE messages for DDEML and holds all the other windows for a DDEML instance. * History: * 12-29-92 sanfords Created. */ { switch (message) { case UM_REGISTER: case UM_UNREGISTER: return(ProcessRegistrationMessage(hwnd, message, wParam, lParam)); case WM_DDE_INITIATE: ProcessDDEMLInitiate((PCL_INSTANCE_INFO)GetWindowLongPtr(hwnd, GWLP_PCI), (HWND)wParam, (ATOM)LOWORD(lParam), (ATOM)HIWORD(lParam)); return(0); } return(DefWindowProc(hwnd, message, wParam, lParam)); } VOID ProcessDDEMLInitiate(PCL_INSTANCE_INFO pcii, HWND hwndClient, GATOM aServer, GATOM aTopic) /* * Description: * WM_DDE_INITIATE messages are processed here. * History: * 12-29-92 sanfords Created. */ { CONVCONTEXT cc = { sizeof(CONVCONTEXT), 0, 0, CP_WINANSI, 0L, 0L, { sizeof(SECURITY_QUALITY_OF_SERVICE), SecurityImpersonation, SECURITY_STATIC_TRACKING, TRUE } }; BOOL flags = ST_INLIST; BOOL fWild; HDDEDATA hData; HWND hwndServer; PSERVER_LOOKUP psl; PHSZPAIR php; HSZPAIR hp[2]; LATOM laService, laFree1 = 0; LATOM laTopic, laFree2 = 0; PSVR_CONV_INFO psi; LATOM* plaNameService; PWND pwndClient; PCLS pcls; if (pcii == NULL) { return; // we aren't done being initiated yet. } EnterDDECrit; if (pcii->afCmd & CBF_FAIL_CONNECTIONS || !IsWindow(hwndClient)) { goto Exit; } pwndClient = ValidateHwnd(hwndClient); if (pwndClient == NULL) goto Exit; pcls = (PCLS)REBASEALWAYS(pwndClient, pcls); if (!TestWF(pwndClient, WFANSIPROC)) { if (pcls->atomClassName == gpsi->atomSysClass[ICLS_DDEMLCLIENTW]) { flags |= ST_ISLOCAL; } } else { if (pcls->atomClassName == gpsi->atomSysClass[ICLS_DDEMLCLIENTA]) { flags |= ST_ISLOCAL; } } if (flags & ST_ISLOCAL) { /* * Make sure other guy allows self-connections if that's what this is. */ if (pcii->hInstServer == (HANDLE)GetWindowLongPtr(hwndClient, GWLP_SHINST)) { if (pcii->afCmd & CBF_FAIL_SELFCONNECTIONS) { goto Exit; } flags |= ST_ISSELF; } GetConvContext(hwndClient, (LONG*)&cc); if (GetWindowLong(hwndClient, GWL_CONVSTATE) & CLST_SINGLE_INITIALIZING) { flags &= ~ST_INLIST; } } else { NtUserDdeGetQualityOfService(hwndClient, NULL, &cc.qos); } /* * Server window creation is minimized by only creating one window per * Instance/Service/Topic set. This should be all that is needed and * duplicate connections (ie where the server/client window pair is identical * to another conversation) should not happen. However, if some dumb * server app attempts to create a duplicate conversation by having * duplicate service/topic pairs passed back from a XTYP_WILD_CONNECT * callback we will not honor the request. * The INSTANCE_INFO structure holds a pointer to an array of SERVERLOOKUP * structures each entry of which references the hwndServer that supports * all conversations on that service/topic pair. The hwndServer windows * in turn have window words that reference the first member in a linked * list of SVR_CONV_INFO structures, one for each conversation on that * service/topic pair. */ laFree1 = laService = GlobalToLocalAtom(aServer); laFree2 = laTopic = GlobalToLocalAtom(aTopic); plaNameService = pcii->plaNameService; if (!laService && pcii->afCmd & APPCMD_FILTERINITS && *plaNameService == 0) { /* * no WILDCONNECTS to servers with no registered names while filtering. */ goto Exit; } if ((pcii->afCmd & APPCMD_FILTERINITS) && laService) { /* * if we can't find the aServer in this instance's service name * list, don't bother the server. */ while (*plaNameService != 0 && *plaNameService != laService) { plaNameService++; } if (*plaNameService == 0) { goto Exit; } } hp[0].hszSvc = NORMAL_HSZ_FROM_LATOM(laService); hp[0].hszTopic = NORMAL_HSZ_FROM_LATOM(laTopic); hp[1].hszSvc = 0; hp[1].hszTopic = 0; fWild = !laService || !laTopic; hData = DoCallback(pcii, (WORD)(fWild ? XTYP_WILDCONNECT : XTYP_CONNECT), 0, (HCONV)0, hp[0].hszTopic, hp[0].hszSvc, (HDDEDATA)0, flags & ST_ISLOCAL ? (ULONG_PTR)&cc : 0, (DWORD)(flags & ST_ISSELF) ? 1 : 0); if (!hData) { goto Exit; } if (fWild) { php = (PHSZPAIR)DdeAccessData(hData, NULL); if (php == NULL) { goto Exit; } } else { php = hp; } while (php->hszSvc && php->hszTopic) { psi = (PSVR_CONV_INFO)DDEMLAlloc(sizeof(SVR_CONV_INFO)); if (psi == NULL) { break; } laService = LATOM_FROM_HSZ(php->hszSvc); laTopic = LATOM_FROM_HSZ(php->hszTopic); hwndServer = 0; if (pcii->cServerLookupAlloc) { int i; /* * See if there already exists a server window for this * aServer/aTopic pair */ for (i = pcii->cServerLookupAlloc; i; i--) { if (pcii->aServerLookup[i - 1].laService == laService && pcii->aServerLookup[i - 1].laTopic == laTopic) { PSVR_CONV_INFO psiT; PCONV_INFO pcoi; hwndServer = pcii->aServerLookup[i - 1].hwndServer; /* * Now make sure this window isn't some bogus idiot * trying to create a second conversation from the * same client window that is already talking to * our existing server window. */ psiT = (PSVR_CONV_INFO)GetWindowLongPtr(hwndServer, GWLP_PSI); for (pcoi = &psiT->ci; pcoi != NULL; pcoi = pcoi->next) { if (pcoi->hwndPartner == hwndClient) { hwndServer = NULL; break; } } break; } } } if (hwndServer == 0) { // no server window exists - make one. LeaveDDECrit; if (pcii->flags & IIF_UNICODE) { hwndServer = CreateWindowW((LPWSTR)(gpsi->atomSysClass[ICLS_DDEMLSERVERW]), L"", WS_CHILD, 0, 0, 0, 0, pcii->hwndMother, (HMENU)0, 0, (LPVOID)NULL); } else { hwndServer = CreateWindowA((LPSTR)(gpsi->atomSysClass[ICLS_DDEMLSERVERA]), "", WS_CHILD, 0, 0, 0, 0, pcii->hwndMother, (HMENU)0, 0, (LPVOID)NULL); } EnterDDECrit; if (hwndServer == 0) { DDEMLFree(psi); break; } // SetWindowLongPtr(hwndServer, GWLP_PSI, (LONG)NULL); // Zero init. // put the window into the lookup list if (pcii->aServerLookup == NULL) { psl = (PSERVER_LOOKUP)DDEMLAlloc(sizeof(SERVER_LOOKUP)); } else { psl = (PSERVER_LOOKUP)DDEMLReAlloc(pcii->aServerLookup, sizeof(SERVER_LOOKUP) * (pcii->cServerLookupAlloc + 1)); } if (psl == NULL) { RIPMSG1(RIP_WARNING, "ProcessDDEMLInitiate:hwndServer (%x) destroyed due to low memory.", hwndServer); NtUserDestroyWindow(hwndServer); DDEMLFree(psi); break; } IncLocalAtomCount(laService); // for SERVER_LOOKUP psl[pcii->cServerLookupAlloc].laService = laService; IncLocalAtomCount(laTopic); // for SERVER_LOOKUP psl[pcii->cServerLookupAlloc].laTopic = laTopic; psl[pcii->cServerLookupAlloc].hwndServer = hwndServer; pcii->aServerLookup = psl; pcii->cServerLookupAlloc++; // DumpServerLookupTable("After addition:", hwndServer, psl, pcii->cServerLookupAlloc); } psi->ci.next = (PCONV_INFO)GetWindowLongPtr(hwndServer, GWLP_PSI); SetWindowLongPtr(hwndServer, GWLP_PSI, (LONG_PTR)psi); psi->ci.pcii = pcii; // psi->ci.hUser = 0; psi->ci.hConv = (HCONV)CreateHandle((ULONG_PTR)psi, HTYPE_SERVER_CONVERSATION, InstFromHandle(pcii->hInstClient)); psi->ci.laService = laService; IncLocalAtomCount(laService); // for server window psi->ci.laTopic = laTopic; IncLocalAtomCount(laTopic); // for server window psi->ci.hwndPartner = hwndClient; psi->ci.hwndConv = hwndServer; psi->ci.state = (WORD)(flags | ST_CONNECTED | pcii->ConvStartupState); SetCommonStateFlags(hwndClient, hwndServer, &psi->ci.state); psi->ci.laServiceRequested = laFree1; IncLocalAtomCount(psi->ci.laServiceRequested); // for server window // psi->ci.pxiIn = NULL; // psi->ci.pxiOut = NULL; // psi->ci.dmqIn = NULL; // psi->ci.dmqOut = NULL; // psi->ci.aLinks = NULL; // psi->ci.cLinks = 0; // psi->ci.cLocks = 0; LeaveDDECrit; CheckDDECritOut; SendMessage(hwndClient, WM_DDE_ACK, (WPARAM)hwndServer, MAKELONG(LocalToGlobalAtom(laService), LocalToGlobalAtom(laTopic))); EnterDDECrit; if (!(pcii->afCmd & CBF_SKIP_CONNECT_CONFIRMS)) { DoCallback(pcii, (WORD)XTYP_CONNECT_CONFIRM, 0, psi->ci.hConv, (HSZ)laTopic, (HSZ)laService, (HDDEDATA)0, 0, (flags & ST_ISSELF) ? 1L : 0L); } MONCONV((PCONV_INFO)psi, TRUE); if (!(flags & ST_INLIST)) { break; // our partner's only gonna take the first one anyway. } php++; } if (fWild) { DdeUnaccessData(hData); InternalFreeDataHandle(hData, FALSE); } Exit: DeleteAtom(laFree1); DeleteAtom(laFree2); LeaveDDECrit; return; } LRESULT DDEMLClientWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) /* * Description: * Handles DDE client messages for DDEML. * History: * 11-12-91 sanfords Created. */ { PCL_CONV_INFO pci, pciNew; LONG lState; LRESULT lRet = 0; PWND pwnd; PCLS pcls; EnterDDECrit; pci = (PCL_CONV_INFO)GetWindowLongPtr(hwnd, GWLP_PCI); UserAssert(pci == NULL || pci->ci.hwndConv == hwnd); switch (message) { case WM_DDE_ACK: lState = GetWindowLong(hwnd, GWL_CONVSTATE); if (lState != CLST_CONNECTED) { // Initiation mode pciNew = (PCL_CONV_INFO)DDEMLAlloc(sizeof(CL_CONV_INFO)); if (pciNew == NULL || (pci != NULL && lState == CLST_SINGLE_INITIALIZING)) { PostMessage((HWND)wParam, WM_DDE_TERMINATE, (WPARAM)hwnd, 0); goto Exit; } // PCL_CONV_INFO initialization pciNew->ci.pcii = ValidateInstance((HANDLE)GetWindowLongPtr(hwnd, GWLP_CHINST)); if (pciNew->ci.pcii == NULL) { DDEMLFree(pciNew); goto Exit; } pciNew->ci.next = (PCONV_INFO)pci; // pci may be NULL // Seting GWLP_PCI gives feedback to ConnectConv() which issued // the WM_DDE_INITIATE message. SetWindowLongPtr(hwnd, GWLP_PCI, (LONG_PTR)pciNew); // pciNew->hUser = 0; // Zero init. // BUG: If this fails we can have some nasty problems pciNew->ci.hConv = (HCONV)CreateHandle((ULONG_PTR)pciNew, HTYPE_CLIENT_CONVERSATION, InstFromHandle(pciNew->ci.pcii->hInstClient)); pciNew->ci.laService = GlobalToLocalAtom(LOWORD(lParam)); // pci copy GlobalDeleteAtom(LOWORD(lParam)); pciNew->ci.laTopic = GlobalToLocalAtom(HIWORD(lParam)); // pci copy GlobalDeleteAtom(HIWORD(lParam)); pciNew->ci.hwndPartner = (HWND)wParam; pciNew->ci.hwndConv = hwnd; pciNew->ci.state = (WORD)(ST_CONNECTED | ST_CLIENT | pciNew->ci.pcii->ConvStartupState); SetCommonStateFlags(hwnd, (HWND)wParam, &pciNew->ci.state); pwnd = ValidateHwnd((HWND)wParam); if (pwnd == NULL) goto Exit; pcls = (PCLS)REBASEALWAYS(pwnd, pcls); if (!TestWF(pwnd, WFANSIPROC)) { if (pcls->atomClassName == gpsi->atomSysClass[ICLS_DDEMLSERVERW]) { pciNew->ci.state |= ST_ISLOCAL; } } else { if (pcls->atomClassName == gpsi->atomSysClass[ICLS_DDEMLSERVERA]) { pciNew->ci.state |= ST_ISLOCAL; } } // pciNew->ci.laServiceRequested = 0; // Set by InitiateEnumerationProc() // pciNew->ci.pxiIn = 0; // pciNew->ci.pxiOut = 0; // pciNew->ci.dmqIn = 0; // pciNew->ci.dmqOut = 0; // pciNew->ci.aLinks = NULL; // pciNew->ci.cLinks = 0; // pciNew->ci.cLocks = 0; goto Exit; } // fall through to handle posted messages here. case WM_DDE_DATA: ProcessAsyncDDEMsg((PCONV_INFO)pci, message, (HWND)wParam, lParam); goto Exit; case WM_DDE_TERMINATE: case WM_DESTROY: { ProcessTerminateMsg((PCONV_INFO)pci, (HWND)wParam); break; } } lRet = DefWindowProc(hwnd, message, wParam, lParam); Exit: LeaveDDECrit; return (lRet); } LRESULT DDEMLServerWndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) /* * Description: * Handles DDE server messages. * History: * 11-12-91 sanfords Created. */ { PSVR_CONV_INFO psi; LRESULT lRet = 0; EnterDDECrit; psi = (PSVR_CONV_INFO)GetWindowLongPtr(hwnd, GWLP_PSI); UserAssert(psi == NULL || psi->ci.hwndConv == hwnd); switch (message) { case WM_DDE_REQUEST: case WM_DDE_POKE: case WM_DDE_ADVISE: case WM_DDE_EXECUTE: case WM_DDE_ACK: case WM_DDE_UNADVISE: ProcessAsyncDDEMsg((PCONV_INFO)psi, message, (HWND)wParam, lParam); goto Exit; case WM_DDE_TERMINATE: case WM_DESTROY: ProcessTerminateMsg((PCONV_INFO)psi, (HWND)wParam); break; } lRet = DefWindowProc(hwnd, message, wParam, lParam); Exit: LeaveDDECrit; return (lRet); } PCONV_INFO ProcessTerminateMsg(PCONV_INFO pcoi, HWND hwndFrom) /* * Description: * Handles WM_DDE_TERMINATE messages for both sides. * History: * 11-26-91 sanfords Created. */ { while (pcoi != NULL && pcoi->hwndPartner != hwndFrom) { pcoi = pcoi->next; } if (pcoi != NULL) { pcoi->state |= ST_TERMINATE_RECEIVED; ShutdownConversation(pcoi, TRUE); } return (pcoi); } VOID ProcessAsyncDDEMsg(PCONV_INFO pcoi, UINT msg, HWND hwndFrom, LPARAM lParam) /* * Description: * Handles incoming DDE messages by either calling ProcessSyncDDEMessage() * if the conversation is able to handle callbacks, or by queuing the * incoming message into the conversations message queue. Doing this * allows simpler code in that no message is processed unless the code * can perform synchronous callbacks. * History: * 11-26-91 sanfords Created. */ { PDDE_MESSAGE_QUEUE pdmq; #if DBG HWND hwndT = pcoi->hwndConv; #endif // DBG while (pcoi != NULL && pcoi->hwndPartner != hwndFrom) { pcoi = pcoi->next; } if (pcoi == NULL) { RIPMSG3(RIP_WARNING, "Bogus DDE message %x received from %x by %x. Dumping.", msg, hwndFrom, hwndT); DumpDDEMessage(FALSE, msg, lParam); return; } if (pcoi->state & ST_CONNECTED) { if (pcoi->dmqOut == NULL && !(pcoi->state & ST_BLOCKED) // && !PctiCurrent()->cInDDEMLCallback ) { if (ProcessSyncDDEMessage(pcoi, msg, lParam)) { return; // not blocked, ok to return. } } // enter into queue pdmq = DDEMLAlloc(sizeof(DDE_MESSAGE_QUEUE)); if (pdmq == NULL) { // insufficient memory - we can't process this msg - we MUST terminate. if (pcoi->state & ST_CONNECTED) { PostMessage(pcoi->hwndPartner, WM_DDE_TERMINATE, (WPARAM)pcoi->hwndConv, 0); pcoi->state &= ~ST_CONNECTED; } DumpDDEMessage(!(pcoi->state & ST_INTRA_PROCESS), msg, lParam); return; } pdmq->pcoi = pcoi; pdmq->msg = msg; pdmq->lParam = lParam; pdmq->next = NULL; // dmqOut->next->next->next->dmqIn->NULL if (pcoi->dmqIn != NULL) { pcoi->dmqIn->next = pdmq; } pcoi->dmqIn = pdmq; if (pcoi->dmqOut == NULL) { pcoi->dmqOut = pcoi->dmqIn; } pcoi->cLocks++; CheckForQueuedMessages(pcoi); pcoi->cLocks--; if (pcoi->cLocks == 0 && pcoi->state & ST_FREE_CONV_RES_NOW) { FreeConversationResources(pcoi); } } else { DumpDDEMessage(!(pcoi->state & ST_INTRA_PROCESS), msg, lParam); } } BOOL CheckForQueuedMessages(PCONV_INFO pcoi) /* * Description: * Handles processing of DDE messages held in the given conversaion's * DDE message queue. * Returns: fProcessed. * History: * 11-12-91 sanfords Created. */ { PDDE_MESSAGE_QUEUE pdmq; BOOL fRet = FALSE; PCLIENTINFO pci; CheckDDECritIn; if (pcoi->state & ST_PROCESSING) { // recursion prevention return(FALSE); } UserAssert(pcoi->cLocks); pci = GetClientInfo(); pcoi->state |= ST_PROCESSING; while (!(pcoi->state & ST_BLOCKED) && pcoi->dmqOut != NULL && !pci->cInDDEMLCallback) { pci->CI_flags |= CI_PROCESSING_QUEUE; if (ProcessSyncDDEMessage(pcoi, pcoi->dmqOut->msg, pcoi->dmqOut->lParam)) { fRet = TRUE; pdmq = pcoi->dmqOut; pcoi->dmqOut = pcoi->dmqOut->next; if (pcoi->dmqOut == NULL) { pcoi->dmqIn = NULL; } DDEMLFree(pdmq); } pci->CI_flags &= ~CI_PROCESSING_QUEUE; } pcoi->state &= ~ST_PROCESSING; return(fRet); } VOID DumpDDEMessage(BOOL fFreeData, UINT msg, LPARAM lParam) /* * Description: * Used to clean up resources referenced by DDE messages that for some * reason could not be processed. * History: * 11-12-91 sanfords Created. */ { UINT_PTR uiLo, uiHi; RIPMSG2(RIP_WARNING, "Dump DDE msg %x lParam %x", msg, lParam); switch (msg) { case WM_DDE_ACK: case WM_DDE_DATA: case WM_DDE_POKE: case WM_DDE_ADVISE: UnpackDDElParam(msg, lParam, &uiLo, &uiHi); switch (msg) { case WM_DDE_DATA: case WM_DDE_POKE: if (uiLo) { if (fFreeData) { FreeDDEData((HANDLE)uiLo, FALSE, TRUE); } GlobalDeleteAtom((ATOM)uiHi); } break; case WM_DDE_ADVISE: if (uiLo) { if (fFreeData) { FreeDDEData((HANDLE)uiLo, FALSE, TRUE); } GlobalDeleteAtom((ATOM)uiHi); } break; case WM_DDE_ACK: // could be EXEC Ack - cant know what to do exactly. break; } FreeDDElParam(msg, lParam); break; case WM_DDE_EXECUTE: if (fFreeData) { WOWGLOBALFREE((HANDLE)lParam); } break; case WM_DDE_REQUEST: case WM_DDE_UNADVISE: GlobalDeleteAtom((ATOM)HIWORD(lParam)); break; } } BOOL ProcessSyncDDEMessage(PCONV_INFO pcoi, UINT msg, LPARAM lParam) /* * Description: * Handles processing of a received DDE message. TRUE is returned if * the message was handled. FALSE implies CBR_BLOCK. * History: * 11-19-91 sanfords Created. */ { BOOL fNotBlocked = TRUE; PCL_INSTANCE_INFO pcii; ENABLE_ENUM_STRUCT ees; BOOL fRet; CheckDDECritIn; /* * lock the conversation so its resources don't go away till we are * done with them. This function could generate a callback which could * disconnect the conversation. */ pcoi->cLocks++; if (pcoi->state & ST_BLOCKNEXT) { pcoi->state ^= ST_BLOCKNEXT | ST_BLOCKED; } if (pcoi->state & ST_BLOCKALLNEXT) { ees.pfRet = &fRet; ees.wCmd = EC_DISABLE; ees.wCmd2 = 0; EnumChildWindows(pcoi->pcii->hwndMother, (WNDENUMPROC)EnableEnumProc, (LPARAM)&ees); } if (pcoi->state & ST_CONNECTED) { if (pcoi->pxiOut == NULL) { if (pcoi->state & ST_CLIENT) { fNotBlocked = SpontaneousClientMessage((PCL_CONV_INFO)pcoi, msg, lParam); } else { fNotBlocked = SpontaneousServerMessage((PSVR_CONV_INFO)pcoi, msg, lParam); } } else { UserAssert(pcoi->pxiOut->hXact == (HANDLE)0 || ValidateCHandle(pcoi->pxiOut->hXact, HTYPE_TRANSACTION, HINST_ANY) == (ULONG_PTR)pcoi->pxiOut); fNotBlocked = (pcoi->pxiOut->pfnResponse)(pcoi->pxiOut, msg, lParam); } } else { DumpDDEMessage(!(pcoi->state & ST_INTRA_PROCESS), msg, lParam); } if (!fNotBlocked) { pcoi->state |= ST_BLOCKED; pcoi->state &= ~ST_BLOCKNEXT; } pcii = pcoi->pcii; // save this incase unlocking makes pcoi go away. pcoi->cLocks--; if (pcoi->cLocks == 0 && pcoi->state & ST_FREE_CONV_RES_NOW) { FreeConversationResources(pcoi); } /* * Because callbacks are capable of blocking DdeUninitialize(), we check * before exit to see if it needs to be called. */ if (pcii->afCmd & APPCMD_UNINIT_ASAP && !(pcii->flags & IIF_IN_SYNC_XACT) && !pcii->cInDDEMLCallback) { DdeUninitialize(HandleToUlong(pcii->hInstClient)); return(FALSE); } return (fNotBlocked); }