added support for broadcasting to UDP sockets (patch 1740266)

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@47126 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Vadim Zeitlin 2007-07-04 21:33:11 +00:00
parent bf511d579f
commit 60edcf4535
14 changed files with 180 additions and 23 deletions

View File

@ -109,6 +109,7 @@ All:
- Added functions for atomically inc/decrementing integers (Armel Asselin) - Added functions for atomically inc/decrementing integers (Armel Asselin)
- wxLogInterposer has been added to replace wxLogPassThrough and new - wxLogInterposer has been added to replace wxLogPassThrough and new
wxLogInterposerTemp was added wxLogInterposerTemp was added
- Added support for broadcasting to UDP sockets (Andrew Vincent)
All (GUI): All (GUI):

View File

@ -98,6 +98,23 @@ On IPV6 implementations, ::
Returns true on success, false if something went wrong. Returns true on success, false if something went wrong.
%
% BroadcastAddress
%
\membersection{wxIPaddress::BroadcastAddress}\label{wxIPaddressbroadcastaddress}
\func{virtual bool}{BroadcastAddress}{\void}
Internally, this is the same as setting the IP address
to {\bf INADDR\_BROADCAST}.
On IPV4 implementations, 255.255.255.255
\wxheading{Return value}
Returns true on success, false if something went wrong.
% %
% LocalHost % LocalHost
% %

View File

@ -524,6 +524,8 @@ The following flags can be used:
\twocolitem{{\bf wxSOCKET\_WAITALL}}{Wait for all required data to be read/written unless an error occurs.} \twocolitem{{\bf wxSOCKET\_WAITALL}}{Wait for all required data to be read/written unless an error occurs.}
\twocolitem{{\bf wxSOCKET\_BLOCK}}{Block the GUI (do not yield) while reading/writing data.} \twocolitem{{\bf wxSOCKET\_BLOCK}}{Block the GUI (do not yield) while reading/writing data.}
\twocolitem{{\bf wxSOCKET\_REUSEADDR}}{Allows the use of an in-use port (wxServerSocket only)} \twocolitem{{\bf wxSOCKET\_REUSEADDR}}{Allows the use of an in-use port (wxServerSocket only)}
\twocolitem{{\bf wxSOCKET\_BROADCAST}}{Switches the socket to broadcast mode}
\twocolitem{{\bf wxSOCKET\_NOBIND}}{Stops the socket from being bound to a specific adapter (normally used in conjunction with {\bf wxSOCKET\_BROADCAST})}
\end{twocollist} \end{twocollist}
A brief overview on how to use these flags follows. A brief overview on how to use these flags follows.
@ -568,6 +570,10 @@ your platform's implementation of setsockopt(). Note that on BSD-based systems (
use of wxSOCKET\_REUSEADDR implies SO\_REUSEPORT in addition to SO\_REUSEADDR to be consistent use of wxSOCKET\_REUSEADDR implies SO\_REUSEPORT in addition to SO\_REUSEADDR to be consistent
with Windows. with Windows.
The {\bf wxSOCKET\_BROADCAST} flag controls the use of the SO\_BROADCAST standard
setsockopt() flag. This flag allows the socket to use the broadcast address, and is generally
used in conjunction with {\bf wxSOCKET\_NOBIND} and \helpref{wxIPaddress::BroadcastAddress}{wxipaddressbroadcastaddress}.
So: So:
{\bf wxSOCKET\_NONE} will try to read at least SOME data, no matter how much. {\bf wxSOCKET\_NONE} will try to read at least SOME data, no matter how much.

View File

@ -150,6 +150,7 @@ GAddressType GAddress_GetFamily(GAddress *address);
*/ */
GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname); GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname);
GSocketError GAddress_INET_SetBroadcastAddress(GAddress *address);
GSocketError GAddress_INET_SetAnyAddress(GAddress *address); GSocketError GAddress_INET_SetAnyAddress(GAddress *address);
GSocketError GAddress_INET_SetHostAddress(GAddress *address, GSocketError GAddress_INET_SetHostAddress(GAddress *address,
unsigned long hostaddr); unsigned long hostaddr);

View File

@ -63,6 +63,8 @@ public:
GSocketError SetServer(); GSocketError SetServer();
GSocket *WaitConnection(); GSocket *WaitConnection();
bool SetReusable(); bool SetReusable();
bool SetBroadcast();
bool DontDoBind();
GSocketError Connect(GSocketStream stream); GSocketError Connect(GSocketStream stream);
GSocketError SetNonOriented(); GSocketError SetNonOriented();
int Read(char *buffer, int size); int Read(char *buffer, int size);
@ -101,6 +103,8 @@ public:
bool m_stream; bool m_stream;
bool m_establishing; bool m_establishing;
bool m_reusable; bool m_reusable;
bool m_broadcast;
bool m_dobind;
struct timeval m_timeout; struct timeval m_timeout;
/* Callbacks */ /* Callbacks */

