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
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
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.
53 * idInst [I] Instance identifier
54 * hszTopic [I] Handle to topic name string
55 * hszItem [I] Handle to item name string
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
;
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
)
82 atom
= WDML_MakeAtomFromHsz(hszItem
);
83 if (!atom
) goto theError
;
85 /* first compute the number of links which will trigger a message */
87 for (pLink
= pInstance
->links
[WDML_SERVER_SIDE
]; pLink
!= NULL
; pLink
= pLink
->next
)
89 if (DdeCmpStringHandles(hszItem
, pLink
->hszItem
) == 0)
94 if (count
>= CADV_LATEACK
)
96 FIXME("too high value for count\n");
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");
115 if (pLink
->transactionType
& XTYPF_NODATA
)
122 TRACE("with data\n");
124 hItemData
= WDML_DataHandle2Global(hDdeData
, FALSE
, FALSE
, FALSE
, FALSE
);
127 pConv
= WDML_GetConv(pLink
->hConv
, TRUE
);
131 if (!WDML_IsAppOwned(hDdeData
)) DdeFreeDataHandle(hDdeData
);
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
);
144 if (!WDML_IsAppOwned(hDdeData
)) DdeFreeDataHandle(hDdeData
);
148 LeaveCriticalSection(&WDML_CritSect
);
151 LeaveCriticalSection(&WDML_CritSect
);
152 if (atom
) GlobalDeleteAtom(atom
);
157 /******************************************************************************
158 * DdeNameService [USER32.@] {Un}registers service name of DDE server
161 * idInst [I] Instance identifier
162 * hsz1 [I] Handle to service name string
164 * afCmd [I] Service name flags
170 HDDEDATA WINAPI
DdeNameService(DWORD idInst
, HSZ hsz1
, HSZ hsz2
, UINT afCmd
)
172 WDML_SERVER
* pServer
;
173 WDML_INSTANCE
* pInstance
;
176 WNDCLASSEXW wndclass
;
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 */
196 /* Illegal, reserved parameter
198 pInstance
->lastError
= DMLERR_INVALIDPARAMETER
;
199 WARN("Reserved parameter no-zero !!\n");
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
;
212 switch (afCmd
& (DNS_REGISTER
| DNS_UNREGISTER
))
215 pServer
= WDML_FindServer(pInstance
, hsz1
, 0);
218 ERR("Trying to register already registered service!\n");
219 pInstance
->lastError
= DMLERR_DLL_USAGE
;
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
);
234 wndclass
.lpfnWndProc
= WDML_ServerNameProc
;
235 wndclass
.cbClsExtra
= 0;
236 wndclass
.cbWndExtra
= 2 * sizeof(ULONG_PTR
);
237 wndclass
.hInstance
= 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,
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
;
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");
273 WDML_RemoveServer(pInstance
, hsz1
, 0L);
278 if (afCmd
& (DNS_FILTERON
| DNS_FILTEROFF
))
280 /* Set filter flags on to hold notifications of connection
282 pServer
= WDML_FindServer(pInstance
, hsz1
, 0);
285 /* trying to filter where no service names !!
287 pInstance
->lastError
= DMLERR_DLL_USAGE
;
292 pServer
->filterOn
= (afCmd
& DNS_FILTERON
) != 0;
295 LeaveCriticalSection(&WDML_CritSect
);
296 return (HDDEDATA
)TRUE
;
299 LeaveCriticalSection(&WDML_CritSect
);
303 /******************************************************************
304 * WDML_CreateServerConv
308 static WDML_CONV
* WDML_CreateServerConv(WDML_INSTANCE
* pInstance
, HWND hwndClient
,
309 HWND hwndServerName
, HSZ hszApp
, HSZ hszTopic
)
313 WNDCLASSEXW wndclass
;
315 wndclass
.cbSize
= sizeof(wndclass
);
317 wndclass
.lpfnWndProc
= WDML_ServerConvProc
;
318 wndclass
.cbClsExtra
= 0;
319 wndclass
.cbWndExtra
= 2 * sizeof(ULONG_PTR
);
320 wndclass
.hInstance
= 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
);
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
;
356 DestroyWindow(hwndServerConv
);
361 /******************************************************************
362 * WDML_ServerNameProc
366 static LRESULT CALLBACK
WDML_ServerNameProc(HWND hwndServer
, UINT iMsg
, WPARAM wParam
, LPARAM lParam
)
370 HDDEDATA hDdeData
= 0;
371 WDML_INSTANCE
* pInstance
;
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
))
399 CONVCONTEXT
* pcc
= NULL
;
403 if (GetWindowThreadProcessId(hwndClient
, NULL
) == GetWindowThreadProcessId(hwndServer
, NULL
) &&
404 WDML_GetInstanceFromWnd(hwndClient
) == WDML_GetInstanceFromWnd(hwndServer
))
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)
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
,
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)
461 hszp
= (HSZPAIR
*)DdeAccessData(hDdeData
, NULL
);
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
);
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
);
486 FIXME("WM_DDE_REQUEST message received!\n");
489 FIXME("WM_DDE_ADVISE message received!\n");
491 case WM_DDE_UNADVISE
:
492 FIXME("WM_DDE_UNADVISE message received!\n");
495 FIXME("WM_DDE_EXECUTE message received!\n");
498 FIXME("WM_DDE_POKE message received!\n");
500 case WM_DDE_TERMINATE
:
501 FIXME("WM_DDE_TERMINATE message received!\n");
507 return DefWindowProcW(hwndServer
, iMsg
, wParam
, lParam
);
510 /******************************************************************
511 * WDML_ServerQueueRequest
515 static WDML_XACT
* WDML_ServerQueueRequest(WDML_CONV
* pConv
, LPARAM lParam
)
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
;
528 /******************************************************************
529 * WDML_ServerHandleRequest
533 static WDML_QUEUE_STATE
WDML_ServerHandleRequest(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
535 HDDEDATA hDdeData
= 0;
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
)
548 TRACE("No data returned from the Callback\n");
552 case (ULONG_PTR
)CBR_BLOCK
:
553 return WDML_QS_BLOCK
;
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
);
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
)
587 /* XTYP_ADVSTART transaction:
588 establish link and save link info to InstanceInfoTable */
590 if (!UnpackDDElParam(WM_DDE_ADVISE
, lParam
, &uiLo
, &uiHi
))
593 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_ADVISE
,
594 0, WDML_MakeHszFromAtom(pConv
->instance
, uiHi
));
597 pXAct
->hMem
= (HGLOBAL
)uiLo
;
603 /******************************************************************
604 * WDML_ServerHandleAdvise
608 static WDML_QUEUE_STATE
WDML_ServerHandleAdvise(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
612 DDEADVISE
* pDdeAdvise
;
613 HDDEDATA hDdeData
= 0;
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
)
630 TRACE("No data returned from the Callback\n");
634 case (ULONG_PTR
)CBR_BLOCK
:
635 return WDML_QS_BLOCK
;
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
);
644 /* we found a link, and only need to modify it in case it changes */
645 pLink
->transactionType
= uType
;
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
);
656 GlobalUnlock(pXAct
->hMem
);
659 GlobalFree(pXAct
->hMem
);
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
)
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
;
688 /******************************************************************
689 * WDML_ServerHandleUnadvise
693 static WDML_QUEUE_STATE
WDML_ServerHandleUnadvise(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
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
);
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
);
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 /******************************************************************
735 static WDML_XACT
* WDML_ServerQueueExecute(WDML_CONV
* pConv
, LPARAM lParam
)
739 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_EXECUTE
, 0, 0);
742 pXAct
->hMem
= (HGLOBAL
)lParam
;
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
);
763 hDdeData
= DdeCreateDataHandle(0, ptr
, GlobalSize(pXAct
->hMem
),
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
;
783 FIXME("Unsupported returned value %p\n", hDdeData
);
785 case DDE_FNOTPROCESSED
:
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
)
803 UnpackDDElParam(WM_DDE_POKE
, lParam
, &uiLo
, &uiHi
);
805 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_POKE
,
806 0, WDML_MakeHszFromAtom(pConv
->instance
, uiHi
));
810 pXAct
->hMem
= (HGLOBAL
)uiLo
;
815 /******************************************************************
816 * WDML_ServerHandlePoke
820 static WDML_QUEUE_STATE
WDML_ServerHandlePoke(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
824 BOOL fBusy
= FALSE
, fAck
= FALSE
;
826 pDdePoke
= (DDEPOKE
*)GlobalLock(pXAct
->hMem
);
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);
839 HDDEDATA hDdeDataOut
;
841 hDdeDataOut
= WDML_InvokeCallback(pConv
->instance
, XTYP_POKE
, pDdePoke
->cfFormat
,
842 (HCONV
)pConv
, pConv
->hszTopic
, pXAct
->hszItem
,
844 switch ((ULONG_PTR
)hDdeDataOut
)
853 FIXME("Unsupported returned value %p\n", hDdeDataOut
);
855 case DDE_FNOTPROCESSED
:
858 DdeFreeDataHandle(hDdeData
);
861 GlobalUnlock(pXAct
->hMem
);
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
)
883 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_TERMINATE
, 0, 0);
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 /******************************************************************
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");
923 qs
= WDML_ServerHandleRequest(pConv
, pXAct
);
927 qs
= WDML_ServerHandleAdvise(pConv
, pXAct
);
930 case WM_DDE_UNADVISE
:
931 qs
= WDML_ServerHandleUnadvise(pConv
, pXAct
);
935 qs
= WDML_ServerHandleExecute(pConv
, pXAct
);
939 qs
= WDML_ServerHandlePoke(pConv
, pXAct
);
942 case WM_DDE_TERMINATE
:
943 qs
= WDML_ServerHandleTerminate(pConv
, pXAct
);
947 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
951 FIXME("Unsupported message %d\n", pXAct
->ddeMsg
);
956 /******************************************************************
957 * WDML_ServerConvProc
961 static LRESULT CALLBACK
WDML_ServerConvProc(HWND hwndServer
, UINT iMsg
, WPARAM wParam
, LPARAM lParam
)
963 WDML_INSTANCE
* pInstance
;
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
);
991 ERR("Got a message (%x) on a not known conversation, dropping request\n", iMsg
);
994 if (pConv
->hwndClient
!= WIN_GetFullHandle( (HWND
)wParam
) || pConv
->hwndServer
!= hwndServer
)
996 ERR("mismatch between C/S windows and converstation\n");
999 if (pConv
->instance
!= pInstance
|| pConv
->instance
== NULL
)
1001 ERR("mismatch in instances\n");
1007 case WM_DDE_INITIATE
:
1008 FIXME("WM_DDE_INITIATE message received!\n");
1011 case WM_DDE_REQUEST
:
1012 pXAct
= WDML_ServerQueueRequest(pConv
, lParam
);
1016 pXAct
= WDML_ServerQueueAdvise(pConv
, lParam
);
1019 case WM_DDE_UNADVISE
:
1020 pXAct
= WDML_ServerQueueUnadvise(pConv
, lParam
);
1023 case WM_DDE_EXECUTE
:
1024 pXAct
= WDML_ServerQueueExecute(pConv
, lParam
);
1028 pXAct
= WDML_ServerQueuePoke(pConv
, lParam
);
1031 case WM_DDE_TERMINATE
:
1032 pXAct
= WDML_ServerQueueTerminate(pConv
, lParam
);
1036 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1040 FIXME("Unsupported message %x\n", iMsg
);
1045 pXAct
->lParam
= lParam
;
1046 if (WDML_ServerHandle(pConv
, pXAct
) == WDML_QS_BLOCK
)
1048 WDML_QueueTransaction(pConv
, pXAct
);
1052 WDML_FreeTransaction(pInstance
, pXAct
, TRUE
);
1056 LeaveCriticalSection(&WDML_CritSect
);