1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: DDE classes
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
21 #pragma implementation "dde.h"
24 // For compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
38 #include "wx/module.h"
43 #include "wx/msw/private.h"
50 #define PCONVCONTEXT CONVCONTEXT*
53 #if defined(__TWIN32__) || defined(__GNUWIN32_OLD__)
54 #include "wx/msw/gnuwin32/extra.h"
57 // some compilers headers don't define this one (mingw32)
58 #ifndef DMLERR_NO_ERROR
59 #define DMLERR_NO_ERROR (0)
61 // this one is also missing from some mingw32 headers, but there is no way
62 // to test for it (I know of) - the test for DMLERR_NO_ERROR works for me,
63 // but is surely not the right thing to do
65 HDDEDATA STDCALL
DdeClientTransaction(LPBYTE pData
,
73 #endif // no DMLERR_NO_ERROR
75 // ----------------------------------------------------------------------------
76 // macros and constants
77 // ----------------------------------------------------------------------------
82 #define _EXPORT _export
86 #define DDE_CP CP_WINUNICODE
88 #define DDE_CP CP_WINANSI
91 #define GetHConv() ((HCONV)m_hConv)
93 // default timeout for DDE operations (5sec)
94 #define DDE_TIMEOUT 5000
96 // ----------------------------------------------------------------------------
98 // ----------------------------------------------------------------------------
100 static wxDDEConnection
*DDEFindConnection(HCONV hConv
);
101 static void DDEDeleteConnection(HCONV hConv
);
102 static wxDDEServer
*DDEFindServer(const wxString
& s
);
104 extern "C" HDDEDATA EXPENTRY _EXPORT
_DDECallback(WORD wType
,
113 // Add topic name to atom table before using in conversations
114 static HSZ
DDEAddAtom(const wxString
& string
);
115 static HSZ
DDEGetAtom(const wxString
& string
);
118 static HSZ
DDEAtomFromString(const wxString
& s
);
119 static wxString
DDEStringFromAtom(HSZ hsz
);
122 static wxString
DDEGetErrorMsg(UINT error
);
123 static void DDELogError(const wxString
& s
, UINT error
= DMLERR_NO_ERROR
);
125 // ----------------------------------------------------------------------------
127 // ----------------------------------------------------------------------------
129 static DWORD DDEIdInst
= 0L;
130 static wxDDEConnection
*DDECurrentlyConnecting
= NULL
;
132 static wxList
wxAtomTable(wxKEY_STRING
);
133 static wxList wxDDEClientObjects
;
134 static wxList wxDDEServerObjects
;
136 static bool DDEInitialized
= FALSE
;
138 // ----------------------------------------------------------------------------
140 // ----------------------------------------------------------------------------
142 // A module to allow DDE cleanup without calling these functions
143 // from app.cpp or from the user's application.
145 class wxDDEModule
: public wxModule
149 bool OnInit() { return TRUE
; }
150 void OnExit() { wxDDECleanUp(); }
153 DECLARE_DYNAMIC_CLASS(wxDDEModule
)
156 // ----------------------------------------------------------------------------
158 // ----------------------------------------------------------------------------
160 IMPLEMENT_DYNAMIC_CLASS(wxDDEServer
, wxServerBase
)
161 IMPLEMENT_DYNAMIC_CLASS(wxDDEClient
, wxClientBase
)
162 IMPLEMENT_CLASS(wxDDEConnection
, wxConnectionBase
)
163 IMPLEMENT_DYNAMIC_CLASS(wxDDEModule
, wxModule
)
165 // ============================================================================
167 // ============================================================================
169 // ----------------------------------------------------------------------------
170 // initialization and cleanup
171 // ----------------------------------------------------------------------------
173 extern void wxDDEInitialize()
175 if ( !DDEInitialized
)
177 // Should insert filter flags
178 PFNCALLBACK callback
= (PFNCALLBACK
)
179 MakeProcInstance((FARPROC
)_DDECallback
, wxGetInstance());
180 UINT rc
= DdeInitialize(&DDEIdInst
, callback
, APPCLASS_STANDARD
, 0L);
181 if ( rc
!= DMLERR_NO_ERROR
)
183 DDELogError(_T("Failed to initialize DDE"), rc
);
187 DDEInitialized
= TRUE
;
194 if ( DDEIdInst
!= 0 )
196 DdeUninitialize(DDEIdInst
);
201 // ----------------------------------------------------------------------------
202 // functions working with the global connection list(s)
203 // ----------------------------------------------------------------------------
205 // Global find connection
206 static wxDDEConnection
*DDEFindConnection(HCONV hConv
)
208 wxNode
*node
= wxDDEServerObjects
.First();
209 wxDDEConnection
*found
= NULL
;
210 while (node
&& !found
)
212 wxDDEServer
*object
= (wxDDEServer
*)node
->Data();
213 found
= object
->FindConnection((WXHCONV
) hConv
);
219 node
= wxDDEClientObjects
.First();
220 while (node
&& !found
)
222 wxDDEClient
*object
= (wxDDEClient
*)node
->Data();
223 found
= object
->FindConnection((WXHCONV
) hConv
);
229 // Global delete connection
230 static void DDEDeleteConnection(HCONV hConv
)
232 wxNode
*node
= wxDDEServerObjects
.First();
234 while (node
&& !found
)
236 wxDDEServer
*object
= (wxDDEServer
*)node
->Data();
237 found
= object
->DeleteConnection((WXHCONV
) hConv
);
243 node
= wxDDEClientObjects
.First();
244 while (node
&& !found
)
246 wxDDEClient
*object
= (wxDDEClient
*)node
->Data();
247 found
= object
->DeleteConnection((WXHCONV
) hConv
);
252 // Find a server from a service name
253 static wxDDEServer
*DDEFindServer(const wxString
& s
)
255 wxNode
*node
= wxDDEServerObjects
.First();
256 wxDDEServer
*found
= NULL
;
257 while (node
&& !found
)
259 wxDDEServer
*object
= (wxDDEServer
*)node
->Data();
261 if (object
->GetServiceName() == s
)
263 else node
= node
->Next();
268 // ----------------------------------------------------------------------------
270 // ----------------------------------------------------------------------------
272 wxDDEServer::wxDDEServer()
276 wxDDEServerObjects
.Append(this);
279 bool wxDDEServer::Create(const wxString
& server
)
281 m_serviceName
= server
;
283 if ( !DdeNameService(DDEIdInst
, DDEAtomFromString(server
), (HSZ
)NULL
, DNS_REGISTER
) )
285 DDELogError(wxString::Format(_("Failed to register DDE server '%s'"),
294 wxDDEServer::~wxDDEServer()
296 if ( !!m_serviceName
)
298 if ( !DdeNameService(DDEIdInst
, DDEAtomFromString(m_serviceName
),
299 (HSZ
)NULL
, DNS_UNREGISTER
) )
301 DDELogError(wxString::Format(_("Failed to unregister DDE server '%s'"),
302 m_serviceName
.c_str()));
306 wxDDEServerObjects
.DeleteObject(this);
308 wxNode
*node
= m_connections
.First();
311 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
312 wxNode
*next
= node
->Next();
313 connection
->SetConnected(false);
314 connection
->OnDisconnect(); // May delete the node implicitly
318 // If any left after this, delete them
319 node
= m_connections
.First();
322 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
323 wxNode
*next
= node
->Next();
329 wxConnectionBase
*wxDDEServer::OnAcceptConnection(const wxString
& /* topic */)
331 return new wxDDEConnection
;
334 wxDDEConnection
*wxDDEServer::FindConnection(WXHCONV conv
)
336 wxNode
*node
= m_connections
.First();
337 wxDDEConnection
*found
= NULL
;
338 while (node
&& !found
)
340 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
341 if (connection
->m_hConv
== conv
)
343 else node
= node
->Next();
348 // Only delete the entry in the map, not the actual connection
349 bool wxDDEServer::DeleteConnection(WXHCONV conv
)
351 wxNode
*node
= m_connections
.First();
353 while (node
&& !found
)
355 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
356 if (connection
->m_hConv
== conv
)
361 else node
= node
->Next();
366 // ----------------------------------------------------------------------------
368 // ----------------------------------------------------------------------------
370 wxDDEClient::wxDDEClient()
374 wxDDEClientObjects
.Append(this);
377 wxDDEClient::~wxDDEClient()
379 wxDDEClientObjects
.DeleteObject(this);
380 wxNode
*node
= m_connections
.First();
383 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
384 delete connection
; // Deletes the node implicitly (see ~wxDDEConnection)
385 node
= m_connections
.First();
389 bool wxDDEClient::ValidHost(const wxString
& /* host */)
394 wxConnectionBase
*wxDDEClient::MakeConnection(const wxString
& WXUNUSED(host
),
395 const wxString
& server
,
396 const wxString
& topic
)
398 HCONV hConv
= DdeConnect(DDEIdInst
, DDEAtomFromString(server
), DDEAtomFromString(topic
),
402 DDELogError(wxString::Format(_("Failed to create connection to server '%s' on topic '%s'"),
403 server
.c_str(), topic
.c_str()));
407 wxDDEConnection
*connection
= (wxDDEConnection
*) OnMakeConnection();
410 connection
->m_hConv
= (WXHCONV
) hConv
;
411 connection
->m_topicName
= topic
;
412 connection
->m_client
= this;
413 m_connections
.Append(connection
);
418 return (wxConnectionBase
*) NULL
;
421 wxConnectionBase
*wxDDEClient::OnMakeConnection()
423 return new wxDDEConnection
;
426 wxDDEConnection
*wxDDEClient::FindConnection(WXHCONV conv
)
428 wxNode
*node
= m_connections
.First();
429 wxDDEConnection
*found
= NULL
;
430 while (node
&& !found
)
432 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
433 if (connection
->m_hConv
== conv
)
435 else node
= node
->Next();
440 // Only delete the entry in the map, not the actual connection
441 bool wxDDEClient::DeleteConnection(WXHCONV conv
)
443 wxNode
*node
= m_connections
.First();
445 while (node
&& !found
)
447 wxDDEConnection
*connection
= (wxDDEConnection
*)node
->Data();
448 if (connection
->m_hConv
== conv
)
453 else node
= node
->Next();
458 // ----------------------------------------------------------------------------
460 // ----------------------------------------------------------------------------
462 wxDDEConnection::wxDDEConnection(char *buffer
, int size
)
463 : wxConnectionBase(buffer
, size
)
469 m_sendingData
= NULL
;
472 wxDDEConnection::wxDDEConnection()
476 m_sendingData
= NULL
;
481 wxDDEConnection::~wxDDEConnection()
485 m_server
->GetConnections().DeleteObject(this);
487 m_client
->GetConnections().DeleteObject(this);
490 // Calls that CLIENT can make
491 bool wxDDEConnection::Disconnect()
493 if ( !GetConnected() )
496 DDEDeleteConnection(GetHConv());
498 bool ok
= DdeDisconnect(GetHConv()) != 0;
501 DDELogError(_T("Failed to disconnect from DDE server gracefully"));
504 SetConnected( false ); // so we don't try and disconnect again
509 bool wxDDEConnection::Execute(const wxChar
*data
, int size
, wxIPCFormat format
)
514 size
= wxStrlen(data
) + 1;
517 bool ok
= DdeClientTransaction((LPBYTE
)data
, size
,
527 DDELogError(_T("DDE execute request failed"));
533 char *wxDDEConnection::Request(const wxString
& item
, int *size
, wxIPCFormat format
)
537 HSZ atom
= DDEGetAtom(item
);
539 HDDEDATA returned_data
= DdeClientTransaction(NULL
, 0,
545 if ( !returned_data
)
547 DDELogError(_T("DDE data request failed"));
552 DWORD len
= DdeGetData(returned_data
, NULL
, 0, 0);
554 wxChar
*data
= GetBufferAtLeast( len
);
555 wxASSERT_MSG(data
!= NULL
,
556 _T("Buffer too small in wxDDEConnection::Request") );
557 DdeGetData(returned_data
, (LPBYTE
)data
, len
, 0);
559 DdeFreeDataHandle(returned_data
);
567 bool wxDDEConnection::Poke(const wxString
& item
, wxChar
*data
, int size
, wxIPCFormat format
)
572 size
= wxStrlen(data
) + 1;
575 HSZ item_atom
= DDEGetAtom(item
);
576 bool ok
= DdeClientTransaction((LPBYTE
)data
, size
,
584 DDELogError(_("DDE poke request failed"));
590 bool wxDDEConnection::StartAdvise(const wxString
& item
)
593 HSZ atom
= DDEGetAtom(item
);
595 bool ok
= DdeClientTransaction(NULL
, 0,
603 DDELogError(_("Failed to establish an advise loop with DDE server"));
609 bool wxDDEConnection::StopAdvise(const wxString
& item
)
612 HSZ atom
= DDEGetAtom(item
);
614 bool ok
= DdeClientTransaction(NULL
, 0,
622 DDELogError(_("Failed to terminate the advise loop with DDE server"));
628 // Calls that SERVER can make
629 bool wxDDEConnection::Advise(const wxString
& item
,
636 size
= wxStrlen(data
) + 1;
639 HSZ item_atom
= DDEGetAtom(item
);
640 HSZ topic_atom
= DDEGetAtom(m_topicName
);
641 m_sendingData
= data
; // mrf: potential for scope problems here?
645 bool ok
= DdePostAdvise(DDEIdInst
, topic_atom
, item_atom
) != 0;
648 DDELogError(_("Failed to send DDE advise notification"));
654 bool wxDDEConnection::OnDisconnect()
660 // ----------------------------------------------------------------------------
662 // ----------------------------------------------------------------------------
664 #define DDERETURN HDDEDATA
666 HDDEDATA EXPENTRY _EXPORT
667 _DDECallback(WORD wType
,
673 DWORD
WXUNUSED(lData1
),
674 DWORD
WXUNUSED(lData2
))
680 wxString topic
= DDEStringFromAtom(hsz1
),
681 srv
= DDEStringFromAtom(hsz2
);
682 wxDDEServer
*server
= DDEFindServer(srv
);
685 wxDDEConnection
*connection
=
686 (wxDDEConnection
*) server
->OnAcceptConnection(topic
);
689 connection
->m_server
= server
;
690 server
->GetConnections().Append(connection
);
691 connection
->m_hConv
= 0;
692 connection
->m_topicName
= topic
;
693 DDECurrentlyConnecting
= connection
;
694 return (DDERETURN
)(DWORD
)TRUE
;
700 case XTYP_CONNECT_CONFIRM
:
702 if (DDECurrentlyConnecting
)
704 DDECurrentlyConnecting
->m_hConv
= (WXHCONV
) hConv
;
705 DDECurrentlyConnecting
= NULL
;
706 return (DDERETURN
)(DWORD
)TRUE
;
711 case XTYP_DISCONNECT
:
713 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
716 connection
->SetConnected( false );
717 if (connection
->OnDisconnect())
719 DDEDeleteConnection(hConv
); // Delete mapping: hConv => connection
720 return (DDERETURN
)(DWORD
)TRUE
;
728 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
732 DWORD len
= DdeGetData(hData
, NULL
, 0, 0);
734 wxChar
*data
= connection
->GetBufferAtLeast( len
);
735 wxASSERT_MSG(data
!= NULL
,
736 _T("Buffer too small in _DDECallback (XTYP_EXECUTE)") );
738 DdeGetData(hData
, (LPBYTE
)data
, len
, 0);
740 DdeFreeDataHandle(hData
);
742 if ( connection
->OnExecute(connection
->m_topicName
,
745 (wxIPCFormat
) wFmt
) )
747 return (DDERETURN
)(DWORD
)DDE_FACK
;
751 return (DDERETURN
)DDE_FNOTPROCESSED
;
756 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
760 wxString item_name
= DDEStringFromAtom(hsz2
);
763 char *data
= connection
->OnRequest(connection
->m_topicName
,
770 user_size
= wxStrlen((wxChar
*)data
) + 1;
772 HDDEDATA handle
= DdeCreateDataHandle(DDEIdInst
,
779 return (DDERETURN
)handle
;
787 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
791 wxString item_name
= DDEStringFromAtom(hsz2
);
793 DWORD len
= DdeGetData(hData
, NULL
, 0, 0);
795 wxChar
*data
= connection
->GetBufferAtLeast( len
);
796 wxASSERT_MSG(data
!= NULL
,
797 _T("Buffer too small in _DDECallback (XTYP_EXECUTE)") );
799 DdeGetData(hData
, (LPBYTE
)data
, len
, 0);
801 DdeFreeDataHandle(hData
);
803 connection
->OnPoke(connection
->m_topicName
,
809 return (DDERETURN
)DDE_FACK
;
813 return (DDERETURN
)DDE_FNOTPROCESSED
;
819 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
823 wxString item_name
= DDEStringFromAtom(hsz2
);
825 return (DDERETURN
)connection
->
826 OnStartAdvise(connection
->m_topicName
, item_name
);
834 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
838 wxString item_name
= DDEStringFromAtom(hsz2
);
840 return (DDERETURN
)connection
->
841 OnStopAdvise(connection
->m_topicName
, item_name
);
849 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
851 if (connection
&& connection
->m_sendingData
)
853 HDDEDATA data
= DdeCreateDataHandle
856 (LPBYTE
)connection
->m_sendingData
,
857 connection
->m_dataSize
,
860 connection
->m_dataType
,
864 connection
->m_sendingData
= NULL
;
866 return (DDERETURN
)data
;
874 wxDDEConnection
*connection
= DDEFindConnection(hConv
);
878 wxString item_name
= DDEStringFromAtom(hsz2
);
880 DWORD len
= DdeGetData(hData
, NULL
, 0, 0);
882 wxChar
*data
= connection
->GetBufferAtLeast( len
);
883 wxASSERT_MSG(data
!= NULL
,
884 _T("Buffer too small in _DDECallback (XTYP_ADVDATA)") );
886 DdeGetData(hData
, (LPBYTE
)data
, len
, 0);
888 DdeFreeDataHandle(hData
);
889 if ( connection
->OnAdvise(connection
->m_topicName
,
893 (wxIPCFormat
) wFmt
) )
895 return (DDERETURN
)(DWORD
)DDE_FACK
;
899 return (DDERETURN
)DDE_FNOTPROCESSED
;
906 // ----------------------------------------------------------------------------
907 // DDE strings and atoms
908 // ----------------------------------------------------------------------------
911 static HSZ
DDEAddAtom(const wxString
& string
)
913 HSZ atom
= DDEAtomFromString(string
);
914 wxAtomTable
.Append(string
, (wxObject
*)atom
);
918 static HSZ
DDEGetAtom(const wxString
& string
)
920 wxNode
*node
= wxAtomTable
.Find(string
);
922 return (HSZ
)node
->Data();
926 return (HSZ
)(wxAtomTable
.Find(string
)->Data());
931 static HSZ
DDEAtomFromString(const wxString
& s
)
933 wxASSERT_MSG( DDEIdInst
, _T("DDE not initialized") );
935 HSZ hsz
= DdeCreateStringHandle(DDEIdInst
, (wxChar
*) s
.c_str(), DDE_CP
);
938 DDELogError(_("Failed to create DDE string"));
944 static wxString
DDEStringFromAtom(HSZ hsz
)
946 // all DDE strings are normally limited to 255 bytes
947 static const size_t len
= 256;
950 (void)DdeQueryString(DDEIdInst
, hsz
, s
.GetWriteBuf(len
), len
, DDE_CP
);
956 // ----------------------------------------------------------------------------
958 // ----------------------------------------------------------------------------
960 static void DDELogError(const wxString
& s
, UINT error
)
964 error
= DdeGetLastError(DDEIdInst
);
967 wxLogError(s
+ _T(": ") + DDEGetErrorMsg(error
));
970 static wxString
DDEGetErrorMsg(UINT error
)
975 case DMLERR_NO_ERROR
:
976 err
= _("no DDE error.");
979 case DMLERR_ADVACKTIMEOUT
:
980 err
= _("a request for a synchronous advise transaction has timed out.");
983 err
= _("the response to the transaction caused the DDE_FBUSY bit to be set.");
985 case DMLERR_DATAACKTIMEOUT
:
986 err
= _("a request for a synchronous data transaction has timed out.");
988 case DMLERR_DLL_NOT_INITIALIZED
:
989 err
= _("a DDEML function was called without first calling the DdeInitialize function,\nor an invalid instance identifier\nwas passed to a DDEML function.");
991 case DMLERR_DLL_USAGE
:
992 err
= _("an application initialized as APPCLASS_MONITOR has\nattempted to perform a DDE transaction,\nor an application initialized as APPCMD_CLIENTONLY has \nattempted to perform server transactions.");
994 case DMLERR_EXECACKTIMEOUT
:
995 err
= _("a request for a synchronous execute transaction has timed out.");
997 case DMLERR_INVALIDPARAMETER
:
998 err
= _("a parameter failed to be validated by the DDEML.");
1000 case DMLERR_LOW_MEMORY
:
1001 err
= _("a DDEML application has created a prolonged race condition.");
1003 case DMLERR_MEMORY_ERROR
:
1004 err
= _("a memory allocation failed.");
1006 case DMLERR_NO_CONV_ESTABLISHED
:
1007 err
= _("a client's attempt to establish a conversation has failed.");
1009 case DMLERR_NOTPROCESSED
:
1010 err
= _("a transaction failed.");
1012 case DMLERR_POKEACKTIMEOUT
:
1013 err
= _("a request for a synchronous poke transaction has timed out.");
1015 case DMLERR_POSTMSG_FAILED
:
1016 err
= _("an internal call to the PostMessage function has failed. ");
1018 case DMLERR_REENTRANCY
:
1019 err
= _("reentrancy problem.");
1021 case DMLERR_SERVER_DIED
:
1022 err
= _("a server-side transaction was attempted on a conversation\nthat was terminated by the client, or the server\nterminated before completing a transaction.");
1024 case DMLERR_SYS_ERROR
:
1025 err
= _("an internal error has occurred in the DDEML.");
1027 case DMLERR_UNADVACKTIMEOUT
:
1028 err
= _("a request to end an advise transaction has timed out.");
1030 case DMLERR_UNFOUND_QUEUE_ID
:
1031 err
= _("an invalid transaction identifier was passed to a DDEML function.\nOnce the application has returned from an XTYP_XACT_COMPLETE callback,\nthe transaction identifier for that callback is no longer valid.");
1034 err
.Printf(_("Unknown DDE error %08x"), error
);