View File

@ -64,6 +64,7 @@ public:
virtual bool IsLocalHost() const = 0; virtual bool IsLocalHost() const = 0;
virtual bool AnyAddress() = 0; virtual bool AnyAddress() = 0;
virtual bool BroadcastAddress() = 0;
virtual wxString IPAddress() const = 0; virtual wxString IPAddress() const = 0;
@ -93,6 +94,8 @@ public:
// any (0.0.0.0) // any (0.0.0.0)
virtual bool AnyAddress(); virtual bool AnyAddress();
// all (255.255.255.255)
virtual bool BroadcastAddress();
virtual wxString Hostname() const; virtual wxString Hostname() const;
wxString OrigHostname() { return m_origHostname; } wxString OrigHostname() { return m_origHostname; }
@ -149,6 +152,8 @@ public:
// any (0000:0000:0000:0000:0000:0000:0000:0000 (::)) // any (0000:0000:0000:0000:0000:0000:0000:0000 (::))
virtual bool AnyAddress(); virtual bool AnyAddress();
// all (?)
virtual bool BroadcastAddress();
// 3ffe:ffff:0100:f101:0210:a4ff:fee3:9566 // 3ffe:ffff:0100:f101:0210:a4ff:fee3:9566
virtual wxString IPAddress() const; virtual wxString IPAddress() const;

View File

@ -71,7 +71,9 @@ enum
wxSOCKET_NOWAIT = 1, wxSOCKET_NOWAIT = 1,
wxSOCKET_WAITALL = 2, wxSOCKET_WAITALL = 2,
wxSOCKET_BLOCK = 4, wxSOCKET_BLOCK = 4,
wxSOCKET_REUSEADDR = 8 wxSOCKET_REUSEADDR = 8,
wxSOCKET_BROADCAST = 16,
wxSOCKET_NOBIND = 32
}; };
enum wxSocketType enum wxSocketType

View File

@ -52,6 +52,8 @@ public:
GSocketError SetServer(); GSocketError SetServer();
GSocket *WaitConnection(); GSocket *WaitConnection();
bool SetReusable(); bool SetReusable();
bool SetBroadcast();
bool DontDoBind();
GSocketError Connect(GSocketStream stream); GSocketError Connect(GSocketStream stream);
GSocketError SetNonOriented(); GSocketError SetNonOriented();
int Read(char *buffer, int size); int Read(char *buffer, int size);
@ -91,6 +93,8 @@ public:
bool m_stream; bool m_stream;
bool m_establishing; bool m_establishing;
bool m_reusable; bool m_reusable;
bool m_broadcast;
bool m_dobind;
unsigned long m_timeout; unsigned long m_timeout;
/* Callbacks */ /* Callbacks */

View File

