4 * Copyright 1997 Alexandre Julliard
5 * Copyright 1997 Len White
6 * Copyright 1999 Keith Matthews
8 * Copyright 2001 Eric Pouech
9 * Copyright 2003, 2004, 2005 Dmitry Timoshkov
11 * This library is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU Lesser General Public
13 * License as published by the Free Software Foundation; either
14 * version 2.1 of the License, or (at your option) any later version.
16 * This library is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 * Lesser General Public License for more details.
21 * You should have received a copy of the GNU Lesser General Public
22 * License along with this library; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
36 #include "wine/debug.h"
37 #include "dde_private.h"
39 WINE_DEFAULT_DEBUG_CHANNEL(ddeml);
41 static const WCHAR szServerNameClass[] = {'W','i','n','e','D','d','e','S','e','r','v','e','r','N','a','m','e',0};
42 const WCHAR WDML_szServerConvClass[] = {'W','i','n','e','D','d','e','S','e','r','v','e','r','C','o','n','v',0};
44 static LRESULT CALLBACK WDML_ServerNameProc(HWND, UINT, WPARAM, LPARAM);
45 static LRESULT CALLBACK WDML_ServerConvProc(HWND, UINT, WPARAM, LPARAM);
47 /******************************************************************************
48 * DdePostAdvise [USER32.@] Send transaction to DDE callback function.
51 * idInst [I] Instance identifier
52 * hszTopic [I] Handle to topic name string
53 * hszItem [I] Handle to item name string
59 BOOL WINAPI DdePostAdvise(DWORD idInst, HSZ hszTopic, HSZ hszItem)
61 WDML_INSTANCE* pInstance = NULL;
62 WDML_LINK* pLink = NULL;
63 HDDEDATA hDdeData = 0;
64 HGLOBAL hItemData = 0;
65 WDML_CONV* pConv = NULL;
69 TRACE("(%ld,%p,%p)\n", idInst, hszTopic, hszItem);
71 EnterCriticalSection(&WDML_CritSect);
73 pInstance = WDML_GetInstance(idInst);
75 if (pInstance == NULL || pInstance->links == NULL)
80 atom = WDML_MakeAtomFromHsz(hszItem);
81 if (!atom) goto theError;
83 /* first compute the number of links which will trigger a message */
85 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
87 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
92 if (count >= CADV_LATEACK)
94 FIXME("too high value for count\n");
98 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
100 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
102 hDdeData = WDML_InvokeCallback(pInstance, XTYP_ADVREQ, pLink->uFmt, pLink->hConv,
103 hszTopic, hszItem, 0, --count, 0);
105 if (hDdeData == (HDDEDATA)CBR_BLOCK)
107 /* MS doc is not consistent here */
108 FIXME("CBR_BLOCK returned for ADVREQ\n");
113 if (pLink->transactionType & XTYPF_NODATA)
120 TRACE("with data\n");
122 hItemData = WDML_DataHandle2Global(hDdeData, FALSE, FALSE, FALSE, FALSE);
125 pConv = WDML_GetConv(pLink->hConv, TRUE);
129 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
133 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
134 PackDDElParam(WM_DDE_DATA, (UINT_PTR)hItemData, atom)))
136 ERR("post message failed\n");
137 pConv->wStatus &= ~ST_CONNECTED;
138 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
139 GlobalFree(hItemData);
142 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
146 LeaveCriticalSection(&WDML_CritSect);
149 LeaveCriticalSection(&WDML_CritSect);
150 if (atom) GlobalDeleteAtom(atom);
155 /******************************************************************************
156 * DdeNameService [USER32.@] {Un}registers service name of DDE server
159 * idInst [I] Instance identifier
160 * hsz1 [I] Handle to service name string
162 * afCmd [I] Service name flags
168 HDDEDATA WINAPI DdeNameService(DWORD idInst, HSZ hsz1, HSZ hsz2, UINT afCmd)
170 WDML_SERVER* pServer;
171 WDML_INSTANCE* pInstance;
174 WNDCLASSEXW wndclass;
178 TRACE("(%ld,%p,%p,%x)\n", idInst, hsz1, hsz2, afCmd);
180 EnterCriticalSection(&WDML_CritSect);
182 /* First check instance
184 pInstance = WDML_GetInstance(idInst);
185 if (pInstance == NULL)
187 TRACE("Instance not found as initialised\n");
188 /* Nothing has been initialised - exit now ! can return TRUE since effect is the same */
194 /* Illegal, reserved parameter
196 pInstance->lastError = DMLERR_INVALIDPARAMETER;
197 WARN("Reserved parameter no-zero !!\n");
200 if (hsz1 == 0 && !(afCmd & DNS_UNREGISTER))
202 /* don't know if we should check this but it makes sense
203 * why supply REGISTER or filter flags if de-registering all
205 TRACE("General unregister unexpected flags\n");
206 pInstance->lastError = DMLERR_INVALIDPARAMETER;
210 switch (afCmd & (DNS_REGISTER | DNS_UNREGISTER))
213 pServer = WDML_FindServer(pInstance, hsz1, 0);
216 ERR("Trying to register already registered service!\n");
217 pInstance->lastError = DMLERR_DLL_USAGE;
221 TRACE("Adding service name\n");
223 WDML_IncHSZ(pInstance, hsz1);
225 pServer = WDML_AddServer(pInstance, hsz1, 0);
227 WDML_BroadcastDDEWindows(WDML_szEventClass, WM_WDML_REGISTER,
228 pServer->atomService, pServer->atomServiceSpec);
230 wndclass.cbSize = sizeof(wndclass);
232 wndclass.lpfnWndProc = WDML_ServerNameProc;
233 wndclass.cbClsExtra = 0;
234 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
235 wndclass.hInstance = 0;
237 wndclass.hCursor = 0;
238 wndclass.hbrBackground = 0;
239 wndclass.lpszMenuName = NULL;
240 wndclass.lpszClassName = szServerNameClass;
241 wndclass.hIconSm = 0;
243 RegisterClassExW(&wndclass);
245 LeaveCriticalSection(&WDML_CritSect);
246 hwndServer = CreateWindowW(szServerNameClass, NULL,
247 WS_POPUP, 0, 0, 0, 0,
249 EnterCriticalSection(&WDML_CritSect);
251 SetWindowLongPtrW(hwndServer, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
252 SetWindowLongPtrW(hwndServer, GWL_WDML_SERVER, (ULONG_PTR)pServer);
253 TRACE("Created nameServer=%p for instance=%08lx\n", hwndServer, idInst);
255 pServer->hwndServer = hwndServer;
261 /* General unregister situation
262 * terminate all server side pending conversations
264 while (pInstance->servers)
265 WDML_RemoveServer(pInstance, pInstance->servers->hszService, 0);
266 pInstance->servers = NULL;
267 TRACE("General de-register - finished\n");
271 WDML_RemoveServer(pInstance, hsz1, 0L);
276 if (afCmd & (DNS_FILTERON | DNS_FILTEROFF))
278 /* Set filter flags on to hold notifications of connection
280 pServer = WDML_FindServer(pInstance, hsz1, 0);
283 /* trying to filter where no service names !!
285 pInstance->lastError = DMLERR_DLL_USAGE;
290 pServer->filterOn = (afCmd & DNS_FILTERON) != 0;
293 LeaveCriticalSection(&WDML_CritSect);
294 return (HDDEDATA)TRUE;
297 LeaveCriticalSection(&WDML_CritSect);
301 /******************************************************************
302 * WDML_CreateServerConv
306 static WDML_CONV* WDML_CreateServerConv(WDML_INSTANCE* pInstance, HWND hwndClient,
307 HWND hwndServerName, HSZ hszApp, HSZ hszTopic)
311 WNDCLASSEXW wndclass;
313 wndclass.cbSize = sizeof(wndclass);
315 wndclass.lpfnWndProc = WDML_ServerConvProc;
316 wndclass.cbClsExtra = 0;
317 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
318 wndclass.hInstance = 0;
320 wndclass.hCursor = 0;
321 wndclass.hbrBackground = 0;
322 wndclass.lpszMenuName = NULL;
323 wndclass.lpszClassName = WDML_szServerConvClass;
324 wndclass.hIconSm = 0;
326 RegisterClassExW(&wndclass);
328 hwndServerConv = CreateWindowW(WDML_szServerConvClass, 0,
329 WS_CHILD, 0, 0, 0, 0,
330 hwndServerName, 0, 0, 0);
332 TRACE("Created convServer=%p (nameServer=%p) for instance=%08lx\n",
333 hwndServerConv, hwndServerName, pInstance->instanceID);
335 pConv = WDML_AddConv(pInstance, WDML_SERVER_SIDE, hszApp, hszTopic,
336 hwndClient, hwndServerConv);
339 SetWindowLongPtrW(hwndServerConv, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
340 SetWindowLongPtrW(hwndServerConv, GWL_WDML_CONVERSATION, (ULONG_PTR)pConv);
342 /* this should be the only place using SendMessage for WM_DDE_ACK */
343 /* note: sent messages shall not use packing */
344 SendMessageW(hwndClient, WM_DDE_ACK, (WPARAM)hwndServerConv,
345 MAKELPARAM(WDML_MakeAtomFromHsz(hszApp), WDML_MakeAtomFromHsz(hszTopic)));
346 /* we assume we're connected since we've sent an answer...
347 * I'm not sure what we can do... it doesn't look like the return value
348 * of SendMessage is used... sigh...
350 pConv->wStatus |= ST_CONNECTED;
354 DestroyWindow(hwndServerConv);
359 /******************************************************************
360 * WDML_ServerNameProc
364 static LRESULT CALLBACK WDML_ServerNameProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
368 HDDEDATA hDdeData = 0;
369 WDML_INSTANCE* pInstance;
374 case WM_DDE_INITIATE:
376 /* wParam -- sending window handle
377 LOWORD(lParam) -- application atom
378 HIWORD(lParam) -- topic atom */
380 TRACE("WM_DDE_INITIATE message received!\n");
381 hwndClient = (HWND)wParam;
383 pInstance = WDML_GetInstanceFromWnd(hwndServer);
384 TRACE("idInst=%ld, threadID=0x%lx\n", pInstance->instanceID, GetCurrentThreadId());
385 if (!pInstance) return 0;
387 /* don't free DDEParams, since this is a broadcast */
388 UnpackDDElParam(WM_DDE_INITIATE, lParam, &uiLo, &uiHi);
390 hszApp = WDML_MakeHszFromAtom(pInstance, uiLo);
391 hszTop = WDML_MakeHszFromAtom(pInstance, uiHi);
393 if (!(pInstance->CBFflags & CBF_FAIL_CONNECTIONS))
397 CONVCONTEXT* pcc = NULL;
401 if (GetWindowThreadProcessId(hwndClient, NULL) == GetWindowThreadProcessId(hwndServer, NULL) &&
402 WDML_GetInstanceFromWnd(hwndClient) == WDML_GetInstanceFromWnd(hwndServer))
406 /* FIXME: so far, we don't grab distant convcontext, so only check if remote is
407 * handled under DDEML, and if so build a default context
409 if (GetClassNameW(hwndClient, buf, sizeof(buf)/sizeof(WCHAR)) &&
410 lstrcmpiW(buf, WDML_szClientConvClass) == 0)
413 memset(pcc, 0, sizeof(*pcc));
414 pcc->cb = sizeof(*pcc);
415 pcc->iCodePage = CP_WINUNICODE;
417 if ((pInstance->CBFflags & CBF_FAIL_SELFCONNECTIONS) && self)
419 TRACE("Don't do self connection as requested\n");
421 else if (hszApp && hszTop)
423 WDML_SERVER* pServer = (WDML_SERVER*)GetWindowLongPtrW(hwndServer, GWL_WDML_SERVER);
425 /* check filters for name service */
426 if (!pServer->filterOn || DdeCmpStringHandles(pServer->hszService, hszApp) == 0)
428 /* pass on to the callback */
429 hDdeData = WDML_InvokeCallback(pInstance, XTYP_CONNECT,
430 0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
431 if ((ULONG_PTR)hDdeData)
433 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
437 if (pcc) pConv->wStatus |= ST_ISLOCAL;
438 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
439 hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
444 else if (pInstance->servers)
446 /* pass on to the callback */
447 hDdeData = WDML_InvokeCallback(pInstance, XTYP_WILDCONNECT,
448 0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
450 if (hDdeData == (HDDEDATA)CBR_BLOCK)
452 /* MS doc is not consistent here */
453 FIXME("CBR_BLOCK returned for WILDCONNECT\n");
455 else if ((ULONG_PTR)hDdeData != 0)
459 hszp = (HSZPAIR*)DdeAccessData(hDdeData, NULL);
463 for (i = 0; hszp[i].hszSvc && hszp[i].hszTopic; i++)
465 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
466 hszp[i].hszSvc, hszp[i].hszTopic);
469 if (pcc) pConv->wStatus |= ST_ISLOCAL;
470 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
471 hszp[i].hszTopic, hszp[i].hszSvc, 0, (ULONG_PTR)pcc, self);
474 DdeUnaccessData(hDdeData);
476 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
484 FIXME("WM_DDE_REQUEST message received!\n");
487 FIXME("WM_DDE_ADVISE message received!\n");
489 case WM_DDE_UNADVISE:
490 FIXME("WM_DDE_UNADVISE message received!\n");
493 FIXME("WM_DDE_EXECUTE message received!\n");
496 FIXME("WM_DDE_POKE message received!\n");
498 case WM_DDE_TERMINATE:
499 FIXME("WM_DDE_TERMINATE message received!\n");
505 return DefWindowProcW(hwndServer, iMsg, wParam, lParam);
508 /******************************************************************
509 * WDML_ServerQueueRequest
513 static WDML_XACT* WDML_ServerQueueRequest(WDML_CONV* pConv, LPARAM lParam)
518 UnpackDDElParam(WM_DDE_REQUEST, lParam, &uiLo, &uiHi);
520 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_REQUEST,
521 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
522 if (pXAct) pXAct->atom = uiHi;
526 /******************************************************************
527 * WDML_ServerHandleRequest
531 static WDML_QUEUE_STATE WDML_ServerHandleRequest(WDML_CONV* pConv, WDML_XACT* pXAct)
533 HDDEDATA hDdeData = 0;
536 if (!(pConv->instance->CBFflags & CBF_FAIL_REQUESTS))
539 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_REQUEST, pXAct->wFmt, (HCONV)pConv,
540 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
543 switch ((ULONG_PTR)hDdeData)
546 TRACE("No data returned from the Callback\n");
550 case (ULONG_PTR)CBR_BLOCK:
551 return WDML_QS_BLOCK;
555 HGLOBAL hMem = WDML_DataHandle2Global(hDdeData, TRUE, FALSE, FALSE, FALSE);
556 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
557 ReuseDDElParam(pXAct->lParam, WM_DDE_REQUEST, WM_DDE_DATA,
558 (UINT_PTR)hMem, (UINT_PTR)pXAct->atom)))
560 DdeFreeDataHandle(hDdeData);
568 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_REQUEST);
570 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
572 return WDML_QS_HANDLED;
575 /******************************************************************
576 * WDML_ServerQueueAdvise
580 static WDML_XACT* WDML_ServerQueueAdvise(WDML_CONV* pConv, LPARAM lParam)
585 /* XTYP_ADVSTART transaction:
586 establish link and save link info to InstanceInfoTable */
588 if (!UnpackDDElParam(WM_DDE_ADVISE, lParam, &uiLo, &uiHi))
591 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_ADVISE,
592 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
595 pXAct->hMem = (HGLOBAL)uiLo;
601 /******************************************************************
602 * WDML_ServerHandleAdvise
606 static WDML_QUEUE_STATE WDML_ServerHandleAdvise(WDML_CONV* pConv, WDML_XACT* pXAct)
610 DDEADVISE* pDdeAdvise;
611 HDDEDATA hDdeData = 0;
614 pDdeAdvise = (DDEADVISE*)GlobalLock(pXAct->hMem);
615 uType = XTYP_ADVSTART |
616 (pDdeAdvise->fDeferUpd ? XTYPF_NODATA : 0) |
617 (pDdeAdvise->fAckReq ? XTYPF_ACKREQ : 0);
619 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
621 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_ADVSTART, pDdeAdvise->cfFormat,
622 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
625 switch ((ULONG_PTR)hDdeData)
628 TRACE("No data returned from the Callback\n");
632 case (ULONG_PTR)CBR_BLOCK:
633 return WDML_QS_BLOCK;
636 /* billx: first to see if the link is already created. */
637 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
638 pXAct->hszItem, TRUE, pDdeAdvise->cfFormat);
642 /* we found a link, and only need to modify it in case it changes */
643 pLink->transactionType = uType;
647 TRACE("Adding Link with hConv %p\n", pConv);
648 WDML_AddLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
649 uType, pXAct->hszItem, pDdeAdvise->cfFormat);
654 GlobalUnlock(pXAct->hMem);
657 GlobalFree(pXAct->hMem);
661 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_ADVISE);
663 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
665 return WDML_QS_HANDLED;
668 /******************************************************************
669 * WDML_ServerQueueUnadvise
673 static WDML_XACT* WDML_ServerQueueUnadvise(WDML_CONV* pConv, LPARAM lParam)
678 UnpackDDElParam(WM_DDE_UNADVISE, lParam, &uiLo, &uiHi);
680 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_UNADVISE,
681 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
682 if (pXAct) pXAct->atom = uiHi;
686 /******************************************************************
687 * WDML_ServerHandleUnadvise
691 static WDML_QUEUE_STATE WDML_ServerHandleUnadvise(WDML_CONV* pConv, WDML_XACT* pXAct)
695 if (pXAct->hszItem == NULL || pXAct->wFmt == 0)
697 ERR("Unsupported yet options (null item or clipboard format)\n");
698 return WDML_QS_ERROR;
701 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
702 pXAct->hszItem, TRUE, pXAct->wFmt);
705 ERR("Couln'd find link for %p, dropping request\n", pXAct->hszItem);
706 FreeDDElParam(WM_DDE_UNADVISE, pXAct->lParam);
707 return WDML_QS_ERROR;
710 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
712 WDML_InvokeCallback(pConv->instance, XTYP_ADVSTOP, pXAct->wFmt, (HCONV)pConv,
713 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
716 WDML_RemoveLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
717 pXAct->hszItem, pXAct->wFmt);
720 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, TRUE, pXAct->atom,
721 pXAct->lParam, WM_DDE_UNADVISE);
723 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
725 return WDML_QS_HANDLED;
728 /******************************************************************
733 static WDML_XACT* WDML_ServerQueueExecute(WDML_CONV* pConv, LPARAM lParam)
737 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_EXECUTE, 0, 0);
740 pXAct->hMem = (HGLOBAL)lParam;
745 /******************************************************************
746 * WDML_ServerHandleExecute
750 static WDML_QUEUE_STATE WDML_ServerHandleExecute(WDML_CONV* pConv, WDML_XACT* pXAct)
752 HDDEDATA hDdeData = DDE_FNOTPROCESSED;
753 BOOL fAck = FALSE, fBusy = FALSE;
755 if (!(pConv->instance->CBFflags & CBF_FAIL_EXECUTES))
757 LPVOID ptr = GlobalLock(pXAct->hMem);
761 hDdeData = DdeCreateDataHandle(0, ptr, GlobalSize(pXAct->hMem),
763 GlobalUnlock(pXAct->hMem);
765 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_EXECUTE, 0, (HCONV)pConv,
766 pConv->hszTopic, 0, hDdeData, 0L, 0L);
769 switch ((ULONG_PTR)hDdeData)
771 case (ULONG_PTR)CBR_BLOCK:
772 return WDML_QS_BLOCK;
781 FIXME("Unsupported returned value %p\n", hDdeData);
783 case DDE_FNOTPROCESSED:
786 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, (UINT)pXAct->hMem, 0, 0);
788 return WDML_QS_HANDLED;
791 /******************************************************************
792 * WDML_ServerQueuePoke
796 static WDML_XACT* WDML_ServerQueuePoke(WDML_CONV* pConv, LPARAM lParam)
801 UnpackDDElParam(WM_DDE_POKE, lParam, &uiLo, &uiHi);
803 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_POKE,
804 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
808 pXAct->hMem = (HGLOBAL)uiLo;
813 /******************************************************************
814 * WDML_ServerHandlePoke
818 static WDML_QUEUE_STATE WDML_ServerHandlePoke(WDML_CONV* pConv, WDML_XACT* pXAct)
822 BOOL fBusy = FALSE, fAck = FALSE;
824 pDdePoke = (DDEPOKE*)GlobalLock(pXAct->hMem);
827 return WDML_QS_ERROR;
830 if (!(pConv->instance->CBFflags & CBF_FAIL_POKES))
832 hDdeData = DdeCreateDataHandle(pConv->instance->instanceID, pDdePoke->Value,
833 GlobalSize(pXAct->hMem) - sizeof(DDEPOKE) + 1,
834 0, 0, pDdePoke->cfFormat, 0);
837 HDDEDATA hDdeDataOut;
839 hDdeDataOut = WDML_InvokeCallback(pConv->instance, XTYP_POKE, pDdePoke->cfFormat,
840 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem,
842 switch ((ULONG_PTR)hDdeDataOut)
851 FIXME("Unsupported returned value %p\n", hDdeDataOut);
853 case DDE_FNOTPROCESSED:
856 DdeFreeDataHandle(hDdeData);
859 GlobalUnlock(pXAct->hMem);
863 GlobalFree(pXAct->hMem);
865 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, pXAct->atom, pXAct->lParam, WM_DDE_POKE);
867 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
869 return WDML_QS_HANDLED;
872 /******************************************************************
873 * WDML_ServerQueueTerminate
877 static WDML_XACT* WDML_ServerQueueTerminate(WDML_CONV* pConv, LPARAM lParam)
881 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_TERMINATE, 0, 0);
885 /******************************************************************
886 * WDML_ServerHandleTerminate
890 static WDML_QUEUE_STATE WDML_ServerHandleTerminate(WDML_CONV* pConv, WDML_XACT* pXAct)
892 /* billx: two things to remove: the conv, and associated links.
893 * Respond with another WM_DDE_TERMINATE iMsg.
895 if (!(pConv->instance->CBFflags & CBF_SKIP_DISCONNECTS))
897 WDML_InvokeCallback(pConv->instance, XTYP_DISCONNECT, 0, (HCONV)pConv, 0, 0,
898 0, 0, (pConv->wStatus & ST_ISSELF) ? 1 : 0);
900 PostMessageW(pConv->hwndClient, WM_DDE_TERMINATE, (WPARAM)pConv->hwndServer, 0);
901 WDML_RemoveConv(pConv, WDML_SERVER_SIDE);
903 return WDML_QS_HANDLED;
906 /******************************************************************
911 WDML_QUEUE_STATE WDML_ServerHandle(WDML_CONV* pConv, WDML_XACT* pXAct)
913 WDML_QUEUE_STATE qs = WDML_QS_ERROR;
915 switch (pXAct->ddeMsg)
917 case WM_DDE_INITIATE:
918 FIXME("WM_DDE_INITIATE shouldn't be there!\n");
921 qs = WDML_ServerHandleRequest(pConv, pXAct);
925 qs = WDML_ServerHandleAdvise(pConv, pXAct);
928 case WM_DDE_UNADVISE:
929 qs = WDML_ServerHandleUnadvise(pConv, pXAct);
933 qs = WDML_ServerHandleExecute(pConv, pXAct);
937 qs = WDML_ServerHandlePoke(pConv, pXAct);
940 case WM_DDE_TERMINATE:
941 qs = WDML_ServerHandleTerminate(pConv, pXAct);
945 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
949 FIXME("Unsupported message %d\n", pXAct->ddeMsg);
954 /******************************************************************
955 * WDML_ServerConvProc
959 static LRESULT CALLBACK WDML_ServerConvProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
961 WDML_INSTANCE* pInstance;
963 WDML_XACT* pXAct = NULL;
965 TRACE("%p %04x %08x %08lx\n", hwndServer, iMsg, wParam , lParam);
967 if (iMsg == WM_DESTROY)
969 EnterCriticalSection(&WDML_CritSect);
970 pConv = WDML_GetConvFromWnd(hwndServer);
971 if (pConv && !(pConv->wStatus & ST_TERMINATED))
973 WDML_ServerHandleTerminate(pConv, NULL);
975 LeaveCriticalSection(&WDML_CritSect);
977 if (iMsg < WM_DDE_FIRST || iMsg > WM_DDE_LAST)
979 return DefWindowProcW(hwndServer, iMsg, wParam, lParam);
982 EnterCriticalSection(&WDML_CritSect);
984 pInstance = WDML_GetInstanceFromWnd(hwndServer);
985 pConv = WDML_GetConvFromWnd(hwndServer);
989 ERR("Got a message (%x) on a not known conversation, dropping request\n", iMsg);
992 if (pConv->hwndClient != WIN_GetFullHandle( (HWND)wParam ) || pConv->hwndServer != hwndServer)
994 ERR("mismatch between C/S windows and converstation\n");
997 if (pConv->instance != pInstance || pConv->instance == NULL)
999 ERR("mismatch in instances\n");
1005 case WM_DDE_INITIATE:
1006 FIXME("WM_DDE_INITIATE message received!\n");
1009 case WM_DDE_REQUEST:
1010 pXAct = WDML_ServerQueueRequest(pConv, lParam);
1014 pXAct = WDML_ServerQueueAdvise(pConv, lParam);
1017 case WM_DDE_UNADVISE:
1018 pXAct = WDML_ServerQueueUnadvise(pConv, lParam);
1021 case WM_DDE_EXECUTE:
1022 pXAct = WDML_ServerQueueExecute(pConv, lParam);
1026 pXAct = WDML_ServerQueuePoke(pConv, lParam);
1029 case WM_DDE_TERMINATE:
1030 pXAct = WDML_ServerQueueTerminate(pConv, lParam);
1034 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1038 FIXME("Unsupported message %x\n", iMsg);
1043 pXAct->lParam = lParam;
1044 if (WDML_ServerHandle(pConv, pXAct) == WDML_QS_BLOCK)
1046 WDML_QueueTransaction(pConv, pXAct);
1050 WDML_FreeTransaction(pInstance, pXAct, TRUE);
1054 LeaveCriticalSection(&WDML_CritSect);