Commit | Line | Data |
---|---|---|
f4ada568 | 1 | ///////////////////////////////////////////////////////////////////////////// |
d5da0ce7 | 2 | // Name: src/common/sckipc.cpp |
f4ada568 | 3 | // Purpose: Interprocess communication implementation (wxSocket version) |
0834112f | 4 | // Author: Julian Smart |
f4ada568 | 5 | // Modified by: Guilhem Lavaux (big rewrite) May 1997, 1998 |
0834112f | 6 | // Guillermo Rodriguez (updated for wxSocket v2) Jan 2000 |
cdc59bb6 | 7 | // (callbacks deprecated) Mar 2000 |
0dbfd66d | 8 | // Vadim Zeitlin (added support for Unix sockets) Apr 2002 |
8aea37a9 | 9 | // (use buffering, many fixes/cleanup) Oct 2008 |
f4ada568 GL |
10 | // Created: 1993 |
11 | // RCS-ID: $Id$ | |
0834112f GRG |
12 | // Copyright: (c) Julian Smart 1993 |
13 | // (c) Guilhem Lavaux 1997, 1998 | |
14 | // (c) 2000 Guillermo Rodriguez <guille@iies.es> | |
65571936 | 15 | // Licence: wxWindows licence |
f4ada568 GL |
16 | ///////////////////////////////////////////////////////////////////////////// |
17 | ||
cdc59bb6 GRG |
18 | // ========================================================================== |
19 | // declarations | |
20 | // ========================================================================== | |
21 | ||
22 | // -------------------------------------------------------------------------- | |
23 | // headers | |
24 | // -------------------------------------------------------------------------- | |
25 | ||
fcc6dddd JS |
26 | // For compilers that support precompilation, includes "wx.h". |
27 | #include "wx/wxprec.h" | |
f4ada568 | 28 | |
fcc6dddd | 29 | #ifdef __BORLANDC__ |
d5da0ce7 | 30 | #pragma hdrstop |
f4ada568 GL |
31 | #endif |
32 | ||
d5da0ce7 WS |
33 | #if wxUSE_SOCKETS && wxUSE_IPC && wxUSE_STREAMS |
34 | ||
35 | #include "wx/sckipc.h" | |
36 | ||
fcc6dddd | 37 | #ifndef WX_PRECOMP |
d5da0ce7 WS |
38 | #include "wx/log.h" |
39 | #include "wx/event.h" | |
02761f6c | 40 | #include "wx/module.h" |
fcc6dddd JS |
41 | #endif |
42 | ||
43 | #include <stdlib.h> | |
44 | #include <stdio.h> | |
0dbfd66d | 45 | #include <errno.h> |
fcc6dddd | 46 | |
f4ada568 | 47 | #include "wx/socket.h" |
f4ada568 | 48 | |
cdc59bb6 GRG |
49 | // -------------------------------------------------------------------------- |
50 | // macros and constants | |
51 | // -------------------------------------------------------------------------- | |
f4ada568 | 52 | |
8aea37a9 VZ |
53 | namespace |
54 | { | |
55 | ||
f4ada568 | 56 | // Message codes |
8aea37a9 | 57 | enum IPCCode |
cdc59bb6 | 58 | { |
7e73fb9c VZ |
59 | IPC_EXECUTE = 1, |
60 | IPC_REQUEST, | |
61 | IPC_POKE, | |
62 | IPC_ADVISE_START, | |
63 | IPC_ADVISE_REQUEST, | |
64 | IPC_ADVISE, | |
65 | IPC_ADVISE_STOP, | |
66 | IPC_REQUEST_REPLY, | |
67 | IPC_FAIL, | |
68 | IPC_CONNECT, | |
69 | IPC_DISCONNECT | |
f4ada568 | 70 | }; |
f4ada568 | 71 | |
8aea37a9 | 72 | } // anonymous namespace |
d3ea6527 | 73 | |
0dbfd66d VZ |
74 | // headers needed for umask() |
75 | #ifdef __UNIX_LIKE__ | |
76 | #include <sys/types.h> | |
77 | #include <sys/stat.h> | |
78 | #endif // __UNIX_LIKE__ | |
79 | ||
80 | // ---------------------------------------------------------------------------- | |
81 | // private functions | |
82 | // ---------------------------------------------------------------------------- | |
83 | ||
84 | // get the address object for the given server name, the caller must delete it | |
85 | static wxSockAddress * | |
7e73fb9c VZ |
86 | GetAddressFromName(const wxString& serverName, |
87 | const wxString& host = wxEmptyString) | |
0dbfd66d VZ |
88 | { |
89 | // we always use INET sockets under non-Unix systems | |
0ad76eea | 90 | #if defined(__UNIX__) && !defined(__WINDOWS__) && !defined(__WINE__) |
0dbfd66d VZ |
91 | // under Unix, if the server name looks like a path, create a AF_UNIX |
92 | // socket instead of AF_INET one | |
93 | if ( serverName.Find(_T('/')) != wxNOT_FOUND ) | |
94 | { | |
95 | wxUNIXaddress *addr = new wxUNIXaddress; | |
96 | addr->Filename(serverName); | |
97 | ||
98 | return addr; | |
99 | } | |
100 | #endif // Unix/!Unix | |
101 | { | |
102 | wxIPV4address *addr = new wxIPV4address; | |
103 | addr->Service(serverName); | |
104 | if ( !host.empty() ) | |
105 | { | |
106 | addr->Hostname(host); | |
107 | } | |
108 | ||
109 | return addr; | |
110 | } | |
111 | } | |
112 | ||
cdc59bb6 GRG |
113 | // -------------------------------------------------------------------------- |
114 | // wxTCPEventHandler stuff (private class) | |
115 | // -------------------------------------------------------------------------- | |
116 | ||
117 | class wxTCPEventHandler : public wxEvtHandler | |
118 | { | |
119 | public: | |
7e73fb9c | 120 | wxTCPEventHandler() : wxEvtHandler() {} |
cdc59bb6 | 121 | |
7e73fb9c VZ |
122 | void Client_OnRequest(wxSocketEvent& event); |
123 | void Server_OnRequest(wxSocketEvent& event); | |
cdc59bb6 | 124 | |
7e73fb9c VZ |
125 | DECLARE_EVENT_TABLE() |
126 | DECLARE_NO_COPY_CLASS(wxTCPEventHandler) | |
cdc59bb6 GRG |
127 | }; |
128 | ||
129 | enum | |
130 | { | |
7e73fb9c VZ |
131 | _CLIENT_ONREQUEST_ID = 1000, |
132 | _SERVER_ONREQUEST_ID | |
cdc59bb6 GRG |
133 | }; |
134 | ||
135 | static wxTCPEventHandler *gs_handler = NULL; | |
136 | ||
8aea37a9 VZ |
137 | // -------------------------------------------------------------------------- |
138 | // wxIPCSocketStreams | |
139 | // -------------------------------------------------------------------------- | |
140 | ||
141 | #define USE_BUFFER | |
142 | ||
143 | // this class contains the various (related) streams used by wxTCPConnection | |
144 | // and also provides a way to read from the socket stream directly | |
145 | // | |
146 | // for writing to the stream use the IPCOutput class below | |
147 | class wxIPCSocketStreams | |
148 | { | |
149 | public: | |
150 | // ctor initializes all the streams on top of the given socket | |
151 | // | |
152 | // note that we use a bigger than default buffer size which matches the | |
153 | // typical Ethernet MTU | |
154 | wxIPCSocketStreams(wxSocketBase& sock) | |
155 | : m_socketStream(sock), | |
156 | #ifdef USE_BUFFER | |
157 | m_bufferedOut(m_socketStream, 1500), | |
158 | #else | |
159 | m_bufferedOut(m_socketStream), | |
160 | #endif | |
161 | m_dataIn(m_socketStream), | |
162 | m_dataOut(m_bufferedOut) | |
163 | { | |
164 | } | |
165 | ||
166 | // expose the IO methods needed by IPC code (notice that writing is only | |
167 | // done via IPCOutput) | |
168 | ||
169 | // flush output | |
170 | void Flush() | |
171 | { | |
172 | #ifdef USE_BUFFER | |
173 | m_bufferedOut.Sync(); | |
174 | #endif | |
175 | } | |
176 | ||
177 | // simple wrappers around the functions with the same name in | |
178 | // wxDataInputStream | |
179 | wxUint8 Read8() | |
180 | { | |
181 | Flush(); | |
182 | return m_dataIn.Read8(); | |
183 | } | |
184 | ||
185 | wxUint32 Read32() | |
186 | { | |
187 | Flush(); | |
188 | return m_dataIn.Read32(); | |
189 | } | |
190 | ||
191 | wxString ReadString() | |
192 | { | |
193 | Flush(); | |
194 | return m_dataIn.ReadString(); | |
195 | } | |
196 | ||
197 | // read arbitrary (size-prepended) data | |
198 | // | |
199 | // connection parameter is needed to call its GetBufferAtLeast() method | |
200 | void *ReadData(wxConnectionBase *conn, size_t *size) | |
201 | { | |
202 | Flush(); | |
203 | ||
204 | wxCHECK_MSG( conn, NULL, "NULL connection parameter" ); | |
205 | wxCHECK_MSG( size, NULL, "NULL size parameter" ); | |
206 | ||
207 | *size = Read32(); | |
208 | ||
209 | void * const data = conn->GetBufferAtLeast(*size); | |
210 | wxCHECK_MSG( data, NULL, "IPC buffer allocation failed" ); | |
211 | ||
212 | m_socketStream.Read(data, *size); | |
213 | ||
214 | return data; | |
215 | } | |
216 | ||
217 | // same as above but for data preceded by the format | |
218 | void * | |
219 | ReadFormatData(wxConnectionBase *conn, wxIPCFormat *format, size_t *size) | |
220 | { | |
221 | wxCHECK_MSG( format, NULL, "NULL format parameter" ); | |
222 | ||
223 | *format = static_cast<wxIPCFormat>(Read8()); | |
224 | ||
225 | return ReadData(conn, size); | |
226 | } | |
227 | ||
228 | ||
229 | // these methods are only used by IPCOutput and not directly | |
230 | wxDataOutputStream& GetDataOut() { return m_dataOut; } | |
231 | wxOutputStream& GetUnformattedOut() { return m_bufferedOut; } | |
232 | ||
233 | private: | |
234 | // this is the low-level underlying stream using the connection socket | |
235 | wxSocketStream m_socketStream; | |
236 | ||
237 | // the buffered stream is used to avoid writing all pieces of an IPC | |
238 | // request to the socket one by one but to instead do it all at once when | |
239 | // we're done with it | |
240 | #ifdef USE_BUFFER | |
241 | wxBufferedOutputStream m_bufferedOut; | |
242 | #else | |
243 | wxOutputStream& m_bufferedOut; | |
244 | #endif | |
245 | ||
246 | // finally the data streams are used to be able to write typed data into | |
247 | // the above streams easily | |
248 | wxDataInputStream m_dataIn; | |
249 | wxDataOutputStream m_dataOut; | |
250 | ||
251 | DECLARE_NO_COPY_CLASS(wxIPCSocketStreams) | |
252 | }; | |
253 | ||
254 | namespace | |
255 | { | |
256 | ||
257 | // an object of this class should be instantiated on the stack to write to the | |
258 | // underlying socket stream | |
259 | // | |
260 | // this class is intentionally separated from wxIPCSocketStreams to ensure that | |
261 | // Flush() is always called | |
262 | class IPCOutput | |
263 | { | |
264 | public: | |
265 | // construct an object associated with the given streams (which must have | |
266 | // life time greater than ours as we keep a reference to it) | |
267 | IPCOutput(wxIPCSocketStreams *streams) | |
268 | : m_streams(*streams) | |
269 | { | |
270 | wxASSERT_MSG( streams, "NULL streams pointer" ); | |
271 | } | |
272 | ||
273 | // dtor calls Flush() really sending the IPC data to the network | |
274 | ~IPCOutput() { m_streams.Flush(); } | |
275 | ||
276 | ||
277 | // write a byte | |
278 | void Write8(wxUint8 i) | |
279 | { | |
280 | m_streams.GetDataOut().Write8(i); | |
281 | } | |
282 | ||
283 | // write the reply code and a string | |
284 | void Write(IPCCode code, const wxString& str) | |
285 | { | |
286 | Write8(code); | |
287 | m_streams.GetDataOut().WriteString(str); | |
288 | } | |
289 | ||
290 | // write the reply code, a string and a format in this order | |
291 | void Write(IPCCode code, const wxString& str, wxIPCFormat format) | |
292 | { | |
293 | Write(code, str); | |
294 | Write8(format); | |
295 | } | |
296 | ||
297 | // write arbitrary data | |
298 | void WriteData(const void *data, size_t size) | |
299 | { | |
300 | m_streams.GetDataOut().Write32(size); | |
301 | m_streams.GetUnformattedOut().Write(data, size); | |
302 | } | |
303 | ||
304 | ||
305 | private: | |
306 | wxIPCSocketStreams& m_streams; | |
307 | ||
308 | DECLARE_NO_COPY_CLASS(IPCOutput) | |
309 | }; | |
310 | ||
311 | } // anonymous namespace | |
312 | ||
cdc59bb6 GRG |
313 | // ========================================================================== |
314 | // implementation | |
315 | // ========================================================================== | |
316 | ||
317 | IMPLEMENT_DYNAMIC_CLASS(wxTCPServer, wxServerBase) | |
318 | IMPLEMENT_DYNAMIC_CLASS(wxTCPClient, wxClientBase) | |
319 | IMPLEMENT_CLASS(wxTCPConnection, wxConnectionBase) | |
320 | ||
321 | // -------------------------------------------------------------------------- | |
f4ada568 | 322 | // wxTCPClient |
cdc59bb6 | 323 | // -------------------------------------------------------------------------- |
f4ada568 | 324 | |
7e73fb9c VZ |
325 | wxTCPClient::wxTCPClient() |
326 | : wxClientBase() | |
f4ada568 GL |
327 | { |
328 | } | |
329 | ||
330 | bool wxTCPClient::ValidHost(const wxString& host) | |
331 | { | |
7e73fb9c | 332 | wxIPV4address addr; |
f4ada568 | 333 | |
7e73fb9c | 334 | return addr.Hostname(host); |
f4ada568 GL |
335 | } |
336 | ||
8aea37a9 VZ |
337 | wxConnectionBase *wxTCPClient::MakeConnection(const wxString& host, |
338 | const wxString& serverName, | |
339 | const wxString& topic) | |
f4ada568 | 340 | { |
7e73fb9c VZ |
341 | wxSockAddress *addr = GetAddressFromName(serverName, host); |
342 | if ( !addr ) | |
343 | return NULL; | |
0834112f | 344 | |
8aea37a9 VZ |
345 | wxSocketClient * const client = new wxSocketClient(wxSOCKET_WAITALL); |
346 | wxIPCSocketStreams * const streams = new wxIPCSocketStreams(*client); | |
f4ada568 | 347 | |
7e73fb9c VZ |
348 | bool ok = client->Connect(*addr); |
349 | delete addr; | |
26a25f95 | 350 | |
7e73fb9c VZ |
351 | if ( ok ) |
352 | { | |
7e73fb9c | 353 | // Send topic name, and enquire whether this has succeeded |
8aea37a9 | 354 | IPCOutput(streams).Write(IPC_CONNECT, topic); |
0834112f | 355 | |
8aea37a9 | 356 | unsigned char msg = streams->Read8(); |
0834112f | 357 | |
7e73fb9c VZ |
358 | // OK! Confirmation. |
359 | if (msg == IPC_CONNECT) | |
3adb47a9 | 360 | { |
7e73fb9c VZ |
361 | wxTCPConnection * |
362 | connection = (wxTCPConnection *)OnMakeConnection (); | |
363 | ||
364 | if (connection) | |
365 | { | |
366 | if (connection->IsKindOf(CLASSINFO(wxTCPConnection))) | |
367 | { | |
368 | connection->m_topic = topic; | |
369 | connection->m_sock = client; | |
8aea37a9 | 370 | connection->m_streams = streams; |
7e73fb9c VZ |
371 | client->SetEventHandler(*gs_handler, _CLIENT_ONREQUEST_ID); |
372 | client->SetClientData(connection); | |
373 | client->SetNotify(wxSOCKET_INPUT_FLAG | wxSOCKET_LOST_FLAG); | |
374 | client->Notify(true); | |
375 | return connection; | |
376 | } | |
377 | else | |
378 | { | |
379 | delete connection; | |
380 | // and fall through to delete everything else | |
381 | } | |
382 | } | |
3adb47a9 | 383 | } |
f4ada568 | 384 | } |
0834112f | 385 | |
7e73fb9c | 386 | // Something went wrong, delete everything |
8aea37a9 | 387 | delete streams; |
7e73fb9c | 388 | client->Destroy(); |
3adb47a9 | 389 | |
7e73fb9c | 390 | return NULL; |
f4ada568 GL |
391 | } |
392 | ||
393 | wxConnectionBase *wxTCPClient::OnMakeConnection() | |
394 | { | |
7e73fb9c | 395 | return new wxTCPConnection(); |
f4ada568 GL |
396 | } |
397 | ||
cdc59bb6 | 398 | // -------------------------------------------------------------------------- |
f4ada568 | 399 | // wxTCPServer |
cdc59bb6 | 400 | // -------------------------------------------------------------------------- |
f4ada568 | 401 | |
7e73fb9c VZ |
402 | wxTCPServer::wxTCPServer() |
403 | : wxServerBase() | |
f4ada568 | 404 | { |
7e73fb9c | 405 | m_server = NULL; |
f4ada568 GL |
406 | } |
407 | ||
f6bcfd97 | 408 | bool wxTCPServer::Create(const wxString& serverName) |
f4ada568 | 409 | { |
7e73fb9c VZ |
410 | // Destroy previous server, if any |
411 | if (m_server) | |
412 | { | |
413 | m_server->SetClientData(NULL); | |
414 | m_server->Destroy(); | |
415 | m_server = NULL; | |
416 | } | |
f4ada568 | 417 | |
7e73fb9c VZ |
418 | wxSockAddress *addr = GetAddressFromName(serverName); |
419 | if ( !addr ) | |
420 | return false; | |
0dbfd66d VZ |
421 | |
422 | #ifdef __UNIX_LIKE__ | |
7e73fb9c VZ |
423 | mode_t umaskOld; |
424 | if ( addr->Type() == wxSockAddress::UNIX ) | |
425 | { | |
426 | // ensure that the file doesn't exist as otherwise calling socket() | |
427 | // would fail | |
428 | int rc = remove(serverName.fn_str()); | |
429 | if ( rc < 0 && errno != ENOENT ) | |
430 | { | |
431 | delete addr; | |
432 | ||
433 | return false; | |
434 | } | |
435 | ||
436 | // also set the umask to prevent the others from reading our file | |
437 | umaskOld = umask(077); | |
438 | } | |
439 | else | |
440 | { | |
441 | // unused anyhow but shut down the compiler warnings | |
442 | umaskOld = 0; | |
443 | } | |
0dbfd66d | 444 | #endif // __UNIX_LIKE__ |
f4ada568 | 445 | |
8aea37a9 VZ |
446 | // Create a socket listening on the specified port (reusing it to allow |
447 | // restarting the server listening on the same port as was used by the | |
448 | // previous instance of this server) | |
449 | m_server = new wxSocketServer(*addr, wxSOCKET_WAITALL | wxSOCKET_REUSEADDR); | |
0dbfd66d VZ |
450 | |
451 | #ifdef __UNIX_LIKE__ | |
7e73fb9c VZ |
452 | if ( addr->Type() == wxSockAddress::UNIX ) |
453 | { | |
454 | // restore the umask | |
455 | umask(umaskOld); | |
456 | ||
457 | // save the file name to remove it later | |
458 | m_filename = serverName; | |
459 | } | |
0dbfd66d VZ |
460 | #endif // __UNIX_LIKE__ |
461 | ||
7e73fb9c | 462 | delete addr; |
f6bcfd97 | 463 | |
7e73fb9c VZ |
464 | if (!m_server->Ok()) |
465 | { | |
466 | m_server->Destroy(); | |
467 | m_server = NULL; | |
f6bcfd97 | 468 | |
7e73fb9c VZ |
469 | return false; |
470 | } | |
f6bcfd97 | 471 | |
7e73fb9c VZ |
472 | m_server->SetEventHandler(*gs_handler, _SERVER_ONREQUEST_ID); |
473 | m_server->SetClientData(this); | |
474 | m_server->SetNotify(wxSOCKET_CONNECTION_FLAG); | |
475 | m_server->Notify(true); | |
f4ada568 | 476 | |
7e73fb9c | 477 | return true; |
f4ada568 GL |
478 | } |
479 | ||
6e31e940 | 480 | wxTCPServer::~wxTCPServer() |
f4ada568 | 481 | { |
7e73fb9c | 482 | if ( m_server ) |
0dbfd66d VZ |
483 | { |
484 | m_server->SetClientData(NULL); | |
485 | m_server->Destroy(); | |
486 | } | |
487 | ||
488 | #ifdef __UNIX_LIKE__ | |
489 | if ( !m_filename.empty() ) | |
490 | { | |
401eb3de | 491 | if ( remove(m_filename.fn_str()) != 0 ) |
0dbfd66d VZ |
492 | { |
493 | wxLogDebug(_T("Stale AF_UNIX file '%s' left."), m_filename.c_str()); | |
494 | } | |
495 | } | |
496 | #endif // __UNIX_LIKE__ | |
f4ada568 GL |
497 | } |
498 | ||
7e73fb9c VZ |
499 | wxConnectionBase * |
500 | wxTCPServer::OnAcceptConnection(const wxString& WXUNUSED(topic)) | |
f4ada568 | 501 | { |
7e73fb9c | 502 | return new wxTCPConnection(); |
f4ada568 GL |
503 | } |
504 | ||
cdc59bb6 | 505 | // -------------------------------------------------------------------------- |
f4ada568 | 506 | // wxTCPConnection |
cdc59bb6 | 507 | // -------------------------------------------------------------------------- |
f4ada568 | 508 | |
7e73fb9c | 509 | void wxTCPConnection::Init() |
f4ada568 | 510 | { |
8aea37a9 VZ |
511 | m_sock = NULL; |
512 | m_streams = NULL; | |
f4ada568 GL |
513 | } |
514 | ||
7e73fb9c | 515 | wxTCPConnection::~wxTCPConnection() |
7921cf2b | 516 | { |
7e73fb9c | 517 | Disconnect(); |
7921cf2b | 518 | |
7e73fb9c VZ |
519 | if ( m_sock ) |
520 | { | |
521 | m_sock->SetClientData(NULL); | |
522 | m_sock->Destroy(); | |
523 | } | |
524 | ||
8aea37a9 | 525 | delete m_streams; |
f4ada568 GL |
526 | } |
527 | ||
cb43b372 | 528 | void wxTCPConnection::Compress(bool WXUNUSED(on)) |
f4ada568 | 529 | { |
7e73fb9c | 530 | // TODO |
f4ada568 GL |
531 | } |
532 | ||
533 | // Calls that CLIENT can make. | |
7e73fb9c | 534 | bool wxTCPConnection::Disconnect() |
f4ada568 | 535 | { |
7e73fb9c VZ |
536 | if ( !GetConnected() ) |
537 | return true; | |
538 | ||
539 | // Send the disconnect message to the peer. | |
8aea37a9 | 540 | IPCOutput(m_streams).Write8(IPC_DISCONNECT); |
82c91ef5 | 541 | |
7e73fb9c VZ |
542 | if ( m_sock ) |
543 | { | |
544 | m_sock->Notify(false); | |
545 | m_sock->Close(); | |
546 | } | |
82c91ef5 | 547 | |
7e73fb9c | 548 | SetConnected(false); |
f4ada568 | 549 | |
7e73fb9c | 550 | return true; |
f4ada568 GL |
551 | } |
552 | ||
7e73fb9c VZ |
553 | bool wxTCPConnection::DoExecute(const void *data, |
554 | size_t size, | |
555 | wxIPCFormat format) | |
f4ada568 | 556 | { |
7e73fb9c VZ |
557 | if ( !m_sock->IsConnected() ) |
558 | return false; | |
f4ada568 | 559 | |
7e73fb9c | 560 | // Prepare EXECUTE message |
8aea37a9 VZ |
561 | IPCOutput out(m_streams); |
562 | out.Write8(IPC_EXECUTE); | |
563 | out.Write8(format); | |
0834112f | 564 | |
8aea37a9 | 565 | out.WriteData(data, size); |
f4ada568 | 566 | |
7e73fb9c | 567 | return true; |
f4ada568 GL |
568 | } |
569 | ||
7e73fb9c VZ |
570 | const void *wxTCPConnection::Request(const wxString& item, |
571 | size_t *size, | |
572 | wxIPCFormat format) | |
f4ada568 | 573 | { |
7e73fb9c VZ |
574 | if ( !m_sock->IsConnected() ) |
575 | return NULL; | |
f4ada568 | 576 | |
8aea37a9 | 577 | IPCOutput(m_streams).Write(IPC_REQUEST, item, format); |
f4ada568 | 578 | |
8aea37a9 | 579 | int ret = m_streams->Read8(); |
7e73fb9c VZ |
580 | if ( ret == IPC_FAIL ) |
581 | return NULL; | |
f4ada568 | 582 | |
8aea37a9 | 583 | return m_streams->ReadData(this, size); |
f4ada568 GL |
584 | } |
585 | ||
7e73fb9c VZ |
586 | bool wxTCPConnection::DoPoke(const wxString& item, |
587 | const void *data, | |
588 | size_t size, | |
589 | wxIPCFormat format) | |
f4ada568 | 590 | { |
7e73fb9c VZ |
591 | if ( !m_sock->IsConnected() ) |
592 | return false; | |
f4ada568 | 593 | |
8aea37a9 VZ |
594 | IPCOutput out(m_streams); |
595 | out.Write(IPC_POKE, item, format); | |
596 | out.WriteData(data, size); | |
f4ada568 | 597 | |
7e73fb9c | 598 | return true; |
f4ada568 GL |
599 | } |
600 | ||
8aea37a9 | 601 | bool wxTCPConnection::StartAdvise(const wxString& item) |
f4ada568 | 602 | { |
7e73fb9c VZ |
603 | if ( !m_sock->IsConnected() ) |
604 | return false; | |
f4ada568 | 605 | |
8aea37a9 | 606 | IPCOutput(m_streams).Write(IPC_ADVISE_START, item); |
f4ada568 | 607 | |
8aea37a9 | 608 | int ret = m_streams->Read8(); |
7e73fb9c VZ |
609 | if (ret != IPC_FAIL) |
610 | return true; | |
611 | else | |
612 | return false; | |
f4ada568 GL |
613 | } |
614 | ||
615 | bool wxTCPConnection::StopAdvise (const wxString& item) | |
616 | { | |
7e73fb9c VZ |
617 | if ( !m_sock->IsConnected() ) |
618 | return false; | |
f4ada568 | 619 | |
8aea37a9 | 620 | IPCOutput(m_streams).Write(IPC_ADVISE_STOP, item); |
f4ada568 | 621 | |
8aea37a9 | 622 | int ret = m_streams->Read8(); |
f4ada568 | 623 | |
7e73fb9c VZ |
624 | if (ret != IPC_FAIL) |
625 | return true; | |
626 | else | |
627 | return false; | |
f4ada568 GL |
628 | } |
629 | ||
630 | // Calls that SERVER can make | |
7e73fb9c VZ |
631 | bool wxTCPConnection::DoAdvise(const wxString& item, |
632 | const void *data, | |
633 | size_t size, | |
634 | wxIPCFormat format) | |
f4ada568 | 635 | { |
7e73fb9c VZ |
636 | if ( !m_sock->IsConnected() ) |
637 | return false; | |
f4ada568 | 638 | |
8aea37a9 VZ |
639 | IPCOutput out(m_streams); |
640 | out.Write(IPC_ADVISE, item, format); | |
641 | out.WriteData(data, size); | |
f4ada568 | 642 | |
7e73fb9c | 643 | return true; |
f4ada568 GL |
644 | } |
645 | ||
cdc59bb6 GRG |
646 | // -------------------------------------------------------------------------- |
647 | // wxTCPEventHandler (private class) | |
648 | // -------------------------------------------------------------------------- | |
649 | ||
650 | BEGIN_EVENT_TABLE(wxTCPEventHandler, wxEvtHandler) | |
7e73fb9c VZ |
651 | EVT_SOCKET(_CLIENT_ONREQUEST_ID, wxTCPEventHandler::Client_OnRequest) |
652 | EVT_SOCKET(_SERVER_ONREQUEST_ID, wxTCPEventHandler::Server_OnRequest) | |
cdc59bb6 GRG |
653 | END_EVENT_TABLE() |
654 | ||
655 | void wxTCPEventHandler::Client_OnRequest(wxSocketEvent &event) | |
f4ada568 | 656 | { |
7e73fb9c VZ |
657 | wxSocketBase *sock = event.GetSocket(); |
658 | if (!sock) | |
659 | return ; | |
f4ada568 | 660 | |
7e73fb9c VZ |
661 | wxSocketNotify evt = event.GetSocketEvent(); |
662 | wxTCPConnection *connection = (wxTCPConnection *)(sock->GetClientData()); | |
f4ada568 | 663 | |
7e73fb9c VZ |
664 | // This socket is being deleted; skip this event |
665 | if (!connection) | |
666 | return; | |
f4ada568 | 667 | |
7e73fb9c VZ |
668 | // We lost the connection: destroy everything |
669 | if (evt == wxSOCKET_LOST) | |
670 | { | |
671 | sock->Notify(false); | |
672 | sock->Close(); | |
673 | connection->OnDisconnect(); | |
674 | return; | |
675 | } | |
f4ada568 | 676 | |
7e73fb9c | 677 | // Receive message number. |
8aea37a9 | 678 | wxIPCSocketStreams * const streams = connection->m_streams; |
f4ada568 | 679 | |
8aea37a9 VZ |
680 | const wxString topic = connection->m_topic; |
681 | wxString item; | |
682 | ||
428dca1c VZ |
683 | bool error = false; |
684 | ||
22a9029e VZ |
685 | const int msg = streams->Read8(); |
686 | switch ( msg ) | |
0834112f | 687 | { |
7e73fb9c VZ |
688 | case IPC_EXECUTE: |
689 | { | |
7e73fb9c | 690 | wxIPCFormat format; |
428dca1c | 691 | size_t size wxDUMMY_INITIALIZE(0); |
8aea37a9 VZ |
692 | void * const |
693 | data = streams->ReadFormatData(connection, &format, &size); | |
428dca1c VZ |
694 | if ( data ) |
695 | connection->OnExecute(topic, data, size, format); | |
696 | else | |
697 | error = true; | |
7e73fb9c | 698 | } |
8aea37a9 VZ |
699 | break; |
700 | ||
7e73fb9c VZ |
701 | case IPC_ADVISE: |
702 | { | |
8aea37a9 | 703 | item = streams->ReadString(); |
7e73fb9c | 704 | |
8aea37a9 | 705 | wxIPCFormat format; |
428dca1c | 706 | size_t size wxDUMMY_INITIALIZE(0); |
8aea37a9 VZ |
707 | void * const |
708 | data = streams->ReadFormatData(connection, &format, &size); | |
7e73fb9c | 709 | |
428dca1c VZ |
710 | if ( data ) |
711 | connection->OnAdvise(topic, item, data, size, format); | |
712 | else | |
713 | error = true; | |
7e73fb9c | 714 | } |
8aea37a9 VZ |
715 | break; |
716 | ||
7e73fb9c VZ |
717 | case IPC_ADVISE_START: |
718 | { | |
8aea37a9 | 719 | item = streams->ReadString(); |
7e73fb9c | 720 | |
8aea37a9 VZ |
721 | IPCOutput(streams).Write8(connection->OnStartAdvise(topic, item) |
722 | ? IPC_ADVISE_START | |
723 | : IPC_FAIL); | |
7e73fb9c | 724 | } |
8aea37a9 VZ |
725 | break; |
726 | ||
7e73fb9c VZ |
727 | case IPC_ADVISE_STOP: |
728 | { | |
8aea37a9 | 729 | item = streams->ReadString(); |
7e73fb9c | 730 | |
8aea37a9 VZ |
731 | IPCOutput(streams).Write8(connection->OnStopAdvise(topic, item) |
732 | ? IPC_ADVISE_STOP | |
733 | : IPC_FAIL); | |
7e73fb9c | 734 | } |
8aea37a9 VZ |
735 | break; |
736 | ||
7e73fb9c VZ |
737 | case IPC_POKE: |
738 | { | |
8aea37a9 VZ |
739 | item = streams->ReadString(); |
740 | wxIPCFormat format = (wxIPCFormat)streams->Read8(); | |
7e73fb9c | 741 | |
428dca1c | 742 | size_t size wxDUMMY_INITIALIZE(0); |
8aea37a9 | 743 | void * const data = streams->ReadData(connection, &size); |
7e73fb9c | 744 | |
428dca1c VZ |
745 | if ( data ) |
746 | connection->OnPoke(topic, item, data, size, format); | |
747 | else | |
748 | error = true; | |
7e73fb9c | 749 | } |
8aea37a9 VZ |
750 | break; |
751 | ||
7e73fb9c VZ |
752 | case IPC_REQUEST: |
753 | { | |
8aea37a9 | 754 | item = streams->ReadString(); |
7e73fb9c | 755 | |
8aea37a9 | 756 | wxIPCFormat format = (wxIPCFormat)streams->Read8(); |
7e73fb9c VZ |
757 | |
758 | size_t user_size = wxNO_LEN; | |
8aea37a9 | 759 | const void *user_data = connection->OnRequest(topic, |
7e73fb9c VZ |
760 | item, |
761 | &user_size, | |
762 | format); | |
763 | ||
8aea37a9 | 764 | if ( !user_data ) |
7e73fb9c | 765 | { |
8aea37a9 VZ |
766 | IPCOutput(streams).Write8(IPC_FAIL); |
767 | break; | |
768 | } | |
769 | ||
770 | IPCOutput out(streams); | |
771 | out.Write8(IPC_REQUEST_REPLY); | |
7e73fb9c | 772 | |
8aea37a9 VZ |
773 | if ( user_size == wxNO_LEN ) |
774 | { | |
775 | switch ( format ) | |
7e73fb9c | 776 | { |
8aea37a9 VZ |
777 | case wxIPC_TEXT: |
778 | case wxIPC_UTF8TEXT: | |
779 | user_size = strlen((const char *)user_data) + 1; // includes final NUL | |
780 | break; | |
781 | case wxIPC_UNICODETEXT: | |
782 | user_size = (wcslen((const wchar_t *)user_data) + 1) * sizeof(wchar_t); // includes final NUL | |
783 | break; | |
784 | default: | |
785 | user_size = 0; | |
7e73fb9c | 786 | } |
7e73fb9c | 787 | } |
7e73fb9c | 788 | |
8aea37a9 | 789 | out.WriteData(user_data, user_size); |
7e73fb9c | 790 | } |
8aea37a9 VZ |
791 | break; |
792 | ||
7e73fb9c | 793 | case IPC_DISCONNECT: |
8aea37a9 VZ |
794 | sock->Notify(false); |
795 | sock->Close(); | |
796 | connection->SetConnected(false); | |
797 | connection->OnDisconnect(); | |
798 | break; | |
799 | ||
7e73fb9c | 800 | default: |
8aea37a9 | 801 | wxLogDebug("Unknown message code %d received.", msg); |
428dca1c | 802 | error = true; |
50c549b9 | 803 | break; |
0834112f | 804 | } |
428dca1c VZ |
805 | |
806 | if ( error ) | |
807 | IPCOutput(streams).Write8(IPC_FAIL); | |
f4ada568 GL |
808 | } |
809 | ||
cdc59bb6 | 810 | void wxTCPEventHandler::Server_OnRequest(wxSocketEvent &event) |
f4ada568 | 811 | { |
7e73fb9c VZ |
812 | wxSocketServer *server = (wxSocketServer *) event.GetSocket(); |
813 | if (!server) | |
814 | return ; | |
815 | wxTCPServer *ipcserv = (wxTCPServer *) server->GetClientData(); | |
f4ada568 | 816 | |
7e73fb9c VZ |
817 | // This socket is being deleted; skip this event |
818 | if (!ipcserv) | |
819 | return; | |
820 | ||
821 | if (event.GetSocketEvent() != wxSOCKET_CONNECTION) | |
822 | return; | |
f4ada568 | 823 | |
7e73fb9c VZ |
824 | // Accept the connection, getting a new socket |
825 | wxSocketBase *sock = server->Accept(); | |
826 | if (!sock) | |
827 | return ; | |
828 | if (!sock->Ok()) | |
829 | { | |
830 | sock->Destroy(); | |
831 | return; | |
832 | } | |
f4ada568 | 833 | |
8aea37a9 | 834 | wxIPCSocketStreams *streams = new wxIPCSocketStreams(*sock); |
3adb47a9 | 835 | |
0834112f | 836 | { |
8aea37a9 | 837 | IPCOutput out(streams); |
7e73fb9c | 838 | |
8aea37a9 VZ |
839 | const int msg = streams->Read8(); |
840 | if ( msg == IPC_CONNECT ) | |
7e73fb9c | 841 | { |
8aea37a9 VZ |
842 | const wxString topic = streams->ReadString(); |
843 | ||
844 | wxTCPConnection *new_connection = | |
845 | (wxTCPConnection *)ipcserv->OnAcceptConnection (topic); | |
846 | ||
847 | if (new_connection) | |
7e73fb9c | 848 | { |
8aea37a9 VZ |
849 | if (new_connection->IsKindOf(CLASSINFO(wxTCPConnection))) |
850 | { | |
851 | // Acknowledge success | |
852 | out.Write8(IPC_CONNECT); | |
853 | ||
854 | new_connection->m_sock = sock; | |
855 | new_connection->m_streams = streams; | |
856 | new_connection->m_topic = topic; | |
857 | sock->SetEventHandler(*gs_handler, _CLIENT_ONREQUEST_ID); | |
858 | sock->SetClientData(new_connection); | |
859 | sock->SetNotify(wxSOCKET_INPUT_FLAG | wxSOCKET_LOST_FLAG); | |
860 | sock->Notify(true); | |
861 | return; | |
862 | } | |
863 | else | |
864 | { | |
865 | delete new_connection; | |
866 | // and fall through to delete everything else | |
867 | } | |
7e73fb9c VZ |
868 | } |
869 | } | |
3adb47a9 | 870 | |
8aea37a9 VZ |
871 | // Something went wrong, send failure message and delete everything |
872 | out.Write8(IPC_FAIL); | |
873 | } // IPCOutput object is destroyed here, before destroying stream | |
3adb47a9 | 874 | |
8aea37a9 | 875 | delete streams; |
7e73fb9c | 876 | sock->Destroy(); |
f4ada568 | 877 | } |
35a4dab7 | 878 | |
cdc59bb6 GRG |
879 | // -------------------------------------------------------------------------- |
880 | // wxTCPEventHandlerModule (private class) | |
881 | // -------------------------------------------------------------------------- | |
882 | ||
ed4c6c69 | 883 | class wxTCPEventHandlerModule: public wxModule |
cdc59bb6 | 884 | { |
cdc59bb6 | 885 | public: |
7e73fb9c VZ |
886 | virtual bool OnInit() { gs_handler = new wxTCPEventHandler; return true; } |
887 | virtual void OnExit() { wxDELETE(gs_handler); } | |
888 | ||
889 | DECLARE_DYNAMIC_CLASS(wxTCPEventHandlerModule) | |
cdc59bb6 GRG |
890 | }; |
891 | ||
892 | IMPLEMENT_DYNAMIC_CLASS(wxTCPEventHandlerModule, wxModule) | |
893 | ||
7e73fb9c | 894 | #endif // wxUSE_SOCKETS && wxUSE_IPC && wxUSE_STREAMS |