@ -178,6 +178,11 @@ bool wxIPV4address::IsLocalHost() const
return (Hostname() == wxT("localhost") || IPAddress() == wxT("127.0.0.1")); return (Hostname() == wxT("localhost") || IPAddress() == wxT("127.0.0.1"));
} }
bool wxIPV4address::BroadcastAddress()
{
return (GAddress_INET_SetBroadcastAddress(m_address) == GSOCK_NOERROR);
}
bool wxIPV4address::AnyAddress() bool wxIPV4address::AnyAddress()
{ {
return (GAddress_INET_SetAnyAddress(m_address) == GSOCK_NOERROR); return (GAddress_INET_SetAnyAddress(m_address) == GSOCK_NOERROR);
@ -275,6 +280,11 @@ bool wxIPV6address::IsLocalHost() const
return (Hostname() == wxT("localhost") || IPAddress() == wxT("127.0.0.1")); return (Hostname() == wxT("localhost") || IPAddress() == wxT("127.0.0.1"));
} }
bool wxIPV6address::BroadcastAddress()
{
return (GAddress_INET_SetBroadcastAddress(m_address) == GSOCK_NOERROR);
}
bool wxIPV6address::AnyAddress() bool wxIPV6address::AnyAddress()
{ {
return (GAddress_INET_SetAnyAddress(m_address) == GSOCK_NOERROR); return (GAddress_INET_SetAnyAddress(m_address) == GSOCK_NOERROR);

View File

@ -1105,6 +1105,12 @@ wxSocketServer::wxSocketServer(const wxSockAddress& addr_man,
if (GetFlags() & wxSOCKET_REUSEADDR) { if (GetFlags() & wxSOCKET_REUSEADDR) {
m_socket->SetReusable(); m_socket->SetReusable();
} }
if (GetFlags() & wxSOCKET_BROADCAST) {
m_socket->SetBroadcast();
}
if (GetFlags() & wxSOCKET_NOBIND) {
m_socket->DontDoBind();
}
if (m_socket->SetServer() != GSOCK_NOERROR) if (m_socket->SetServer() != GSOCK_NOERROR)
{ {
@ -1275,6 +1281,14 @@ bool wxSocketClient::DoConnect(wxSockAddress& addr_man, wxSockAddress* local, bo
{ {
m_socket->SetReusable(); m_socket->SetReusable();
} }
if (GetFlags() & wxSOCKET_BROADCAST)
{
m_socket->SetBroadcast();
}
if (GetFlags() & wxSOCKET_NOBIND)
{
m_socket->DontDoBind();
}
// If no local address was passed and one has been set, use the one that was Set // If no local address was passed and one has been set, use the one that was Set
if (!local && m_localAddress.GetAddress()) if (!local && m_localAddress.GetAddress())
@ -1355,6 +1369,14 @@ wxDatagramSocket::wxDatagramSocket( const wxSockAddress& addr,
{ {
m_socket->SetReusable(); m_socket->SetReusable();
} }
if (GetFlags() & wxSOCKET_BROADCAST)
{
m_socket->SetBroadcast();
}
if (GetFlags() & wxSOCKET_NOBIND)
{
m_socket->DontDoBind();
}
if ( m_socket->SetNonOriented() != GSOCK_NOERROR ) if ( m_socket->SetNonOriented() != GSOCK_NOERROR )
{ {
delete m_socket; delete m_socket;

View File

@ -1562,6 +1562,11 @@ GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname)
return GSOCK_NOERROR; return GSOCK_NOERROR;
} }
GSocketError GAddress_INET_SetBroadcastAddress(GAddress *address)
{
return GAddress_INET_SetHostAddress(address, INADDR_BROADCAST);
}
GSocketError GAddress_INET_SetAnyAddress(GAddress *address) GSocketError GAddress_INET_SetAnyAddress(GAddress *address)
{ {
return GAddress_INET_SetHostAddress(address, INADDR_ANY); return GAddress_INET_SetHostAddress(address, INADDR_ANY);

View File

@ -1314,6 +1314,11 @@ GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname)
return GSOCK_NOERROR; return GSOCK_NOERROR;
} }
GSocketError GAddress_INET_SetBroadcastAddress(GAddress *address)
{
return GAddress_INET_SetHostAddress(address, INADDR_BROADCAST);
}
GSocketError GAddress_INET_SetAnyAddress(GAddress *address) GSocketError GAddress_INET_SetAnyAddress(GAddress *address)
{ {
return GAddress_INET_SetHostAddress(address, INADDR_ANY); return GAddress_INET_SetHostAddress(address, INADDR_ANY);

View File

@ -177,6 +177,8 @@ GSocket::GSocket()
m_timeout.tv_usec = 0; m_timeout.tv_usec = 0;
m_establishing = false; m_establishing = false;
m_reusable = false; m_reusable = false;
m_broadcast = false;
m_dobind = true;
assert(gs_gui_functions); assert(gs_gui_functions);
/* Per-socket GUI-specific initialization */ /* Per-socket GUI-specific initialization */
@ -529,6 +531,34 @@ bool GSocket::SetReusable()
return false; return false;
} }
/* GSocket_SetBroadcast:
* Simply sets the m_broadcast flag on the socket. GSocket_SetServer will
* make the appropriate setsockopt() call.
* Implemented as a GSocket function because clients (ie, wxSocketServer)
* don't have access to the GSocket struct information.
* Returns true if the flag was set correctly, false if an error occurred
* (ie, if the parameter was NULL)
*/
bool GSocket::SetBroadcast()
{
/* socket must not be in use/already bound */
if (m_fd == INVALID_SOCKET) {
m_broadcast = true;
return true;
}
return false;
}
bool GSocket::DontDoBind()
{
/* socket must not be in use/already bound */
if (m_fd == INVALID_SOCKET) {
m_dobind = false;
return true;
}
return false;
}
/* Client specific parts */ /* Client specific parts */
/* GSocket_Connect: /* GSocket_Connect:
@ -706,18 +736,24 @@ GSocketError GSocket::SetNonOriented()
{ {
setsockopt(m_fd, SOL_SOCKET, SO_REUSEADDR, (const char*)&arg, sizeof(arg)); setsockopt(m_fd, SOL_SOCKET, SO_REUSEADDR, (const char*)&arg, sizeof(arg));
} }
if (m_broadcast)
/* Bind to the local address,
* and retrieve the actual address bound.
*/
if ((bind(m_fd, m_local->m_addr, m_local->m_len) != 0) ||
(getsockname(m_fd,
m_local->m_addr,
(WX_SOCKLEN_T *)&m_local->m_len) != 0))
{ {
Close(); setsockopt(m_fd, SOL_SOCKET, SO_BROADCAST, (const char*)&arg, sizeof(arg));
m_error = GSOCK_IOERR; }
return GSOCK_IOERR; if (m_dobind)
{
/* Bind to the local address,
* and retrieve the actual address bound.
*/
if ((bind(m_fd, m_local->m_addr, m_local->m_len) != 0) ||
(getsockname(m_fd,
m_local->m_addr,
(WX_SOCKLEN_T *)&m_local->m_len) != 0))
{
Close();
m_error = GSOCK_IOERR;
return GSOCK_IOERR;
}
} }
return GSOCK_NOERROR; return GSOCK_NOERROR;
@ -1413,6 +1449,11 @@ GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname)
return GSOCK_NOERROR; return GSOCK_NOERROR;
} }
GSocketError GAddress_INET_SetBroadcastAddress(GAddress *address)
{
return GAddress_INET_SetHostAddress(address, INADDR_BROADCAST);
}
GSocketError GAddress_INET_SetAnyAddress(GAddress *address) GSocketError GAddress_INET_SetAnyAddress(GAddress *address)
{ {
return GAddress_INET_SetHostAddress(address, INADDR_ANY); return GAddress_INET_SetHostAddress(address, INADDR_ANY);

View File

@ -533,6 +533,8 @@ GSocket::GSocket()
m_gui_dependent = NULL; m_gui_dependent = NULL;
m_non_blocking = false; m_non_blocking = false;
m_reusable = false; m_reusable = false;
m_broadcast = false;
m_dobind = true;
m_timeout = 10*60*1000; m_timeout = 10*60*1000;
/* 10 minutes * 60 sec * 1000 millisec */ /* 10 minutes * 60 sec * 1000 millisec */
m_establishing = false; m_establishing = false;
@ -903,6 +905,26 @@ bool GSocket::SetReusable()
return false; return false;
} }
bool GSocket::SetBroadcast()
{
/* socket must not be in use/already bound */
if (m_fd == INVALID_SOCKET) {
m_broadcast = true;
return true;
}
return false;
}
bool GSocket::DontDoBind()
{
/* socket must not be in use/already bound */
if (m_fd == INVALID_SOCKET) {
m_dobind = false;
return true;
}
return false;
}
/* Client specific parts */ /* Client specific parts */
/* GSocket_Connect: /* GSocket_Connect:
@ -1119,19 +1141,25 @@ GSocketError GSocket::SetNonOriented()
#endif #endif
} }
/* Bind to the local address, if (m_broadcast)
* and retrieve the actual address bound.
*/
if ((bind(m_fd, m_local->m_addr, m_local->m_len) != 0) ||
(getsockname(m_fd,
m_local->m_addr,
(WX_SOCKLEN_T *) &m_local->m_len) != 0))
{ {
Close(); setsockopt(m_fd, SOL_SOCKET, SO_BROADCAST, (const char*)&arg, sizeof(arg));
m_error = GSOCK_IOERR; }
return GSOCK_IOERR; if (m_dobind)
{
/* Bind to the local address,
* and retrieve the actual address bound.
*/
if ((bind(m_fd, m_local->m_addr, m_local->m_len) != 0) ||
(getsockname(m_fd,
m_local->m_addr,
(WX_SOCKLEN_T *) &m_local->m_len) != 0))
{
Close();
m_error = GSOCK_IOERR;
return GSOCK_IOERR;
}
} }
return GSOCK_NOERROR; return GSOCK_NOERROR;
} }
@ -2068,6 +2096,12 @@ GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname)
return GSOCK_NOERROR; return GSOCK_NOERROR;
} }
GSocketError GAddress_INET_SetBroadcastAddress(GAddress *address)
{
return GAddress_INET_SetHostAddress(address, INADDR_BROADCAST);
}
GSocketError GAddress_INET_SetAnyAddress(GAddress *address) GSocketError GAddress_INET_SetAnyAddress(GAddress *address)
{ {
return GAddress_INET_SetHostAddress(address, INADDR_ANY); return GAddress_INET_SetHostAddress(address, INADDR_ANY);