]>
Commit | Line | Data |
---|---|---|
1 | ///////////////////////////////////////////////////////////////////////////// | |
2 | // Name: src/unix/sockunix.cpp | |
3 | // Purpose: wxSocketImpl implementation for Unix systems | |
4 | // Authors: Guilhem Lavaux, Guillermo Rodriguez Garcia, David Elliott, | |
5 | // Vadim Zeitlin | |
6 | // Created: April 1997 | |
7 | // Copyright: (c) 1997 Guilhem Lavaux | |
8 | // (c) 2008 Vadim Zeitlin | |
9 | // Licence: wxWindows licence | |
10 | ///////////////////////////////////////////////////////////////////////////// | |
11 | ||
12 | ||
13 | #include "wx/wxprec.h" | |
14 | ||
15 | #if wxUSE_SOCKETS | |
16 | ||
17 | #include "wx/private/fd.h" | |
18 | #include "wx/private/socket.h" | |
19 | #include "wx/unix/private/sockunix.h" | |
20 | ||
21 | #include <errno.h> | |
22 | ||
23 | #if defined(__WATCOMC__) | |
24 | #include <nerrno.h> | |
25 | #endif | |
26 | ||
27 | #include <sys/types.h> | |
28 | ||
29 | #ifdef HAVE_SYS_SELECT_H | |
30 | # include <sys/select.h> | |
31 | #endif | |
32 | ||
33 | #ifdef __EMX__ | |
34 | #include <sys/select.h> | |
35 | #endif | |
36 | ||
37 | #ifndef WX_SOCKLEN_T | |
38 | ||
39 | #ifdef VMS | |
40 | # define WX_SOCKLEN_T unsigned int | |
41 | #else | |
42 | # ifdef __GLIBC__ | |
43 | # if __GLIBC__ == 2 | |
44 | # define WX_SOCKLEN_T socklen_t | |
45 | # endif | |
46 | # elif defined(__WXMAC__) | |
47 | # define WX_SOCKLEN_T socklen_t | |
48 | # else | |
49 | # define WX_SOCKLEN_T int | |
50 | # endif | |
51 | #endif | |
52 | ||
53 | #endif /* SOCKLEN_T */ | |
54 | ||
55 | #ifndef SOCKOPTLEN_T | |
56 | #define SOCKOPTLEN_T WX_SOCKLEN_T | |
57 | #endif | |
58 | ||
59 | // UnixWare reportedly needs this for FIONBIO definition | |
60 | #ifdef __UNIXWARE__ | |
61 | #include <sys/filio.h> | |
62 | #endif | |
63 | ||
64 | // ============================================================================ | |
65 | // wxSocketImpl implementation | |
66 | // ============================================================================ | |
67 | ||
68 | wxSocketError wxSocketImplUnix::GetLastError() const | |
69 | { | |
70 | switch ( errno ) | |
71 | { | |
72 | case 0: | |
73 | return wxSOCKET_NOERROR; | |
74 | ||
75 | case ENOTSOCK: | |
76 | return wxSOCKET_INVSOCK; | |
77 | ||
78 | // unfortunately EAGAIN only has the "would block" meaning for read(), | |
79 | // not for connect() for which it means something rather different but | |
80 | // we can't distinguish between these two situations currently... | |
81 | // | |
82 | // also notice that EWOULDBLOCK can be different from EAGAIN on some | |
83 | // systems (HP-UX being the only known example) while it's defined as | |
84 | // EAGAIN on most others (e.g. Linux) | |
85 | case EAGAIN: | |
86 | #ifdef EWOULDBLOCK | |
87 | #if EWOULDBLOCK != EAGAIN | |
88 | case EWOULDBLOCK: | |
89 | #endif | |
90 | #endif // EWOULDBLOCK | |
91 | case EINPROGRESS: | |
92 | return wxSOCKET_WOULDBLOCK; | |
93 | ||
94 | default: | |
95 | return wxSOCKET_IOERR; | |
96 | } | |
97 | } | |
98 | ||
99 | void wxSocketImplUnix::DoEnableEvents(int flags, bool enable) | |
100 | { | |
101 | wxSocketManager * const manager = wxSocketManager::Get(); | |
102 | if (!manager) | |
103 | return; | |
104 | ||
105 | if ( enable ) | |
106 | { | |
107 | if ( flags & wxSOCKET_INPUT_FLAG ) | |
108 | manager->Install_Callback(this, wxSOCKET_INPUT); | |
109 | if ( flags & wxSOCKET_OUTPUT_FLAG ) | |
110 | manager->Install_Callback(this, wxSOCKET_OUTPUT); | |
111 | } | |
112 | else // off | |
113 | { | |
114 | if ( flags & wxSOCKET_INPUT_FLAG ) | |
115 | manager->Uninstall_Callback(this, wxSOCKET_INPUT); | |
116 | if ( flags & wxSOCKET_OUTPUT_FLAG ) | |
117 | manager->Uninstall_Callback(this, wxSOCKET_OUTPUT); | |
118 | } | |
119 | } | |
120 | ||
121 | int wxSocketImplUnix::CheckForInput() | |
122 | { | |
123 | char c; | |
124 | int rc; | |
125 | do | |
126 | { | |
127 | rc = recv(m_fd, &c, 1, MSG_PEEK); | |
128 | } while ( rc == -1 && errno == EINTR ); | |
129 | ||
130 | return rc; | |
131 | } | |
132 | ||
133 | void wxSocketImplUnix::OnStateChange(wxSocketNotify event) | |
134 | { | |
135 | NotifyOnStateChange(event); | |
136 | ||
137 | if ( event == wxSOCKET_LOST ) | |
138 | Shutdown(); | |
139 | } | |
140 | ||
141 | void wxSocketImplUnix::OnReadWaiting() | |
142 | { | |
143 | wxASSERT_MSG( m_fd != INVALID_SOCKET, "invalid socket ready for reading?" ); | |
144 | ||
145 | // we need to disable the read notifications until we read all the data | |
146 | // already available for the socket, otherwise we're going to keep getting | |
147 | // them continuously which is worse than inefficient: as IO notifications | |
148 | // have higher priority than idle events in e.g. GTK+, our pending events | |
149 | // whose handlers typically call Read() which would consume the data and so | |
150 | // stop the notifications flood would never be dispatched at all if the | |
151 | // notifications were not disabled | |
152 | DisableEvents(wxSOCKET_INPUT_FLAG); | |
153 | ||
154 | ||
155 | // find out what are we going to notify about exactly | |
156 | wxSocketNotify notify; | |
157 | ||
158 | // TCP listening sockets become ready for reading when there is a pending | |
159 | // connection | |
160 | if ( m_server && m_stream ) | |
161 | { | |
162 | notify = wxSOCKET_CONNECTION; | |
163 | } | |
164 | else // check if there is really any input available | |
165 | { | |
166 | switch ( CheckForInput() ) | |
167 | { | |
168 | case 1: | |
169 | notify = wxSOCKET_INPUT; | |
170 | break; | |
171 | ||
172 | case 0: | |
173 | // reading 0 bytes for a TCP socket means that the connection | |
174 | // was closed by peer but for UDP it just means that we got an | |
175 | // empty datagram | |
176 | notify = m_stream ? wxSOCKET_LOST : wxSOCKET_INPUT; | |
177 | break; | |
178 | ||
179 | default: | |
180 | wxFAIL_MSG( "unexpected CheckForInput() return value" ); | |
181 | // fall through | |
182 | ||
183 | case -1: | |
184 | if ( GetLastError() == wxSOCKET_WOULDBLOCK ) | |
185 | { | |
186 | // just a spurious wake up | |
187 | EnableEvents(wxSOCKET_INPUT_FLAG); | |
188 | return; | |
189 | } | |
190 | ||
191 | notify = wxSOCKET_LOST; | |
192 | } | |
193 | } | |
194 | ||
195 | OnStateChange(notify); | |
196 | } | |
197 | ||
198 | void wxSocketImplUnix::OnWriteWaiting() | |
199 | { | |
200 | wxASSERT_MSG( m_fd != INVALID_SOCKET, "invalid socket ready for writing?" ); | |
201 | ||
202 | // see comment in the beginning of OnReadWaiting() above | |
203 | DisableEvents(wxSOCKET_OUTPUT_FLAG); | |
204 | ||
205 | ||
206 | // check whether this is a notification for the completion of a | |
207 | // non-blocking connect() | |
208 | if ( m_establishing && !m_server ) | |
209 | { | |
210 | m_establishing = false; | |
211 | ||
212 | // check whether we connected successfully | |
213 | int error; | |
214 | SOCKOPTLEN_T len = sizeof(error); | |
215 | ||
216 | getsockopt(m_fd, SOL_SOCKET, SO_ERROR, (char*)&error, &len); | |
217 | ||
218 | if ( error ) | |
219 | { | |
220 | OnStateChange(wxSOCKET_LOST); | |
221 | return; | |
222 | } | |
223 | ||
224 | OnStateChange(wxSOCKET_CONNECTION); | |
225 | } | |
226 | ||
227 | OnStateChange(wxSOCKET_OUTPUT); | |
228 | } | |
229 | ||
230 | void wxSocketImplUnix::OnExceptionWaiting() | |
231 | { | |
232 | // when using epoll() this is called when an error occurred on the socket | |
233 | // so close it if it hadn't been done yet -- what else can we do? | |
234 | // | |
235 | // notice that we shouldn't be called at all when using select() as we | |
236 | // don't use wxFDIO_EXCEPTION when registering the socket for monitoring | |
237 | // and this is good because select() would call this for any OOB data which | |
238 | // is not necessarily an error | |
239 | if ( m_fd != INVALID_SOCKET ) | |
240 | OnStateChange(wxSOCKET_LOST); | |
241 | } | |
242 | ||
243 | #endif /* wxUSE_SOCKETS */ |