1 /* -*- tab-width: 8; c-basic-offset: 4 -*- */
6 * Copyright 1997 Alexandre Julliard
7 * Copyright 1997 Len White
8 * Copyright 1999 Keith Matthews
10 * Copyright 2001 Eric Pouech
12 * This library is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU Lesser General Public
14 * License as published by the Free Software Foundation; either
15 * version 2.1 of the License, or (at your option) any later version.
17 * This library is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * Lesser General Public License for more details.
22 * You should have received a copy of the GNU Lesser General Public
23 * License along with this library; if not, write to the Free Software
24 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
36 #include "wine/debug.h"
37 #include "dde/dde_private.h"
39 WINE_DEFAULT_DEBUG_CHANNEL(ddeml);
41 static const char szServerNameClassA[] = "DdeServerNameAnsi";
42 const char WDML_szServerConvClassA[] = "DdeServerConvAnsi";
43 const WCHAR WDML_szServerConvClassW[] = {'D','d','e','S','e','r','v','e','r','C','o','n','v','U','n','i','c','o','d','e',0};
45 static LRESULT CALLBACK WDML_ServerNameProc(HWND, UINT, WPARAM, LPARAM);
46 static LRESULT CALLBACK WDML_ServerConvProc(HWND, UINT, WPARAM, LPARAM);
48 /******************************************************************************
49 * DdePostAdvise [USER32.@] Send transaction to DDE callback function.
52 * idInst [I] Instance identifier
53 * hszTopic [I] Handle to topic name string
54 * hszItem [I] Handle to item name string
60 BOOL WINAPI DdePostAdvise(DWORD idInst, HSZ hszTopic, HSZ hszItem)
62 WDML_INSTANCE* pInstance = NULL;
63 WDML_LINK* pLink = NULL;
64 HDDEDATA hDdeData = 0;
65 HGLOBAL hItemData = 0;
66 WDML_CONV* pConv = NULL;
70 TRACE("(%ld,0x%x,0x%x)\n", idInst, hszTopic, hszItem);
72 EnterCriticalSection(&WDML_CritSect);
74 pInstance = WDML_GetInstance(idInst);
76 if (pInstance == NULL || pInstance->links == NULL)
81 atom = WDML_MakeAtomFromHsz(hszItem);
82 if (!atom) goto theError;
84 /* first compute the number of links which will trigger a message */
86 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
88 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
93 if (count >= CADV_LATEACK)
95 FIXME("too high value for count\n");
99 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
101 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
103 hDdeData = WDML_InvokeCallback(pInstance, XTYP_ADVREQ, pLink->uFmt, pLink->hConv,
104 hszTopic, hszItem, 0, count--, 0);
106 if (hDdeData == (HDDEDATA)CBR_BLOCK)
108 /* MS doc is not consistent here */
109 FIXME("CBR_BLOCK returned for ADVREQ\n");
114 if (pLink->transactionType & XTYPF_NODATA)
121 TRACE("with data\n");
123 hItemData = WDML_DataHandle2Global(hDdeData, FALSE, FALSE, FALSE, FALSE);
126 pConv = WDML_GetConv(pLink->hConv, TRUE);
130 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
134 if (!PostMessageA(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
135 PackDDElParam(WM_DDE_DATA, (UINT)hItemData, atom)))
137 ERR("post message failed\n");
138 pConv->wStatus &= ~ST_CONNECTED;
139 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
140 GlobalFree(hItemData);
143 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
147 LeaveCriticalSection(&WDML_CritSect);
150 LeaveCriticalSection(&WDML_CritSect);
151 if (atom) GlobalDeleteAtom(atom);
156 /******************************************************************************
157 * DdeNameService [USER32.@] {Un}registers service name of DDE server
160 * idInst [I] Instance identifier
161 * hsz1 [I] Handle to service name string
163 * afCmd [I] Service name flags
169 HDDEDATA WINAPI DdeNameService(DWORD idInst, HSZ hsz1, HSZ hsz2, UINT afCmd)
171 WDML_SERVER* pServer;
172 WDML_INSTANCE* pInstance;
175 WNDCLASSEXA wndclass;
177 hDdeData = (HDDEDATA)NULL;
179 TRACE("(%ld,0x%x,0x%x,%d)\n", idInst, hsz1, hsz2, afCmd);
181 EnterCriticalSection(&WDML_CritSect);
183 /* First check instance
185 pInstance = WDML_GetInstance(idInst);
186 if (pInstance == NULL)
188 TRACE("Instance not found as initialised\n");
189 /* Nothing has been initialised - exit now ! can return TRUE since effect is the same */
195 /* Illegal, reserved parameter
197 pInstance->lastError = DMLERR_INVALIDPARAMETER;
198 WARN("Reserved parameter no-zero !!\n");
201 if (hsz1 == 0 && afCmd != DNS_UNREGISTER)
203 /* don't know if we should check this but it makes sense
204 * why supply REGISTER or filter flags if de-registering all
206 TRACE("General unregister unexpected flags\n");
207 pInstance->lastError = DMLERR_INVALIDPARAMETER;
214 pServer = WDML_FindServer(pInstance, hsz1, 0);
217 ERR("Trying to register already registered service!\n");
218 pInstance->lastError = DMLERR_DLL_USAGE;
222 TRACE("Adding service name\n");
224 WDML_IncHSZ(pInstance, hsz1);
226 pServer = WDML_AddServer(pInstance, hsz1, 0);
228 WDML_BroadcastDDEWindows(WDML_szEventClass, WM_WDML_REGISTER,
229 pServer->atomService, pServer->atomServiceSpec);
231 wndclass.cbSize = sizeof(wndclass);
233 wndclass.lpfnWndProc = WDML_ServerNameProc;
234 wndclass.cbClsExtra = 0;
235 wndclass.cbWndExtra = 2 * sizeof(DWORD);
236 wndclass.hInstance = 0;
238 wndclass.hCursor = 0;
239 wndclass.hbrBackground = 0;
240 wndclass.lpszMenuName = NULL;
241 wndclass.lpszClassName = szServerNameClassA;
242 wndclass.hIconSm = 0;
244 RegisterClassExA(&wndclass);
246 LeaveCriticalSection(&WDML_CritSect);
247 hwndServer = CreateWindowA(szServerNameClassA, NULL,
248 WS_POPUP, 0, 0, 0, 0,
250 EnterCriticalSection(&WDML_CritSect);
252 SetWindowLongA(hwndServer, GWL_WDML_INSTANCE, (DWORD)pInstance);
253 SetWindowLongA(hwndServer, GWL_WDML_SERVER, (DWORD)pServer);
254 TRACE("Created nameServer=%04x for instance=%08lx\n", hwndServer, idInst);
256 pServer->hwndServer = hwndServer;
262 /* General unregister situation
263 * terminate all server side pending conversations
265 while (pInstance->servers)
266 WDML_RemoveServer(pInstance, pInstance->servers->hszService, 0);
267 pInstance->servers = NULL;
268 TRACE("General de-register - finished\n");
272 WDML_RemoveServer(pInstance, hsz1, 0L);
277 /* Set filter flags on to hold notifications of connection
279 pServer = WDML_FindServer(pInstance, hsz1, 0);
282 /* trying to filter where no service names !!
284 pInstance->lastError = DMLERR_DLL_USAGE;
289 pServer->filterOn = (afCmd == DNS_FILTERON);
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)
312 if (pInstance->unicode)
314 WNDCLASSEXW wndclass;
316 wndclass.cbSize = sizeof(wndclass);
318 wndclass.lpfnWndProc = WDML_ServerConvProc;
319 wndclass.cbClsExtra = 0;
320 wndclass.cbWndExtra = 2 * sizeof(DWORD);
321 wndclass.hInstance = 0;
323 wndclass.hCursor = 0;
324 wndclass.hbrBackground = 0;
325 wndclass.lpszMenuName = NULL;
326 wndclass.lpszClassName = WDML_szServerConvClassW;
327 wndclass.hIconSm = 0;
329 RegisterClassExW(&wndclass);
331 hwndServerConv = CreateWindowW(WDML_szServerConvClassW, 0,
332 WS_CHILD, 0, 0, 0, 0,
333 hwndServerName, 0, 0, 0);
337 WNDCLASSEXA wndclass;
339 wndclass.cbSize = sizeof(wndclass);
341 wndclass.lpfnWndProc = WDML_ServerConvProc;
342 wndclass.cbClsExtra = 0;
343 wndclass.cbWndExtra = 2 * sizeof(DWORD);
344 wndclass.hInstance = 0;
346 wndclass.hCursor = 0;
347 wndclass.hbrBackground = 0;
348 wndclass.lpszMenuName = NULL;
349 wndclass.lpszClassName = WDML_szServerConvClassA;
350 wndclass.hIconSm = 0;
352 RegisterClassExA(&wndclass);
354 hwndServerConv = CreateWindowA(WDML_szServerConvClassA, 0,
355 WS_CHILD, 0, 0, 0, 0,
356 hwndServerName, 0, 0, 0);
359 TRACE("Created convServer=%04x (nameServer=%04x) for instance=%08lx\n",
360 hwndServerConv, hwndServerName, pInstance->instanceID);
362 pConv = WDML_AddConv(pInstance, WDML_SERVER_SIDE, hszApp, hszTopic,
363 hwndClient, hwndServerConv);
366 SetWindowLongA(hwndServerConv, GWL_WDML_INSTANCE, (DWORD)pInstance);
367 SetWindowLongA(hwndServerConv, GWL_WDML_CONVERSATION, (DWORD)pConv);
369 /* this should be the only place using SendMessage for WM_DDE_ACK */
370 /* note: sent messages shall not use packing */
371 SendMessageA(hwndClient, WM_DDE_ACK, (WPARAM)hwndServerConv,
372 MAKELPARAM(WDML_MakeAtomFromHsz(hszApp), WDML_MakeAtomFromHsz(hszTopic)));
373 /* we assume we're connected since we've sent an answer...
374 * I'm not sure what we can do... it doesn't look like the return value
375 * of SendMessage is used... sigh...
377 pConv->wStatus |= ST_CONNECTED;
381 DestroyWindow(hwndServerConv);
386 /******************************************************************
387 * WDML_ServerNameProc
391 static LRESULT CALLBACK WDML_ServerNameProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
395 HDDEDATA hDdeData = 0;
396 WDML_INSTANCE* pInstance;
401 case WM_DDE_INITIATE:
403 /* wParam -- sending window handle
404 LOWORD(lParam) -- application atom
405 HIWORD(lParam) -- topic atom */
407 TRACE("WM_DDE_INITIATE message received!\n");
408 hwndClient = (HWND)wParam;
410 pInstance = WDML_GetInstanceFromWnd(hwndServer);
411 TRACE("idInst=%ld, threadID=0x%lx\n", pInstance->instanceID, GetCurrentThreadId());
412 if (!pInstance) return 0;
414 /* don't free DDEParams, since this is a broadcast */
415 UnpackDDElParam(WM_DDE_INITIATE, lParam, &uiLo, &uiHi);
417 hszApp = WDML_MakeHszFromAtom(pInstance, uiLo);
418 hszTop = WDML_MakeHszFromAtom(pInstance, uiHi);
420 if (!(pInstance->CBFflags & CBF_FAIL_CONNECTIONS))
424 CONVCONTEXT* pcc = NULL;
428 if (GetWindowThreadProcessId(hwndClient, NULL) == GetWindowThreadProcessId(hwndServer, NULL) &&
429 WDML_GetInstanceFromWnd(hwndClient) == WDML_GetInstanceFromWnd(hwndServer))
433 /* FIXME: so far, we don't grab distant convcontext, so only check if remote is
434 * handled under DDEML, and if so build a default context
436 if ((GetClassNameA(hwndClient, buf, sizeof(buf)) &&
437 strcmp(buf, WDML_szClientConvClassA) == 0) ||
438 (GetClassNameW(hwndClient, (LPWSTR)buf, sizeof(buf)/sizeof(WCHAR)) &&
439 lstrcmpW((LPWSTR)buf, WDML_szClientConvClassW) == 0))
442 memset(pcc, 0, sizeof(*pcc));
443 pcc->cb = sizeof(*pcc);
444 pcc->iCodePage = IsWindowUnicode(hwndClient) ? CP_WINUNICODE : CP_WINANSI;
446 if ((pInstance->CBFflags & CBF_FAIL_SELFCONNECTIONS) && self)
448 TRACE("Don't do self connection as requested\n");
450 else if (hszApp && hszTop)
452 WDML_SERVER* pServer = (WDML_SERVER*)GetWindowLongA(hwndServer, GWL_WDML_SERVER);
454 /* check filters for name service */
455 if (!pServer->filterOn || DdeCmpStringHandles(pServer->hszService, hszApp) == 0)
457 /* pass on to the callback */
458 hDdeData = WDML_InvokeCallback(pInstance, XTYP_CONNECT,
459 0, 0, hszTop, hszApp, 0, (DWORD)pcc, self);
462 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
464 if (pConv && pcc) pConv->wStatus |= ST_ISLOCAL;
468 else if (pInstance->servers)
470 /* pass on to the callback */
471 hDdeData = WDML_InvokeCallback(pInstance, XTYP_WILDCONNECT,
472 0, 0, hszTop, hszApp, 0, (DWORD)pcc, self);
474 if (hDdeData == (HDDEDATA)CBR_BLOCK)
476 /* MS doc is not consistent here */
477 FIXME("CBR_BLOCK returned for WILDCONNECT\n");
479 else if ((UINT)hDdeData != 0)
483 hszp = (HSZPAIR*)DdeAccessData(hDdeData, NULL);
487 for (i = 0; hszp[i].hszSvc && hszp[i].hszTopic; i++)
489 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
490 hszp[i].hszSvc, hszp[i].hszTopic);
491 if (pConv && pcc) pConv->wStatus |= ST_ISLOCAL;
493 DdeUnaccessData(hDdeData);
495 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
504 FIXME("WM_DDE_REQUEST message received!\n");
507 FIXME("WM_DDE_ADVISE message received!\n");
509 case WM_DDE_UNADVISE:
510 FIXME("WM_DDE_UNADVISE message received!\n");
513 FIXME("WM_DDE_EXECUTE message received!\n");
516 FIXME("WM_DDE_POKE message received!\n");
518 case WM_DDE_TERMINATE:
519 FIXME("WM_DDE_TERMINATE message received!\n");
524 return DefWindowProcA(hwndServer, iMsg, wParam, lParam);
527 /******************************************************************
528 * WDML_ServerQueueRequest
532 static WDML_XACT* WDML_ServerQueueRequest(WDML_CONV* pConv, LPARAM lParam)
537 UnpackDDElParam(WM_DDE_REQUEST, lParam, &uiLo, &uiHi);
539 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_REQUEST,
540 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
541 if (pXAct) pXAct->atom = uiHi;
545 /******************************************************************
546 * WDML_ServerHandleRequest
550 static WDML_QUEUE_STATE WDML_ServerHandleRequest(WDML_CONV* pConv, WDML_XACT* pXAct)
552 HDDEDATA hDdeData = 0;
553 WDML_QUEUE_STATE ret = WDML_QS_HANDLED;
555 if (!(pConv->instance->CBFflags & CBF_FAIL_REQUESTS))
558 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_REQUEST, pXAct->wFmt, (HCONV)pConv,
559 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
562 switch ((DWORD)hDdeData)
565 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, FALSE, pXAct->atom,
566 pXAct->lParam, WM_DDE_REQUEST);
573 HGLOBAL hMem = WDML_DataHandle2Global(hDdeData, FALSE, FALSE, FALSE, FALSE);
574 if (!PostMessageA(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
575 ReuseDDElParam(pXAct->lParam, WM_DDE_REQUEST, WM_DDE_DATA,
576 (UINT)hMem, (UINT)pXAct->atom)))
578 DdeFreeDataHandle(hDdeData);
584 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
588 /******************************************************************
589 * WDML_ServerQueueAdvise
593 static WDML_XACT* WDML_ServerQueueAdvise(WDML_CONV* pConv, LPARAM lParam)
598 /* XTYP_ADVSTART transaction:
599 establish link and save link info to InstanceInfoTable */
601 if (!UnpackDDElParam(WM_DDE_ADVISE, lParam, &uiLo, &uiHi))
604 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_ADVISE,
605 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
608 pXAct->hMem = (HGLOBAL)uiLo;
614 /******************************************************************
615 * WDML_ServerHandleAdvise
619 static WDML_QUEUE_STATE WDML_ServerHandleAdvise(WDML_CONV* pConv, WDML_XACT* pXAct)
623 DDEADVISE* pDdeAdvise;
627 pDdeAdvise = (DDEADVISE*)GlobalLock(pXAct->hMem);
628 uType = XTYP_ADVSTART |
629 (pDdeAdvise->fDeferUpd ? XTYPF_NODATA : 0) |
630 (pDdeAdvise->fAckReq ? XTYPF_ACKREQ : 0);
632 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
634 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_ADVSTART, pDdeAdvise->cfFormat,
635 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
646 /* billx: first to see if the link is already created. */
647 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
648 pXAct->hszItem, TRUE, pDdeAdvise->cfFormat);
652 /* we found a link, and only need to modify it in case it changes */
653 pLink->transactionType = uType;
657 TRACE("Adding Link with hConv=0x%lx\n", (DWORD)pConv);
658 WDML_AddLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
659 uType, pXAct->hszItem, pDdeAdvise->cfFormat);
664 TRACE("No data returned from the Callback\n");
668 GlobalUnlock(pXAct->hMem);
671 GlobalFree(pXAct->hMem);
675 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_ADVISE);
677 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
679 return WDML_QS_HANDLED;
682 /******************************************************************
683 * WDML_ServerQueueUnadvise
687 static WDML_XACT* WDML_ServerQueueUnadvise(WDML_CONV* pConv, LPARAM lParam)
692 UnpackDDElParam(WM_DDE_UNADVISE, lParam, &uiLo, &uiHi);
694 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_UNADVISE,
695 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
696 if (pXAct) pXAct->atom = uiHi;
700 /******************************************************************
701 * WDML_ServerHandleUnadvise
705 static WDML_QUEUE_STATE WDML_ServerHandleUnadvise(WDML_CONV* pConv, WDML_XACT* pXAct)
709 if (pXAct->hszItem == (HSZ)0 || pXAct->wFmt == 0)
711 ERR("Unsupported yet options (null item or clipboard format)\n");
712 return WDML_QS_ERROR;
715 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
716 pXAct->hszItem, TRUE, pXAct->wFmt);
719 ERR("Couln'd find link for %08lx, dropping request\n", (DWORD)pXAct->hszItem);
720 FreeDDElParam(WM_DDE_UNADVISE, pXAct->lParam);
721 return WDML_QS_ERROR;
724 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
726 WDML_InvokeCallback(pConv->instance, XTYP_ADVSTOP, pXAct->wFmt, (HCONV)pConv,
727 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
730 WDML_RemoveLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
731 pXAct->hszItem, pXAct->wFmt);
734 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, TRUE, pXAct->atom,
735 pXAct->lParam, WM_DDE_UNADVISE);
737 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
739 return WDML_QS_HANDLED;
742 /******************************************************************
747 static WDML_XACT* WDML_ServerQueueExecute(WDML_CONV* pConv, LPARAM lParam)
751 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_EXECUTE, 0, 0);
754 pXAct->hMem = (HGLOBAL)lParam;
759 /******************************************************************
760 * WDML_ServerHandleExecute
764 static WDML_QUEUE_STATE WDML_ServerHandleExecute(WDML_CONV* pConv, WDML_XACT* pXAct)
766 HDDEDATA hDdeData = DDE_FNOTPROCESSED;
767 BOOL fAck = FALSE, fBusy = FALSE;
769 if (!(pConv->instance->CBFflags & CBF_FAIL_EXECUTES))
771 LPVOID ptr = GlobalLock(pXAct->hMem);
775 hDdeData = DdeCreateDataHandle(0, ptr, GlobalSize(pXAct->hMem),
777 GlobalUnlock(pXAct->hMem);
779 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_EXECUTE, 0, (HCONV)pConv,
780 pConv->hszTopic, 0, hDdeData, 0L, 0L);
783 switch ((UINT)hDdeData)
792 WARN("Bad result code\n");
794 case DDE_FNOTPROCESSED:
797 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, pXAct->hMem, 0, 0);
799 return WDML_QS_HANDLED;
802 /******************************************************************
803 * WDML_ServerQueuePoke
807 static WDML_XACT* WDML_ServerQueuePoke(WDML_CONV* pConv, LPARAM lParam)
812 UnpackDDElParam(WM_DDE_POKE, lParam, &uiLo, &uiHi);
814 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_POKE,
815 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
819 pXAct->hMem = (HGLOBAL)uiLo;
824 /******************************************************************
825 * WDML_ServerHandlePoke
829 static WDML_QUEUE_STATE WDML_ServerHandlePoke(WDML_CONV* pConv, WDML_XACT* pXAct)
833 BOOL fBusy = FALSE, fAck = FALSE;
835 pDdePoke = (DDEPOKE*)GlobalLock(pXAct->hMem);
838 return WDML_QS_ERROR;
841 if (!(pConv->instance->CBFflags & CBF_FAIL_POKES))
843 hDdeData = DdeCreateDataHandle(pConv->instance->instanceID, pDdePoke->Value,
844 GlobalSize(pXAct->hMem) - sizeof(DDEPOKE) + 1,
845 0, 0, pDdePoke->cfFormat, 0);
848 HDDEDATA hDdeDataOut;
850 hDdeDataOut = WDML_InvokeCallback(pConv->instance, XTYP_POKE, pDdePoke->cfFormat,
851 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem,
853 switch ((UINT)hDdeDataOut)
862 FIXME("Unsupported returned value %08lx\n", (DWORD)hDdeDataOut);
864 case DDE_FNOTPROCESSED:
867 DdeFreeDataHandle(hDdeData);
870 GlobalUnlock(pXAct->hMem);
874 GlobalFree(pXAct->hMem);
876 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, pXAct->atom, pXAct->lParam, WM_DDE_POKE);
878 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
880 return WDML_QS_HANDLED;
883 /******************************************************************
884 * WDML_ServerQueueTerminate
888 static WDML_XACT* WDML_ServerQueueTerminate(WDML_CONV* pConv, LPARAM lParam)
892 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_TERMINATE, 0, 0);
896 /******************************************************************
897 * WDML_ServerHandleTerminate
901 static WDML_QUEUE_STATE WDML_ServerHandleTerminate(WDML_CONV* pConv, WDML_XACT* pXAct)
903 /* billx: two things to remove: the conv, and associated links.
904 * Respond with another WM_DDE_TERMINATE iMsg.
906 if (!(pConv->instance->CBFflags & CBF_SKIP_DISCONNECTS))
908 WDML_InvokeCallback(pConv->instance, XTYP_DISCONNECT, 0, (HCONV)pConv, 0, 0,
909 0, 0, (pConv->wStatus & ST_ISSELF) ? 1 : 0);
911 PostMessageA(pConv->hwndClient, WM_DDE_TERMINATE, (WPARAM)pConv->hwndServer, 0);
912 WDML_RemoveConv(pConv, WDML_SERVER_SIDE);
914 return WDML_QS_HANDLED;
917 /******************************************************************
922 static WDML_QUEUE_STATE WDML_ServerHandle(WDML_CONV* pConv, WDML_XACT* pXAct)
924 WDML_QUEUE_STATE qs = WDML_QS_ERROR;
926 switch (pXAct->ddeMsg)
928 case WM_DDE_INITIATE:
929 FIXME("WM_DDE_INITIATE shouldn't be there!\n");
932 qs = WDML_ServerHandleRequest(pConv, pXAct);
936 qs = WDML_ServerHandleAdvise(pConv, pXAct);
939 case WM_DDE_UNADVISE:
940 qs = WDML_ServerHandleUnadvise(pConv, pXAct);
944 qs = WDML_ServerHandleExecute(pConv, pXAct);
948 qs = WDML_ServerHandlePoke(pConv, pXAct);
951 case WM_DDE_TERMINATE:
952 qs = WDML_ServerHandleTerminate(pConv, pXAct);
956 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
960 FIXME("Unsupported message %d\n", pXAct->ddeMsg);
965 /******************************************************************
966 * WDML_ServerConvProc
970 static LRESULT CALLBACK WDML_ServerConvProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
972 WDML_INSTANCE* pInstance;
974 WDML_XACT* pXAct = NULL;
976 if (iMsg == WM_DESTROY)
978 EnterCriticalSection(&WDML_CritSect);
979 pConv = WDML_GetConvFromWnd(hwndServer);
980 if (pConv && !(pConv->wStatus & ST_TERMINATED))
982 WDML_ServerHandleTerminate(pConv, NULL);
984 LeaveCriticalSection(&WDML_CritSect);
986 if (iMsg < WM_DDE_FIRST || iMsg > WM_DDE_LAST)
988 return DefWindowProcA(hwndServer, iMsg, wParam, lParam);
991 EnterCriticalSection(&WDML_CritSect);
993 pInstance = WDML_GetInstanceFromWnd(hwndServer);
994 pConv = WDML_GetConvFromWnd(hwndServer);
998 ERR("Got a message (%u) on a not known conversation, dropping request\n", iMsg);
1001 if (pConv->hwndClient != WIN_GetFullHandle( (HWND)wParam ) || pConv->hwndServer != hwndServer)
1003 ERR("mismatch between C/S windows and converstation\n");
1006 if (pConv->instance != pInstance || pConv->instance == NULL)
1008 ERR("mismatch in instances\n");
1014 case WM_DDE_INITIATE:
1015 FIXME("WM_DDE_INITIATE message received!\n");
1018 case WM_DDE_REQUEST:
1019 pXAct = WDML_ServerQueueRequest(pConv, lParam);
1023 pXAct = WDML_ServerQueueAdvise(pConv, lParam);
1026 case WM_DDE_UNADVISE:
1027 pXAct = WDML_ServerQueueUnadvise(pConv, lParam);
1030 case WM_DDE_EXECUTE:
1031 pXAct = WDML_ServerQueueExecute(pConv, lParam);
1035 pXAct = WDML_ServerQueuePoke(pConv, lParam);
1038 case WM_DDE_TERMINATE:
1039 pXAct = WDML_ServerQueueTerminate(pConv, lParam);
1043 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1047 FIXME("Unsupported message %d\n", iMsg);
1052 pXAct->lParam = lParam;
1053 if (WDML_ServerHandle(pConv, pXAct) == WDML_QS_BLOCK)
1055 WDML_QueueTransaction(pConv, pXAct);
1059 WDML_FreeTransaction(pInstance, pXAct, TRUE);
1063 LeaveCriticalSection(&WDML_CritSect);