mirror of https://github.com/n7tae/new-xlxd.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.
183 lines
4.7 KiB
183 lines
4.7 KiB
//
|
|
// cudpsocket.cpp
|
|
// xlxd
|
|
//
|
|
// Created by Jean-Luc Deltombe (LX3JL) on 31/10/2015.
|
|
// Copyright © 2015 Jean-Luc Deltombe (LX3JL). All rights reserved.
|
|
// Copyright © 2020 Thomas A. Early, N7TAE
|
|
//
|
|
// ----------------------------------------------------------------------------
|
|
// This file is part of xlxd.
|
|
//
|
|
// xlxd 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.
|
|
//
|
|
// xlxd 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 Foobar. If not, see <http://www.gnu.org/licenses/>.
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#include <string.h>
|
|
#include "main.h"
|
|
#include "cudpsocket.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 vaild 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 (0 > rval)
|
|
{
|
|
std::cerr << "select error on UPD port " << m_addr << ": " << strerror(errno) << std::endl;
|
|
return false;
|
|
}
|
|
else if (0 == rval)
|
|
return false;
|
|
|
|
// 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);
|
|
|
|
// handle
|
|
if (0 >= iRecvLen)
|
|
return false;
|
|
|
|
Buffer.Set(buf, iRecvLen);
|
|
|
|
// done
|
|
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());
|
|
}
|