2008-11-28 12:47:07 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
2008-11-28 13:04:47 +00:00
|
|
|
// Name: wx/unix/private/sockunix.h
|
2008-11-28 12:47:07 +00:00
|
|
|
// Purpose: wxSocketImpl implementation for Unix systems
|
|
|
|
// Authors: Guilhem Lavaux, Vadim Zeitlin
|
|
|
|
// Created: April 1997
|
|
|
|
// RCS-ID: $Id$
|
|
|
|
// Copyright: (c) 1997 Guilhem Lavaux
|
|
|
|
// (c) 2008 Vadim Zeitlin
|
|
|
|
// Licence: wxWindows licence
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
1999-09-30 23:53:10 +00:00
|
|
|
|
2007-12-19 16:45:08 +00:00
|
|
|
#ifndef _WX_UNIX_GSOCKUNX_H_
|
|
|
|
#define _WX_UNIX_GSOCKUNX_H_
|
1999-08-06 10:10:10 +00:00
|
|
|
|
2008-11-25 13:57:19 +00:00
|
|
|
#include <unistd.h>
|
2008-11-28 12:47:07 +00:00
|
|
|
#include <sys/ioctl.h>
|
2008-11-28 14:39:49 +00:00
|
|
|
#include "wx/private/fdiodispatcher.h"
|
2008-11-25 13:57:19 +00:00
|
|
|
|
2008-11-28 14:39:49 +00:00
|
|
|
class wxSocketImplUnix : public wxSocketImpl,
|
|
|
|
public wxFDIOHandler
|
2004-08-03 19:18:39 +00:00
|
|
|
{
|
|
|
|
public:
|
2008-11-28 13:43:41 +00:00
|
|
|
wxSocketImplUnix(wxSocketBase& wxsocket)
|
|
|
|
: wxSocketImpl(wxsocket)
|
|
|
|
{
|
|
|
|
m_fds[0] =
|
|
|
|
m_fds[1] = -1;
|
|
|
|
|
2008-11-28 15:01:41 +00:00
|
|
|
m_enabledCallbacks = 0;
|
2008-11-28 13:43:41 +00:00
|
|
|
}
|
2008-11-23 13:12:46 +00:00
|
|
|
|
2008-12-27 17:15:22 +00:00
|
|
|
virtual wxSocketError GetLastError() const;
|
|
|
|
|
2009-01-02 21:53:30 +00:00
|
|
|
virtual void ReenableEvents(wxSocketEventFlags flags)
|
|
|
|
{
|
|
|
|
// enable the notifications about input/output being available again in
|
|
|
|
// case they were disabled by OnRead/WriteWaiting()
|
|
|
|
//
|
|
|
|
// notice that we'd like to enable the events here only if there is
|
|
|
|
// nothing more left on the socket right now as otherwise we're going
|
|
|
|
// to get a "ready for whatever" notification immediately (well, during
|
|
|
|
// the next event loop iteration) and disable the event back again
|
|
|
|
// which is rather inefficient but unfortunately doing it like this
|
|
|
|
// doesn't work because the existing code (e.g. src/common/sckipc.cpp)
|
|
|
|
// expects to keep getting notifications about the data available from
|
|
|
|
// the socket even if it didn't read all the data the last time, so we
|
|
|
|
// absolutely have to continue generating them
|
|
|
|
EnableEvents(flags);
|
|
|
|
}
|
|
|
|
|
2008-11-28 14:39:49 +00:00
|
|
|
// wxFDIOHandler methods
|
|
|
|
virtual void OnReadWaiting();
|
|
|
|
virtual void OnWriteWaiting();
|
|
|
|
virtual void OnExceptionWaiting();
|
2007-11-17 15:21:26 +00:00
|
|
|
|
2009-01-02 16:59:23 +00:00
|
|
|
// Unix-specific functions used by wxSocketFDIOManager only
|
2008-11-28 15:01:41 +00:00
|
|
|
bool HasAnyEnabledCallbacks() const { return m_enabledCallbacks != 0; }
|
|
|
|
void EnableCallback(wxFDIODispatcherEntryFlags flag)
|
|
|
|
{ m_enabledCallbacks |= flag; }
|
|
|
|
void DisableCallback(wxFDIODispatcherEntryFlags flag)
|
|
|
|
{ m_enabledCallbacks &= ~flag; }
|
|
|
|
int GetEnabledCallbacks() const { return m_enabledCallbacks; }
|
|
|
|
|
2008-11-25 13:33:07 +00:00
|
|
|
private:
|
2008-11-28 12:47:07 +00:00
|
|
|
virtual void DoClose()
|
|
|
|
{
|
2008-12-27 21:56:05 +00:00
|
|
|
DisableEvents();
|
2008-11-28 12:47:07 +00:00
|
|
|
|
|
|
|
close(m_fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void UnblockAndRegisterWithEventLoop()
|
|
|
|
{
|
|
|
|
int trueArg = 1;
|
|
|
|
ioctl(m_fd, FIONBIO, &trueArg);
|
|
|
|
|
|
|
|
EnableEvents();
|
|
|
|
}
|
|
|
|
|
2008-12-26 20:20:46 +00:00
|
|
|
// enable or disable notifications for socket input/output events
|
2009-01-02 21:53:30 +00:00
|
|
|
void EnableEvents(int flags = wxSOCKET_INPUT_FLAG | wxSOCKET_OUTPUT_FLAG)
|
|
|
|
{ DoEnableEvents(flags, true); }
|
|
|
|
void DisableEvents(int flags = wxSOCKET_INPUT_FLAG | wxSOCKET_OUTPUT_FLAG)
|
|
|
|
{ DoEnableEvents(flags, false); }
|
2008-11-25 13:33:07 +00:00
|
|
|
|
2008-12-26 20:20:46 +00:00
|
|
|
// really enable or disable socket input/output events
|
2009-01-02 21:53:30 +00:00
|
|
|
void DoEnableEvents(int flags, bool enable);
|
2008-11-25 13:33:07 +00:00
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
protected:
|
|
|
|
// descriptors for input and output event notification channels associated
|
|
|
|
// with the socket
|
|
|
|
int m_fds[2];
|
2008-11-23 13:12:46 +00:00
|
|
|
|
2008-11-28 15:01:41 +00:00
|
|
|
// the events which are currently enabled for this socket, combination of
|
|
|
|
// wxFDIO_INPUT and wxFDIO_OUTPUT values
|
|
|
|
int m_enabledCallbacks;
|
|
|
|
|
2008-11-23 13:12:46 +00:00
|
|
|
private:
|
|
|
|
// notify the associated wxSocket about a change in socket state and shut
|
2008-11-28 12:47:07 +00:00
|
|
|
// down the socket if the event is wxSOCKET_LOST
|
|
|
|
void OnStateChange(wxSocketNotify event);
|
|
|
|
|
2009-01-02 21:53:30 +00:00
|
|
|
// check if there is any input available, return 1 if yes, 0 if no or -1 on
|
|
|
|
// error
|
|
|
|
int CheckForInput();
|
|
|
|
|
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
// give it access to our m_fds
|
|
|
|
friend class wxSocketFDBasedManager;
|
1999-07-22 17:51:54 +00:00
|
|
|
};
|
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
// A version of wxSocketManager which uses FDs for socket IO
|
|
|
|
class wxSocketFDBasedManager : public wxSocketManager
|
2007-12-19 16:45:08 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
// no special initialization/cleanup needed when using FDs
|
|
|
|
virtual bool OnInit() { return true; }
|
|
|
|
virtual void OnExit() { }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
// identifies either input or output direction
|
|
|
|
//
|
|
|
|
// NB: the values of this enum shouldn't change
|
|
|
|
enum SocketDir
|
|
|
|
{
|
|
|
|
FD_INPUT,
|
|
|
|
FD_OUTPUT
|
|
|
|
};
|
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
// get the FD index corresponding to the given wxSocketNotify
|
|
|
|
SocketDir GetDirForEvent(wxSocketImpl *socket, wxSocketNotify event)
|
2007-12-19 16:45:08 +00:00
|
|
|
{
|
|
|
|
switch ( event )
|
|
|
|
{
|
|
|
|
default:
|
2008-12-30 00:11:04 +00:00
|
|
|
wxFAIL_MSG( "unknown socket event" );
|
|
|
|
return FD_INPUT; // we must return something
|
2007-12-19 16:45:08 +00:00
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
case wxSOCKET_LOST:
|
2008-12-30 00:11:04 +00:00
|
|
|
wxFAIL_MSG( "unexpected socket event" );
|
|
|
|
return FD_INPUT; // as above
|
2007-12-19 16:45:08 +00:00
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
case wxSOCKET_INPUT:
|
2007-12-19 16:45:08 +00:00
|
|
|
return FD_INPUT;
|
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
case wxSOCKET_OUTPUT:
|
2007-12-19 16:45:08 +00:00
|
|
|
return FD_OUTPUT;
|
|
|
|
|
2008-11-28 12:47:07 +00:00
|
|
|
case wxSOCKET_CONNECTION:
|
2008-12-30 00:11:04 +00:00
|
|
|
// for server sockets we're interested in events indicating
|
|
|
|
// that a new connection is pending, i.e. that accept() will
|
|
|
|
// succeed and this is indicated by socket becoming ready for
|
|
|
|
// reading, while for the other ones we're interested in the
|
|
|
|
// completion of non-blocking connect() which is indicated by
|
|
|
|
// the socket becoming ready for writing
|
2008-12-27 17:15:22 +00:00
|
|
|
return socket->IsServer() ? FD_INPUT : FD_OUTPUT;
|
2007-12-19 16:45:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// access the FDs we store
|
2008-11-28 14:39:49 +00:00
|
|
|
int& FD(wxSocketImplUnix *socket, SocketDir d)
|
2007-12-19 16:45:08 +00:00
|
|
|
{
|
2008-11-28 14:39:49 +00:00
|
|
|
return socket->m_fds[d];
|
2007-12-19 16:45:08 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Common base class for all ports using X11-like (and hence implemented in
|
|
|
|
// X11, Motif and GTK) AddInput() and RemoveInput() functions
|
2008-11-28 12:47:07 +00:00
|
|
|
class wxSocketInputBasedManager : public wxSocketFDBasedManager
|
2007-12-19 16:45:08 +00:00
|
|
|
{
|
|
|
|
public:
|
2008-11-28 14:39:49 +00:00
|
|
|
virtual void Install_Callback(wxSocketImpl *socket_, wxSocketNotify event)
|
2007-12-19 16:45:08 +00:00
|
|
|
{
|
2008-11-28 14:39:49 +00:00
|
|
|
wxSocketImplUnix * const
|
|
|
|
socket = static_cast<wxSocketImplUnix *>(socket_);
|
|
|
|
|
2007-12-19 16:45:08 +00:00
|
|
|
wxCHECK_RET( socket->m_fd != -1,
|
|
|
|
"shouldn't be called on invalid socket" );
|
|
|
|
|
|
|
|
const SocketDir d = GetDirForEvent(socket, event);
|
|
|
|
|
|
|
|
int& fd = FD(socket, d);
|
|
|
|
if ( fd != -1 )
|
|
|
|
RemoveInput(fd);
|
|
|
|
|
2008-11-28 14:39:49 +00:00
|
|
|
fd = AddInput(socket, socket->m_fd, d);
|
2007-12-19 16:45:08 +00:00
|
|
|
}
|
|
|
|
|
2008-11-28 14:39:49 +00:00
|
|
|
virtual void Uninstall_Callback(wxSocketImpl *socket_, wxSocketNotify event)
|
2007-12-19 16:45:08 +00:00
|
|
|
{
|
2008-11-28 14:39:49 +00:00
|
|
|
wxSocketImplUnix * const
|
|
|
|
socket = static_cast<wxSocketImplUnix *>(socket_);
|
|
|
|
|
2007-12-19 16:45:08 +00:00
|
|
|
const SocketDir d = GetDirForEvent(socket, event);
|
|
|
|
|
|
|
|
int& fd = FD(socket, d);
|
|
|
|
if ( fd != -1 )
|
|
|
|
{
|
|
|
|
RemoveInput(fd);
|
|
|
|
fd = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
// these functions map directly to XtAdd/RemoveInput() or
|
|
|
|
// gdk_input_add/remove()
|
2008-11-28 14:39:49 +00:00
|
|
|
virtual int AddInput(wxFDIOHandler *handler, int fd, SocketDir d) = 0;
|
2007-12-19 16:45:08 +00:00
|
|
|
virtual void RemoveInput(int fd) = 0;
|
|
|
|
};
|
1999-08-06 10:10:10 +00:00
|
|
|
|
2007-12-19 16:45:08 +00:00
|
|
|
#endif /* _WX_UNIX_GSOCKUNX_H_ */
|