1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     Interprocess communication implementation (wxSocket version) 
   4 // Author:      Julian Smart 
   5 // Modified by: Guilhem Lavaux (big rewrite) May 1997, 1998 
   6 //              Guillermo Rodriguez (updated for wxSocket v2) Jan 2000 
   7 //                                  (callbacks deprecated)    Mar 2000 
   8 //              Vadim Zeitlin (added support for Unix sockets) Apr 2002 
  11 // Copyright:   (c) Julian Smart 1993 
  12 //              (c) Guilhem Lavaux 1997, 1998 
  13 //              (c) 2000 Guillermo Rodriguez <guille@iies.es> 
  14 // Licence:     wxWindows license 
  15 ///////////////////////////////////////////////////////////////////////////// 
  17 // ========================================================================== 
  19 // ========================================================================== 
  21 // -------------------------------------------------------------------------- 
  23 // -------------------------------------------------------------------------- 
  26 #pragma implementation "sckipc.h" 
  29 // For compilers that support precompilation, includes "wx.h". 
  30 #include "wx/wxprec.h" 
  40 #if wxUSE_SOCKETS && wxUSE_IPC && wxUSE_STREAMS 
  46 #include "wx/socket.h" 
  47 #include "wx/sckipc.h" 
  48 #include "wx/module.h" 
  51 // -------------------------------------------------------------------------- 
  52 // macros and constants 
  53 // -------------------------------------------------------------------------- 
  55 // It seems to be already defined somewhere in the Xt includes. 
  74 // All sockets will be created with the following flags 
  75 #define SCKIPC_FLAGS (wxSOCKET_WAITALL) 
  77 // headers needed for umask() 
  79     #include <sys/types.h> 
  81 #endif // __UNIX_LIKE__ 
  83 // ---------------------------------------------------------------------------- 
  85 // ---------------------------------------------------------------------------- 
  87 // get the address object for the given server name, the caller must delete it 
  88 static wxSockAddress 
* 
  89 GetAddressFromName(const wxString
& serverName
, const wxString
& host 
= _T("")) 
  91     // we always use INET sockets under non-Unix systems 
  92 #if defined(__UNIX__) && !defined(__WXMAC__) && !defined(__WINE__) 
  93     // under Unix, if the server name looks like a path, create a AF_UNIX 
  94     // socket instead of AF_INET one 
  95     if ( serverName
.Find(_T('/')) != wxNOT_FOUND 
) 
  97         wxUNIXaddress 
*addr 
= new wxUNIXaddress
; 
  98         addr
->Filename(serverName
); 
 104         wxIPV4address 
*addr 
= new wxIPV4address
; 
 105         addr
->Service(serverName
); 
 108             addr
->Hostname(host
); 
 115 // -------------------------------------------------------------------------- 
 116 // wxTCPEventHandler stuff (private class) 
 117 // -------------------------------------------------------------------------- 
 119 class wxTCPEventHandler 
: public wxEvtHandler
 
 122   wxTCPEventHandler() : wxEvtHandler() {}; 
 124   void Client_OnRequest(wxSocketEvent
& event
); 
 125   void Server_OnRequest(wxSocketEvent
& event
); 
 127   DECLARE_EVENT_TABLE() 
 132   _CLIENT_ONREQUEST_ID 
= 1000, 
 136 static wxTCPEventHandler 
*gs_handler 
= NULL
; 
 138 // ========================================================================== 
 140 // ========================================================================== 
 142 IMPLEMENT_DYNAMIC_CLASS(wxTCPServer
, wxServerBase
) 
 143 IMPLEMENT_DYNAMIC_CLASS(wxTCPClient
, wxClientBase
) 
 144 IMPLEMENT_CLASS(wxTCPConnection
, wxConnectionBase
) 
 146 // -------------------------------------------------------------------------- 
 148 // -------------------------------------------------------------------------- 
 150 wxTCPClient::wxTCPClient () : wxClientBase() 
 154 wxTCPClient::~wxTCPClient () 
 158 bool wxTCPClient::ValidHost(const wxString
& host
) 
 162   return addr
