Implement NtAccessCheck.
[wine/gsoc-2012-control.git] / dlls / user / dde / server.c
blob39b5c5d86ce0cf798f2135f5e29af13d519f00c4
1 /* -*- tab-width: 8; c-basic-offset: 4 -*- */
3 /*
4 * DDEML library
6 * Copyright 1997 Alexandre Julliard
7 * Copyright 1997 Len White
8 * Copyright 1999 Keith Matthews
9 * Copyright 2000 Corel
10 * Copyright 2001 Eric Pouech
11 * Copyright 2003, 2004, 2005 Dmitry Timoshkov
13 * This library is free software; you can redistribute it and/or
14 * modify it under the terms of the GNU Lesser General Public
15 * License as published by the Free Software Foundation; either
16 * version 2.1 of the License, or (at your option) any later version.
18 * This library is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
21 * Lesser General Public License for more details.
23 * You should have received a copy of the GNU Lesser General Public
24 * License along with this library; if not, write to the Free Software
25 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
28 #include <stdarg.h>
29 #include <string.h>
30 #include "windef.h"
31 #include "winbase.h"
32 #include "wingdi.h"
33 #include "winuser.h"
34 #include "winerror.h"
35 #include "dde.h"
36 #include "ddeml.h"
37 #include "win.h"
38 #include "wine/debug.h"
39 #include "dde/dde_private.h"
41 WINE_DEFAULT_DEBUG_CHANNEL(ddeml);
43 static const WCHAR szServerNameClass[] = {'W','i','n','e','D','d','e','S','e','r','v','e','r','N','a','m','e',0};
44 const WCHAR WDML_szServerConvClass[] = {'W','i','n','e','D','d','e','S','e','r','v','e','r','C','o','n','v',0};
46 static LRESULT CALLBACK WDML_ServerNameProc(HWND, UINT, WPARAM, LPARAM);
47 static LRESULT CALLBACK WDML_ServerConvProc(HWND, UINT, WPARAM, LPARAM);
49 /******************************************************************************
50 * DdePostAdvise [USER32.@] Send transaction to DDE callback function.
52 * PARAMS
53 * idInst [I] Instance identifier
54 * hszTopic [I] Handle to topic name string
55 * hszItem [I] Handle to item name string
57 * RETURNS
58 * Success: TRUE
59 * Failure: FALSE
61 BOOL WINAPI DdePostAdvise(DWORD idInst, HSZ hszTopic, HSZ hszItem)
63 WDML_INSTANCE* pInstance = NULL;
64 WDML_LINK* pLink = NULL;
65 HDDEDATA hDdeData = 0;
66 HGLOBAL hItemData = 0;
67 WDML_CONV* pConv = NULL;
68 ATOM atom = 0;
69 UINT count;
71 TRACE("(%ld,%p,%p)\n", idInst, hszTopic, hszItem);
73 EnterCriticalSection(&WDML_CritSect);
75 pInstance = WDML_GetInstance(idInst);
77 if (pInstance == NULL || pInstance->links == NULL)
79 goto theError;
82 atom = WDML_MakeAtomFromHsz(hszItem);
83 if (!atom) goto theError;
85 /* first compute the number of links which will trigger a message */
86 count = 0;
87 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
89 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
91 count++;
94 if (count >= CADV_LATEACK)
96 FIXME("too high value for count\n");
97 count &= 0xFFFF;
100 for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
102 if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
104 hDdeData = WDML_InvokeCallback(pInstance, XTYP_ADVREQ, pLink->uFmt, pLink->hConv,
105 hszTopic, hszItem, 0, --count, 0);
107 if (hDdeData == (HDDEDATA)CBR_BLOCK)
109 /* MS doc is not consistent here */
110 FIXME("CBR_BLOCK returned for ADVREQ\n");
111 continue;
113 if (hDdeData)
115 if (pLink->transactionType & XTYPF_NODATA)
117 TRACE("no data\n");
118 hItemData = 0;
120 else
122 TRACE("with data\n");
124 hItemData = WDML_DataHandle2Global(hDdeData, FALSE, FALSE, FALSE, FALSE);
127 pConv = WDML_GetConv(pLink->hConv, TRUE);
129 if (pConv == NULL)
131 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
132 goto theError;
135 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
136 PackDDElParam(WM_DDE_DATA, (UINT_PTR)hItemData, atom)))
138 ERR("post message failed\n");
139 pConv->wStatus &= ~ST_CONNECTED;
140 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
141 GlobalFree(hItemData);
142 goto theError;
144 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
148 LeaveCriticalSection(&WDML_CritSect);
149 return TRUE;
150 theError:
151 LeaveCriticalSection(&WDML_CritSect);
152 if (atom) GlobalDeleteAtom(atom);
153 return FALSE;
157 /******************************************************************************
158 * DdeNameService [USER32.@] {Un}registers service name of DDE server
160 * PARAMS
161 * idInst [I] Instance identifier
162 * hsz1 [I] Handle to service name string
163 * hsz2 [I] Reserved
164 * afCmd [I] Service name flags
166 * RETURNS
167 * Success: Non-zero
168 * Failure: 0
170 HDDEDATA WINAPI DdeNameService(DWORD idInst, HSZ hsz1, HSZ hsz2, UINT afCmd)
172 WDML_SERVER* pServer;
173 WDML_INSTANCE* pInstance;
174 HDDEDATA hDdeData;
175 HWND hwndServer;
176 WNDCLASSEXW wndclass;
178 hDdeData = NULL;
180 TRACE("(%ld,%p,%p,%x)\n", idInst, hsz1, hsz2, afCmd);
182 EnterCriticalSection(&WDML_CritSect);
184 /* First check instance
186 pInstance = WDML_GetInstance(idInst);
187 if (pInstance == NULL)
189 TRACE("Instance not found as initialised\n");
190 /* Nothing has been initialised - exit now ! can return TRUE since effect is the same */
191 goto theError;
194 if (hsz2 != 0L)
196 /* Illegal, reserved parameter
198 pInstance->lastError = DMLERR_INVALIDPARAMETER;
199 WARN("Reserved parameter no-zero !!\n");
200 goto theError;
202 if (hsz1 == 0 && !(afCmd & DNS_UNREGISTER))
204 /* don't know if we should check this but it makes sense
205 * why supply REGISTER or filter flags if de-registering all
207 TRACE("General unregister unexpected flags\n");
208 pInstance->lastError = DMLERR_INVALIDPARAMETER;
209 goto theError;
212 switch (afCmd & (DNS_REGISTER | DNS_UNREGISTER))
214 case DNS_REGISTER:
215 pServer = WDML_FindServer(pInstance, hsz1, 0);
216 if (pServer)
218 ERR("Trying to register already registered service!\n");
219 pInstance->lastError = DMLERR_DLL_USAGE;
220 goto theError;
223 TRACE("Adding service name\n");
225 WDML_IncHSZ(pInstance, hsz1);
227 pServer = WDML_AddServer(pInstance, hsz1, 0);
229 WDML_BroadcastDDEWindows(WDML_szEventClass, WM_WDML_REGISTER,
230 pServer->atomService, pServer->atomServiceSpec);
232 wndclass.cbSize = sizeof(wndclass);
233 wndclass.style = 0;
234 wndclass.lpfnWndProc = WDML_ServerNameProc;
235 wndclass.cbClsExtra = 0;
236 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
237 wndclass.hInstance = 0;
238 wndclass.hIcon = 0;
239 wndclass.hCursor = 0;
240 wndclass.hbrBackground = 0;
241 wndclass.lpszMenuName = NULL;
242 wndclass.lpszClassName = szServerNameClass;
243 wndclass.hIconSm = 0;
245 RegisterClassExW(&wndclass);
247 LeaveCriticalSection(&WDML_CritSect);
248 hwndServer = CreateWindowW(szServerNameClass, NULL,
249 WS_POPUP, 0, 0, 0, 0,
250 0, 0, 0, 0);
251 EnterCriticalSection(&WDML_CritSect);
253 SetWindowLongPtrW(hwndServer, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
254 SetWindowLongPtrW(hwndServer, GWL_WDML_SERVER, (ULONG_PTR)pServer);
255 TRACE("Created nameServer=%p for instance=%08lx\n", hwndServer, idInst);
257 pServer->hwndServer = hwndServer;
258 break;
260 case DNS_UNREGISTER:
261 if (hsz1 == 0L)
263 /* General unregister situation
264 * terminate all server side pending conversations
266 while (pInstance->servers)
267 WDML_RemoveServer(pInstance, pInstance->servers->hszService, 0);
268 pInstance->servers = NULL;
269 TRACE("General de-register - finished\n");
271 else
273 WDML_RemoveServer(pInstance, hsz1, 0L);
275 break;
278 if (afCmd & (DNS_FILTERON | DNS_FILTEROFF))
280 /* Set filter flags on to hold notifications of connection
282 pServer = WDML_FindServer(pInstance, hsz1, 0);
283 if (!pServer)
285 /* trying to filter where no service names !!
287 pInstance->lastError = DMLERR_DLL_USAGE;
288 goto theError;
290 else
292 pServer->filterOn = (afCmd & DNS_FILTERON) != 0;
295 LeaveCriticalSection(&WDML_CritSect);
296 return (HDDEDATA)TRUE;
298 theError:
299 LeaveCriticalSection(&WDML_CritSect);
300 return FALSE;
303 /******************************************************************
304 * WDML_CreateServerConv
308 static WDML_CONV* WDML_CreateServerConv(WDML_INSTANCE* pInstance, HWND hwndClient,
309 HWND hwndServerName, HSZ hszApp, HSZ hszTopic)
311 HWND hwndServerConv;
312 WDML_CONV* pConv;
313 WNDCLASSEXW wndclass;
315 wndclass.cbSize = sizeof(wndclass);
316 wndclass.style = 0;
317 wndclass.lpfnWndProc = WDML_ServerConvProc;
318 wndclass.cbClsExtra = 0;
319 wndclass.cbWndExtra = 2 * sizeof(ULONG_PTR);
320 wndclass.hInstance = 0;
321 wndclass.hIcon = 0;
322 wndclass.hCursor = 0;
323 wndclass.hbrBackground = 0;
324 wndclass.lpszMenuName = NULL;
325 wndclass.lpszClassName = WDML_szServerConvClass;
326 wndclass.hIconSm = 0;
328 RegisterClassExW(&wndclass);
330 hwndServerConv = CreateWindowW(WDML_szServerConvClass, 0,
331 WS_CHILD, 0, 0, 0, 0,
332 hwndServerName, 0, 0, 0);
334 TRACE("Created convServer=%p (nameServer=%p) for instance=%08lx\n",
335 hwndServerConv, hwndServerName, pInstance->instanceID);
337 pConv = WDML_AddConv(pInstance, WDML_SERVER_SIDE, hszApp, hszTopic,
338 hwndClient, hwndServerConv);
339 if (pConv)
341 SetWindowLongPtrW(hwndServerConv, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
342 SetWindowLongPtrW(hwndServerConv, GWL_WDML_CONVERSATION, (ULONG_PTR)pConv);
344 /* this should be the only place using SendMessage for WM_DDE_ACK */
345 /* note: sent messages shall not use packing */
346 SendMessageW(hwndClient, WM_DDE_ACK, (WPARAM)hwndServerConv,
347 MAKELPARAM(WDML_MakeAtomFromHsz(hszApp), WDML_MakeAtomFromHsz(hszTopic)));
348 /* we assume we're connected since we've sent an answer...
349 * I'm not sure what we can do... it doesn't look like the return value
350 * of SendMessage is used... sigh...
352 pConv->wStatus |= ST_CONNECTED;
354 else
356 DestroyWindow(hwndServerConv);
358 return pConv;
361 /******************************************************************
362 * WDML_ServerNameProc
366 static LRESULT CALLBACK WDML_ServerNameProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
368 HWND hwndClient;
369 HSZ hszApp, hszTop;
370 HDDEDATA hDdeData = 0;
371 WDML_INSTANCE* pInstance;
372 UINT_PTR uiLo, uiHi;
374 switch (iMsg)
376 case WM_DDE_INITIATE:
378 /* wParam -- sending window handle
379 LOWORD(lParam) -- application atom
380 HIWORD(lParam) -- topic atom */
382 TRACE("WM_DDE_INITIATE message received!\n");
383 hwndClient = (HWND)wParam;
385 pInstance = WDML_GetInstanceFromWnd(hwndServer);
386 TRACE("idInst=%ld, threadID=0x%lx\n", pInstance->instanceID, GetCurrentThreadId());
387 if (!pInstance) return 0;
389 /* don't free DDEParams, since this is a broadcast */
390 UnpackDDElParam(WM_DDE_INITIATE, lParam, &uiLo, &uiHi);
392 hszApp = WDML_MakeHszFromAtom(pInstance, uiLo);
393 hszTop = WDML_MakeHszFromAtom(pInstance, uiHi);
395 if (!(pInstance->CBFflags & CBF_FAIL_CONNECTIONS))
397 BOOL self = FALSE;
398 CONVCONTEXT cc;
399 CONVCONTEXT* pcc = NULL;
400 WDML_CONV* pConv;
401 WCHAR buf[256];
403 if (GetWindowThreadProcessId(hwndClient, NULL) == GetWindowThreadProcessId(hwndServer, NULL) &&
404 WDML_GetInstanceFromWnd(hwndClient) == WDML_GetInstanceFromWnd(hwndServer))
406 self = TRUE;
408 /* FIXME: so far, we don't grab distant convcontext, so only check if remote is
409 * handled under DDEML, and if so build a default context
411 if (GetClassNameW(hwndClient, buf, sizeof(buf)/sizeof(WCHAR)) &&
412 lstrcmpiW(buf, WDML_szClientConvClass) == 0)
414 pcc = &cc;
415 memset(pcc, 0, sizeof(*pcc));
416 pcc->cb = sizeof(*pcc);
417 pcc->iCodePage = CP_WINUNICODE;
419 if ((pInstance->CBFflags & CBF_FAIL_SELFCONNECTIONS) && self)
421 TRACE("Don't do self connection as requested\n");
423 else if (hszApp && hszTop)
425 WDML_SERVER* pServer = (WDML_SERVER*)GetWindowLongPtrW(hwndServer, GWL_WDML_SERVER);
427 /* check filters for name service */
428 if (!pServer->filterOn || DdeCmpStringHandles(pServer->hszService, hszApp) == 0)
430 /* pass on to the callback */
431 hDdeData = WDML_InvokeCallback(pInstance, XTYP_CONNECT,
432 0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
433 if ((ULONG_PTR)hDdeData)
435 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
436 hszApp, hszTop);
437 if (pConv)
439 if (pcc) pConv->wStatus |= ST_ISLOCAL;
440 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
441 hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
446 else if (pInstance->servers)
448 /* pass on to the callback */
449 hDdeData = WDML_InvokeCallback(pInstance, XTYP_WILDCONNECT,
450 0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
452 if (hDdeData == (HDDEDATA)CBR_BLOCK)
454 /* MS doc is not consistent here */
455 FIXME("CBR_BLOCK returned for WILDCONNECT\n");
457 else if ((ULONG_PTR)hDdeData != 0)
459 HSZPAIR* hszp;
461 hszp = (HSZPAIR*)DdeAccessData(hDdeData, NULL);
462 if (hszp)
464 int i;
465 for (i = 0; hszp[i].hszSvc && hszp[i].hszTopic; i++)
467 pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
468 hszp[i].hszSvc, hszp[i].hszTopic);
469 if (pConv)
471 if (pcc) pConv->wStatus |= ST_ISLOCAL;
472 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
473 hszp[i].hszTopic, hszp[i].hszSvc, 0, (ULONG_PTR)pcc, self);
476 DdeUnaccessData(hDdeData);
478 if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
483 return 0;
485 case WM_DDE_REQUEST:
486 FIXME("WM_DDE_REQUEST message received!\n");
487 return 0;
488 case WM_DDE_ADVISE:
489 FIXME("WM_DDE_ADVISE message received!\n");
490 return 0;
491 case WM_DDE_UNADVISE:
492 FIXME("WM_DDE_UNADVISE message received!\n");
493 return 0;
494 case WM_DDE_EXECUTE:
495 FIXME("WM_DDE_EXECUTE message received!\n");
496 return 0;
497 case WM_DDE_POKE:
498 FIXME("WM_DDE_POKE message received!\n");
499 return 0;
500 case WM_DDE_TERMINATE:
501 FIXME("WM_DDE_TERMINATE message received!\n");
502 return 0;
503 default:
504 break;
507 return DefWindowProcW(hwndServer, iMsg, wParam, lParam);
510 /******************************************************************
511 * WDML_ServerQueueRequest
515 static WDML_XACT* WDML_ServerQueueRequest(WDML_CONV* pConv, LPARAM lParam)
517 UINT_PTR uiLo, uiHi;
518 WDML_XACT* pXAct;
520 UnpackDDElParam(WM_DDE_REQUEST, lParam, &uiLo, &uiHi);
522 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_REQUEST,
523 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
524 if (pXAct) pXAct->atom = uiHi;
525 return pXAct;
528 /******************************************************************
529 * WDML_ServerHandleRequest
533 static WDML_QUEUE_STATE WDML_ServerHandleRequest(WDML_CONV* pConv, WDML_XACT* pXAct)
535 HDDEDATA hDdeData = 0;
536 BOOL fAck = TRUE;
538 if (!(pConv->instance->CBFflags & CBF_FAIL_REQUESTS))
541 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_REQUEST, pXAct->wFmt, (HCONV)pConv,
542 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
545 switch ((ULONG_PTR)hDdeData)
547 case 0:
548 TRACE("No data returned from the Callback\n");
549 fAck = FALSE;
550 break;
552 case (ULONG_PTR)CBR_BLOCK:
553 return WDML_QS_BLOCK;
555 default:
557 HGLOBAL hMem = WDML_DataHandle2Global(hDdeData, TRUE, FALSE, FALSE, FALSE);
558 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
559 ReuseDDElParam(pXAct->lParam, WM_DDE_REQUEST, WM_DDE_DATA,
560 (UINT_PTR)hMem, (UINT_PTR)pXAct->atom)))
562 DdeFreeDataHandle(hDdeData);
563 GlobalFree(hMem);
564 fAck = FALSE;
567 break;
570 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_REQUEST);
572 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
574 return WDML_QS_HANDLED;
577 /******************************************************************
578 * WDML_ServerQueueAdvise
582 static WDML_XACT* WDML_ServerQueueAdvise(WDML_CONV* pConv, LPARAM lParam)
584 UINT_PTR uiLo, uiHi;
585 WDML_XACT* pXAct;
587 /* XTYP_ADVSTART transaction:
588 establish link and save link info to InstanceInfoTable */
590 if (!UnpackDDElParam(WM_DDE_ADVISE, lParam, &uiLo, &uiHi))
591 return NULL;
593 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_ADVISE,
594 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
595 if (pXAct)
597 pXAct->hMem = (HGLOBAL)uiLo;
598 pXAct->atom = uiHi;
600 return pXAct;
603 /******************************************************************
604 * WDML_ServerHandleAdvise
608 static WDML_QUEUE_STATE WDML_ServerHandleAdvise(WDML_CONV* pConv, WDML_XACT* pXAct)
610 UINT uType;
611 WDML_LINK* pLink;
612 DDEADVISE* pDdeAdvise;
613 HDDEDATA hDdeData = 0;
614 BOOL fAck = TRUE;
616 pDdeAdvise = (DDEADVISE*)GlobalLock(pXAct->hMem);
617 uType = XTYP_ADVSTART |
618 (pDdeAdvise->fDeferUpd ? XTYPF_NODATA : 0) |
619 (pDdeAdvise->fAckReq ? XTYPF_ACKREQ : 0);
621 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
623 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_ADVSTART, pDdeAdvise->cfFormat,
624 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
627 switch ((ULONG_PTR)hDdeData)
629 case 0:
630 TRACE("No data returned from the Callback\n");
631 fAck = FALSE;
632 break;
634 case (ULONG_PTR)CBR_BLOCK:
635 return WDML_QS_BLOCK;
637 default:
638 /* billx: first to see if the link is already created. */
639 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
640 pXAct->hszItem, TRUE, pDdeAdvise->cfFormat);
642 if (pLink != NULL)
644 /* we found a link, and only need to modify it in case it changes */
645 pLink->transactionType = uType;
647 else
649 TRACE("Adding Link with hConv %p\n", pConv);
650 WDML_AddLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
651 uType, pXAct->hszItem, pDdeAdvise->cfFormat);
653 break;
656 GlobalUnlock(pXAct->hMem);
657 if (fAck)
659 GlobalFree(pXAct->hMem);
661 pXAct->hMem = 0;
663 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_ADVISE);
665 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
667 return WDML_QS_HANDLED;
670 /******************************************************************
671 * WDML_ServerQueueUnadvise
675 static WDML_XACT* WDML_ServerQueueUnadvise(WDML_CONV* pConv, LPARAM lParam)
677 UINT_PTR uiLo, uiHi;
678 WDML_XACT* pXAct;
680 UnpackDDElParam(WM_DDE_UNADVISE, lParam, &uiLo, &uiHi);
682 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_UNADVISE,
683 uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
684 if (pXAct) pXAct->atom = uiHi;
685 return pXAct;
688 /******************************************************************
689 * WDML_ServerHandleUnadvise
693 static WDML_QUEUE_STATE WDML_ServerHandleUnadvise(WDML_CONV* pConv, WDML_XACT* pXAct)
695 WDML_LINK* pLink;
697 if (pXAct->hszItem == NULL || pXAct->wFmt == 0)
699 ERR("Unsupported yet options (null item or clipboard format)\n");
700 return WDML_QS_ERROR;
703 pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
704 pXAct->hszItem, TRUE, pXAct->wFmt);
705 if (pLink == NULL)
707 ERR("Couln'd find link for %p, dropping request\n", pXAct->hszItem);
708 FreeDDElParam(WM_DDE_UNADVISE, pXAct->lParam);
709 return WDML_QS_ERROR;
712 if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
714 WDML_InvokeCallback(pConv->instance, XTYP_ADVSTOP, pXAct->wFmt, (HCONV)pConv,
715 pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
718 WDML_RemoveLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
719 pXAct->hszItem, pXAct->wFmt);
721 /* send back ack */
722 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, TRUE, pXAct->atom,
723 pXAct->lParam, WM_DDE_UNADVISE);
725 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
727 return WDML_QS_HANDLED;
730 /******************************************************************
731 * WDML_QueueExecute
735 static WDML_XACT* WDML_ServerQueueExecute(WDML_CONV* pConv, LPARAM lParam)
737 WDML_XACT* pXAct;
739 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_EXECUTE, 0, 0);
740 if (pXAct)
742 pXAct->hMem = (HGLOBAL)lParam;
744 return pXAct;
747 /******************************************************************
748 * WDML_ServerHandleExecute
752 static WDML_QUEUE_STATE WDML_ServerHandleExecute(WDML_CONV* pConv, WDML_XACT* pXAct)
754 HDDEDATA hDdeData = DDE_FNOTPROCESSED;
755 BOOL fAck = FALSE, fBusy = FALSE;
757 if (!(pConv->instance->CBFflags & CBF_FAIL_EXECUTES))
759 LPVOID ptr = GlobalLock(pXAct->hMem);
761 if (ptr)
763 hDdeData = DdeCreateDataHandle(0, ptr, GlobalSize(pXAct->hMem),
764 0, 0, CF_TEXT, 0);
765 GlobalUnlock(pXAct->hMem);
767 hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_EXECUTE, 0, (HCONV)pConv,
768 pConv->hszTopic, 0, hDdeData, 0L, 0L);
771 switch ((ULONG_PTR)hDdeData)
773 case (ULONG_PTR)CBR_BLOCK:
774 return WDML_QS_BLOCK;
776 case DDE_FACK:
777 fAck = TRUE;
778 break;
779 case DDE_FBUSY:
780 fBusy = TRUE;
781 break;
782 default:
783 FIXME("Unsupported returned value %p\n", hDdeData);
784 /* fall through */
785 case DDE_FNOTPROCESSED:
786 break;
788 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, (UINT)pXAct->hMem, 0, 0);
790 return WDML_QS_HANDLED;
793 /******************************************************************
794 * WDML_ServerQueuePoke
798 static WDML_XACT* WDML_ServerQueuePoke(WDML_CONV* pConv, LPARAM lParam)
800 UINT_PTR uiLo, uiHi;
801 WDML_XACT* pXAct;
803 UnpackDDElParam(WM_DDE_POKE, lParam, &uiLo, &uiHi);
805 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_POKE,
806 0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
807 if (pXAct)
809 pXAct->atom = uiHi;
810 pXAct->hMem = (HGLOBAL)uiLo;
812 return pXAct;
815 /******************************************************************
816 * WDML_ServerHandlePoke
820 static WDML_QUEUE_STATE WDML_ServerHandlePoke(WDML_CONV* pConv, WDML_XACT* pXAct)
822 DDEPOKE* pDdePoke;
823 HDDEDATA hDdeData;
824 BOOL fBusy = FALSE, fAck = FALSE;
826 pDdePoke = (DDEPOKE*)GlobalLock(pXAct->hMem);
827 if (!pDdePoke)
829 return WDML_QS_ERROR;
832 if (!(pConv->instance->CBFflags & CBF_FAIL_POKES))
834 hDdeData = DdeCreateDataHandle(pConv->instance->instanceID, pDdePoke->Value,
835 GlobalSize(pXAct->hMem) - sizeof(DDEPOKE) + 1,
836 0, 0, pDdePoke->cfFormat, 0);
837 if (hDdeData)
839 HDDEDATA hDdeDataOut;
841 hDdeDataOut = WDML_InvokeCallback(pConv->instance, XTYP_POKE, pDdePoke->cfFormat,
842 (HCONV)pConv, pConv->hszTopic, pXAct->hszItem,
843 hDdeData, 0, 0);
844 switch ((ULONG_PTR)hDdeDataOut)
846 case DDE_FACK:
847 fAck = TRUE;
848 break;
849 case DDE_FBUSY:
850 fBusy = TRUE;
851 break;
852 default:
853 FIXME("Unsupported returned value %p\n", hDdeDataOut);
854 /* fal through */
855 case DDE_FNOTPROCESSED:
856 break;
858 DdeFreeDataHandle(hDdeData);
861 GlobalUnlock(pXAct->hMem);
863 if (!fAck)
865 GlobalFree(pXAct->hMem);
867 WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, pXAct->atom, pXAct->lParam, WM_DDE_POKE);
869 WDML_DecHSZ(pConv->instance, pXAct->hszItem);
871 return WDML_QS_HANDLED;
874 /******************************************************************
875 * WDML_ServerQueueTerminate
879 static WDML_XACT* WDML_ServerQueueTerminate(WDML_CONV* pConv, LPARAM lParam)
881 WDML_XACT* pXAct;
883 pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_TERMINATE, 0, 0);
884 return pXAct;
887 /******************************************************************
888 * WDML_ServerHandleTerminate
892 static WDML_QUEUE_STATE WDML_ServerHandleTerminate(WDML_CONV* pConv, WDML_XACT* pXAct)
894 /* billx: two things to remove: the conv, and associated links.
895 * Respond with another WM_DDE_TERMINATE iMsg.
897 if (!(pConv->instance->CBFflags & CBF_SKIP_DISCONNECTS))
899 WDML_InvokeCallback(pConv->instance, XTYP_DISCONNECT, 0, (HCONV)pConv, 0, 0,
900 0, 0, (pConv->wStatus & ST_ISSELF) ? 1 : 0);
902 PostMessageW(pConv->hwndClient, WM_DDE_TERMINATE, (WPARAM)pConv->hwndServer, 0);
903 WDML_RemoveConv(pConv, WDML_SERVER_SIDE);
905 return WDML_QS_HANDLED;
908 /******************************************************************
909 * WDML_ServerHandle
913 WDML_QUEUE_STATE WDML_ServerHandle(WDML_CONV* pConv, WDML_XACT* pXAct)
915 WDML_QUEUE_STATE qs = WDML_QS_ERROR;
917 switch (pXAct->ddeMsg)
919 case WM_DDE_INITIATE:
920 FIXME("WM_DDE_INITIATE shouldn't be there!\n");
921 break;
922 case WM_DDE_REQUEST:
923 qs = WDML_ServerHandleRequest(pConv, pXAct);
924 break;
926 case WM_DDE_ADVISE:
927 qs = WDML_ServerHandleAdvise(pConv, pXAct);
928 break;
930 case WM_DDE_UNADVISE:
931 qs = WDML_ServerHandleUnadvise(pConv, pXAct);
932 break;
934 case WM_DDE_EXECUTE:
935 qs = WDML_ServerHandleExecute(pConv, pXAct);
936 break;
938 case WM_DDE_POKE:
939 qs = WDML_ServerHandlePoke(pConv, pXAct);
940 break;
942 case WM_DDE_TERMINATE:
943 qs = WDML_ServerHandleTerminate(pConv, pXAct);
944 break;
946 case WM_DDE_ACK:
947 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
948 break;
950 default:
951 FIXME("Unsupported message %d\n", pXAct->ddeMsg);
953 return qs;
956 /******************************************************************
957 * WDML_ServerConvProc
961 static LRESULT CALLBACK WDML_ServerConvProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
963 WDML_INSTANCE* pInstance;
964 WDML_CONV* pConv;
965 WDML_XACT* pXAct = NULL;
967 TRACE("%p %04x %08x %08lx\n", hwndServer, iMsg, wParam , lParam);
969 if (iMsg == WM_DESTROY)
971 EnterCriticalSection(&WDML_CritSect);
972 pConv = WDML_GetConvFromWnd(hwndServer);
973 if (pConv && !(pConv->wStatus & ST_TERMINATED))
975 WDML_ServerHandleTerminate(pConv, NULL);
977 LeaveCriticalSection(&WDML_CritSect);
979 if (iMsg < WM_DDE_FIRST || iMsg > WM_DDE_LAST)
981 return DefWindowProcW(hwndServer, iMsg, wParam, lParam);
984 EnterCriticalSection(&WDML_CritSect);
986 pInstance = WDML_GetInstanceFromWnd(hwndServer);
987 pConv = WDML_GetConvFromWnd(hwndServer);
989 if (!pConv)
991 ERR("Got a message (%x) on a not known conversation, dropping request\n", iMsg);
992 goto theError;
994 if (pConv->hwndClient != WIN_GetFullHandle( (HWND)wParam ) || pConv->hwndServer != hwndServer)
996 ERR("mismatch between C/S windows and converstation\n");
997 goto theError;
999 if (pConv->instance != pInstance || pConv->instance == NULL)
1001 ERR("mismatch in instances\n");
1002 goto theError;
1005 switch (iMsg)
1007 case WM_DDE_INITIATE:
1008 FIXME("WM_DDE_INITIATE message received!\n");
1009 break;
1011 case WM_DDE_REQUEST:
1012 pXAct = WDML_ServerQueueRequest(pConv, lParam);
1013 break;
1015 case WM_DDE_ADVISE:
1016 pXAct = WDML_ServerQueueAdvise(pConv, lParam);
1017 break;
1019 case WM_DDE_UNADVISE:
1020 pXAct = WDML_ServerQueueUnadvise(pConv, lParam);
1021 break;
1023 case WM_DDE_EXECUTE:
1024 pXAct = WDML_ServerQueueExecute(pConv, lParam);
1025 break;
1027 case WM_DDE_POKE:
1028 pXAct = WDML_ServerQueuePoke(pConv, lParam);
1029 break;
1031 case WM_DDE_TERMINATE:
1032 pXAct = WDML_ServerQueueTerminate(pConv, lParam);
1033 break;
1035 case WM_DDE_ACK:
1036 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1037 break;
1039 default:
1040 FIXME("Unsupported message %x\n", iMsg);
1043 if (pXAct)
1045 pXAct->lParam = lParam;
1046 if (WDML_ServerHandle(pConv, pXAct) == WDML_QS_BLOCK)
1048 WDML_QueueTransaction(pConv, pXAct);
1050 else
1052 WDML_FreeTransaction(pInstance, pXAct, TRUE);
1055 theError:
1056 LeaveCriticalSection(&WDML_CritSect);
1057 return 0;