mirror of https://github.com/nostar/urfd.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
181 lines
4.6 KiB
181 lines
4.6 KiB
// Copyright © 2015 Jean-Luc Deltombe (LX3JL). All rights reserved.
|
|
|
|
// urfd -- The universal reflector
|
|
// Copyright © 2021 Thomas A. Early N7TAE
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
#include <string.h>
|
|
|
|
#include "UDPSocket.h"
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
// constructor
|
|
|
|
CUdpSocket::CUdpSocket() : m_fd(-1) {}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
// destructor
|
|
|
|
CUdpSocket::~CUdpSocket()
|
|
{
|
|
Close();
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
// open & close
|
|
|
|
// returns true on error
|
|
bool CUdpSocket::Open(const CIp &Ip)
|
|
{
|
|
// check for a valid family
|
|
if (AF_UNSPEC == Ip.GetFamily())
|
|
return true;
|
|
|
|
// create socket
|
|
m_fd = socket(Ip.GetFamily(), SOCK_DGRAM, 0);
|
|
if ( m_fd < 0 )
|
|
{
|
|
std::cerr << "Unable to open socket on " << Ip << ", " << strerror(errno) << std::endl;
|
|
return false;
|
|
}
|
|
// initialize sockaddr struct
|
|
m_addr = Ip;
|
|
|
|
int reuse = 1;
|
|
if ( 0 > setsockopt(m_fd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(int)))
|
|
{
|
|
std::cerr << "Cannot set the UDP socket option on " << m_addr << ", " << strerror(errno) << std::endl;
|
|
Close();
|
|
return false;
|
|
}
|
|
|
|
if (fcntl(m_fd, F_SETFL, O_NONBLOCK))
|
|
{
|
|
std::cerr << "fcntl set non-blocking failed on " << m_addr << ", " << strerror(errno) << std::endl;
|
|
Close();
|
|
return false;
|
|
}
|
|
|
|
if ( bind(m_fd, m_addr.GetCPointer(), m_addr.GetSize()) )
|
|
{
|
|
std::cerr << "bind failed on " << m_addr << ", " << strerror(errno) << std::endl;
|
|
Close();
|
|
return false;
|
|
}
|
|
|
|
if (0 == m_addr.GetPort()) // get the assigned port for an ephemeral port request
|
|
{
|
|
CIp a;
|
|
socklen_t len = sizeof(struct sockaddr_storage);
|
|
if (getsockname(m_fd, a.GetPointer(), &len))
|
|
{
|
|
std::cerr << "getsockname error " << m_addr << ", " << strerror(errno) << std::endl;
|
|
Close();
|
|
return false;
|
|
}
|
|
if (a != m_addr)
|
|
std::cout << "getsockname didn't return the same address as set: returned " << a << ", should have been " << m_addr << std::endl;
|
|
|
|
m_addr.SetPort(a.GetPort());
|
|
}
|
|
|
|
// done
|
|
return true;
|
|
}
|
|
|
|
void CUdpSocket::Close(void)
|
|
{
|
|
if ( m_fd >= 0 )
|
|
{
|
|
close(m_fd);
|
|
m_fd = -1;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
// read
|
|
|
|
bool CUdpSocket::Receive(CBuffer &Buffer, CIp &Ip, int timeout)
|
|
{
|
|
// socket valid ?
|
|
if ( 0 > m_fd )
|
|
return false;
|
|
|
|
// control socket
|
|
fd_set FdSet;
|
|
FD_ZERO(&FdSet);
|
|
FD_SET(m_fd, &FdSet);
|
|
struct timeval tv;
|
|
tv.tv_sec = timeout / 1000;
|
|
tv.tv_usec = (timeout % 1000) * 1000;
|
|
|
|
auto rval = select(m_fd + 1, &FdSet, 0, 0, &tv);
|
|
if (rval > 0)
|
|
return ReceiveFrom(Buffer, Ip);
|
|
|
|
if (rval < 0)
|
|
std::cerr << "select error on UPD port " << m_addr << ": " << strerror(errno) << std::endl;
|
|
|
|
return false;
|
|
}
|
|
|
|
bool CUdpSocket::ReceiveFrom(CBuffer &Buffer, CIp &ip)
|
|
{
|
|
// read
|
|
uint8_t buf[UDP_BUFFER_LENMAX];
|
|
unsigned int fromsize = sizeof(struct sockaddr_storage);
|
|
auto iRecvLen = recvfrom(m_fd, buf, UDP_BUFFER_LENMAX, 0, ip.GetPointer(), &fromsize);
|
|
|
|
if (0 >= iRecvLen)
|
|
return false;
|
|
|
|
Buffer.Set(buf, iRecvLen);
|
|
|
|
return true;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
// write
|
|
|
|
void CUdpSocket::Send(const CBuffer &Buffer, const CIp &Ip) const
|
|
{
|
|
sendto(m_fd, Buffer.data(), Buffer.size(), 0, Ip.GetCPointer(), Ip.GetSize());
|
|
}
|
|
|
|
void CUdpSocket::Send(const char *Buffer, const CIp &Ip) const
|
|
{
|
|
sendto(m_fd, Buffer, ::strlen(Buffer), 0, Ip.GetCPointer(), Ip.GetSize());
|
|
}
|
|
|
|
void CUdpSocket::Send(const CBuffer &Buffer, const CIp &Ip, uint16_t destport) const
|
|
{
|
|
CIp temp(Ip);
|
|
temp.SetPort(destport);
|
|
sendto(m_fd, Buffer.data(), Buffer.size(), 0, temp.GetCPointer(), temp.GetSize());
|
|
}
|
|
|
|
void CUdpSocket::Send(const char *Buffer, const CIp &Ip, uint16_t destport) const
|
|
{
|
|
CIp temp(Ip);
|
|
temp.SetPort(destport);
|
|
sendto(m_fd, Buffer, ::strlen(Buffer), 0, temp.GetCPointer(), temp.GetSize());
|
|
}
|
|
|
|
void CUdpSocket::Send(const uint8_t *data, size_t size, const CIp &Ip) const
|
|
{
|
|
sendto(m_fd, data, size, 0, Ip.GetCPointer(), Ip.GetSize());
|
|
}
|