2009-10-07 15:37:31 +00:00
|
|
|
/*
|
|
|
|
* uhub - A tiny ADC p2p connection hub
|
|
|
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
|
|
|
*
|
|
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "uhub.h"
|
|
|
|
#include "probe.h"
|
|
|
|
|
|
|
|
#define PROBE_RECV_SIZE 12
|
|
|
|
static char probe_recvbuf[PROBE_RECV_SIZE];
|
|
|
|
|
|
|
|
static void probe_net_event(struct net_connection* con, int events, void *arg)
|
|
|
|
{
|
|
|
|
struct hub_probe* probe = (struct hub_probe*) con->ptr;
|
|
|
|
|
|
|
|
if (events == NET_EVENT_DESTROYED)
|
|
|
|
{
|
|
|
|
if (probe)
|
|
|
|
{
|
|
|
|
probe->connection = 0;
|
|
|
|
}
|
|
|
|
hub_free(con);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (events == NET_EVENT_SOCKERROR || events == NET_EVENT_CLOSED || events == NET_EVENT_TIMEOUT)
|
|
|
|
{
|
|
|
|
probe_destroy(probe);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (events & NET_EVENT_READ)
|
|
|
|
{
|
|
|
|
int bytes = net_con_peek(con, probe_recvbuf, PROBE_RECV_SIZE);
|
|
|
|
if (bytes < 0)
|
|
|
|
{
|
|
|
|
probe_destroy(probe);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bytes >= 4)
|
|
|
|
{
|
|
|
|
if (memcmp(probe_recvbuf, "HSUP", 4) == 0)
|
|
|
|
{
|
|
|
|
LOG_TRACE("Probed ADC");
|
|
|
|
if (user_create(probe->hub, probe->connection, &probe->addr))
|
|
|
|
{
|
|
|
|
probe->connection = 0;
|
|
|
|
}
|
|
|
|
probe_destroy(probe);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SSL_SUPPORT
|
2009-10-09 19:11:11 +00:00
|
|
|
if (bytes >= 11 &&
|
2009-10-07 15:37:31 +00:00
|
|
|
probe_recvbuf[0] == 22 &&
|
|
|
|
probe_recvbuf[1] == 3 && /* protocol major version */
|
|
|
|
probe_recvbuf[5] == 1 && /* message type */
|
|
|
|
probe_recvbuf[9] == probe_recvbuf[1] &&
|
|
|
|
probe_recvbuf[10] == probe_recvbuf[2])
|
|
|
|
{
|
2009-10-09 19:11:11 +00:00
|
|
|
if (probe->hub->config->tls_enable)
|
|
|
|
{
|
|
|
|
LOG_TRACE("Probed TLS %d.%d connection", (int) probe_recvbuf[1], (int) probe_recvbuf[2]);
|
2009-10-09 19:38:14 +00:00
|
|
|
if (user_create(probe->hub, probe->connection, &probe->addr))
|
|
|
|
{
|
|
|
|
probe->connection = 0;
|
|
|
|
}
|
2009-10-09 19:11:11 +00:00
|
|
|
net_con_ssl_handshake(con, NET_CON_SSL_MODE_SERVER);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LOG_TRACE("Probed TLS %d.%d connection. TLS disabled in hub.", (int) probe_recvbuf[1], (int) probe_recvbuf[2]);
|
|
|
|
}
|
2009-10-09 19:38:14 +00:00
|
|
|
probe_destroy(probe);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-11-18 16:38:25 +00:00
|
|
|
LOG_TRACE("Probed TLS %d.%d connection", (int) probe_recvbuf[1], (int) probe_recvbuf[2]);
|
|
|
|
|
|
|
|
net_con_ssl_handshake(con, NET_CON_SSL_MODE_SERVER);
|
2009-10-07 15:37:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2009-11-18 16:43:11 +00:00
|
|
|
#endif
|
2009-10-07 15:37:31 +00:00
|
|
|
probe_destroy(probe);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct hub_probe* probe_create(struct hub_info* hub, int sd, struct ip_addr_encap* addr)
|
|
|
|
{
|
|
|
|
struct hub_probe* probe = (struct hub_probe*) hub_malloc_zero(sizeof(struct hub_probe));
|
|
|
|
|
|
|
|
if (probe == NULL)
|
|
|
|
return NULL; /* OOM */
|
|
|
|
|
|
|
|
probe->hub = hub;
|
|
|
|
probe->connection = (struct net_connection*) hub_malloc(sizeof(struct net_connection));
|
|
|
|
net_con_initialize(probe->connection, sd, probe_net_event, probe, NET_EVENT_READ);
|
|
|
|
net_con_set_timeout(probe->connection, TIMEOUT_CONNECTED);
|
|
|
|
|
|
|
|
memcpy(&probe->addr, addr, sizeof(struct ip_addr_encap));
|
|
|
|
return probe;
|
|
|
|
}
|
|
|
|
|
|
|
|
void probe_destroy(struct hub_probe* probe)
|
|
|
|
{
|
|
|
|
if (probe->connection)
|
|
|
|
{
|
|
|
|
if (net_con_close(probe->connection))
|
|
|
|
{
|
|
|
|
hub_free(probe->connection);
|
|
|
|
}
|
|
|
|
probe->connection = 0;
|
|
|
|
}
|
|
|
|
hub_free(probe);
|
|
|
|
}
|