~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~ [ freetext search ] ~ [ file search ] ~

Wine Cross Reference
wine/dlls/user32/dde_server.c

Version: ~ [ wine-1.5.30 ] ~ [ wine-1.5.29 ] ~ [ wine-1.5.28 ] ~ [ wine-1.5.27 ] ~ [ wine-1.5.26 ] ~ [ wine-1.5.25 ] ~ [ wine-1.5.24 ] ~ [ wine-1.5.23 ] ~ [ wine-1.5.22 ] ~ [ wine-1.5.21 ] ~ [ wine-1.5.20 ] ~ [ wine-1.5.19 ] ~ [ wine-1.5.18 ] ~ [ wine-1.5.17 ] ~ [ wine-1.5.16 ] ~ [ wine-1.5.15 ] ~ [ wine-1.5.14 ] ~ [ wine-1.5.13 ] ~ [ wine-1.5.12 ] ~ [ wine-1.5.11 ] ~ [ wine-1.5.10 ] ~ [ wine-1.5.9 ] ~ [ wine-1.5.8 ] ~ [ wine-1.5.7 ] ~ [ wine-1.4.1 ] ~ [ wine-1.5.6 ] ~ [ wine-1.5.5 ] ~ [ wine-1.5.4 ] ~ [ wine-1.5.3 ] ~ [ wine-1.5.2 ] ~ [ wine-1.5.1 ] ~ [ wine-1.5.0 ] ~ [ wine-1.4 ] ~ [ wine-1.4-rc6 ] ~ [ wine-1.4-rc5 ] ~ [ wine-1.4-rc4 ] ~ [ wine-1.4-rc3 ] ~ [ wine-1.4-rc2 ] ~ [ wine-1.4-rc1 ] ~ [ wine-1.3.37 ] ~ [ wine-1.3.36 ] ~ [ wine-1.3.35 ] ~ [ wine-1.3.34 ] ~ [ wine-1.3.33 ] ~ [ wine-1.3.32 ] ~ [ wine-1.3.31 ] ~ [ wine-1.3.30 ] ~ [ wine-1.3.29 ] ~ [ wine-1.3.28 ] ~ [ wine-1.3.27 ] ~ [ wine-1.3.26 ] ~ [ wine-1.3.25 ] ~ [ wine-1.3.24 ] ~ [ wine-1.3.23 ] ~ [ wine-1.3.22 ] ~ [ wine-1.3.21 ] ~ [ wine-1.3.20 ] ~ [ wine-1.3.19 ] ~ [ wine-1.3.18 ] ~ [ wine-1.2.3 ] ~ [ wine-1.3.17 ] ~ [ wine-1.3.16 ] ~ [ wine-1.3.15 ] ~ [ wine-1.3.14 ] ~ [ wine-1.3.13 ] ~ [ wine-1.3.12 ] ~ [ wine-1.3.11 ] ~ [ wine-1.3.10 ] ~ [ wine-1.3.9 ] ~ [ wine-1.2.2 ] ~ [ wine-1.3.8 ] ~ [ wine-1.3.7 ] ~ [ wine-1.3.6 ] ~ [ wine-1.3.5 ] ~ [ wine-1.2.1 ] ~ [ wine-1.3.4 ] ~ [ wine-1.3.3 ] ~ [ wine-1.3.2 ] ~ [ wine-1.3.1 ] ~ [ wine-1.3.0 ] ~ [ wine-1.2 ] ~ [ wine-1.2-rc7 ] ~ [ wine-1.2-rc6 ] ~ [ wine-1.2-rc5 ] ~ [ wine-1.2-rc4 ] ~ [ wine-1.2-rc3 ] ~ [ wine-1.2-rc2 ] ~ [ wine-1.2-rc1 ] ~ [ wine-1.1.44 ] ~ [ wine-1.1.43 ] ~ [ wine-1.1.42 ] ~ [ wine-1.1.41 ] ~ [ wine-1.1.40 ] ~ [ wine-1.1.39 ] ~ [ wine-1.1.38 ] ~ [ wine-1.1.37 ] ~ [ wine-1.1.36 ] ~ [ wine-1.1.35 ] ~ [ wine-1.1.34 ] ~ [ wine-1.1.33 ] ~ [ wine-1.1.32 ] ~ [ wine-1.1.31 ] ~ [ wine-1.1.30 ] ~ [ wine-1.1.29 ] ~ [ wine-1.1.28 ] ~ [ wine-1.1.27 ] ~ [ wine-1.1.26 ] ~ [ wine-1.1.25 ] ~ [ wine-1.1.24 ] ~ [ wine-1.1.23 ] ~ [ wine-1.1.22 ] ~ [ wine-1.1.21 ] ~ [ wine-1.1.20 ] ~ [ wine-1.1.19 ] ~ [ wine-1.1.18 ] ~ [ wine-1.1.17 ] ~ [ wine-1.1.16 ] ~ [ wine-1.1.15 ] ~ [ wine-1.1.14 ] ~ [ wine-1.1.13 ] ~ [ wine-1.1.12 ] ~ [ wine-1.1.11 ] ~ [ wine-1.1.10 ] ~ [ wine-1.1.9 ] ~ [ wine-1.1.8 ] ~ [ wine-1.1.7 ] ~ [ wine-1.0.1 ] ~ [ wine-1.1.6 ] ~ [ wine-1.1.5 ] ~ [ wine-1.1.4 ] ~ [ wine-1.1.3 ] ~ [ wine-1.1.2 ] ~ [ wine-1.1.1 ] ~ [ wine-1.1.0 ] ~ [ wine-1.0 ] ~

  1 /*
  2  * DDEML library
  3  *
  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
 10  *
 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.
 15  *
 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.
 20  *
 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
 24  */
 25 
 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"
 37 
 38 WINE_DEFAULT_DEBUG_CHANNEL(ddeml);
 39 
 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};
 43 
 44 static LRESULT CALLBACK WDML_ServerNameProc(HWND, UINT, WPARAM, LPARAM);
 45 static LRESULT CALLBACK WDML_ServerConvProc(HWND, UINT, WPARAM, LPARAM);
 46 
 47 /******************************************************************************
 48  * DdePostAdvise [USER32.@]  Send transaction to DDE callback function.
 49  *
 50  * PARAMS
 51  *      idInst    [I] Instance identifier
 52  *      hszTopic  [I] Handle to topic name string
 53  *      hszItem   [I] Handle to item name string
 54  *
 55  * RETURNS
 56  *    Success: TRUE
 57  *    Failure: FALSE
 58  */
 59 BOOL WINAPI DdePostAdvise(DWORD idInst, HSZ hszTopic, HSZ hszItem)
 60 {
 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;
 68 
 69     TRACE("(%d,%p,%p)\n", idInst, hszTopic, hszItem);
 70 
 71     pInstance = WDML_GetInstance(idInst);
 72 
 73     if (pInstance == NULL || pInstance->links == NULL)
 74         return FALSE;
 75 
 76     atom = WDML_MakeAtomFromHsz(hszItem);
 77     if (!atom) return FALSE;
 78 
 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)
 82     {
 83         if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
 84         {
 85             count++;
 86         }
 87     }
 88     if (count >= CADV_LATEACK)
 89     {
 90         FIXME("too high value for count\n");
 91         count &= 0xFFFF;
 92     }
 93 
 94     for (pLink = pInstance->links[WDML_SERVER_SIDE]; pLink != NULL; pLink = pLink->next)
 95     {
 96         if (DdeCmpStringHandles(hszItem, pLink->hszItem) == 0)
 97         {
 98             hDdeData = WDML_InvokeCallback(pInstance, XTYP_ADVREQ, pLink->uFmt, pLink->hConv,
 99                                            hszTopic, hszItem, 0, --count, 0);
100 
101             if (hDdeData == CBR_BLOCK)
102             {
103                 /* MS doc is not consistent here */
104                 FIXME("CBR_BLOCK returned for ADVREQ\n");
105                 continue;
106             }
107             if (hDdeData)
108             {
109                 if (pLink->transactionType & XTYPF_NODATA)
110                 {
111                     TRACE("no data\n");
112                     hItemData = 0;
113                 }
114                 else
115                 {
116                     TRACE("with data\n");
117 
118                     hItemData = WDML_DataHandle2Global(hDdeData, FALSE, FALSE, FALSE, FALSE);
119                 }
120 
121                 pConv = WDML_GetConv(pLink->hConv, TRUE);
122 
123                 if (pConv == NULL)
124                 {
125                     if (!WDML_IsAppOwned(hDdeData))  DdeFreeDataHandle(hDdeData);
126                     goto theError;
127                 }
128 
129                 if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
130                                   PackDDElParam(WM_DDE_DATA, (UINT_PTR)hItemData, atom)))
131                 {
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;
138                 }
139                 if (!WDML_IsAppOwned(hDdeData))  DdeFreeDataHandle(hDdeData);
140             }
141         }
142     }
143     return TRUE;
144 
145  theError:
146     if (atom) GlobalDeleteAtom(atom);
147     return FALSE;
148 }
149 
150 
151 /******************************************************************************
152  * DdeNameService [USER32.@]  {Un}registers service name of DDE server
153  *
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
159  *
160  * RETURNS
161  *    Success: Non-zero
162  *    Failure: 0
163  */
164 HDDEDATA WINAPI DdeNameService(DWORD idInst, HSZ hsz1, HSZ hsz2, UINT afCmd)
165 {
166     WDML_SERVER*        pServer;
167     WDML_INSTANCE*      pInstance;
168     HWND                hwndServer;
169     WNDCLASSEXW         wndclass;
170 
171     TRACE("(%d,%p,%p,%x)\n", idInst, hsz1, hsz2, afCmd);
172 
173     /*  First check instance
174      */
175     pInstance = WDML_GetInstance(idInst);
176     if  (pInstance == NULL)
177     {
178         TRACE("Instance not found as initialised\n");
179         /*  Nothing has been initialised - exit now ! can return TRUE since effect is the same */
180         return NULL;
181     }
182 
183     if (hsz2 != 0L)
184     {
185         /*      Illegal, reserved parameter
186          */
187         pInstance->lastError = DMLERR_INVALIDPARAMETER;
188         WARN("Reserved parameter no-zero !!\n");
189         return NULL;
190     }
191     if (hsz1 == 0 && !(afCmd & DNS_UNREGISTER))
192     {
193         /*      don't know if we should check this but it makes sense
194          *      why supply REGISTER or filter flags if de-registering all
195          */
196         TRACE("General unregister unexpected flags\n");
197         pInstance->lastError = DMLERR_INVALIDPARAMETER;
198         return NULL;
199     }
200 
201     switch (afCmd & (DNS_REGISTER | DNS_UNREGISTER))
202     {
203     case DNS_REGISTER:
204         pServer = WDML_FindServer(pInstance, hsz1, 0);
205         if (pServer)
206         {
207             ERR("Trying to register already registered service!\n");
208             pInstance->lastError = DMLERR_DLL_USAGE;
209             return NULL;
210         }
211 
212         TRACE("Adding service name\n");
213 
214         WDML_IncHSZ(pInstance, hsz1);
215 
216         pServer = WDML_AddServer(pInstance, hsz1, 0);
217 
218         WDML_BroadcastDDEWindows(WDML_szEventClass, WM_WDML_REGISTER,
219                                  pServer->atomService, pServer->atomServiceSpec);
220 
221         wndclass.cbSize        = sizeof(wndclass);
222         wndclass.style         = 0;
223         wndclass.lpfnWndProc   = WDML_ServerNameProc;
224         wndclass.cbClsExtra    = 0;
225         wndclass.cbWndExtra    = 2 * sizeof(ULONG_PTR);
226         wndclass.hInstance     = 0;
227         wndclass.hIcon         = 0;
228         wndclass.hCursor       = 0;
229         wndclass.hbrBackground = 0;
230         wndclass.lpszMenuName  = NULL;
231         wndclass.lpszClassName = szServerNameClass;
232         wndclass.hIconSm       = 0;
233 
234         RegisterClassExW(&wndclass);
235 
236         hwndServer = CreateWindowW(szServerNameClass, NULL,
237                                    WS_POPUP, 0, 0, 0, 0,
238                                    0, 0, 0, 0);
239 
240         SetWindowLongPtrW(hwndServer, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
241         SetWindowLongPtrW(hwndServer, GWL_WDML_SERVER, (ULONG_PTR)pServer);
242         TRACE("Created nameServer=%p for instance=%08x\n", hwndServer, idInst);
243 
244         pServer->hwndServer = hwndServer;
245         break;
246 
247     case DNS_UNREGISTER:
248         if (hsz1 == 0L)
249         {
250             /* General unregister situation
251              * terminate all server side pending conversations
252              */
253             while (pInstance->servers)
254                 WDML_RemoveServer(pInstance, pInstance->servers->hszService, 0);
255             pInstance->servers = NULL;
256             TRACE("General de-register - finished\n");
257         }
258         else
259         {
260             WDML_RemoveServer(pInstance, hsz1, 0L);
261         }
262         break;
263     }
264 
265     if (afCmd & (DNS_FILTERON | DNS_FILTEROFF))
266     {
267         /*      Set filter flags on to hold notifications of connection
268          */
269         pServer = WDML_FindServer(pInstance, hsz1, 0);
270         if (!pServer)
271         {
272             /*  trying to filter where no service names !!
273              */
274             pInstance->lastError = DMLERR_DLL_USAGE;
275             return NULL;
276         }
277         else
278         {
279             pServer->filterOn = (afCmd & DNS_FILTERON) != 0;
280         }
281     }
282     return (HDDEDATA)TRUE;
283 }
284 
285 /******************************************************************
286  *              WDML_CreateServerConv
287  *
288  *
289  */
290 static WDML_CONV* WDML_CreateServerConv(WDML_INSTANCE* pInstance, HWND hwndClient,
291                                         HWND hwndServerName, HSZ hszApp, HSZ hszTopic)
292 {
293     HWND        hwndServerConv;
294     WDML_CONV*  pConv;
295 
296     if (pInstance->unicode)
297     {
298         WNDCLASSEXW wndclass;
299 
300         wndclass.cbSize        = sizeof(wndclass);
301         wndclass.style         = 0;
302         wndclass.lpfnWndProc   = WDML_ServerConvProc;
303         wndclass.cbClsExtra    = 0;
304         wndclass.cbWndExtra    = 2 * sizeof(ULONG_PTR);
305         wndclass.hInstance     = 0;
306         wndclass.hIcon         = 0;
307         wndclass.hCursor       = 0;
308         wndclass.hbrBackground = 0;
309         wndclass.lpszMenuName  = NULL;
310         wndclass.lpszClassName = WDML_szServerConvClassW;
311         wndclass.hIconSm       = 0;
312 
313         RegisterClassExW(&wndclass);
314 
315         hwndServerConv = CreateWindowW(WDML_szServerConvClassW, 0,
316                                        WS_CHILD, 0, 0, 0, 0,
317                                        hwndServerName, 0, 0, 0);
318     }
319     else
320     {
321         WNDCLASSEXA wndclass;
322 
323         wndclass.cbSize        = sizeof(wndclass);
324         wndclass.style         = 0;
325         wndclass.lpfnWndProc   = WDML_ServerConvProc;
326         wndclass.cbClsExtra    = 0;
327         wndclass.cbWndExtra    = 2 * sizeof(ULONG_PTR);
328         wndclass.hInstance     = 0;
329         wndclass.hIcon         = 0;
330         wndclass.hCursor       = 0;
331         wndclass.hbrBackground = 0;
332         wndclass.lpszMenuName  = NULL;
333         wndclass.lpszClassName = WDML_szServerConvClassA;
334         wndclass.hIconSm       = 0;
335 
336         RegisterClassExA(&wndclass);
337 
338         hwndServerConv = CreateWindowA(WDML_szServerConvClassA, 0,
339                                       WS_CHILD, 0, 0, 0, 0,
340                                       hwndServerName, 0, 0, 0);
341     }
342 
343     TRACE("Created convServer=%p (nameServer=%p) for instance=%08x unicode=%d\n",
344           hwndServerConv, hwndServerName, pInstance->instanceID, pInstance->unicode);
345 
346     pConv = WDML_AddConv(pInstance, WDML_SERVER_SIDE, hszApp, hszTopic,
347                          hwndClient, hwndServerConv);
348     if (pConv)
349     {
350         SetWindowLongPtrW(hwndServerConv, GWL_WDML_INSTANCE, (ULONG_PTR)pInstance);
351         SetWindowLongPtrW(hwndServerConv, GWL_WDML_CONVERSATION, (ULONG_PTR)pConv);
352 
353         /* this should be the only place using SendMessage for WM_DDE_ACK */
354         /* note: sent messages shall not use packing */
355         SendMessageW(hwndClient, WM_DDE_ACK, (WPARAM)hwndServerConv,
356                      MAKELPARAM(WDML_MakeAtomFromHsz(hszApp), WDML_MakeAtomFromHsz(hszTopic)));
357         /* we assume we're connected since we've sent an answer...
358          * I'm not sure what we can do... it doesn't look like the return value
359          * of SendMessage is used... sigh...
360          */
361         pConv->wStatus |= ST_CONNECTED;
362     }
363     else
364     {
365         DestroyWindow(hwndServerConv);
366     }
367     return pConv;
368 }
369 
370 /******************************************************************
371  *              WDML_ServerNameProc
372  *
373  *
374  */
375 static LRESULT CALLBACK WDML_ServerNameProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
376 {
377     HWND                hwndClient;
378     HSZ                 hszApp, hszTop;
379     HDDEDATA            hDdeData = 0;
380     WDML_INSTANCE*      pInstance;
381     UINT_PTR            uiLo, uiHi;
382 
383     switch (iMsg)
384     {
385     case WM_DDE_INITIATE:
386 
387         /* wParam         -- sending window handle
388            LOWORD(lParam) -- application atom
389            HIWORD(lParam) -- topic atom */
390 
391         TRACE("WM_DDE_INITIATE message received!\n");
392         hwndClient = (HWND)wParam;
393 
394         pInstance = WDML_GetInstanceFromWnd(hwndServer);
395         TRACE("idInst=%d, threadID=0x%x\n", pInstance->instanceID, GetCurrentThreadId());
396         if (!pInstance) return 0;
397 
398         /* don't free DDEParams, since this is a broadcast */
399         UnpackDDElParam(WM_DDE_INITIATE, lParam, &uiLo, &uiHi);
400 
401         hszApp = WDML_MakeHszFromAtom(pInstance, uiLo);
402         hszTop = WDML_MakeHszFromAtom(pInstance, uiHi);
403 
404         if (!(pInstance->CBFflags & CBF_FAIL_CONNECTIONS))
405         {
406             BOOL                self = FALSE;
407             CONVCONTEXT         cc;
408             CONVCONTEXT*        pcc = NULL;
409             WDML_CONV*          pConv;
410             char                buf[256];
411 
412             if (GetWindowThreadProcessId(hwndClient, NULL) == GetWindowThreadProcessId(hwndServer, NULL) &&
413                 WDML_GetInstanceFromWnd(hwndClient) == WDML_GetInstanceFromWnd(hwndServer))
414             {
415                 self = TRUE;
416             }
417             /* FIXME: so far, we don't grab distant convcontext, so only check if remote is
418              * handled under DDEML, and if so build a default context
419              */
420            if ((GetClassNameA(hwndClient, buf, sizeof(buf)) &&
421                 lstrcmpiA(buf, WDML_szClientConvClassA) == 0) ||
422                (GetClassNameW(hwndClient, (LPWSTR)buf, sizeof(buf)/sizeof(WCHAR)) &&
423                 lstrcmpiW((LPWSTR)buf, WDML_szClientConvClassW) == 0))
424             {
425                 pcc = &cc;
426                 memset(pcc, 0, sizeof(*pcc));
427                 pcc->cb = sizeof(*pcc);
428                 pcc->iCodePage = IsWindowUnicode(hwndClient) ? CP_WINUNICODE : CP_WINANSI;
429             }
430             if ((pInstance->CBFflags & CBF_FAIL_SELFCONNECTIONS) && self)
431             {
432                 TRACE("Don't do self connection as requested\n");
433             }
434             else if (hszApp && hszTop)
435             {
436                 WDML_SERVER*    pServer = (WDML_SERVER*)GetWindowLongPtrW(hwndServer, GWL_WDML_SERVER);
437 
438                 /* check filters for name service */
439                 if (!pServer->filterOn || DdeCmpStringHandles(pServer->hszService, hszApp) == 0)
440                 {
441                     /* pass on to the callback  */
442                     hDdeData = WDML_InvokeCallback(pInstance, XTYP_CONNECT,
443                                                    0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
444                     if ((ULONG_PTR)hDdeData)
445                     {
446                         pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
447                                                       hszApp, hszTop);
448                         if (pConv)
449                         {
450                             if (pcc) pConv->wStatus |= ST_ISLOCAL;
451                             WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
452                                                 hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
453                         }
454                     }
455                 }
456             }
457             else if (pInstance->servers)
458             {
459                 /* pass on to the callback  */
460                 hDdeData = WDML_InvokeCallback(pInstance, XTYP_WILDCONNECT,
461                                                0, 0, hszTop, hszApp, 0, (ULONG_PTR)pcc, self);
462 
463                 if (hDdeData == CBR_BLOCK)
464                 {
465                     /* MS doc is not consistent here */
466                     FIXME("CBR_BLOCK returned for WILDCONNECT\n");
467                 }
468                 else if ((ULONG_PTR)hDdeData != 0)
469                 {
470                     HSZPAIR*    hszp;
471 
472                     hszp = (HSZPAIR*)DdeAccessData(hDdeData, NULL);
473                     if (hszp)
474                     {
475                         int     i;
476                         for (i = 0; hszp[i].hszSvc && hszp[i].hszTopic; i++)
477                         {
478                             pConv = WDML_CreateServerConv(pInstance, hwndClient, hwndServer,
479                                                           hszp[i].hszSvc, hszp[i].hszTopic);
480                             if (pConv)
481                             {
482                                 if (pcc) pConv->wStatus |= ST_ISLOCAL;
483                                 WDML_InvokeCallback(pInstance, XTYP_CONNECT_CONFIRM, 0, (HCONV)pConv,
484                                                     hszp[i].hszTopic, hszp[i].hszSvc, 0, (ULONG_PTR)pcc, self);
485                             }
486                         }
487                         DdeUnaccessData(hDdeData);
488                     }
489                     if (!WDML_IsAppOwned(hDdeData)) DdeFreeDataHandle(hDdeData);
490                 }
491             }
492         }
493 
494         return 0;
495 
496     case WM_DDE_REQUEST:
497         FIXME("WM_DDE_REQUEST message received!\n");
498         return 0;
499     case WM_DDE_ADVISE:
500         FIXME("WM_DDE_ADVISE message received!\n");
501         return 0;
502     case WM_DDE_UNADVISE:
503         FIXME("WM_DDE_UNADVISE message received!\n");
504         return 0;
505     case WM_DDE_EXECUTE:
506         FIXME("WM_DDE_EXECUTE message received!\n");
507         return 0;
508     case WM_DDE_POKE:
509         FIXME("WM_DDE_POKE message received!\n");
510         return 0;
511     case WM_DDE_TERMINATE:
512         FIXME("WM_DDE_TERMINATE message received!\n");
513         return 0;
514     default:
515         break;
516     }
517 
518     return DefWindowProcW(hwndServer, iMsg, wParam, lParam);
519 }
520 
521 /******************************************************************
522  *              WDML_ServerQueueRequest
523  *
524  *
525  */
526 static  WDML_XACT*      WDML_ServerQueueRequest(WDML_CONV* pConv, LPARAM lParam)
527 {
528     UINT_PTR            uiLo, uiHi;
529     WDML_XACT*          pXAct;
530 
531     UnpackDDElParam(WM_DDE_REQUEST, lParam, &uiLo, &uiHi);
532 
533     pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_REQUEST,
534                                   uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
535     if (pXAct) pXAct->atom = uiHi;
536     return pXAct;
537 }
538 
539 /******************************************************************
540  *              WDML_ServerHandleRequest
541  *
542  *
543  */
544 static  WDML_QUEUE_STATE WDML_ServerHandleRequest(WDML_CONV* pConv, WDML_XACT* pXAct)
545 {
546     HDDEDATA            hDdeData = 0;
547     BOOL                fAck = TRUE;
548 
549     if (!(pConv->instance->CBFflags & CBF_FAIL_REQUESTS))
550     {
551 
552         hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_REQUEST, pXAct->wFmt, (HCONV)pConv,
553                                        pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
554     }
555 
556     switch ((ULONG_PTR)hDdeData)
557     {
558     case 0:
559         TRACE("No data returned from the Callback\n");
560         fAck = FALSE;
561         break;
562 
563     case (ULONG_PTR)CBR_BLOCK:
564         return WDML_QS_BLOCK;
565 
566     default:
567         {
568             HGLOBAL     hMem = WDML_DataHandle2Global(hDdeData, TRUE, FALSE, FALSE, FALSE);
569             if (!PostMessageW(pConv->hwndClient, WM_DDE_DATA, (WPARAM)pConv->hwndServer,
570                               ReuseDDElParam(pXAct->lParam, WM_DDE_REQUEST, WM_DDE_DATA,
571                                              (UINT_PTR)hMem, (UINT_PTR)pXAct->atom)))
572             {
573                 pConv->instance->lastError = DMLERR_POSTMSG_FAILED;
574                 DdeFreeDataHandle(hDdeData);
575                 GlobalFree(hMem);
576                 fAck = FALSE;
577             }
578         }
579         break;
580     }
581 
582     WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_REQUEST);
583 
584     WDML_DecHSZ(pConv->instance, pXAct->hszItem);
585 
586     return WDML_QS_HANDLED;
587 }
588 
589 /******************************************************************
590  *              WDML_ServerQueueAdvise
591  *
592  *
593  */
594 static  WDML_XACT*      WDML_ServerQueueAdvise(WDML_CONV* pConv, LPARAM lParam)
595 {
596     UINT_PTR            uiLo, uiHi;
597     WDML_XACT*          pXAct;
598 
599     /* XTYP_ADVSTART transaction:
600        establish link and save link info to InstanceInfoTable */
601 
602     if (!UnpackDDElParam(WM_DDE_ADVISE, lParam, &uiLo, &uiHi))
603         return NULL;
604 
605     pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_ADVISE,
606                                   0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
607     if (pXAct)
608     {
609         pXAct->hMem = (HGLOBAL)uiLo;
610         pXAct->atom = uiHi;
611     }
612     return pXAct;
613 }
614 
615 /******************************************************************
616  *              WDML_ServerHandleAdvise
617  *
618  *
619  */
620 static  WDML_QUEUE_STATE WDML_ServerHandleAdvise(WDML_CONV* pConv, WDML_XACT* pXAct)
621 {
622     UINT                uType;
623     WDML_LINK*          pLink;
624     DDEADVISE*          pDdeAdvise;
625     HDDEDATA            hDdeData = 0;
626     BOOL                fAck = TRUE;
627 
628     pDdeAdvise = GlobalLock(pXAct->hMem);
629     uType = XTYP_ADVSTART |
630             (pDdeAdvise->fDeferUpd ? XTYPF_NODATA : 0) |
631             (pDdeAdvise->fAckReq ? XTYPF_ACKREQ : 0);
632 
633     if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
634     {
635         hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_ADVSTART, pDdeAdvise->cfFormat,
636                                        (HCONV)pConv, pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
637     }
638 
639     switch ((ULONG_PTR)hDdeData)
640     {
641     case 0:
642         TRACE("No data returned from the Callback\n");
643         fAck = FALSE;
644         break;
645 
646     case (ULONG_PTR)CBR_BLOCK:
647         return WDML_QS_BLOCK;
648 
649     default:
650         /* billx: first to see if the link is already created. */
651         pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
652                               pXAct->hszItem, TRUE, pDdeAdvise->cfFormat);
653 
654         if (pLink != NULL)
655         {
656             /* we found a link, and only need to modify it in case it changes */
657             pLink->transactionType = uType;
658         }
659         else
660         {
661             TRACE("Adding Link with hConv %p\n", pConv);
662             WDML_AddLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
663                          uType, pXAct->hszItem, pDdeAdvise->cfFormat);
664         }
665         break;
666     }
667 
668     GlobalUnlock(pXAct->hMem);
669     if (fAck)
670     {
671         GlobalFree(pXAct->hMem);
672     }
673     pXAct->hMem = 0;
674 
675     WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, fAck, pXAct->atom, pXAct->lParam, WM_DDE_ADVISE);
676 
677     WDML_DecHSZ(pConv->instance, pXAct->hszItem);
678 
679     return WDML_QS_HANDLED;
680 }
681 
682 /******************************************************************
683  *              WDML_ServerQueueUnadvise
684  *
685  *
686  */
687 static  WDML_XACT* WDML_ServerQueueUnadvise(WDML_CONV* pConv, LPARAM lParam)
688 {
689     UINT_PTR            uiLo, uiHi;
690     WDML_XACT*          pXAct;
691 
692     UnpackDDElParam(WM_DDE_UNADVISE, lParam, &uiLo, &uiHi);
693 
694     pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_UNADVISE,
695                                   uiLo, WDML_MakeHszFromAtom(pConv->instance, uiHi));
696     if (pXAct) pXAct->atom = uiHi;
697     return pXAct;
698 }
699 
700 /******************************************************************
701  *              WDML_ServerHandleUnadvise
702  *
703  *
704  */
705 static  WDML_QUEUE_STATE WDML_ServerHandleUnadvise(WDML_CONV* pConv, WDML_XACT* pXAct)
706 {
707     WDML_LINK*  pLink;
708 
709     if (pXAct->hszItem == NULL || pXAct->wFmt == 0)
710     {
711         ERR("Unsupported yet options (null item or clipboard format)\n");
712         return WDML_QS_ERROR;
713     }
714 
715     pLink = WDML_FindLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
716                           pXAct->hszItem, TRUE, pXAct->wFmt);
717     if (pLink == NULL)
718     {
719         ERR("Couln'd find link for %p, dropping request\n", pXAct->hszItem);
720         FreeDDElParam(WM_DDE_UNADVISE, pXAct->lParam);
721         return WDML_QS_ERROR;
722     }
723 
724     if (!(pConv->instance->CBFflags & CBF_FAIL_ADVISES))
725     {
726         WDML_InvokeCallback(pConv->instance, XTYP_ADVSTOP, pXAct->wFmt, (HCONV)pConv,
727                             pConv->hszTopic, pXAct->hszItem, 0, 0, 0);
728     }
729 
730     WDML_RemoveLink(pConv->instance, (HCONV)pConv, WDML_SERVER_SIDE,
731                     pXAct->hszItem, pXAct->wFmt);
732 
733     /* send back ack */
734     WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, FALSE, TRUE, pXAct->atom,
735                  pXAct->lParam, WM_DDE_UNADVISE);
736 
737     WDML_DecHSZ(pConv->instance, pXAct->hszItem);
738 
739     return WDML_QS_HANDLED;
740 }
741 
742 /******************************************************************
743  *              WDML_QueueExecute
744  *
745  *
746  */
747 static  WDML_XACT* WDML_ServerQueueExecute(WDML_CONV* pConv, LPARAM lParam)
748 {
749     WDML_XACT*  pXAct;
750 
751     pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_EXECUTE, 0, 0);
752     if (pXAct)
753     {
754         pXAct->hMem    = (HGLOBAL)lParam;
755     }
756     return pXAct;
757 }
758 
759  /******************************************************************
760  *              WDML_ServerHandleExecute
761  *
762  *
763  */
764 static  WDML_QUEUE_STATE WDML_ServerHandleExecute(WDML_CONV* pConv, WDML_XACT* pXAct)
765 {
766     HDDEDATA    hDdeData = DDE_FNOTPROCESSED;
767     BOOL        fAck = FALSE, fBusy = FALSE;
768 
769     if (!(pConv->instance->CBFflags & CBF_FAIL_EXECUTES))
770     {
771         LPVOID  ptr = GlobalLock(pXAct->hMem);
772 
773         if (ptr)
774         {
775             hDdeData = DdeCreateDataHandle(0, ptr, GlobalSize(pXAct->hMem),
776                                            0, 0, CF_TEXT, 0);
777             GlobalUnlock(pXAct->hMem);
778         }
779         hDdeData = WDML_InvokeCallback(pConv->instance, XTYP_EXECUTE, 0, (HCONV)pConv,
780                                        pConv->hszTopic, 0, hDdeData, 0L, 0L);
781     }
782 
783     switch ((ULONG_PTR)hDdeData)
784     {
785     case (ULONG_PTR)CBR_BLOCK:
786         return WDML_QS_BLOCK;
787 
788     case DDE_FACK:
789         fAck = TRUE;
790         break;
791     case DDE_FBUSY:
792         fBusy = TRUE;
793         break;
794     default:
795         FIXME("Unsupported returned value %p\n", hDdeData);
796         /* fall through */
797     case DDE_FNOTPROCESSED:
798         break;
799     }
800     WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, (UINT_PTR)pXAct->hMem, 0, 0);
801 
802     return WDML_QS_HANDLED;
803 }
804 
805 /******************************************************************
806  *              WDML_ServerQueuePoke
807  *
808  *
809  */
810 static  WDML_XACT* WDML_ServerQueuePoke(WDML_CONV* pConv, LPARAM lParam)
811 {
812     UINT_PTR            uiLo, uiHi;
813     WDML_XACT*          pXAct;
814 
815     UnpackDDElParam(WM_DDE_POKE, lParam, &uiLo, &uiHi);
816 
817     pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_POKE,
818                                   0, WDML_MakeHszFromAtom(pConv->instance, uiHi));
819     if (pXAct)
820     {
821         pXAct->atom = uiHi;
822         pXAct->hMem = (HGLOBAL)uiLo;
823     }
824     return pXAct;
825 }
826 
827 /******************************************************************
828  *              WDML_ServerHandlePoke
829  *
830  *
831  */
832 static  WDML_QUEUE_STATE WDML_ServerHandlePoke(WDML_CONV* pConv, WDML_XACT* pXAct)
833 {
834     DDEPOKE*            pDdePoke;
835     HDDEDATA            hDdeData;
836     BOOL                fBusy = FALSE, fAck = FALSE;
837 
838     pDdePoke = GlobalLock(pXAct->hMem);
839     if (!pDdePoke)
840     {
841         return WDML_QS_ERROR;
842     }
843 
844     if (!(pConv->instance->CBFflags & CBF_FAIL_POKES))
845     {
846         hDdeData = DdeCreateDataHandle(pConv->instance->instanceID, pDdePoke->Value,
847                                        GlobalSize(pXAct->hMem) - FIELD_OFFSET(DDEPOKE, Value),
848                                        0, 0, pDdePoke->cfFormat, 0);
849         if (hDdeData)
850         {
851             HDDEDATA    hDdeDataOut;
852 
853             hDdeDataOut = WDML_InvokeCallback(pConv->instance, XTYP_POKE, pDdePoke->cfFormat,
854                                               (HCONV)pConv, pConv->hszTopic, pXAct->hszItem,
855                                               hDdeData, 0, 0);
856             switch ((ULONG_PTR)hDdeDataOut)
857             {
858             case DDE_FACK:
859                 fAck = TRUE;
860                 break;
861             case DDE_FBUSY:
862                 fBusy = TRUE;
863                 break;
864             default:
865                 FIXME("Unsupported returned value %p\n", hDdeDataOut);
866                 /* fal through */
867             case DDE_FNOTPROCESSED:
868                 break;
869             }
870             DdeFreeDataHandle(hDdeData);
871         }
872     }
873     GlobalUnlock(pXAct->hMem);
874 
875     if (!fAck)
876     {
877         GlobalFree(pXAct->hMem);
878     }
879     WDML_PostAck(pConv, WDML_SERVER_SIDE, 0, fBusy, fAck, pXAct->atom, pXAct->lParam, WM_DDE_POKE);
880 
881     WDML_DecHSZ(pConv->instance, pXAct->hszItem);
882 
883     return WDML_QS_HANDLED;
884 }
885 
886 /******************************************************************
887  *              WDML_ServerQueueTerminate
888  *
889  *
890  */
891 static  WDML_XACT*      WDML_ServerQueueTerminate(WDML_CONV* pConv, LPARAM lParam)
892 {
893     WDML_XACT*  pXAct;
894 
895     pXAct = WDML_AllocTransaction(pConv->instance, WM_DDE_TERMINATE, 0, 0);
896     return pXAct;
897 }
898 
899 /******************************************************************
900  *              WDML_ServerHandleTerminate
901  *
902  *
903  */
904 static  WDML_QUEUE_STATE WDML_ServerHandleTerminate(WDML_CONV* pConv, WDML_XACT* pXAct)
905 {
906     /* billx: two things to remove: the conv, and associated links.
907      * Respond with another WM_DDE_TERMINATE iMsg.
908      */
909     if (!(pConv->instance->CBFflags & CBF_SKIP_DISCONNECTS))
910     {
911         WDML_InvokeCallback(pConv->instance, XTYP_DISCONNECT, 0, (HCONV)pConv, 0, 0,
912                             0, 0, (pConv->wStatus & ST_ISSELF) ? 1 : 0);
913     }
914     PostMessageW(pConv->hwndClient, WM_DDE_TERMINATE, (WPARAM)pConv->hwndServer, 0);
915     WDML_RemoveConv(pConv, WDML_SERVER_SIDE);
916 
917     return WDML_QS_HANDLED;
918 }
919 
920 /******************************************************************
921  *              WDML_ServerHandle
922  *
923  *
924  */
925 WDML_QUEUE_STATE WDML_ServerHandle(WDML_CONV* pConv, WDML_XACT* pXAct)
926 {
927     WDML_QUEUE_STATE    qs = WDML_QS_ERROR;
928 
929     switch (pXAct->ddeMsg)
930     {
931     case WM_DDE_INITIATE:
932         FIXME("WM_DDE_INITIATE shouldn't be there!\n");
933         break;
934     case WM_DDE_REQUEST:
935         qs = WDML_ServerHandleRequest(pConv, pXAct);
936         break;
937 
938     case WM_DDE_ADVISE:
939         qs = WDML_ServerHandleAdvise(pConv, pXAct);
940         break;
941 
942     case WM_DDE_UNADVISE:
943         qs = WDML_ServerHandleUnadvise(pConv, pXAct);
944         break;
945 
946     case WM_DDE_EXECUTE:
947         qs = WDML_ServerHandleExecute(pConv, pXAct);
948         break;
949 
950     case WM_DDE_POKE:
951         qs = WDML_ServerHandlePoke(pConv, pXAct);
952         break;
953 
954     case WM_DDE_TERMINATE:
955         qs = WDML_ServerHandleTerminate(pConv, pXAct);
956         break;
957 
958     case WM_DDE_ACK:
959         WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
960         break;
961 
962     default:
963         FIXME("Unsupported message %d\n", pXAct->ddeMsg);
964     }
965     return qs;
966 }
967 
968 /******************************************************************
969  *              WDML_ServerConvProc
970  *
971  *
972  */
973 static LRESULT CALLBACK WDML_ServerConvProc(HWND hwndServer, UINT iMsg, WPARAM wParam, LPARAM lParam)
974 {
975     WDML_INSTANCE*      pInstance;
976     WDML_CONV*          pConv;
977     WDML_XACT*          pXAct = NULL;
978 
979     TRACE("%p %04x %08lx %08lx\n", hwndServer, iMsg, wParam, lParam);
980 
981     if (iMsg == WM_DESTROY)
982     {
983         pConv = WDML_GetConvFromWnd(hwndServer);
984         if (pConv && !(pConv->wStatus & ST_TERMINATED))
985         {
986             WDML_ServerHandleTerminate(pConv, NULL);
987         }
988     }
989     if (iMsg < WM_DDE_FIRST || iMsg > WM_DDE_LAST)
990     {
991         return IsWindowUnicode(hwndServer) ? DefWindowProcW(hwndServer, iMsg, wParam, lParam) :
992                                              DefWindowProcA(hwndServer, iMsg, wParam, lParam);
993     }
994 
995     pInstance = WDML_GetInstanceFromWnd(hwndServer);
996     pConv = WDML_GetConvFromWnd(hwndServer);
997 
998     if (!pConv)
999     {
1000         ERR("Got a message (%x) on a not known conversation, dropping request\n", iMsg);
1001         return 0;
1002     }
1003     if (pConv->hwndClient != WIN_GetFullHandle( (HWND)wParam ) || pConv->hwndServer != hwndServer)
1004     {
1005         ERR("mismatch between C/S windows and conversation\n");
1006         return 0;
1007     }
1008     if (pConv->instance != pInstance || pConv->instance == NULL)
1009     {
1010         ERR("mismatch in instances\n");
1011         return 0;
1012     }
1013 
1014     switch (iMsg)
1015     {
1016     case WM_DDE_INITIATE:
1017         FIXME("WM_DDE_INITIATE message received!\n");
1018         break;
1019 
1020     case WM_DDE_REQUEST:
1021         pXAct = WDML_ServerQueueRequest(pConv, lParam);
1022         break;
1023 
1024     case WM_DDE_ADVISE:
1025         pXAct = WDML_ServerQueueAdvise(pConv, lParam);
1026         break;
1027 
1028     case WM_DDE_UNADVISE:
1029         pXAct = WDML_ServerQueueUnadvise(pConv, lParam);
1030         break;
1031 
1032     case WM_DDE_EXECUTE:
1033         pXAct = WDML_ServerQueueExecute(pConv, lParam);
1034         break;
1035 
1036     case WM_DDE_POKE:
1037         pXAct = WDML_ServerQueuePoke(pConv, lParam);
1038         break;
1039 
1040     case WM_DDE_TERMINATE:
1041         pXAct = WDML_ServerQueueTerminate(pConv, lParam);
1042         break;
1043 
1044     case WM_DDE_ACK:
1045         WARN("Shouldn't receive a ACK message (never requests them). Ignoring it\n");
1046         break;
1047 
1048     default:
1049         FIXME("Unsupported message %x\n", iMsg);
1050         break;
1051     }
1052 
1053     if (pXAct)
1054     {
1055         pXAct->lParam = lParam;
1056 
1057         if ((pConv->wStatus & ST_BLOCKED) || WDML_ServerHandle(pConv, pXAct) == WDML_QS_BLOCK)
1058         {
1059             TRACE("Transactions are blocked, add to the queue and exit\n");
1060             WDML_QueueTransaction(pConv, pXAct);
1061         }
1062         else
1063         {
1064             WDML_FreeTransaction(pInstance, pXAct, TRUE);
1065         }
1066     }
1067     else
1068         pConv->instance->lastError = DMLERR_MEMORY_ERROR;
1069 
1070     return 0;
1071 }
1072 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~ [ freetext search ] ~ [ file search ] ~

This page was automatically generated by the LXR engine.
Visit the LXR main site for more information.