.Hostname(host
); 
 165 wxConnectionBase 
*wxTCPClient::MakeConnection (const wxString
& host
, 
 166                                                const wxString
& serverName
, 
 167                                                const wxString
& topic
) 
 169   wxSocketClient 
*client 
= new wxSocketClient(SCKIPC_FLAGS
); 
 170   wxSocketStream 
*stream 
= new wxSocketStream(*client
); 
 171   wxDataInputStream 
*data_is 
= new wxDataInputStream(*stream
); 
 172   wxDataOutputStream 
*data_os 
= new wxDataOutputStream(*stream
); 
 174   wxSockAddress 
*addr 
= GetAddressFromName(serverName
, host
); 
 178   bool ok 
= client
->Connect(*addr
); 
 185     // Send topic name, and enquire whether this has succeeded 
 186     data_os
->Write8(IPC_CONNECT
); 
 187     data_os
->WriteString(topic
); 
 189     msg 
= data_is
->Read8(); 
 192     if (msg 
== IPC_CONNECT
) 
 194       wxTCPConnection 
*connection 
= (wxTCPConnection 
*)OnMakeConnection (); 
 198         if (connection
->IsKindOf(CLASSINFO(wxTCPConnection
))) 
 200           connection
->m_topic 
= topic
; 
 201           connection
->m_sock  
= client
; 
 202           connection
->m_sockstrm 
= stream
; 
 203           connection
->m_codeci 
= data_is
; 
 204           connection
->m_codeco 
= data_os
; 
 205           client
->SetEventHandler(*gs_handler
, _CLIENT_ONREQUEST_ID
); 
 206           client
->SetClientData(connection
); 
 207           client
->SetNotify(wxSOCKET_INPUT_FLAG 
| wxSOCKET_LOST_FLAG
); 
 208           client
->Notify(TRUE
); 
 214           // and fall through to delete everything else 
 220   // Something went wrong, delete everything 
 229 wxConnectionBase 
*wxTCPClient::OnMakeConnection() 
 231   return new wxTCPConnection(); 
 234 // -------------------------------------------------------------------------- 
 236 // -------------------------------------------------------------------------- 
 238 wxTCPServer::wxTCPServer () : wxServerBase() 
 243 bool wxTCPServer::Create(const wxString
& serverName
) 
 245   // Destroy previous server, if any 
 248     m_server
->SetClientData(NULL
); 
 253   wxSockAddress 
*addr 
= GetAddressFromName(serverName
); 
 259   if ( addr
->Type() == wxSockAddress::UNIX 
) 
 261       // ensure that the file doesn't exist as otherwise calling socket() would 
 263       int rc 
= remove(serverName
.fn_str()); 
 264       if ( rc 
< 0 && errno 
!= ENOENT 
) 
 271       // also set the umask to prevent the others from reading our file 
 272       umaskOld 
= umask(077); 
 276       // unused anyhow but shut down the compiler warnings 
 279 #endif // __UNIX_LIKE__ 
 281   // Create a socket listening on the specified port 
 282   m_server 
= new wxSocketServer(*addr
, SCKIPC_FLAGS
); 
 285   if ( addr
->Type() == wxSockAddress::UNIX 
) 
 290       // save the file name to remove it later 
 291       m_filename 
= serverName
; 
 293 #endif // __UNIX_LIKE__ 
 305   m_server
->SetEventHandler(*gs_handler
, _SERVER_ONREQUEST_ID
); 
 306   m_server
->SetClientData(this); 
 307   m_server
->SetNotify(wxSOCKET_CONNECTION_FLAG
); 
 308   m_server
->Notify(TRUE
); 
 313 wxTCPServer::~wxTCPServer() 
 317         m_server
