push b59ba84f7e04af9ef068bd4c6e96701941f0256e
[wine/hacks.git] / dlls / user32 / dde_server.c
bloba1c2425dce1f78c584ab3dc02e3c8d7ef44f6037
1 /*
2 * DDEML library
4 * Copyright 1997 Alexandre Julliard
5 * Copyright 1997 Len White
6 * Copyright 1999 Keith Matthews
7 * Copyright 2000 Corel
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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
26 #include <stdarg.h>
27 #include <string.h>
28 #include "windef.h"
29 #include "winbase.h"
30 #include "wingdi.h"
31 #include "winuser.h"
32 #include "dde.h"
33 #include "ddeml.h"
34 #include "win.h"
35 #include "wine/debug.h"
36 #include "dde_private.h"
38 WINE_DEFAULT_DEBUG_CHANNEL(ddeml);
40 static const WCHAR szServerNameClass[] = {'W','i','n','e','D','d','e','S','e','r','v','e','r','N','a','m','e',0};
41 const char WDML_szServerConvClassA[] = "WineDdeServerConvA";
42 const WCHAR WDML_szServerConvClassW[] = {'W','i','n','e','D','d','e','S','e','r','v','e','r','C','o','n','v','W',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.
50 * PARAMS
51 * idInst [I] Instance identifier
52 * hszTopic [I] Handle to topic name string
53 * hszItem [I] Handle to item name string
55 * RETURNS
56 * Success: TRUE
57 * Failure: FALSE
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;
66 ATOM atom = 0;
67 UINT count;
69 TRACE("(%d,%p,%p)\n", idInst, hszTopic, hszItem);
71 pInstance = WDML_GetInstance(idInst);
73 if (pInstance == NULL || pInstance->links == NULL)
74 return FALSE;
76 atom = WDML_MakeAtomFromHsz(hszItem);
77 if (!atom) return FALSE;
79 /* first compute the number of links which will trigger a message */
80 count = 0;
81 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
83 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
85 count++;
88 if (count >= CADV_LATEACK)
90 FIXME("too high value for count\n");
91 count &= 0xFFFF;
94 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
96 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
98 hDdeData = WDML_InvokeCallback(pInstance, XTYP_ADVREQ, pLink->uFmt, pLink->hConv,
99 hszTopic, hszItem, 0, --count, 0);
101 if (hDdeData == CBR_BLOCK)
103 /* MS doc is not consistent here */
104 FIXME("CBR_BLOCK returned for ADVREQ\n");
105 continue;
107 if (hDdeData)
109 if (pLink->transactionType & XTYPF_NODATA)
111 TRACE("no data\n");
112 hItemData = 0;
114 else
116 TRACE("with data\n");
118 hItemData = WDML_DataHandle2Global(hDdeData, FALSE, FALSE, FALSE, FALSE);
121 pConv = WDML_GetConv(pLink->hConv, TRUE);
123 if (pConv == NULL)
125 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
126 goto theError;
129 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
130 PackDDElParam(WM_DDE_DATA, (UINT_PTR)hItemData, atom)))
132 ERR("post message failed\n");
133 pConv->wStatus &= ~ST_CONNECTED;
134 pConv->instance->lastError = DMLERR_POSTMSG_FAILED;
135 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
136 GlobalFree(hItemData);
137 goto theError;
139 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
143 return TRUE;
145 theError:
146 if (atom) GlobalDeleteAtom(atom);
147 return FALSE;
151 /******************************************************************************
152 * DdeNameService [USER32.@] {Un}registers service name of DDE server
154 * PARAMS
155 * idInst [I] Instance identifier
156 * hsz1 [I] Handle to service name string
157 * hsz2 [I] Reserved
158 * afCmd [I] Service name flags
160 * RETURNS
161 * Success: Non-zero
162 * Failure: 0
164 HDDEDATA WINAPI DdeNameService(DWORD idInst, HSZ hsz1, HSZ hsz2, UINT afCmd)
166 WDML_SERVER* pServer;
167 WDML_INSTANCE* pInstance;
168 HDDEDATA hDdeData;
169 HWND hwndServer;
170 WNDCLASSEXW wndclass;
172 hDdeData = NULL;
174 TRACE("(%d,%p,%p,%x)\n", idInst, hsz1, hsz2, afCmd);
176 /* First check instance
178 pInstance = WDML_GetInstance(idInst);
179 if (pInstance == NULL)
181 TRACE("Instance not found as initialised\n");
182 /* Nothing has been initialised - exit now ! can return TRUE since effect is the same */
183 return NULL;
186 if (hsz2 != 0L)
188 /* Illegal, reserved parameter
190 pInstance->lastError = DMLERR_INVALIDPARAMETER;
191 WARN("Reserved parameter no-zero !!\n");
192 return NULL;
194 if (hsz1 == 0 && !(afCmd & DNS_UNREGISTER))
196 /* don't know if we should check this but it makes sense
197 * why supply REGISTER or filter flags if de-registering all
199 TRACE("General unregister unexpected flags\n");
200 pInstance->lastError = DMLERR_INVALIDPARAMETER;
201 return NULL;
204 switch (afCmd & (DNS_REGISTER | DNS_UNREGISTER))
206 case DNS_REGISTER:
207 pServer = WDML_FindServer(pInstance, hsz1, 0);
208 if (pServer)
210 ERR("Trying to register already registered service!\n");
211 pInstance->lastError = DMLERR_DLL_USAGE;
212 return NULL;
215 TRACE("Adding service name\n");
217 WDML_IncHSZ(pInstance, hsz1);
219 pServer = WDML_AddServer(pInstance, hsz1, 0);
221 WDML_BroadcastDDEWindows(WDML_szEventClass, WM_WDML_REGISTER,
222 pServer->atomService, pServer->atomServiceSpec);
224 wndclass.cbSize = sizeof(wndclass);
225 wndclass.style = 0;
226 wndclass.lpfnWndProc = WDML_ServerNameProc;
227 wndclass.cbClsExtra = 0;
228 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
229 wndclass.hInstance = 0;
230 wndclass.hIcon = 0;
231 wndclass.hCursor = 0;
232 wndclass.hbrBackground = 0;
233 wndclass.lpszMenuName = NULL;
234 wndclass.lpszClassName = szServerNameClass;
235 wndclass.hIconSm = 0;
237 RegisterClassExW(&wndclass);
239 hwndServer = CreateWindowW(szServerNameClass, NULL,
240 WS_POPUP, 0, 0, 0, 0,
241 0, 0, 0, 0);
243 SetWindowLongPtrW(hwndServer, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
244 SetWindowLongPtrW(hwndServer, GWL_WDML_SERVER, (ULONG_PTR)pServer);
245 TRACE("Created nameServer=%p for instance=%08x\n", hwndServer, idInst);
247 pServer->hwndServer = hwndServer;
248 break;
250 case DNS_UNREGISTER:
251 if (hsz1 == 0L)
253 /* General unregister situation
254 * terminate all server side pending conversations
256 while (pInstance->servers)
257 WDML_RemoveServer(pInstance, pInstance->servers->hszService, 0);
258 pInstance->servers = NULL;
259 TRACE("General de-register - finished\n");
261 else
263 WDML_RemoveServer(pInstance, hsz1, 0L);
265 break;
268 if (afCmd & (DNS_FILTERON | DNS_FILTEROFF))
270 /* Set filter flags on to hold notifications of connection
272 pServer = WDML_FindServer(pInstance, hsz1, 0);
273 if (!pServer)
275 /* trying to filter where no service names !!
277 pInstance->lastError = DMLERR_DLL_USAGE;
278 return NULL;
280 else
282 pServer->filterOn = (afCmd & DNS_FILTERON) != 0;
285 return (HDDEDATA)TRUE;
288 /******************************************************************
289 * WDML_CreateServerConv
293 static WDML_CONV* WDML_CreateServerConv(WDML_INSTANCE* pInstance, HWND hwndClient,
294 HWND hwndServerName, HSZ hszApp, HSZ hszTopic)
296 HWND hwndServerConv;
297 WDML_CONV* pConv;
299 if (pInstance->unicode)
301 WNDCLASSEXW wndclass;
303 wndclass.cbSize = sizeof(wndclass);
304 wndclass.style = 0;
305 wndclass.lpfnWndProc = WDML_ServerConvProc;
306 wndclass.cbClsExtra = 0;
307 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
308 wndclass.hInstance = 0;
309 wndclass.hIcon = 0;
310 wndclass.hCursor = 0;
311 wndclass.hbrBackground = 0;
312 wndclass.lpszMenuName = NULL;
313 wndclass.lpszClassName = WDML_szServerConvClassW;
314 wndclass.hIconSm = 0;
316 RegisterClassExW(&wndclass);
318 hwndServerConv = CreateWindowW(WDML_szServerConvClassW, 0,
319 WS_CHILD, 0, 0, 0, 0,
320 hwndServerName, 0, 0, 0);
322 else
324 WNDCLASSEXA wndclass;
326 wndclass.cbSize = sizeof(wndclass);
327 wndclass.style = 0;
328 wndclass.lpfnWndProc = WDML_ServerConvProc;
329 wndclass.cbClsExtra = 0;
330 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
331 wndclass.hInstance = 0;
332 wndclass.hIcon = 0;
333 wndclass.hCursor = 0;
334 wndclass.hbrBackground = 0;
335 wndclass.lpszMenuName = NULL;
336 wndclass.lpszClassName = WDML_szServerConvClassA;
337 wndclass.hIconSm = 0;
339 RegisterClassExA(&wndclass);
341 hwndServerConv = CreateWindowA(WDML_szServerConvClassA, 0,
342 WS_CHILD, 0, 0, 0, 0,
343 hwndServerName, 0, 0, 0);
346 TRACE("Created convServer=%p (nameServer=%p) for instance=%08x\n",
347 hwndServerConv, hwndServerName, pInstance->instanceID);
349 pConv = WDML_AddConv(pInstance, WDML_SERVER_SIDE, hszApp, hszTopic,
350 hwndClient, hwndServerConv);
351 if (pConv)
353 SetWindowLongPtrW(hwndServerConv, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
354 SetWindowLongPtrW(hwndServerConv, GWL_WDML_CONVERSATION, (ULONG_PTR)pConv);
356 /* this should be the only place using SendMessage for WM_DDE_ACK */
357 /* note: sent messages shall not use packing */
358 SendMessageW(hwndClient, WM_DDE_ACK, (WPARAM)hwndServerConv,
359 MAKELPARAM(WDML_MakeAtomFromHsz(hszApp), WDML_MakeAtomFromHsz(hszTopic)));
360 /* we assume we're connected since we've sent an answer...
361 * I'm not sure what we can do... it doesn't look like the return value
362 * of SendMessage is used... sigh...
364 pConv->wStatus |= ST_CONNECTED;
366 else
368 DestroyWindow(hwndServerConv);
370 return pConv;
373 /******************************************************************
374 * WDML_ServerNameProc
378 static LRESULT CALLBACK WDML_ServerNameProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
380 HWND hwndClient;
381 HSZ hszApp, hszTop;
382 HDDEDATA hDdeData = 0;
383 WDML_INSTANCE* pInstance;
384 UINT_PTR uiLo, uiHi;
386 switch (iMsg)
388 case WM_DDE_INITIATE:
390 /* wParam -- sending window handle
391 LOWORD(lParam) -- application atom
392 HIWORD(lParam) -- topic atom */
394 TRACE("WM_DDE_INITIATE message received!\n");
395 hwndClient = (HWND)wParam;
397 pInstance = WDML_GetInstanceFromWnd(hwndServer);
398 TRACE("idInst=%d, threadID=0x%x\n", pInstance->instanceID, GetCurrentThreadId());
399 if (!pInstance) return 0;
401 /* don't free DDEParams, since this is a broadcast */
402 UnpackDDElParam(WM_DDE_INITIATE, lParam, &uiLo, &uiHi);
404 hszApp = WDML_MakeHszFromAtom(pInstance, uiLo);
405 hszTop = WDML_MakeHszFromAtom(pInstance, uiHi);
407 if (!(pInstance->CBFflags & CBF_FAIL_CONNECTIONS))
409 BOOL self = FALSE;
410 CONVCONTEXT cc;
411 CONVCONTEXT* pcc = NULL;
412 WDML_CONV* pConv;
413 char buf[256];
415 if (GetWindowThreadProcessId(hwndClient, NULL) == GetWindowThreadProcessId(hwndServer, NULL) &&
416 WDML_GetInstanceFromWnd(hwndClient) == WDML_GetInstanceFromWnd(hwndServer))
418 self = TRUE;
420 /* FIXME: so far, we don't grab distant convcontext, so only check if remote is
421 * handled under DDEML, and if so build a default context
423 if ((GetClassNameA(hwndClient, buf, sizeof(buf)) &&
424 lstrcmpiA(buf, WDML_szClientConvClassA) == 0) ||
425 (GetClassNameW(hwndClient, (LPWSTR)buf, sizeof(buf)/sizeof(WCHAR)) &&
426 lstrcmpiW((LPWSTR)buf, WDML_szClientConvClassW) == 0))
428 pcc = &cc;
429 memset(pcc, 0, sizeof(*pcc));
430 pcc->cb = sizeof(*pcc);
431 pcc->iCodePage = IsWindowUnicode(hwndClient) ? CP_WINUNICODE : CP_WINANSI;
433 if ((pInstance->CBFflags & CBF_FAIL_SELFCONNECTIONS) && self)
435 TRACE("Don't do self connection as requested\n");
437 else if (hszApp && hszTop)
439 WDML_SERVER* pServer = (WDML_SERVER*)GetWindowLongPtrW(hwndServer, GWL_WDML_SERVER);
441 /* check filters for name service */
442 if (!pServer->filterOn || DdeCmpStringHandles(pServer->hszService, hszApp) == 0)
444 /* pass on to the callback */
445 hDdeData = WDML_InvokeCallback(pInstance, XTYP_CONNECT,
446 0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
447 if ((ULONG_PTR)hDdeData)
449 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
450 hszApp, hszTop);
451 if (pConv)
453 if (pcc) pConv->wStatus |= ST_ISLOCAL;
454 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
455 hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
460 else if (pInstance->servers)
462 /* pass on to the callback */
463 hDdeData = WDML_InvokeCallback(pInstance, XTYP_WILDCONNECT,
464 0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
466 if (hDdeData == CBR_BLOCK)
468 /* MS doc is not consistent here */
469 FIXME("CBR_BLOCK returned for WILDCONNECT\n");
471 else if ((ULONG_PTR)hDdeData != 0)
473 HSZPAIR* hszp;
475 hszp = (HSZPAIR*)DdeAccessData(hDdeData, NULL);
476 if (hszp)
478 int i;
479 for (i = 0; hszp[i].hszSvc && hszp[i].hszTopic; i++)
481 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
482 hszp[i].hszSvc, hszp[i].hszTopic);
483 if (pConv)
485 if (pcc) pConv->wStatus |= ST_ISLOCAL;
486 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
487 hszp[i].hszTopic, hszp[i].hszSvc, 0, (ULONG_PTR)pcc, self);
490 DdeUnaccessData(hDdeData);
492 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
497 return 0;
499 case WM_DDE_REQUEST:
500 FIXME("WM_DDE_REQUEST message received!\n");
501 return 0;
502 case WM_DDE_ADVISE:
503 FIXME("WM_DDE_ADVISE message received!\n");
504 return 0;
505 case WM_DDE_UNADVISE:
506 FIXME("WM_DDE_UNADVISE message received!\n");
507 return 0;
508 case WM_DDE_EXECUTE:
509 FIXME("WM_DDE_EXECUTE message received!\n");
510 return 0;
511 case WM_DDE_POKE:
512 FIXME("WM_DDE_POKE message received!\n");
513 return 0;
514 case WM_DDE_TERMINATE:
515 FIXME("WM_DDE_TERMINATE message received!\n");
516 return 0;
517 default:
518 break;
521 return DefWindowProcW(hwndServer, iMsg, wParam, lParam);
524 /******************************************************************
525 * WDML_ServerQueueRequest
529 static WDML_XACT* WDML_ServerQueueRequest(WDML_CONV* pConv, LPARAM lParam)
531 UINT_PTR uiLo, uiHi;
532 WDML_XACT* pXAct;
534 UnpackDDElParam(WM_DDE_REQUEST, lParam, &uiLo, &uiHi);
536 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_REQUEST,
537 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
538 if (pXAct) pXAct->atom = uiHi;
539 return pXAct;
542 /******************************************************************
543 * WDML_ServerHandleRequest
547 static WDML_QUEUE_STATE WDML_ServerHandleRequest(WDML_CONV* pConv, WDML_XACT* pXAct)
549 HDDEDATA hDdeData = 0;
550 BOOL fAck = TRUE;
552 if (!(pConv->instance->CBFflags & CBF_FAIL_REQUESTS))
555 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_REQUEST, pXAct->wFmt, (HCONV)pConv,
556 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
559 switch ((ULONG_PTR)hDdeData)
561 case 0:
562 TRACE("No data returned from the Callback\n");
563 fAck = FALSE;
564 break;
566 case (ULONG_PTR)CBR_BLOCK:
567 return WDML_QS_BLOCK;
569 default:
571 HGLOBAL hMem = WDML_DataHandle2Global(hDdeData, TRUE, FALSE, FALSE, FALSE);
572 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
573 ReuseDDElParam(pXAct->lParam, WM_DDE_REQUEST, WM_DDE_DATA,
574 (UINT_PTR)hMem, (UINT_PTR)pXAct->atom)))
576 pConv->instance->lastError = DMLERR_POSTMSG_FAILED;
577 DdeFreeDataHandle(hDdeData);
578 GlobalFree(hMem);
579 fAck = FALSE;
582 break;
585 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_REQUEST);
587 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
589 return WDML_QS_HANDLED;
592 /******************************************************************
593 * WDML_ServerQueueAdvise
597 static WDML_XACT* WDML_ServerQueueAdvise(WDML_CONV* pConv, LPARAM lParam)
599 UINT_PTR uiLo, uiHi;
600 WDML_XACT* pXAct;
602 /* XTYP_ADVSTART transaction:
603 establish link and save link info to InstanceInfoTable */
605 if (!UnpackDDElParam(WM_DDE_ADVISE, lParam, &uiLo, &uiHi))
606 return NULL;
608 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_ADVISE,
609 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
610 if (pXAct)
612 pXAct->hMem = (HGLOBAL)uiLo;
613 pXAct->atom = uiHi;
615 return pXAct;
618 /******************************************************************
619 * WDML_ServerHandleAdvise
623 static WDML_QUEUE_STATE WDML_ServerHandleAdvise(WDML_CONV* pConv, WDML_XACT* pXAct)
625 UINT uType;
626 WDML_LINK* pLink;
627 DDEADVISE* pDdeAdvise;
628 HDDEDATA hDdeData = 0;
629 BOOL fAck = TRUE;
631 pDdeAdvise = (DDEADVISE*)GlobalLock(pXAct->hMem);
632 uType = XTYP_ADVSTART |
633 (pDdeAdvise->fDeferUpd ? XTYPF_NODATA : 0) |
634 (pDdeAdvise->fAckReq ? XTYPF_ACKREQ : 0);
636 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
638 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_ADVSTART, pDdeAdvise->cfFormat,
639 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
642 switch ((ULONG_PTR)hDdeData)
644 case 0:
645 TRACE("No data returned from the Callback\n");
646 fAck = FALSE;
647 break;
649 case (ULONG_PTR)CBR_BLOCK:
650 return WDML_QS_BLOCK;
652 default:
653 /* billx: first to see if the link is already created. */
654 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
655 pXAct->hszItem, TRUE, pDdeAdvise->cfFormat);
657 if (pLink != NULL)
659 /* we found a link, and only need to modify it in case it changes */
660 pLink->transactionType = uType;
662 else
664 TRACE("Adding Link with hConv %p\n", pConv);
665 WDML_AddLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
666 uType, pXAct->hszItem, pDdeAdvise->cfFormat);
668 break;
671 GlobalUnlock(pXAct->hMem);
672 if (fAck)
674 GlobalFree(pXAct->hMem);
676 pXAct->hMem = 0;
678 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_ADVISE);
680 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
682 return WDML_QS_HANDLED;
685 /******************************************************************
686 * WDML_ServerQueueUnadvise
690 static WDML_XACT* WDML_ServerQueueUnadvise(WDML_CONV* pConv, LPARAM lParam)
692 UINT_PTR uiLo, uiHi;
693 WDML_XACT* pXAct;
695 UnpackDDElParam(WM_DDE_UNADVISE, lParam, &uiLo, &uiHi);
697 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_UNADVISE,
698 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
699 if (pXAct) pXAct->atom = uiHi;
700 return pXAct;
703 /******************************************************************
704 * WDML_ServerHandleUnadvise
708 static WDML_QUEUE_STATE WDML_ServerHandleUnadvise(WDML_CONV* pConv, WDML_XACT* pXAct)
710 WDML_LINK* pLink;
712 if (pXAct->hszItem == NULL || pXAct->wFmt == 0)
714 ERR("Unsupported yet options (null item or clipboard format)\n");
715 return WDML_QS_ERROR;
718 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
719 pXAct->hszItem, TRUE, pXAct->wFmt);
720 if (pLink == NULL)
722 ERR("Couln'd find link for %p, dropping request\n", pXAct->hszItem);
723 FreeDDElParam(WM_DDE_UNADVISE, pXAct->lParam);
724 return WDML_QS_ERROR;
727 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
729 WDML_InvokeCallback(pConv->instance, XTYP_ADVSTOP, pXAct->wFmt, (HCONV)pConv,
730 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
733 WDML_RemoveLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
734 pXAct->hszItem, pXAct->wFmt);
736 /* send back ack */
737 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, TRUE, pXAct->atom,
738 pXAct->lParam, WM_DDE_UNADVISE);
740 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
742 return WDML_QS_HANDLED;
745 /******************************************************************
746 * WDML_QueueExecute
750 static WDML_XACT* WDML_ServerQueueExecute(WDML_CONV* pConv, LPARAM lParam)
752 WDML_XACT* pXAct;
754 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_EXECUTE, 0, 0);
755 if (pXAct)
757 pXAct->hMem = (HGLOBAL)lParam;
759 return pXAct;
762 /******************************************************************
763 * WDML_ServerHandleExecute
767 static WDML_QUEUE_STATE WDML_ServerHandleExecute(WDML_CONV* pConv, WDML_XACT* pXAct)
769 HDDEDATA hDdeData = DDE_FNOTPROCESSED;
770 BOOL fAck = FALSE, fBusy = FALSE;
772 if (!(pConv->instance->CBFflags & CBF_FAIL_EXECUTES))
774 LPVOID ptr = GlobalLock(pXAct->hMem);
776 if (ptr)
778 hDdeData = DdeCreateDataHandle(0, ptr, GlobalSize(pXAct->hMem),
779 0, 0, CF_TEXT, 0);
780 GlobalUnlock(pXAct->hMem);
782 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_EXECUTE, 0, (HCONV)pConv,
783 pConv->hszTopic, 0, hDdeData, 0L, 0L);
786 switch ((ULONG_PTR)hDdeData)
788 case (ULONG_PTR)CBR_BLOCK:
789 return WDML_QS_BLOCK;
791 case DDE_FACK:
792 fAck = TRUE;
793 break;
794 case DDE_FBUSY:
795 fBusy = TRUE;
796 break;
797 default:
798 FIXME("Unsupported returned value %p\n", hDdeData);
799 /* fall through */
800 case DDE_FNOTPROCESSED:
801 break;
803 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, (UINT_PTR)pXAct->hMem, 0, 0);
805 return WDML_QS_HANDLED;
808 /******************************************************************
809 * WDML_ServerQueuePoke
813 static WDML_XACT* WDML_ServerQueuePoke(WDML_CONV* pConv, LPARAM lParam)
815 UINT_PTR uiLo, uiHi;
816 WDML_XACT* pXAct;
818 UnpackDDElParam(WM_DDE_POKE, lParam, &uiLo, &uiHi);
820 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_POKE,
821 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
822 if (pXAct)
824 pXAct->atom = uiHi;
825 pXAct->hMem = (HGLOBAL)uiLo;
827 return pXAct;
830 /******************************************************************
831 * WDML_ServerHandlePoke
835 static WDML_QUEUE_STATE WDML_ServerHandlePoke(WDML_CONV* pConv, WDML_XACT* pXAct)
837 DDEPOKE* pDdePoke;
838 HDDEDATA hDdeData;
839 BOOL fBusy = FALSE, fAck = FALSE;
841 pDdePoke = (DDEPOKE*)GlobalLock(pXAct->hMem);
842 if (!pDdePoke)
844 return WDML_QS_ERROR;
847 if (!(pConv->instance->CBFflags & CBF_FAIL_POKES))
849 hDdeData = DdeCreateDataHandle(pConv->instance->instanceID, pDdePoke->Value,
850 GlobalSize(pXAct->hMem) - sizeof(DDEPOKE) + 1,
851 0, 0, pDdePoke->cfFormat, 0);
852 if (hDdeData)
854 HDDEDATA hDdeDataOut;
856 hDdeDataOut = WDML_InvokeCallback(pConv->instance, XTYP_POKE, pDdePoke->cfFormat,
857 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem,
858 hDdeData, 0, 0);
859 switch ((ULONG_PTR)hDdeDataOut)
861 case DDE_FACK:
862 fAck = TRUE;
863 break;
864 case DDE_FBUSY:
865 fBusy = TRUE;
866 break;
867 default:
868 FIXME("Unsupported returned value %p\n", hDdeDataOut);
869 /* fal through */
870 case DDE_FNOTPROCESSED:
871 break;
873 DdeFreeDataHandle(hDdeData);
876 GlobalUnlock(pXAct->hMem);
878 if (!fAck)
880 GlobalFree(pXAct->hMem);
882 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, pXAct->atom, pXAct->lParam, WM_DDE_POKE);
884 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
886 return WDML_QS_HANDLED;
889 /******************************************************************
890 * WDML_ServerQueueTerminate
894 static WDML_XACT* WDML_ServerQueueTerminate(WDML_CONV* pConv, LPARAM lParam)
896 WDML_XACT* pXAct;
898 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_TERMINATE, 0, 0);
899 return pXAct;
902 /******************************************************************
903 * WDML_ServerHandleTerminate
907 static WDML_QUEUE_STATE WDML_ServerHandleTerminate(WDML_CONV* pConv, WDML_XACT* pXAct)
909 /* billx: two things to remove: the conv, and associated links.
910 * Respond with another WM_DDE_TERMINATE iMsg.
912 if (!(pConv->instance->CBFflags & CBF_SKIP_DISCONNECTS))
914 WDML_InvokeCallback(pConv->instance, XTYP_DISCONNECT, 0, (HCONV)pConv, 0, 0,
915 0, 0, (pConv->wStatus & ST_ISSELF) ? 1 : 0);
917 PostMessageW(pConv->hwndClient, WM_DDE_TERMINATE, (WPARAM)pConv->hwndServer, 0);
918 WDML_RemoveConv(pConv, WDML_SERVER_SIDE);
920 return WDML_QS_HANDLED;
923 /******************************************************************
924 * WDML_ServerHandle
928 WDML_QUEUE_STATE WDML_ServerHandle(WDML_CONV* pConv, WDML_XACT* pXAct)
930 WDML_QUEUE_STATE qs = WDML_QS_ERROR;
932 switch (pXAct->ddeMsg)
934 case WM_DDE_INITIATE:
935 FIXME("WM_DDE_INITIATE shouldn't be there!\n");
936 break;
937 case WM_DDE_REQUEST:
938 qs = WDML_ServerHandleRequest(pConv, pXAct);
939 break;
941 case WM_DDE_ADVISE:
942 qs = WDML_ServerHandleAdvise(pConv, pXAct);
943 break;
945 case WM_DDE_UNADVISE:
946 qs = WDML_ServerHandleUnadvise(pConv, pXAct);
947 break;
949 case WM_DDE_EXECUTE:
950 qs = WDML_ServerHandleExecute(pConv, pXAct);
951 break;
953 case WM_DDE_POKE:
954 qs = WDML_ServerHandlePoke(pConv, pXAct);
955 break;
957 case WM_DDE_TERMINATE:
958 qs = WDML_ServerHandleTerminate(pConv, pXAct);
959 break;
961 case WM_DDE_ACK:
962 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
963 break;
965 default:
966 FIXME("Unsupported message %d\n", pXAct->ddeMsg);
968 return qs;
971 /******************************************************************
972 * WDML_ServerConvProc
976 static LRESULT CALLBACK WDML_ServerConvProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
978 WDML_INSTANCE* pInstance;
979 WDML_CONV* pConv;
980 WDML_XACT* pXAct = NULL;
982 TRACE("%p %04x %08lx %08lx\n", hwndServer, iMsg, wParam, lParam);
984 if (iMsg == WM_DESTROY)
986 pConv = WDML_GetConvFromWnd(hwndServer);
987 if (pConv && !(pConv->wStatus & ST_TERMINATED))
989 WDML_ServerHandleTerminate(pConv, NULL);
992 if (iMsg < WM_DDE_FIRST || iMsg > WM_DDE_LAST)
994 return IsWindowUnicode(hwndServer) ? DefWindowProcW(hwndServer, iMsg, wParam, lParam) :
995 DefWindowProcA(hwndServer, iMsg, wParam, lParam);
998 pInstance = WDML_GetInstanceFromWnd(hwndServer);
999 pConv = WDML_GetConvFromWnd(hwndServer);
1001 if (!pConv)
1003 ERR("Got a message (%x) on a not known conversation, dropping request\n", iMsg);
1004 return 0;
1006 if (pConv->hwndClient != WIN_GetFullHandle( (HWND)wParam ) || pConv->hwndServer != hwndServer)
1008 ERR("mismatch between C/S windows and conversation\n");
1009 return 0;
1011 if (pConv->instance != pInstance || pConv->instance == NULL)
1013 ERR("mismatch in instances\n");
1014 return 0;
1017 switch (iMsg)
1019 case WM_DDE_INITIATE:
1020 FIXME("WM_DDE_INITIATE message received!\n");
1021 break;
1023 case WM_DDE_REQUEST:
1024 pXAct = WDML_ServerQueueRequest(pConv, lParam);
1025 break;
1027 case WM_DDE_ADVISE:
1028 pXAct = WDML_ServerQueueAdvise(pConv, lParam);
1029 break;
1031 case WM_DDE_UNADVISE:
1032 pXAct = WDML_ServerQueueUnadvise(pConv, lParam);
1033 break;
1035 case WM_DDE_EXECUTE:
1036 pXAct = WDML_ServerQueueExecute(pConv, lParam);
1037 break;
1039 case WM_DDE_POKE:
1040 pXAct = WDML_ServerQueuePoke(pConv, lParam);
1041 break;
1043 case WM_DDE_TERMINATE:
1044 pXAct = WDML_ServerQueueTerminate(pConv, lParam);
1045 break;
1047 case WM_DDE_ACK:
1048 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1049 break;
1051 default:
1052 FIXME("Unsupported message %x\n", iMsg);
1053 break;
1056 if (pXAct)
1058 pXAct->lParam = lParam;
1060 if ((pConv->wStatus & ST_BLOCKED) || WDML_ServerHandle(pConv, pXAct) == WDML_QS_BLOCK)
1062 TRACE("Transactions are blocked, add to the queue and exit\n");
1063 WDML_QueueTransaction(pConv, pXAct);
1065 else
1067 WDML_FreeTransaction(pInstance, pXAct, TRUE);
1070 else
1071 pConv->instance->lastError = DMLERR_MEMORY_ERROR;
1073 return 0;