4 * Copyright 1997 Alexandre Julliard
5 * Copyright 1997 Len White
6 * Copyright 1999 Keith Matthews
8 * Copyright 2001 Eric Pouech
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Lesser General Public
12 * License as published by the Free Software Foundation; either
13 * version 2.1 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Lesser General Public License for more details.
20 * You should have received a copy of the GNU Lesser General Public
21 * License along with this library; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
29 WINE_DEFAULT_DEBUG_CHANNEL(ddeml
);
31 static const char szServerNameClassA
[] = "DdeServerNameAnsi";
32 const char WDML_szServerConvClassA
[] = "DdeServerConvAnsi";
33 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};
35 static LRESULT CALLBACK
WDML_ServerNameProc(HWND
, UINT
, WPARAM
, LPARAM
);
36 static LRESULT CALLBACK
WDML_ServerConvProc(HWND
, UINT
, WPARAM
, LPARAM
);
38 /******************************************************************************
39 * DdePostAdvise [USER32.@] Send transaction to DDE callback function.
42 * idInst [I] Instance identifier
43 * hszTopic [I] Handle to topic name string
44 * hszItem [I] Handle to item name string
50 BOOL WINAPI
DdePostAdvise(DWORD idInst
, HSZ hszTopic
, HSZ hszItem
)
52 WDML_INSTANCE
* pInstance
= NULL
;
53 WDML_LINK
* pLink
= NULL
;
54 HDDEDATA hDdeData
= 0;
55 HGLOBAL hItemData
= 0;
56 WDML_CONV
* pConv
= NULL
;
60 TRACE("(%ld,%p,%p)\n", idInst
, hszTopic
, hszItem
);
62 EnterCriticalSection(&WDML_CritSect
);
64 pInstance
= WDML_GetInstance(idInst
);
66 if (pInstance
== NULL
|| pInstance
->links
== NULL
)
71 atom
= WDML_MakeAtomFromHsz(hszItem
);
72 if (!atom
) goto theError
;
74 /* first compute the number of links which will trigger a message */
76 for (pLink
= pInstance
->links
[WDML_SERVER_SIDE
]; pLink
!= NULL
; pLink
= pLink
->next
)
78 if (DdeCmpStringHandles(hszItem
, pLink
->hszItem
) == 0)
83 if (count
>= CADV_LATEACK
)
85 FIXME("too high value for count\n");
89 for (pLink
= pInstance
->links
[WDML_SERVER_SIDE
]; pLink
!= NULL
; pLink
= pLink
->next
)
91 if (DdeCmpStringHandles(hszItem
, pLink
->hszItem
) == 0)
93 hDdeData
= WDML_InvokeCallback(pInstance
, XTYP_ADVREQ
, pLink
->uFmt
, pLink
->hConv
,
94 hszTopic
, hszItem
, 0, --count
, 0);
96 if (hDdeData
== (HDDEDATA
)CBR_BLOCK
)
98 /* MS doc is not consistent here */
99 FIXME("CBR_BLOCK returned for ADVREQ\n");
104 if (pLink
->transactionType
& XTYPF_NODATA
)
111 TRACE("with data\n");
113 hItemData
= WDML_DataHandle2Global(hDdeData
, FALSE
, FALSE
, FALSE
, FALSE
);
116 pConv
= WDML_GetConv(pLink
->hConv
, TRUE
);
120 if (!WDML_IsAppOwned(hDdeData
)) DdeFreeDataHandle(hDdeData
);
124 if (!PostMessageA(pConv
->hwndClient
, WM_DDE_DATA
, (WPARAM
)pConv
->hwndServer
,
125 PackDDElParam(WM_DDE_DATA
, (UINT_PTR
)hItemData
, atom
)))
127 ERR("post message failed\n");
128 pConv
->wStatus
&= ~ST_CONNECTED
;
129 if (!WDML_IsAppOwned(hDdeData
)) DdeFreeDataHandle(hDdeData
);
130 GlobalFree(hItemData
);
133 if (!WDML_IsAppOwned(hDdeData
)) DdeFreeDataHandle(hDdeData
);
137 LeaveCriticalSection(&WDML_CritSect
);
140 LeaveCriticalSection(&WDML_CritSect
);
141 if (atom
) GlobalDeleteAtom(atom
);
146 /******************************************************************************
147 * DdeNameService [USER32.@] {Un}registers service name of DDE server
150 * idInst [I] Instance identifier
151 * hsz1 [I] Handle to service name string
153 * afCmd [I] Service name flags
159 HDDEDATA WINAPI
DdeNameService(DWORD idInst
, HSZ hsz1
, HSZ hsz2
, UINT afCmd
)
161 WDML_SERVER
* pServer
;
162 WDML_INSTANCE
* pInstance
;
165 WNDCLASSEXA wndclass
;
169 TRACE("(%ld,%p,%p,%x)\n", idInst
, hsz1
, hsz2
, afCmd
);
171 EnterCriticalSection(&WDML_CritSect
);
173 /* First check instance
175 pInstance
= WDML_GetInstance(idInst
);
176 if (pInstance
== NULL
)
178 TRACE("Instance not found as initialised\n");
179 /* Nothing has been initialised - exit now ! can return TRUE since effect is the same */
185 /* Illegal, reserved parameter
187 pInstance
->lastError
= DMLERR_INVALIDPARAMETER
;
188 WARN("Reserved parameter no-zero !!\n");
191 if (hsz1
== 0 && !(afCmd
& DNS_UNREGISTER
))
193 /* don't know if we should check this but it makes sense
194 * why supply REGISTER or filter flags if de-registering all
196 TRACE("General unregister unexpected flags\n");
197 pInstance
->lastError
= DMLERR_INVALIDPARAMETER
;
201 switch (afCmd
& (DNS_REGISTER
| DNS_UNREGISTER
))
204 pServer
= WDML_FindServer(pInstance
, hsz1
, 0);
207 ERR("Trying to register already registered service!\n");
208 pInstance
->lastError
= DMLERR_DLL_USAGE
;
212 TRACE("Adding service name\n");
214 WDML_IncHSZ(pInstance
, hsz1
);
216 pServer
= WDML_AddServer(pInstance
, hsz1
, 0);
218 WDML_BroadcastDDEWindows(WDML_szEventClass
, WM_WDML_REGISTER
,
219 pServer
->atomService
, pServer
->atomServiceSpec
);
221 wndclass
.cbSize
= sizeof(wndclass
);
223 wndclass
.lpfnWndProc
= WDML_ServerNameProc
;
224 wndclass
.cbClsExtra
= 0;
225 wndclass
.cbWndExtra
= 2 * sizeof(DWORD
);
226 wndclass
.hInstance
= 0;
228 wndclass
.hCursor
= 0;
229 wndclass
.hbrBackground
= 0;
230 wndclass
.lpszMenuName
= NULL
;
231 wndclass
.lpszClassName
= szServerNameClassA
;
232 wndclass
.hIconSm
= 0;
234 RegisterClassExA(&wndclass
);
236 LeaveCriticalSection(&WDML_CritSect
);
237 hwndServer
= CreateWindowA(szServerNameClassA
, NULL
,
238 WS_POPUP
, 0, 0, 0, 0,
240 EnterCriticalSection(&WDML_CritSect
);
242 SetWindowLongA(hwndServer
, GWL_WDML_INSTANCE
, (DWORD
)pInstance
);
243 SetWindowLongA(hwndServer
, GWL_WDML_SERVER
, (DWORD
)pServer
);
244 TRACE("Created nameServer=%p for instance=%08lx\n", hwndServer
, idInst
);
246 pServer
->hwndServer
= hwndServer
;
252 /* General unregister situation
253 * terminate all server side pending conversations
255 while (pInstance
->servers
)
256 WDML_RemoveServer(pInstance
, pInstance
->servers
->hszService
, 0);
257 pInstance
->servers
= NULL
;
258 TRACE("General de-register - finished\n");
262 WDML_RemoveServer(pInstance
, hsz1
, 0L);
267 if (afCmd
& (DNS_FILTERON
| DNS_FILTEROFF
))
269 /* Set filter flags on to hold notifications of connection
271 pServer
= WDML_FindServer(pInstance
, hsz1
, 0);
274 /* trying to filter where no service names !!
276 pInstance
->lastError
= DMLERR_DLL_USAGE
;
281 pServer
->filterOn
= (afCmd
& DNS_FILTERON
) != 0;
284 LeaveCriticalSection(&WDML_CritSect
);
285 return (HDDEDATA
)TRUE
;
288 LeaveCriticalSection(&WDML_CritSect
);
292 /******************************************************************
293 * WDML_CreateServerConv
297 static WDML_CONV
* WDML_CreateServerConv(WDML_INSTANCE
* pInstance
, HWND hwndClient
,
298 HWND hwndServerName
, HSZ hszApp
, HSZ hszTopic
)
303 if (pInstance
->unicode
)
305 WNDCLASSEXW wndclass
;
307 wndclass
.cbSize
= sizeof(wndclass
);
309 wndclass
.lpfnWndProc
= WDML_ServerConvProc
;
310 wndclass
.cbClsExtra
= 0;
311 wndclass
.cbWndExtra
= 2 * sizeof(DWORD
);
312 wndclass
.hInstance
= 0;
314 wndclass
.hCursor
= 0;
315 wndclass
.hbrBackground
= 0;
316 wndclass
.lpszMenuName
= NULL
;
317 wndclass
.lpszClassName
= WDML_szServerConvClassW
;
318 wndclass
.hIconSm
= 0;
320 RegisterClassExW(&wndclass
);
322 hwndServerConv
= CreateWindowW(WDML_szServerConvClassW
, 0,
323 WS_CHILD
, 0, 0, 0, 0,
324 hwndServerName
, 0, 0, 0);
328 WNDCLASSEXA wndclass
;
330 wndclass
.cbSize
= sizeof(wndclass
);
332 wndclass
.lpfnWndProc
= WDML_ServerConvProc
;
333 wndclass
.cbClsExtra
= 0;
334 wndclass
.cbWndExtra
= 2 * sizeof(DWORD
);
335 wndclass
.hInstance
= 0;
337 wndclass
.hCursor
= 0;
338 wndclass
.hbrBackground
= 0;
339 wndclass
.lpszMenuName
= NULL
;
340 wndclass
.lpszClassName
= WDML_szServerConvClassA
;
341 wndclass
.hIconSm
= 0;
343 RegisterClassExA(&wndclass
);
345 hwndServerConv
= CreateWindowA(WDML_szServerConvClassA
, 0,
346 WS_CHILD
, 0, 0, 0, 0,
347 hwndServerName
, 0, 0, 0);
350 TRACE("Created convServer=%p (nameServer=%p) for instance=%08lx\n",
351 hwndServerConv
, hwndServerName
, pInstance
->instanceID
);
353 pConv
= WDML_AddConv(pInstance
, WDML_SERVER_SIDE
, hszApp
, hszTopic
,
354 hwndClient
, hwndServerConv
);
357 SetWindowLongA(hwndServerConv
, GWL_WDML_INSTANCE
, (DWORD
)pInstance
);
358 SetWindowLongA(hwndServerConv
, GWL_WDML_CONVERSATION
, (DWORD
)pConv
);
360 /* this should be the only place using SendMessage for WM_DDE_ACK */
361 /* note: sent messages shall not use packing */
362 SendMessageA(hwndClient
, WM_DDE_ACK
, (WPARAM
)hwndServerConv
,
363 MAKELPARAM(WDML_MakeAtomFromHsz(hszApp
), WDML_MakeAtomFromHsz(hszTopic
)));
364 /* we assume we're connected since we've sent an answer...
365 * I'm not sure what we can do... it doesn't look like the return value
366 * of SendMessage is used... sigh...
368 pConv
->wStatus
|= ST_CONNECTED
;
372 DestroyWindow(hwndServerConv
);
377 /******************************************************************
378 * WDML_ServerNameProc
382 static LRESULT CALLBACK
WDML_ServerNameProc(HWND hwndServer
, UINT iMsg
, WPARAM wParam
, LPARAM lParam
)
386 HDDEDATA hDdeData
= 0;
387 WDML_INSTANCE
* pInstance
;
392 case WM_DDE_INITIATE
:
394 /* wParam -- sending window handle
395 LOWORD(lParam) -- application atom
396 HIWORD(lParam) -- topic atom */
398 TRACE("WM_DDE_INITIATE message received!\n");
399 hwndClient
= (HWND
)wParam
;
401 pInstance
= WDML_GetInstanceFromWnd(hwndServer
);
402 TRACE("idInst=%ld, threadID=0x%lx\n", pInstance
->instanceID
, GetCurrentThreadId());
403 if (!pInstance
) return 0;
405 /* don't free DDEParams, since this is a broadcast */
406 UnpackDDElParam(WM_DDE_INITIATE
, lParam
, &uiLo
, &uiHi
);
408 hszApp
= WDML_MakeHszFromAtom(pInstance
, uiLo
);
409 hszTop
= WDML_MakeHszFromAtom(pInstance
, uiHi
);
411 if (!(pInstance
->CBFflags
& CBF_FAIL_CONNECTIONS
))
415 CONVCONTEXT
* pcc
= NULL
;
419 if (GetWindowThreadProcessId(hwndClient
, NULL
) == GetWindowThreadProcessId(hwndServer
, NULL
) &&
420 WDML_GetInstanceFromWnd(hwndClient
) == WDML_GetInstanceFromWnd(hwndServer
))
424 /* FIXME: so far, we don't grab distant convcontext, so only check if remote is
425 * handled under DDEML, and if so build a default context
427 if ((GetClassNameA(hwndClient
, buf
, sizeof(buf
)) &&
428 strcmp(buf
, WDML_szClientConvClassA
) == 0) ||
429 (GetClassNameW(hwndClient
, (LPWSTR
)buf
, sizeof(buf
)/sizeof(WCHAR
)) &&
430 lstrcmpW((LPWSTR
)buf
, WDML_szClientConvClassW
) == 0))
433 memset(pcc
, 0, sizeof(*pcc
));
434 pcc
->cb
= sizeof(*pcc
);
435 pcc
->iCodePage
= IsWindowUnicode(hwndClient
) ? CP_WINUNICODE
: CP_WINANSI
;
437 if ((pInstance
->CBFflags
& CBF_FAIL_SELFCONNECTIONS
) && self
)
439 TRACE("Don't do self connection as requested\n");
441 else if (hszApp
&& hszTop
)
443 WDML_SERVER
* pServer
= (WDML_SERVER
*)GetWindowLongA(hwndServer
, GWL_WDML_SERVER
);
445 /* check filters for name service */
446 if (!pServer
->filterOn
|| DdeCmpStringHandles(pServer
->hszService
, hszApp
) == 0)
448 /* pass on to the callback */
449 hDdeData
= WDML_InvokeCallback(pInstance
, XTYP_CONNECT
,
450 0, 0, hszTop
, hszApp
, 0, (DWORD
)pcc
, self
);
453 pConv
= WDML_CreateServerConv(pInstance
, hwndClient
, hwndServer
,
457 if (pcc
) pConv
->wStatus
|= ST_ISLOCAL
;
458 WDML_InvokeCallback(pInstance
, XTYP_CONNECT_CONFIRM
, 0, (HCONV
)pConv
,
459 hszTop
, hszApp
, 0, (DWORD
)pcc
, self
);
464 else if (pInstance
->servers
)
466 /* pass on to the callback */
467 hDdeData
= WDML_InvokeCallback(pInstance
, XTYP_WILDCONNECT
,
468 0, 0, hszTop
, hszApp
, 0, (DWORD
)pcc
, self
);
470 if (hDdeData
== (HDDEDATA
)CBR_BLOCK
)
472 /* MS doc is not consistent here */
473 FIXME("CBR_BLOCK returned for WILDCONNECT\n");
475 else if ((UINT
)hDdeData
!= 0)
479 hszp
= (HSZPAIR
*)DdeAccessData(hDdeData
, NULL
);
483 for (i
= 0; hszp
[i
].hszSvc
&& hszp
[i
].hszTopic
; i
++)
485 pConv
= WDML_CreateServerConv(pInstance
, hwndClient
, hwndServer
,
486 hszp
[i
].hszSvc
, hszp
[i
].hszTopic
);
489 if (pcc
) pConv
->wStatus
|= ST_ISLOCAL
;
490 WDML_InvokeCallback(pInstance
, XTYP_CONNECT_CONFIRM
, 0, (HCONV
)pConv
,
491 hszp
[i
].hszTopic
, hszp
[i
].hszSvc
, 0, (DWORD
)pcc
, self
);
494 DdeUnaccessData(hDdeData
);
496 if (!WDML_IsAppOwned(hDdeData
)) DdeFreeDataHandle(hDdeData
);
505 FIXME("WM_DDE_REQUEST message received!\n");
508 FIXME("WM_DDE_ADVISE message received!\n");
510 case WM_DDE_UNADVISE
:
511 FIXME("WM_DDE_UNADVISE message received!\n");
514 FIXME("WM_DDE_EXECUTE message received!\n");
517 FIXME("WM_DDE_POKE message received!\n");
519 case WM_DDE_TERMINATE
:
520 FIXME("WM_DDE_TERMINATE message received!\n");
525 return DefWindowProcA(hwndServer
, iMsg
, wParam
, lParam
);
528 /******************************************************************
529 * WDML_ServerQueueRequest
533 static WDML_XACT
* WDML_ServerQueueRequest(WDML_CONV
* pConv
, LPARAM lParam
)
538 UnpackDDElParam(WM_DDE_REQUEST
, lParam
, &uiLo
, &uiHi
);
540 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_REQUEST
,
541 uiLo
, WDML_MakeHszFromAtom(pConv
->instance
, uiHi
));
542 if (pXAct
) pXAct
->atom
= uiHi
;
546 /******************************************************************
547 * WDML_ServerHandleRequest
551 static WDML_QUEUE_STATE
WDML_ServerHandleRequest(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
553 HDDEDATA hDdeData
= 0;
554 WDML_QUEUE_STATE ret
= WDML_QS_HANDLED
;
556 if (!(pConv
->instance
->CBFflags
& CBF_FAIL_REQUESTS
))
559 hDdeData
= WDML_InvokeCallback(pConv
->instance
, XTYP_REQUEST
, pXAct
->wFmt
, (HCONV
)pConv
,
560 pConv
->hszTopic
, pXAct
->hszItem
, 0, 0, 0);
563 switch ((ULONG_PTR
)hDdeData
)
566 WDML_PostAck(pConv
, WDML_SERVER_SIDE
, 0, FALSE
, FALSE
, pXAct
->atom
,
567 pXAct
->lParam
, WM_DDE_REQUEST
);
569 case (ULONG_PTR
)CBR_BLOCK
:
574 HGLOBAL hMem
= WDML_DataHandle2Global(hDdeData
, TRUE
, FALSE
, FALSE
, FALSE
);
575 if (!PostMessageA(pConv
->hwndClient
, WM_DDE_DATA
, (WPARAM
)pConv
->hwndServer
,
576 ReuseDDElParam(pXAct
->lParam
, WM_DDE_REQUEST
, WM_DDE_DATA
,
577 (UINT_PTR
)hMem
, (UINT_PTR
)pXAct
->atom
)))
579 DdeFreeDataHandle(hDdeData
);
585 WDML_DecHSZ(pConv
->instance
, pXAct
->hszItem
);
589 /******************************************************************
590 * WDML_ServerQueueAdvise
594 static WDML_XACT
* WDML_ServerQueueAdvise(WDML_CONV
* pConv
, LPARAM lParam
)
599 /* XTYP_ADVSTART transaction:
600 establish link and save link info to InstanceInfoTable */
602 if (!UnpackDDElParam(WM_DDE_ADVISE
, lParam
, &uiLo
, &uiHi
))
605 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_ADVISE
,
606 0, WDML_MakeHszFromAtom(pConv
->instance
, uiHi
));
609 pXAct
->hMem
= (HGLOBAL
)uiLo
;
615 /******************************************************************
616 * WDML_ServerHandleAdvise
620 static WDML_QUEUE_STATE
WDML_ServerHandleAdvise(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
624 DDEADVISE
* pDdeAdvise
;
628 pDdeAdvise
= (DDEADVISE
*)GlobalLock(pXAct
->hMem
);
629 uType
= XTYP_ADVSTART
|
630 (pDdeAdvise
->fDeferUpd
? XTYPF_NODATA
: 0) |
631 (pDdeAdvise
->fAckReq
? XTYPF_ACKREQ
: 0);
633 if (!(pConv
->instance
->CBFflags
& CBF_FAIL_ADVISES
))
635 hDdeData
= WDML_InvokeCallback(pConv
->instance
, XTYP_ADVSTART
, pDdeAdvise
->cfFormat
,
636 (HCONV
)pConv
, pConv
->hszTopic
, pXAct
->hszItem
, 0, 0, 0);
647 /* billx: first to see if the link is already created. */
648 pLink
= WDML_FindLink(pConv
->instance
, (HCONV
)pConv
, WDML_SERVER_SIDE
,
649 pXAct
->hszItem
, TRUE
, pDdeAdvise
->cfFormat
);
653 /* we found a link, and only need to modify it in case it changes */
654 pLink
->transactionType
= uType
;
658 TRACE("Adding Link with hConv %p\n", pConv
);
659 WDML_AddLink(pConv
->instance
, (HCONV
)pConv
, WDML_SERVER_SIDE
,
660 uType
, pXAct
->hszItem
, pDdeAdvise
->cfFormat
);
665 TRACE("No data returned from the Callback\n");
669 GlobalUnlock(pXAct
->hMem
);
672 GlobalFree(pXAct
->hMem
);
676 WDML_PostAck(pConv
, WDML_SERVER_SIDE
, 0, FALSE
, fAck
, pXAct
->atom
, pXAct
->lParam
, WM_DDE_ADVISE
);
678 WDML_DecHSZ(pConv
->instance
, pXAct
->hszItem
);
680 return WDML_QS_HANDLED
;
683 /******************************************************************
684 * WDML_ServerQueueUnadvise
688 static WDML_XACT
* WDML_ServerQueueUnadvise(WDML_CONV
* pConv
, LPARAM lParam
)
693 UnpackDDElParam(WM_DDE_UNADVISE
, lParam
, &uiLo
, &uiHi
);
695 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_UNADVISE
,
696 uiLo
, WDML_MakeHszFromAtom(pConv
->instance
, uiHi
));
697 if (pXAct
) pXAct
->atom
= uiHi
;
701 /******************************************************************
702 * WDML_ServerHandleUnadvise
706 static WDML_QUEUE_STATE
WDML_ServerHandleUnadvise(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
710 if (pXAct
->hszItem
== NULL
|| pXAct
->wFmt
== 0)
712 ERR("Unsupported yet options (null item or clipboard format)\n");
713 return WDML_QS_ERROR
;
716 pLink
= WDML_FindLink(pConv
->instance
, (HCONV
)pConv
, WDML_SERVER_SIDE
,
717 pXAct
->hszItem
, TRUE
, pXAct
->wFmt
);
720 ERR("Couln'd find link for %p, dropping request\n", pXAct
->hszItem
);
721 FreeDDElParam(WM_DDE_UNADVISE
, pXAct
->lParam
);
722 return WDML_QS_ERROR
;
725 if (!(pConv
->instance
->CBFflags
& CBF_FAIL_ADVISES
))
727 WDML_InvokeCallback(pConv
->instance
, XTYP_ADVSTOP
, pXAct
->wFmt
, (HCONV
)pConv
,
728 pConv
->hszTopic
, pXAct
->hszItem
, 0, 0, 0);
731 WDML_RemoveLink(pConv
->instance
, (HCONV
)pConv
, WDML_SERVER_SIDE
,
732 pXAct
->hszItem
, pXAct
->wFmt
);
735 WDML_PostAck(pConv
, WDML_SERVER_SIDE
, 0, FALSE
, TRUE
, pXAct
->atom
,
736 pXAct
->lParam
, WM_DDE_UNADVISE
);
738 WDML_DecHSZ(pConv
->instance
, pXAct
->hszItem
);
740 return WDML_QS_HANDLED
;
743 /******************************************************************
748 static WDML_XACT
* WDML_ServerQueueExecute(WDML_CONV
* pConv
, LPARAM lParam
)
752 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_EXECUTE
, 0, 0);
755 pXAct
->hMem
= (HGLOBAL
)lParam
;
760 /******************************************************************
761 * WDML_ServerHandleExecute
765 static WDML_QUEUE_STATE
WDML_ServerHandleExecute(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
767 HDDEDATA hDdeData
= DDE_FNOTPROCESSED
;
768 BOOL fAck
= FALSE
, fBusy
= FALSE
;
770 if (!(pConv
->instance
->CBFflags
& CBF_FAIL_EXECUTES
))
772 LPVOID ptr
= GlobalLock(pXAct
->hMem
);
776 hDdeData
= DdeCreateDataHandle(0, ptr
, GlobalSize(pXAct
->hMem
),
778 GlobalUnlock(pXAct
->hMem
);
780 hDdeData
= WDML_InvokeCallback(pConv
->instance
, XTYP_EXECUTE
, 0, (HCONV
)pConv
,
781 pConv
->hszTopic
, 0, hDdeData
, 0L, 0L);
784 switch ((UINT
)hDdeData
)
793 WARN("Bad result code\n");
795 case DDE_FNOTPROCESSED
:
798 WDML_PostAck(pConv
, WDML_SERVER_SIDE
, 0, fBusy
, fAck
, (UINT
)pXAct
->hMem
, 0, 0);
800 return WDML_QS_HANDLED
;
803 /******************************************************************
804 * WDML_ServerQueuePoke
808 static WDML_XACT
* WDML_ServerQueuePoke(WDML_CONV
* pConv
, LPARAM lParam
)
813 UnpackDDElParam(WM_DDE_POKE
, lParam
, &uiLo
, &uiHi
);
815 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_POKE
,
816 0, WDML_MakeHszFromAtom(pConv
->instance
, uiHi
));
820 pXAct
->hMem
= (HGLOBAL
)uiLo
;
825 /******************************************************************
826 * WDML_ServerHandlePoke
830 static WDML_QUEUE_STATE
WDML_ServerHandlePoke(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
834 BOOL fBusy
= FALSE
, fAck
= FALSE
;
836 pDdePoke
= (DDEPOKE
*)GlobalLock(pXAct
->hMem
);
839 return WDML_QS_ERROR
;
842 if (!(pConv
->instance
->CBFflags
& CBF_FAIL_POKES
))
844 hDdeData
= DdeCreateDataHandle(pConv
->instance
->instanceID
, pDdePoke
->Value
,
845 GlobalSize(pXAct
->hMem
) - sizeof(DDEPOKE
) + 1,
846 0, 0, pDdePoke
->cfFormat
, 0);
849 HDDEDATA hDdeDataOut
;
851 hDdeDataOut
= WDML_InvokeCallback(pConv
->instance
, XTYP_POKE
, pDdePoke
->cfFormat
,
852 (HCONV
)pConv
, pConv
->hszTopic
, pXAct
->hszItem
,
854 switch ((ULONG_PTR
)hDdeDataOut
)
863 FIXME("Unsupported returned value %p\n", hDdeDataOut
);
865 case DDE_FNOTPROCESSED
:
868 DdeFreeDataHandle(hDdeData
);
871 GlobalUnlock(pXAct
->hMem
);
875 GlobalFree(pXAct
->hMem
);
877 WDML_PostAck(pConv
, WDML_SERVER_SIDE
, 0, fBusy
, fAck
, pXAct
->atom
, pXAct
->lParam
, WM_DDE_POKE
);
879 WDML_DecHSZ(pConv
->instance
, pXAct
->hszItem
);
881 return WDML_QS_HANDLED
;
884 /******************************************************************
885 * WDML_ServerQueueTerminate
889 static WDML_XACT
* WDML_ServerQueueTerminate(WDML_CONV
* pConv
, LPARAM lParam
)
893 pXAct
= WDML_AllocTransaction(pConv
->instance
, WM_DDE_TERMINATE
, 0, 0);
897 /******************************************************************
898 * WDML_ServerHandleTerminate
902 static WDML_QUEUE_STATE
WDML_ServerHandleTerminate(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
904 /* billx: two things to remove: the conv, and associated links.
905 * Respond with another WM_DDE_TERMINATE iMsg.
907 if (!(pConv
->instance
->CBFflags
& CBF_SKIP_DISCONNECTS
))
909 WDML_InvokeCallback(pConv
->instance
, XTYP_DISCONNECT
, 0, (HCONV
)pConv
, 0, 0,
910 0, 0, (pConv
->wStatus
& ST_ISSELF
) ? 1 : 0);
912 PostMessageA(pConv
->hwndClient
, WM_DDE_TERMINATE
, (WPARAM
)pConv
->hwndServer
, 0);
913 WDML_RemoveConv(pConv
, WDML_SERVER_SIDE
);
915 return WDML_QS_HANDLED
;
918 /******************************************************************
923 WDML_QUEUE_STATE
WDML_ServerHandle(WDML_CONV
* pConv
, WDML_XACT
* pXAct
)
925 WDML_QUEUE_STATE qs
= WDML_QS_ERROR
;
927 switch (pXAct
->ddeMsg
)
929 case WM_DDE_INITIATE
:
930 FIXME("WM_DDE_INITIATE shouldn't be there!\n");
933 qs
= WDML_ServerHandleRequest(pConv
, pXAct
);
937 qs
= WDML_ServerHandleAdvise(pConv
, pXAct
);
940 case WM_DDE_UNADVISE
:
941 qs
= WDML_ServerHandleUnadvise(pConv
, pXAct
);
945 qs
= WDML_ServerHandleExecute(pConv
, pXAct
);
949 qs
= WDML_ServerHandlePoke(pConv
, pXAct
);
952 case WM_DDE_TERMINATE
:
953 qs
= WDML_ServerHandleTerminate(pConv
, pXAct
);
957 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
961 FIXME("Unsupported message %d\n", pXAct
->ddeMsg
);
966 /******************************************************************
967 * WDML_ServerConvProc
971 static LRESULT CALLBACK
WDML_ServerConvProc(HWND hwndServer
, UINT iMsg
, WPARAM wParam
, LPARAM lParam
)
973 WDML_INSTANCE
* pInstance
;
975 WDML_XACT
* pXAct
= NULL
;
977 if (iMsg
== WM_DESTROY
)
979 EnterCriticalSection(&WDML_CritSect
);
980 pConv
= WDML_GetConvFromWnd(hwndServer
);
981 if (pConv
&& !(pConv
->wStatus
& ST_TERMINATED
))
983 WDML_ServerHandleTerminate(pConv
, NULL
);
985 LeaveCriticalSection(&WDML_CritSect
);
987 if (iMsg
< WM_DDE_FIRST
|| iMsg
> WM_DDE_LAST
)
989 return IsWindowUnicode(hwndServer
) ? DefWindowProcW(hwndServer
, iMsg
, wParam
, lParam
) :
990 DefWindowProcA(hwndServer
, iMsg
, wParam
, lParam
);
993 EnterCriticalSection(&WDML_CritSect
);
995 pInstance
= WDML_GetInstanceFromWnd(hwndServer
);
996 pConv
= WDML_GetConvFromWnd(hwndServer
);
1000 ERR("Got a message (%x) on a not known conversation, dropping request\n", iMsg
);
1003 if (pConv
->hwndClient
!= (HWND
)wParam
|| pConv
->hwndServer
!= hwndServer
)
1005 ERR("mismatch between C/S windows and converstation\n");
1008 if (pConv
->instance
!= pInstance
|| pConv
->instance
== NULL
)
1010 ERR("mismatch in instances\n");
1016 case WM_DDE_INITIATE
:
1017 FIXME("WM_DDE_INITIATE message received!\n");
1020 case WM_DDE_REQUEST
:
1021 pXAct
= WDML_ServerQueueRequest(pConv
, lParam
);
1025 pXAct
= WDML_ServerQueueAdvise(pConv
, lParam
);
1028 case WM_DDE_UNADVISE
:
1029 pXAct
= WDML_ServerQueueUnadvise(pConv
, lParam
);
1032 case WM_DDE_EXECUTE
:
1033 pXAct
= WDML_ServerQueueExecute(pConv
, lParam
);
1037 pXAct
= WDML_ServerQueuePoke(pConv
, lParam
);
1040 case WM_DDE_TERMINATE
:
1041 pXAct
= WDML_ServerQueueTerminate(pConv
, lParam
);
1045 WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1049 FIXME("Unsupported message %x\n", iMsg
);
1054 pXAct
->lParam
= lParam
;
1055 if (WDML_ServerHandle(pConv
, pXAct
) == WDML_QS_BLOCK
)
1057 WDML_QueueTransaction(pConv
, pXAct
);
1061 WDML_FreeTransaction(pInstance
, pXAct
, TRUE
);
1065 LeaveCriticalSection(&WDML_CritSect
);