->SetClientData(NULL
); 
 322     if ( !m_filename
.empty() ) 
 324         if ( remove(m_filename
.fn_str()) != 0 ) 
 326             wxLogDebug(_T("Stale AF_UNIX file '%s' left."), m_filename
.c_str()); 
 329 #endif // __UNIX_LIKE__ 
 332 wxConnectionBase 
*wxTCPServer::OnAcceptConnection( const wxString
& WXUNUSED(topic
) ) 
 334   return new wxTCPConnection(); 
 337 // -------------------------------------------------------------------------- 
 339 // -------------------------------------------------------------------------- 
 341 wxTCPConnection::wxTCPConnection () : wxConnectionBase() 
 349 wxTCPConnection::wxTCPConnection(wxChar 
*buffer
, int size
) 
 350        : wxConnectionBase(buffer
, size
) 
 358 wxTCPConnection::~wxTCPConnection () 
 363   wxDELETE(m_sockstrm
); 
 367     m_sock
->SetClientData(NULL
); 
 372 void wxTCPConnection::Compress(bool WXUNUSED(on
)) 
 377 // Calls that CLIENT can make. 
 378 bool wxTCPConnection::Disconnect () 
 380   if ( !GetConnected() ) 
 382   // Send the the disconnect message to the peer. 
 383   m_codeco
->Write8(IPC_DISCONNECT
); 
 384   m_sock
->Notify(FALSE
); 
 391 bool wxTCPConnection::Execute(const wxChar 
*data
, int size
, wxIPCFormat format
) 
 393   if (!m_sock
->IsConnected()) 
 396   // Prepare EXECUTE message 
 397   m_codeco
->Write8(IPC_EXECUTE
); 
 398   m_codeco
->Write8(format
); 
 401     size 
= wxStrlen(data
) + 1;    // includes final NUL 
 403   m_codeco
->Write32(size
); 
 404   m_sockstrm
->Write(data
, size
); 
 409 wxChar 
*wxTCPConnection::Request (const wxString
& item
, int *size
, wxIPCFormat format
) 
 411   if (!m_sock
->IsConnected()) 
 414   m_codeco
->Write8(IPC_REQUEST
); 
 415   m_codeco
->WriteString(item
); 
 416   m_codeco
->Write8(format
); 
 418   // If Unpack doesn't initialize it. 
 421   ret 
= m_codeci
->Read8(); 
 428     s 
= m_codeci
->Read32(); 
 429     wxChar 
*data 
= GetBufferAtLeast( s 
); 
 430     wxASSERT_MSG(data 
!= NULL
, 
 431                  _T("Buffer too small in wxTCPConnection::Request") ); 
 432     m_sockstrm
->Read(data
, s
); 
 440 bool wxTCPConnection::Poke (const wxString
& item
, wxChar 
*data
, int size
, wxIPCFormat format
) 
 442   if (!m_sock
->IsConnected()) 
 445   m_codeco
->Write8(IPC_POKE
); 
 446   m_codeco
->WriteString(item
); 
 447   m_codeco
->Write8(format
); 
 450     size 
= wxStrlen(data
) + 1;    // includes final NUL 
 452   m_codeco
->Write32(size
); 
 453   m_sockstrm
->Write(data
, size
); 
 458 bool wxTCPConnection::StartAdvise (const wxString
& item
) 
 462   if (!m_sock
->IsConnected()) 
 465   m_codeco
->Write8(IPC_ADVISE_START
); 
 466   m_codeco
->WriteString(item
); 
 468   ret 
= m_codeci
->Read8(); 
 476 bool wxTCPConnection::StopAdvise (const wxString
& item
) 
 480   if (!m_sock
->IsConnected()) 
 483   m_codeco
->Write8(IPC_ADVISE_STOP
); 
 484   m_codeco
->WriteString(item
); 
 486   msg 
= m_codeci
->Read8(); 
 494 // Calls that SERVER can make 
 495 bool wxTCPConnection::Advise (const wxString
& item
, 
 496                               wxChar 
*data
, int size
, wxIPCFormat format
) 
 498   if (!m_sock
->IsConnected()) 
 501   m_codeco
