1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/socket.cpp
3 // Purpose: Socket handler classes
4 // Authors: Guilhem Lavaux, Guillermo Rodriguez Garcia
6 // Copyright: (C) 1999-1997, Guilhem Lavaux
7 // (C) 2000-1999, Guillermo Rodriguez Garcia
9 // License: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ==========================================================================
14 // ==========================================================================
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
25 #include "wx/socket.h"
28 #include "wx/object.h"
29 #include "wx/string.h"
36 #include "wx/module.h"
39 #include "wx/apptrait.h"
40 #include "wx/sckaddr.h"
41 #include "wx/stopwatch.h"
42 #include "wx/thread.h"
43 #include "wx/evtloop.h"
45 // DLL options compatibility check:
47 WX_CHECK_BUILD_OPTIONS("wxNet")
49 // --------------------------------------------------------------------------
50 // macros and constants
51 // --------------------------------------------------------------------------
54 #define MAX_DISCARD_SIZE (10 * 1024)
56 #define wxTRACE_Socket _T("wxSocket")
58 // --------------------------------------------------------------------------
60 // --------------------------------------------------------------------------
62 IMPLEMENT_CLASS(wxSocketBase
, wxObject
)
63 IMPLEMENT_CLASS(wxSocketServer
, wxSocketBase
)
64 IMPLEMENT_CLASS(wxSocketClient
, wxSocketBase
)
65 IMPLEMENT_CLASS(wxDatagramSocket
, wxSocketBase
)
66 IMPLEMENT_DYNAMIC_CLASS(wxSocketEvent
, wxEvent
)
68 // --------------------------------------------------------------------------
70 // --------------------------------------------------------------------------
72 class wxSocketState
: public wxObject
75 wxSocketFlags m_flags
;
76 wxSocketEventFlags m_eventmask
;
81 wxSocketState() : wxObject() {}
83 DECLARE_NO_COPY_CLASS(wxSocketState
)
86 // ============================================================================
88 // ============================================================================
90 GSocketManager
*GSocketManager::ms_manager
= NULL
;
93 void GSocketManager::Set(GSocketManager
*manager
)
95 wxASSERT_MSG( !ms_manager
, "too late to set manager now" );
101 void GSocketManager::Init()
103 wxASSERT_MSG( !ms_manager
, "shouldn't be initialized twice" );
106 Details: Initialize() creates a hidden window as a sink for socket
107 events, such as 'read completed'. wxMSW has only one message loop
108 for the main thread. If Initialize is called in a secondary thread,
109 the socket window will be created for the secondary thread, but
110 since there is no message loop on this thread, it will never
111 receive events and all socket operations will time out.
112 BTW, the main thread must not be stopped using sleep or block
113 on a semaphore (a bad idea in any case) or socket operations
116 On the Mac side, Initialize() stores a pointer to the CFRunLoop for
117 the main thread. Because secondary threads do not have run loops,
118 adding event notifications to the "Current" loop would have no
119 effect at all, events would never fire.
121 wxASSERT_MSG( wxIsMainThread(),
122 "sockets must be initialized from the main thread" );
124 wxAppConsole
* const app
= wxAppConsole::GetInstance();
125 wxCHECK_RET( app
, "sockets can't be initialized without wxApp" );
127 ms_manager
= app
->GetTraits()->GetSocketManager();
130 // ==========================================================================
132 // ==========================================================================
134 // --------------------------------------------------------------------------
135 // Initialization and shutdown
136 // --------------------------------------------------------------------------
138 // FIXME-MT: all this is MT-unsafe, of course, we should protect all accesses
139 // to m_countInit with a crit section
140 size_t wxSocketBase::m_countInit
= 0;
142 bool wxSocketBase::IsInitialized()
144 return m_countInit
> 0;
147 bool wxSocketBase::Initialize()
149 if ( !m_countInit
++ )
151 if ( !GSocket_Init() )
162 void wxSocketBase::Shutdown()
164 // we should be initialized
165 wxASSERT_MSG( m_countInit
> 0, _T("extra call to Shutdown()") );
166 if ( --m_countInit
== 0 )
172 // --------------------------------------------------------------------------
174 // --------------------------------------------------------------------------
176 void wxSocketBase::Init()
179 m_type
= wxSOCKET_UNINIT
;
191 m_beingDeleted
= false;
205 if ( !IsInitialized() )
207 // this Initialize() will be undone by wxSocketModule::OnExit(), all
208 // the other calls to it should be matched by a call to Shutdown()
213 wxSocketBase::wxSocketBase()
218 wxSocketBase::wxSocketBase(wxSocketFlags flags
, wxSocketType type
)
226 wxSocketBase::~wxSocketBase()
228 // Just in case the app called Destroy() *and* then deleted the socket
229 // immediately: don't leave dangling pointers.
230 wxAppTraits
*traits
= wxTheApp
? wxTheApp
->GetTraits() : NULL
;
232 traits
->RemoveFromPendingDelete(this);
234 // Shutdown and close the socket
238 // Destroy the GSocket object
242 // Free the pushback buffer
247 bool wxSocketBase::Destroy()
249 // Delayed destruction: the socket will be deleted during the next idle
250 // loop iteration. This ensures that all pending events have been
252 m_beingDeleted
= true;
254 // Shutdown and close the socket
257 // Supress events from now on
260 // schedule this object for deletion
261 wxAppTraits
*traits
= wxTheApp
? wxTheApp
->GetTraits() : NULL
;
264 // let the traits object decide what to do with us
265 traits
->ScheduleForDestroy(this);
267 else // no app or no traits
269 // in wxBase we might have no app object at all, don't leak memory
276 // --------------------------------------------------------------------------
278 // --------------------------------------------------------------------------
280 // The following IO operations update m_error and m_lcount:
281 // {Read, Write, ReadMsg, WriteMsg, Peek, Unread, Discard}
283 // TODO: Should Connect, Accept and AcceptWith update m_error?
285 bool wxSocketBase::Close()
287 // Interrupt pending waits
293 m_socket
->UnsetCallback(
297 GSOCK_CONNECTION_FLAG
300 // Shutdown the connection
301 m_socket
->Shutdown();
305 m_establishing
= false;
309 wxSocketBase
& wxSocketBase::Read(void* buffer
, wxUint32 nbytes
)
314 m_lcount
= _Read(buffer
, nbytes
);
316 // If in wxSOCKET_WAITALL mode, all bytes should have been read.
317 if (m_flags
& wxSOCKET_WAITALL
)
318 m_error
= (m_lcount
!= nbytes
);
320 m_error
= (m_lcount
== 0);
322 // Allow read events from now on
328 wxUint32
wxSocketBase::_Read(void* buffer_
, wxUint32 nbytes
)
330 char *buffer
= (char *)buffer_
;
334 // Try the pushback buffer first
335 total
= GetPushback(buffer
, nbytes
, false);
337 buffer
= (char *)buffer
+ total
;
339 // Return now in one of the following cases:
340 // - the socket is invalid,
341 // - we got all the data
346 // Possible combinations (they are checked in this order)
348 // wxSOCKET_WAITALL (with or without wxSOCKET_BLOCK)
353 if (m_flags
& wxSOCKET_NOWAIT
)
355 m_socket
->SetNonBlocking(1);
356 ret
= m_socket
->Read(buffer
, nbytes
);
357 m_socket
->SetNonBlocking(0);
364 else // blocking socket
368 // dispatch events unless disabled
369 if ( !(m_flags
& wxSOCKET_BLOCK
) && !WaitForRead() )
372 ret
= m_socket
->Read(buffer
, nbytes
);
375 // for connection-oriented (e.g. TCP) sockets we can only read
376 // 0 bytes if the other end has been closed, and for
377 // connectionless ones (UDP) this flag doesn't make sense
378 // anyhow so we can set it to true too without doing any harm
385 // this will be always interpreted as error by Read()
391 // if wxSOCKET_WAITALL is not set, we can leave now as we did read
393 if ( !(m_flags
& wxSOCKET_WAITALL
) )
396 // otherwise check if read the maximal requested amount of data
401 // we didn't, so continue reading
402 buffer
= (char *)buffer
+ ret
;
409 wxSocketBase
& wxSocketBase::ReadMsg(void* buffer
, wxUint32 nbytes
)
411 wxUint32 len
, len2
, sig
, total
;
416 unsigned char sig
[4];
417 unsigned char len
[4];
426 SetFlags((m_flags
& wxSOCKET_BLOCK
) | wxSOCKET_WAITALL
);
428 if (_Read(&msg
, sizeof(msg
)) != sizeof(msg
))
431 sig
= (wxUint32
)msg
.sig
[0];
432 sig
|= (wxUint32
)(msg
.sig
[1] << 8);
433 sig
|= (wxUint32
)(msg
.sig
[2] << 16);
434 sig
|= (wxUint32
)(msg
.sig
[3] << 24);
436 if (sig
!= 0xfeeddead)
438 wxLogWarning(_("wxSocket: invalid signature in ReadMsg."));
442 len
= (wxUint32
)msg
.len
[0];
443 len
|= (wxUint32
)(msg
.len
[1] << 8);
444 len
|= (wxUint32
)(msg
.len
[2] << 16);
445 len
|= (wxUint32
)(msg
.len
[3] << 24);
455 // Don't attempt to read if the msg was zero bytes long.
458 total
= _Read(buffer
, len
);
466 char *discard_buffer
= new char[MAX_DISCARD_SIZE
];
469 // NOTE: discarded bytes don't add to m_lcount.
472 discard_len
= ((len2
> MAX_DISCARD_SIZE
)? MAX_DISCARD_SIZE
: len2
);
473 discard_len
= _Read(discard_buffer
, (wxUint32
)discard_len
);
474 len2
-= (wxUint32
)discard_len
;
476 while ((discard_len
> 0) && len2
);
478 delete [] discard_buffer
;
483 if (_Read(&msg
, sizeof(msg
)) != sizeof(msg
))
486 sig
= (wxUint32
)msg
.sig
[0];
487 sig
|= (wxUint32
)(msg
.sig
[1] << 8);
488 sig
|= (wxUint32
)(msg
.sig
[2] << 16);
489 sig
|= (wxUint32
)(msg
.sig
[3] << 24);
491 if (sig
!= 0xdeadfeed)
493 wxLogWarning(_("wxSocket: invalid signature in ReadMsg."));
509 wxSocketBase
& wxSocketBase::Peek(void* buffer
, wxUint32 nbytes
)
514 m_lcount
= _Read(buffer
, nbytes
);
515 Pushback(buffer
, m_lcount
);
517 // If in wxSOCKET_WAITALL mode, all bytes should have been read.
518 if (m_flags
& wxSOCKET_WAITALL
)
519 m_error
= (m_lcount
!= nbytes
);
521 m_error
= (m_lcount
== 0);
523 // Allow read events again
529 wxSocketBase
& wxSocketBase::Write(const void *buffer
, wxUint32 nbytes
)
534 m_lcount
= _Write(buffer
, nbytes
);
536 // If in wxSOCKET_WAITALL mode, all bytes should have been written.
537 if (m_flags
& wxSOCKET_WAITALL
)
538 m_error
= (m_lcount
!= nbytes
);
540 m_error
= (m_lcount
== 0);
542 // Allow write events again
548 wxUint32
wxSocketBase::_Write(const void *buffer_
, wxUint32 nbytes
)
550 const char *buffer
= (const char *)buffer_
;
554 // If the socket is invalid or parameters are ill, return immediately
555 if (!m_socket
|| !buffer
|| !nbytes
)
558 // Possible combinations (they are checked in this order)
560 // wxSOCKET_WAITALL (with or without wxSOCKET_BLOCK)
565 if (m_flags
& wxSOCKET_NOWAIT
)
567 m_socket
->SetNonBlocking(1);
568 ret
= m_socket
->Write(buffer
, nbytes
);
569 m_socket
->SetNonBlocking(0);
574 else // blocking socket
578 if ( !(m_flags
& wxSOCKET_BLOCK
) && !WaitForWrite() )
581 ret
= m_socket
->Write(buffer
, nbytes
);
583 // see comments for similar logic for ret handling in _Read()
596 if ( !(m_flags
& wxSOCKET_WAITALL
) )
603 buffer
= (const char *)buffer
+ ret
;
610 wxSocketBase
& wxSocketBase::WriteMsg(const void *buffer
, wxUint32 nbytes
)
616 unsigned char sig
[4];
617 unsigned char len
[4];
625 SetFlags((m_flags
& wxSOCKET_BLOCK
) | wxSOCKET_WAITALL
);
627 msg
.sig
[0] = (unsigned char) 0xad;
628 msg
.sig
[1] = (unsigned char) 0xde;
629 msg
.sig
[2] = (unsigned char) 0xed;
630 msg
.sig
[3] = (unsigned char) 0xfe;
632 msg
.len
[0] = (unsigned char) (nbytes
& 0xff);
633 msg
.len
[1] = (unsigned char) ((nbytes
>> 8) & 0xff);
634 msg
.len
[2] = (unsigned char) ((nbytes
>> 16) & 0xff);
635 msg
.len
[3] = (unsigned char) ((nbytes
>> 24) & 0xff);
637 if (_Write(&msg
, sizeof(msg
)) < sizeof(msg
))
640 total
= _Write(buffer
, nbytes
);
645 msg
.sig
[0] = (unsigned char) 0xed;
646 msg
.sig
[1] = (unsigned char) 0xfe;
647 msg
.sig
[2] = (unsigned char) 0xad;
648 msg
.sig
[3] = (unsigned char) 0xde;
652 msg
.len
[3] = (char) 0;
654 if ((_Write(&msg
, sizeof(msg
))) < sizeof(msg
))
668 wxSocketBase
& wxSocketBase::Unread(const void *buffer
, wxUint32 nbytes
)
671 Pushback(buffer
, nbytes
);
679 wxSocketBase
& wxSocketBase::Discard()
681 char *buffer
= new char[MAX_DISCARD_SIZE
];
688 SetFlags(wxSOCKET_NOWAIT
);
692 ret
= _Read(buffer
, MAX_DISCARD_SIZE
);
695 while (ret
== MAX_DISCARD_SIZE
);
701 // Allow read events again
707 // --------------------------------------------------------------------------
709 // --------------------------------------------------------------------------
711 // All Wait functions poll the socket using GSocket_Select() to
712 // check for the specified combination of conditions, until one
713 // of these conditions become true, an error occurs, or the
714 // timeout elapses. The polling loop runs the event loop so that
715 // this won't block the GUI.
717 bool wxSocketBase::_Wait(long seconds
,
719 wxSocketEventFlags flags
)
721 GSocketEventFlags result
;
722 long timeout
; // in ms
724 // Set this to true to interrupt ongoing waits
727 // Check for valid socket
731 // Check for valid timeout value.
733 timeout
= seconds
* 1000 + milliseconds
;
735 timeout
= m_timeout
* 1000;
737 // Get the active event loop
738 wxEventLoopBase
* const eventLoop
= wxEventLoop::GetActive();
741 wxASSERT_MSG( !wxIsMainThread() || eventLoop
,
742 "Sockets won't work without a running event loop" );
745 // Wait in an active polling loop.
747 // NOTE: We duplicate some of the code in OnRequest, but this doesn't
748 // hurt. It has to be here because the (GSocket) event might arrive
749 // a bit delayed, and it has to be in OnRequest as well because we
750 // don't know whether the Wait functions are being used.
752 // Do this at least once (important if timeout == 0, when
753 // we are just polling). Also, if just polling, do not yield.
755 const wxMilliClock_t time_limit
= wxGetLocalTimeMillis() + timeout
;
757 bool valid_result
= false;
761 // This is used to avoid a busy loop on wxBase - having a select
762 // timeout of 50 ms per iteration should be enough.
764 m_socket
->SetTimeout(50);
766 m_socket
->SetTimeout(timeout
);
771 result
= m_socket
->Select(flags
| GSOCK_LOST_FLAG
);
773 // Incoming connection (server) or connection established (client)
774 if (result
& GSOCK_CONNECTION_FLAG
)
777 m_establishing
= false;
782 // Data available or output buffer ready
783 if ((result
& GSOCK_INPUT_FLAG
) || (result
& GSOCK_OUTPUT_FLAG
))
790 if (result
& GSOCK_LOST_FLAG
)
793 m_establishing
= false;
794 valid_result
= ((flags
& GSOCK_LOST_FLAG
) != 0);
799 long time_left
= wxMilliClockToLong(time_limit
- wxGetLocalTimeMillis());
800 if ((!timeout
) || (time_left
<= 0) || (m_interrupt
))
806 // from the main thread itself we have to run the event loop to
807 // let the events (including the GUI events and the low-level
808 // (not wxWidgets) events from GSocket) be processed but from
809 // another thread it is enough to just call wxThread::Yield()
810 // which will give away the rest of our time slice: the
811 // explanation is that the events will be processed by the main
812 // thread anyhow, but we don't want to eat the CPU time
813 // uselessly while sitting in the loop waiting for the data
814 if ( wxIsMainThread() )
816 if ( eventLoop
->Pending() )
817 eventLoop
->Dispatch();
822 #endif // wxUSE_THREADS
826 // If there's less than 50 ms left, just call select with that timeout.
828 m_socket
->SetTimeout(time_left
);
833 // Set timeout back to original value (we overwrote it for polling)
835 m_socket
->SetTimeout(m_timeout
*1000);
840 bool wxSocketBase::Wait(long seconds
, long milliseconds
)
842 return _Wait(seconds
, milliseconds
, GSOCK_INPUT_FLAG
|
844 GSOCK_CONNECTION_FLAG
|
848 bool wxSocketBase::WaitForRead(long seconds
, long milliseconds
)
850 // Check pushback buffer before entering _Wait
854 // Note that GSOCK_INPUT_LOST has to be explicitly passed to
855 // _Wait because of the semantics of WaitForRead: a return
856 // value of true means that a GSocket_Read call will return
857 // immediately, not that there is actually data to read.
859 return _Wait(seconds
, milliseconds
, GSOCK_INPUT_FLAG
| GSOCK_LOST_FLAG
);
863 bool wxSocketBase::WaitForWrite(long seconds
, long milliseconds
)
865 return _Wait(seconds
, milliseconds
, GSOCK_OUTPUT_FLAG
);
868 bool wxSocketBase::WaitForLost(long seconds
, long milliseconds
)
870 return _Wait(seconds
, milliseconds
, GSOCK_LOST_FLAG
);
873 // --------------------------------------------------------------------------
875 // --------------------------------------------------------------------------
878 // Get local or peer address
881 bool wxSocketBase::GetPeer(wxSockAddress
& addr_man
) const
888 peer
= m_socket
->GetPeer();
890 // copying a null address would just trigger an assert anyway
895 addr_man
.SetAddress(peer
);
896 GAddress_destroy(peer
);
901 bool wxSocketBase::GetLocal(wxSockAddress
& addr_man
) const
908 local
= m_socket
->GetLocal();
909 addr_man
.SetAddress(local
);
910 GAddress_destroy(local
);
916 // Save and restore socket state
919 void wxSocketBase::SaveState()
921 wxSocketState
*state
;
923 state
= new wxSocketState();
925 state
->m_flags
= m_flags
;
926 state
->m_notify
= m_notify
;
927 state
->m_eventmask
= m_eventmask
;
928 state
->m_clientData
= m_clientData
;
930 m_states
.Append(state
);
933 void wxSocketBase::RestoreState()
935 wxList::compatibility_iterator node
;
936 wxSocketState
*state
;
938 node
= m_states
.GetLast();
942 state
= (wxSocketState
*)node
->GetData();
944 m_flags
= state
->m_flags
;
945 m_notify
= state
->m_notify
;
946 m_eventmask
= state
->m_eventmask
;
947 m_clientData
= state
->m_clientData
;
949 m_states
.Erase(node
);
957 void wxSocketBase::SetTimeout(long seconds
)
962 m_socket
->SetTimeout(m_timeout
* 1000);
965 void wxSocketBase::SetFlags(wxSocketFlags flags
)
971 // --------------------------------------------------------------------------
973 // --------------------------------------------------------------------------
975 // A note on how events are processed, which is probably the most
976 // difficult thing to get working right while keeping the same API
977 // and functionality for all platforms.
979 // When GSocket detects an event, it calls wx_socket_callback, which in
980 // turn just calls wxSocketBase::OnRequest in the corresponding wxSocket
981 // object. OnRequest does some housekeeping, and if the event is to be
982 // propagated to the user, it creates a new wxSocketEvent object and
983 // posts it. The event is not processed immediately, but delayed with
984 // AddPendingEvent instead. This is necessary in order to decouple the
985 // event processing from wx_socket_callback; otherwise, subsequent IO
986 // calls made from the user event handler would fail, as gtk callbacks
987 // are not reentrant.
989 // Note that, unlike events, user callbacks (now deprecated) are _not_
990 // decoupled from wx_socket_callback and thus they suffer from a variety
991 // of problems. Avoid them where possible and use events instead.
994 void LINKAGEMODE
wx_socket_callback(GSocket
* WXUNUSED(socket
),
995 GSocketEvent notification
,
998 wxSocketBase
*sckobj
= (wxSocketBase
*)cdata
;
1000 sckobj
->OnRequest((wxSocketNotify
) notification
);
1003 void wxSocketBase::OnRequest(wxSocketNotify notification
)
1005 // NOTE: We duplicate some of the code in _Wait, but this doesn't
1006 // hurt. It has to be here because the (GSocket) event might arrive
1007 // a bit delayed, and it has to be in _Wait as well because we don't
1008 // know whether the Wait functions are being used.
1010 switch(notification
)
1012 case wxSOCKET_CONNECTION
:
1013 m_establishing
= false;
1017 // If we are in the middle of a R/W operation, do not
1018 // propagate events to users. Also, filter 'late' events
1019 // which are no longer valid.
1021 case wxSOCKET_INPUT
:
1022 if (m_reading
|| !m_socket
->Select(GSOCK_INPUT_FLAG
))
1026 case wxSOCKET_OUTPUT
:
1027 if (m_writing
|| !m_socket
->Select(GSOCK_OUTPUT_FLAG
))
1032 m_connected
= false;
1033 m_establishing
= false;
1037 // Schedule the event
1039 wxSocketEventFlags flag
= 0;
1041 switch (notification
)
1043 case GSOCK_INPUT
: flag
= GSOCK_INPUT_FLAG
; break;
1044 case GSOCK_OUTPUT
: flag
= GSOCK_OUTPUT_FLAG
; break;
1045 case GSOCK_CONNECTION
: flag
= GSOCK_CONNECTION_FLAG
; break;
1046 case GSOCK_LOST
: flag
= GSOCK_LOST_FLAG
; break;
1048 wxLogWarning(_("wxSocket: unknown event!."));
1052 if (((m_eventmask
& flag
) == flag
) && m_notify
)
1056 wxSocketEvent
event(m_id
);
1057 event
.m_event
= notification
;
1058 event
.m_clientData
= m_clientData
;
1059 event
.SetEventObject(this);
1061 m_handler
->AddPendingEvent(event
);
1066 void wxSocketBase::Notify(bool notify
)
1070 m_socket
->Notify(notify
);
1073 void wxSocketBase::SetNotify(wxSocketEventFlags flags
)
1075 m_eventmask
= flags
;
1078 void wxSocketBase::SetEventHandler(wxEvtHandler
& handler
, int id
)
1080 m_handler
= &handler
;
1084 // --------------------------------------------------------------------------
1086 // --------------------------------------------------------------------------
1088 void wxSocketBase::Pushback(const void *buffer
, wxUint32 size
)
1092 if (m_unread
== NULL
)
1093 m_unread
= malloc(size
);
1098 tmp
= malloc(m_unrd_size
+ size
);
1099 memcpy((char *)tmp
+ size
, m_unread
, m_unrd_size
);
1105 m_unrd_size
+= size
;
1107 memcpy(m_unread
, buffer
, size
);
1110 wxUint32
wxSocketBase::GetPushback(void *buffer
, wxUint32 size
, bool peek
)
1115 if (size
> (m_unrd_size
-m_unrd_cur
))
1116 size
= m_unrd_size
-m_unrd_cur
;
1118 memcpy(buffer
, (char *)m_unread
+ m_unrd_cur
, size
);
1123 if (m_unrd_size
== m_unrd_cur
)
1136 // ==========================================================================
1138 // ==========================================================================
1140 // --------------------------------------------------------------------------
1142 // --------------------------------------------------------------------------
1144 wxSocketServer::wxSocketServer(const wxSockAddress
& addr_man
,
1145 wxSocketFlags flags
)
1146 : wxSocketBase(flags
, wxSOCKET_SERVER
)
1148 wxLogTrace( wxTRACE_Socket
, _T("Opening wxSocketServer") );
1150 m_socket
= GSocket_new();
1154 wxLogTrace( wxTRACE_Socket
, _T("*** GSocket_new failed") );
1158 // Setup the socket as server
1159 m_socket
->Notify(m_notify
);
1160 m_socket
->SetLocal(addr_man
.GetAddress());
1162 if (GetFlags() & wxSOCKET_REUSEADDR
) {
1163 m_socket
->SetReusable();
1165 if (GetFlags() & wxSOCKET_BROADCAST
) {
1166 m_socket
->SetBroadcast();
1168 if (GetFlags() & wxSOCKET_NOBIND
) {
1169 m_socket
->DontDoBind();
1172 if (m_socket
->SetServer() != GSOCK_NOERROR
)
1177 wxLogTrace( wxTRACE_Socket
, _T("*** GSocket_SetServer failed") );
1181 m_socket
->SetTimeout(m_timeout
* 1000);
1182 m_socket
->SetCallback(GSOCK_INPUT_FLAG
| GSOCK_OUTPUT_FLAG
|
1183 GSOCK_LOST_FLAG
| GSOCK_CONNECTION_FLAG
,
1184 wx_socket_callback
, (char *)this);
1186 wxLogTrace( wxTRACE_Socket
, _T("wxSocketServer on fd %d"), m_socket
->m_fd
);
1189 // --------------------------------------------------------------------------
1191 // --------------------------------------------------------------------------
1193 bool wxSocketServer::AcceptWith(wxSocketBase
& sock
, bool wait
)
1195 GSocket
*child_socket
;
1200 // If wait == false, then the call should be nonblocking.
1201 // When we are finished, we put the socket to blocking mode
1205 m_socket
->SetNonBlocking(1);
1207 child_socket
= m_socket
->WaitConnection();
1210 m_socket
->SetNonBlocking(0);
1215 sock
.m_type
= wxSOCKET_BASE
;
1216 sock
.m_socket
= child_socket
;
1217 sock
.m_connected
= true;
1219 sock
.m_socket
->SetTimeout(sock
.m_timeout
* 1000);
1220 sock
.m_socket
->SetCallback(GSOCK_INPUT_FLAG
| GSOCK_OUTPUT_FLAG
|
1221 GSOCK_LOST_FLAG
| GSOCK_CONNECTION_FLAG
,
1222 wx_socket_callback
, (char *)&sock
);
1227 wxSocketBase
*wxSocketServer::Accept(bool wait
)
1229 wxSocketBase
* sock
= new wxSocketBase();
1231 sock
->SetFlags(m_flags
);
1233 if (!AcceptWith(*sock
, wait
))
1242 bool wxSocketServer::WaitForAccept(long seconds
, long milliseconds
)
1244 return _Wait(seconds
, milliseconds
, GSOCK_CONNECTION_FLAG
);
1247 bool wxSocketBase::GetOption(int level
, int optname
, void *optval
, int *optlen
)
1249 wxASSERT_MSG( m_socket
, _T("Socket not initialised") );
1251 if (m_socket
->GetSockOpt(level
, optname
, optval
, optlen
)
1259 bool wxSocketBase::SetOption(int level
, int optname
, const void *optval
,
1262 wxASSERT_MSG( m_socket
, _T("Socket not initialised") );
1264 if (m_socket
->SetSockOpt(level
, optname
, optval
, optlen
)
1272 bool wxSocketBase::SetLocal(const wxIPV4address
& local
)
1274 GAddress
* la
= local
.GetAddress();
1276 // If the address is valid, save it for use when we call Connect
1277 if (la
&& la
->m_addr
)
1279 m_localAddress
= local
;
1287 // ==========================================================================
1289 // ==========================================================================
1291 // --------------------------------------------------------------------------
1293 // --------------------------------------------------------------------------
1295 wxSocketClient::wxSocketClient(wxSocketFlags flags
)
1296 : wxSocketBase(flags
, wxSOCKET_CLIENT
)
1298 m_initialRecvBufferSize
=
1299 m_initialSendBufferSize
= -1;
1302 wxSocketClient::~wxSocketClient()
1306 // --------------------------------------------------------------------------
1308 // --------------------------------------------------------------------------
1310 bool wxSocketClient::DoConnect(const wxSockAddress
& addr_man
,
1311 const wxSockAddress
* local
,
1318 // Shutdown and destroy the socket
1323 m_socket
= GSocket_new();
1324 m_connected
= false;
1325 m_establishing
= false;
1330 m_socket
->SetTimeout(m_timeout
* 1000);
1331 m_socket
->SetCallback(
1335 GSOCK_CONNECTION_FLAG
,
1340 // If wait == false, then the call should be nonblocking. When we are
1341 // finished, we put the socket to blocking mode again.
1344 m_socket
->SetNonBlocking(1);
1346 // Reuse makes sense for clients too, if we are trying to rebind to the same port
1347 if (GetFlags() & wxSOCKET_REUSEADDR
)
1349 m_socket
->SetReusable();
1351 if (GetFlags() & wxSOCKET_BROADCAST
)
1353 m_socket
->SetBroadcast();
1355 if (GetFlags() & wxSOCKET_NOBIND
)
1357 m_socket
->DontDoBind();
1360 // If no local address was passed and one has been set, use the one that was Set
1361 if (!local
&& m_localAddress
.GetAddress())
1363 local
= &m_localAddress
;
1366 // Bind to the local IP address and port, when provided
1369 GAddress
* la
= local
->GetAddress();
1371 if (la
&& la
->m_addr
)
1372 m_socket
->SetLocal(la
);
1375 #if defined(__WXMSW__) || defined(__WXGTK__)
1376 m_socket
->SetInitialSocketBuffers(m_initialRecvBufferSize
, m_initialSendBufferSize
);
1379 m_socket
->SetPeer(addr_man
.GetAddress());
1380 err
= m_socket
->Connect(GSOCK_STREAMED
);
1382 //this will register for callbacks - must be called after m_socket->m_fd was initialized
1383 m_socket
->Notify(m_notify
);
1386 m_socket
->SetNonBlocking(0);
1388 if (err
!= GSOCK_NOERROR
)
1390 if (err
== GSOCK_WOULDBLOCK
)
1391 m_establishing
= true;
1400 bool wxSocketClient::Connect(const wxSockAddress
& addr_man
, bool wait
)
1402 return DoConnect(addr_man
, NULL
, wait
);
1405 bool wxSocketClient::Connect(const wxSockAddress
& addr_man
,
1406 const wxSockAddress
& local
,
1409 return DoConnect(addr_man
, &local
, wait
);
1412 bool wxSocketClient::WaitOnConnect(long seconds
, long milliseconds
)
1414 if (m_connected
) // Already connected
1417 if (!m_establishing
|| !m_socket
) // No connection in progress
1420 return _Wait(seconds
, milliseconds
, GSOCK_CONNECTION_FLAG
|
1424 // ==========================================================================
1426 // ==========================================================================
1428 /* NOTE: experimental stuff - might change */
1430 wxDatagramSocket::wxDatagramSocket( const wxSockAddress
& addr
,
1431 wxSocketFlags flags
)
1432 : wxSocketBase( flags
, wxSOCKET_DATAGRAM
)
1434 // Create the socket
1435 m_socket
= GSocket_new();
1439 wxFAIL_MSG( _T("datagram socket not new'd") );
1442 m_socket
->Notify(m_notify
);
1443 // Setup the socket as non connection oriented
1444 m_socket
->SetLocal(addr
.GetAddress());
1445 if (flags
& wxSOCKET_REUSEADDR
)
1447 m_socket
->SetReusable();
1449 if (GetFlags() & wxSOCKET_BROADCAST
)
1451 m_socket
->SetBroadcast();
1453 if (GetFlags() & wxSOCKET_NOBIND
)
1455 m_socket
->DontDoBind();
1457 if ( m_socket
->SetNonOriented() != GSOCK_NOERROR
)
1464 // Initialize all stuff
1465 m_connected
= false;
1466 m_establishing
= false;
1467 m_socket
->SetTimeout( m_timeout
);
1468 m_socket
->SetCallback( GSOCK_INPUT_FLAG
| GSOCK_OUTPUT_FLAG
|
1469 GSOCK_LOST_FLAG
| GSOCK_CONNECTION_FLAG
,
1470 wx_socket_callback
, (char*)this );
1473 wxDatagramSocket
& wxDatagramSocket::RecvFrom( wxSockAddress
& addr
,
1482 wxDatagramSocket
& wxDatagramSocket::SendTo( const wxSockAddress
& addr
,
1486 wxASSERT_MSG( m_socket
, _T("Socket not initialised") );
1488 m_socket
->SetPeer(addr
.GetAddress());
1493 // ==========================================================================
1495 // ==========================================================================
1497 class wxSocketModule
: public wxModule
1500 virtual bool OnInit()
1502 // wxSocketBase will call GSocket_Init() itself when/if needed
1506 virtual void OnExit()
1508 if ( wxSocketBase::IsInitialized() )
1509 wxSocketBase::Shutdown();
1513 DECLARE_DYNAMIC_CLASS(wxSocketModule
)
1516 IMPLEMENT_DYNAMIC_CLASS(wxSocketModule
, wxModule
)
1518 #endif // wxUSE_SOCKETS