#16 Add G2 Pool, start some IPV6 changes
parent
53b5d469a8
commit
4d4f9f3a52
@ -0,0 +1,169 @@
|
||||
/*
|
||||
* Copyright (c) 2021-2022 by Geoffrey Merck F4FXL / KC3FRA
|
||||
*
|
||||
* 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 2 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, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#include <cassert>
|
||||
|
||||
#include "Log.h"
|
||||
#include "G2ProtocolHandlerPool.h"
|
||||
|
||||
const unsigned int G2_BUFFER_LENGTH = 255U;
|
||||
|
||||
CG2ProtocolHandlerPool::CG2ProtocolHandlerPool(unsigned short port, const std::string& address) :
|
||||
m_address(address),
|
||||
m_basePort(port),
|
||||
m_socket(address, port)
|
||||
{
|
||||
assert(port > 0U);
|
||||
m_index = m_pool.end();
|
||||
}
|
||||
|
||||
CG2ProtocolHandlerPool::~CG2ProtocolHandlerPool()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
bool CG2ProtocolHandlerPool::open()
|
||||
{
|
||||
bool res = m_socket.open();
|
||||
return res;
|
||||
}
|
||||
|
||||
void CG2ProtocolHandlerPool::close()
|
||||
{
|
||||
for(auto handler : m_pool) {
|
||||
delete handler;
|
||||
}
|
||||
m_pool.clear();
|
||||
m_index = m_pool.end();
|
||||
m_socket.close();
|
||||
}
|
||||
|
||||
G2_TYPE CG2ProtocolHandlerPool::read()
|
||||
{
|
||||
bool res = true;
|
||||
while(res)
|
||||
res = readPackets();
|
||||
|
||||
if(m_index == m_pool.end())
|
||||
m_index = m_pool.begin();
|
||||
|
||||
while(m_index != m_pool.end()) {
|
||||
if((*m_index)->getType() != GT_NONE) {
|
||||
return (*m_index)->getType();
|
||||
}
|
||||
m_index++;
|
||||
}
|
||||
|
||||
return GT_NONE;
|
||||
}
|
||||
|
||||
CAMBEData * CG2ProtocolHandlerPool::readAMBE()
|
||||
{
|
||||
if(m_index == m_pool.end() || (*m_index)->getType() != GT_AMBE)
|
||||
return nullptr;
|
||||
|
||||
return (*m_index)->readAMBE();
|
||||
}
|
||||
|
||||
CHeaderData * CG2ProtocolHandlerPool::readHeader()
|
||||
{
|
||||
if(m_index == m_pool.end() || (*m_index)->getType() != GT_HEADER)
|
||||
return nullptr;
|
||||
|
||||
return (*m_index)->readHeader();
|
||||
}
|
||||
|
||||
bool CG2ProtocolHandlerPool::readPackets()
|
||||
{
|
||||
unsigned char buffer[G2_BUFFER_LENGTH];
|
||||
struct sockaddr_storage addr;
|
||||
::memset(&addr, 0, sizeof(sockaddr_storage));
|
||||
|
||||
// No more data?
|
||||
int length = m_socket.read(buffer, G2_BUFFER_LENGTH, addr);
|
||||
if(length <= 0) return false;
|
||||
|
||||
if(length == 1 && buffer[0] == 0U) {
|
||||
CLog::logDebug("G2 Nat traversal packet received");
|
||||
}
|
||||
|
||||
CG2ProtocolHandler * handler = findHandler(addr, IMT_ADDRESS_AND_PORT);
|
||||
if(handler == nullptr) {
|
||||
CLog::logTrace("new incoming G2 %s:%u", inet_ntoa(TOIPV4(addr)->sin_addr), ntohs(TOIPV4(addr)->sin_port));
|
||||
handler = new CG2ProtocolHandler(&m_socket, addr, G2_BUFFER_LENGTH);
|
||||
m_pool.push_back(handler);
|
||||
m_index = m_pool.end();
|
||||
}
|
||||
|
||||
bool res = handler->setBuffer(buffer, length);
|
||||
return res;
|
||||
}
|
||||
|
||||
void CG2ProtocolHandlerPool::traverseNat(const std::string& address)
|
||||
{
|
||||
unsigned char buffer = 0x00U;
|
||||
|
||||
in_addr addr = CUDPReaderWriter::lookup(address);
|
||||
|
||||
CLog::logInfo("G2 Punching hole to %s", address.c_str());
|
||||
|
||||
m_socket.write(&buffer, 1U, addr, G2_DV_PORT);
|
||||
}
|
||||
|
||||
bool CG2ProtocolHandlerPool::writeHeader(const CHeaderData& header)
|
||||
{
|
||||
auto handler = findHandler(header.getDestination(), IMT_ADDRESS_ONLY);
|
||||
if(handler == nullptr) {
|
||||
handler = new CG2ProtocolHandler(&m_socket, header.getDestination(), G2_BUFFER_LENGTH);
|
||||
m_pool.push_back(handler);
|
||||
m_index = m_pool.end();
|
||||
}
|
||||
return handler->writeHeader(header);
|
||||
}
|
||||
|
||||
bool CG2ProtocolHandlerPool::writeAMBE(const CAMBEData& data)
|
||||
{
|
||||
auto handler = findHandler(data.getDestination(), IMT_ADDRESS_ONLY);
|
||||
if(handler == nullptr) {
|
||||
handler = new CG2ProtocolHandler(&m_socket, data.getDestination(), G2_BUFFER_LENGTH);
|
||||
m_pool.push_back(handler);
|
||||
m_index = m_pool.end();
|
||||
}
|
||||
|
||||
return handler->writeAMBE(data);
|
||||
}
|
||||
|
||||
CG2ProtocolHandler * CG2ProtocolHandlerPool::findHandler(const struct sockaddr_storage& addr, IPMATCHTYPE matchType) const
|
||||
{
|
||||
for(auto handler : m_pool) {
|
||||
if(handler != nullptr && CNetUtils::match(addr, handler->getDestination(), matchType))
|
||||
return handler;
|
||||
}
|
||||
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
CG2ProtocolHandler * CG2ProtocolHandlerPool::findHandler(in_addr addr, unsigned int port, IPMATCHTYPE matchType) const
|
||||
{
|
||||
struct sockaddr_storage addrStorage;
|
||||
addrStorage.ss_family = AF_INET;
|
||||
TOIPV4(addrStorage)->sin_addr = addr;
|
||||
TOIPV4(addrStorage)->sin_port = port;
|
||||
|
||||
return findHandler(addrStorage, matchType);
|
||||
}
|
||||
@ -0,0 +1,91 @@
|
||||
/*
|
||||
* Copyright (c) 2021-2022 by Geoffrey Merck F4FXL / KC3FRA
|
||||
*
|
||||
* 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 2 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, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include <sys/socket.h>
|
||||
#include <boost/container_hash/hash.hpp>
|
||||
|
||||
#include "G2ProtocolHandler.h"
|
||||
#include "NetUtils.h"
|
||||
|
||||
struct sockaddr_storage_map {
|
||||
struct compAddrAndPort {
|
||||
bool operator() (const struct sockaddr_storage& a, const struct sockaddr_storage& b) const {
|
||||
return CNetUtils::match(a, b, IMT_ADDRESS_AND_PORT);
|
||||
}
|
||||
};
|
||||
struct hash {
|
||||
std::size_t operator() (const sockaddr_storage& a) const {
|
||||
switch(a.ss_family)
|
||||
{
|
||||
case AF_INET: {
|
||||
auto ptr4 = ((struct sockaddr_in *)&a);
|
||||
size_t res = AF_INET;
|
||||
boost::hash_combine(res, ptr4->sin_port);
|
||||
boost::hash_combine(res, ptr4->sin_addr.s_addr);
|
||||
return res;
|
||||
}
|
||||
case AF_INET6: {
|
||||
auto ptr6 = ((struct sockaddr_in6 *)&a);
|
||||
size_t res = AF_INET6;
|
||||
boost::hash_combine(res, ptr6->sin6_port);
|
||||
auto in6Ptr = (unsigned int *)&(ptr6->sin6_addr);
|
||||
boost::hash_combine(res, in6Ptr[0]);
|
||||
boost::hash_combine(res, in6Ptr[1]);
|
||||
boost::hash_combine(res, in6Ptr[2]);
|
||||
boost::hash_combine(res, in6Ptr[3]);
|
||||
return res;
|
||||
}
|
||||
default:
|
||||
return 0U;
|
||||
}
|
||||
}
|
||||
};
|
||||
};
|
||||
|
||||
class CG2ProtocolHandlerPool
|
||||
{
|
||||
public:
|
||||
CG2ProtocolHandlerPool(unsigned short g2Port, const std::string& address = "");
|
||||
~CG2ProtocolHandlerPool();
|
||||
|
||||
bool open();
|
||||
void close();
|
||||
G2_TYPE read();
|
||||
CAMBEData * readAMBE();
|
||||
CHeaderData * readHeader();
|
||||
|
||||
bool writeAMBE(const CAMBEData& data);
|
||||
bool writeHeader(const CHeaderData& header);
|
||||
|
||||
void traverseNat(const std::string& address);
|
||||
|
||||
private:
|
||||
bool readPackets();
|
||||
CG2ProtocolHandler * findHandler(const struct sockaddr_storage& addr, IPMATCHTYPE matchType) const;
|
||||
CG2ProtocolHandler * findHandler(in_addr addr, unsigned int port, IPMATCHTYPE matchType) const;
|
||||
|
||||
std::string m_address;
|
||||
unsigned int m_basePort;
|
||||
CUDPReaderWriter m_socket;
|
||||
std::vector<CG2ProtocolHandler *> m_pool;
|
||||
std::vector<CG2ProtocolHandler *>::iterator m_index;
|
||||
};
|
||||
Loading…
Reference in new issue