->Write8(IPC_ADVISE
); 
 502   m_codeco
->WriteString(item
); 
 503   m_codeco
->Write8(format
); 
 506     size 
= wxStrlen(data
) + 1;    // includes final NUL 
 508   m_codeco
->Write32(size
); 
 509   m_sockstrm
->Write(data
, size
); 
 514 // -------------------------------------------------------------------------- 
 515 // wxTCPEventHandler (private class) 
 516 // -------------------------------------------------------------------------- 
 518 BEGIN_EVENT_TABLE(wxTCPEventHandler
, wxEvtHandler
) 
 519   EVT_SOCKET(_CLIENT_ONREQUEST_ID
, wxTCPEventHandler::Client_OnRequest
) 
 520   EVT_SOCKET(_SERVER_ONREQUEST_ID
, wxTCPEventHandler::Server_OnRequest
) 
 523 void wxTCPEventHandler::Client_OnRequest(wxSocketEvent 
&event
) 
 525   wxSocketBase 
*sock 
= event
.GetSocket(); 
 526   wxSocketNotify evt 
= event
.GetSocketEvent(); 
 527   wxTCPConnection 
*connection 
= (wxTCPConnection 
*)(sock
->GetClientData()); 
 529   // This socket is being deleted; skip this event 
 534   wxDataInputStream 
*codeci
; 
 535   wxDataOutputStream 
*codeco
; 
 536   wxSocketStream 
*sockstrm
; 
 537   wxString topic_name 
= connection
->m_topic
; 
 540   // We lost the connection: destroy everything 
 541   if (evt 
== wxSOCKET_LOST
) 
 545     connection
->OnDisconnect(); 
 549   // Receive message number. 
 550   codeci 
= connection
->m_codeci
; 
 551   codeco 
= connection
->m_codeco
; 
 552   sockstrm 
= connection
->m_sockstrm
; 
 553   msg 
= codeci
->Read8(); 
 563     format 
= (wxIPCFormat
)codeci
->Read8(); 
 564     size 
= codeci
->Read32(); 
 565     data 
= connection
->GetBufferAtLeast( size 
); 
 566     wxASSERT_MSG(data 
!= NULL
, 
 567                  _T("Buffer too small in wxTCPEventHandler::Client_OnRequest") ); 
 568     sockstrm
->Read(data
, size
); 
 570     connection
->OnExecute (topic_name
, data
, size
, format
); 
 580     item 
= codeci
->ReadString(); 
 581     format 
= (wxIPCFormat
)codeci
->Read8(); 
 582     size 
= codeci
->Read32(); 
 583     data 
= connection
->GetBufferAtLeast( size 
); 
 584     wxASSERT_MSG(data 
!= NULL
, 
 585                  _T("Buffer too small in wxTCPEventHandler::Client_OnRequest") ); 
 586     sockstrm
->Read(data
, size
); 
 588     connection
->OnAdvise (topic_name
, item
, data
, size
, format
); 
 592   case IPC_ADVISE_START
: 
 594     item 
= codeci
->ReadString(); 
 596     bool ok 
= connection
->OnStartAdvise (topic_name
, item
); 
 598       codeco
->Write8(IPC_ADVISE_START
); 
 600       codeco
->Write8(IPC_FAIL
); 
 604   case IPC_ADVISE_STOP
: 
 606     item 
= codeci
->ReadString(); 
 608     bool ok 
= connection
->OnStopAdvise (topic_name
, item
); 
 610       codeco
->Write8(IPC_ADVISE_STOP
); 
 612       codeco
->Write8(IPC_FAIL
); 
 622     item 
= codeci
->ReadString(); 
 623     format 
= (wxIPCFormat
)codeci
->Read8(); 
 624     size 
= codeci
->Read32(); 
 625     data 
= connection
->GetBufferAtLeast( size 
); 
 626     wxASSERT_MSG(data 
!= NULL
, 
 627                  _T("Buffer too small in wxTCPEventHandler::Client_OnRequest") ); 
 628     sockstrm
->Read(data
, size
); 
 630     connection
->OnPoke (topic_name
, item
, data
, size
, format
); 
 638     item 
= codeci
->ReadString(); 
 639     format 
= (wxIPCFormat
)codeci
->Read8(); 
 642     wxChar 
*user_data 
= connection
->OnRequest (topic_name
, item
, &user_size
, format
); 
 646       codeco
->Write8(IPC_REQUEST_REPLY
); 
 649         user_size 
= wxStrlen(user_data
) + 1;      // includes final NUL 
 651       codeco
->Write32(user_size
); 
 652       sockstrm
->Write(user_data
, user_size
); 
 655       codeco
->Write8(IPC_FAIL
); 
 663     connection
->SetConnected(FALSE
); 
 664     connection
->OnDisconnect(); 
 668     codeco
->Write8(IPC_FAIL
); 
 673 void wxTCPEventHandler::Server_OnRequest(wxSocketEvent 
&event
) 
 675   wxSocketServer 
*server 
= (wxSocketServer 
*) event
.GetSocket(); 
 676   wxTCPServer 
*ipcserv 
= (wxTCPServer 
*) server
->GetClientData(); 
 678   // This socket is being deleted; skip this event 
 682   if (event
.GetSocketEvent() != wxSOCKET_CONNECTION
) 
 685   // Accept the connection, getting a new socket 
 686   wxSocketBase 
*sock 
= server
->Accept(); 
 693   wxSocketStream 
*stream     
= new wxSocketStream(*sock
); 
 694   wxDataInputStream 
*codeci  
= new wxDataInputStream(*stream
); 
 695   wxDataOutputStream 
*codeco 
= new wxDataOutputStream(*stream
); 
 698   msg 
= codeci
->Read8(); 
 700   if (msg 
== IPC_CONNECT
) 
 703     topic_name 
= codeci
->ReadString(); 
 705     wxTCPConnection 
*new_connection 
= 
 706          (wxTCPConnection 
*)ipcserv
->OnAcceptConnection (topic_name
); 
 710       if (new_connection
->IsKindOf(CLASSINFO(wxTCPConnection
))) 
 712         // Acknowledge success 
 713         codeco
->Write8(IPC_CONNECT
); 
 714         new_connection
->m_topic 
= topic_name
; 
 715         new_connection
->m_sock 
= sock
; 
 716         new_connection
->m_sockstrm 
= stream
; 
 717         new_connection
->m_codeci 
= codeci
; 
 718         new_connection
->m_codeco 
= codeco
; 
 719         sock
->SetEventHandler(*gs_handler
, _CLIENT_ONREQUEST_ID
); 
 720         sock
->SetClientData(new_connection
); 
 721         sock
->SetNotify(wxSOCKET_INPUT_FLAG 
| wxSOCKET_LOST_FLAG
); 
 727         delete new_connection
; 
 728         // and fall through to delete everything else 
 733   // Something went wrong, send failure message and delete everything 
 734   codeco
->Write8(IPC_FAIL
); 
 742 // -------------------------------------------------------------------------- 
 743 // wxTCPEventHandlerModule (private class) 
 744 // -------------------------------------------------------------------------- 
 746 class WXDLLEXPORT wxTCPEventHandlerModule
: public wxModule
 
 748   DECLARE_DYNAMIC_CLASS(wxTCPEventHandlerModule
) 
 751   bool OnInit() { gs_handler 
= new wxTCPEventHandler(); return TRUE
; } 
 752   void OnExit() { wxDELETE(gs_handler
); } 
 755 IMPLEMENT_DYNAMIC_CLASS(wxTCPEventHandlerModule
, wxModule
) 
 759     // wxUSE_SOCKETS && wxUSE_IPC