2007-12-08 19:34:15 +00:00
|
|
|
/*
|
2011-01-19 13:48:47 +00:00
|
|
|
* This file Copyright (C) Mnemosyne LLC
|
2006-09-25 18:37:45 +00:00
|
|
|
*
|
2010-12-27 19:18:17 +00:00
|
|
|
* This file is licensed by the GPL version 2. Works owned by the
|
2012-12-05 17:29:46 +00:00
|
|
|
* Transmission project are granted a special exemption to clause 2 (b)
|
2008-09-23 19:11:04 +00:00
|
|
|
* so that the bulk of its code can remain under the MIT license.
|
2007-12-08 19:34:15 +00:00
|
|
|
* This exemption does not extend to derived works not owned by
|
|
|
|
* the Transmission project.
|
2006-09-25 18:37:45 +00:00
|
|
|
*
|
2007-12-13 01:20:16 +00:00
|
|
|
* $Id$
|
2007-12-08 19:34:15 +00:00
|
|
|
*/
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2007-07-29 18:11:21 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <time.h>
|
2007-12-08 19:34:15 +00:00
|
|
|
#include <inttypes.h>
|
2007-07-29 18:11:21 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
#include <event2/util.h> /* evutil_inet_ntop () */
|
2011-03-24 22:57:39 +00:00
|
|
|
|
2008-07-15 03:26:53 +00:00
|
|
|
#define ENABLE_STRNATPMPERR
|
2012-02-04 01:28:15 +00:00
|
|
|
#include "natpmp.h"
|
2007-07-14 16:29:21 +00:00
|
|
|
|
2006-09-25 18:37:45 +00:00
|
|
|
#include "transmission.h"
|
2012-02-04 01:28:15 +00:00
|
|
|
#include "natpmp_local.h"
|
2013-01-25 23:34:20 +00:00
|
|
|
#include "log.h"
|
2011-03-24 22:57:39 +00:00
|
|
|
#include "net.h" /* tr_netCloseSocket */
|
2008-04-11 17:01:13 +00:00
|
|
|
#include "port-forwarding.h"
|
2007-07-30 18:04:10 +00:00
|
|
|
#include "utils.h"
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2007-12-08 19:34:15 +00:00
|
|
|
#define LIFETIME_SECS 3600
|
2007-12-13 19:28:51 +00:00
|
|
|
#define COMMAND_WAIT_SECS 8
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
static const char *
|
2012-12-07 01:53:31 +00:00
|
|
|
getKey (void) { return _("Port Forwarding (NAT-PMP)"); }
|
2008-03-07 03:26:59 +00:00
|
|
|
|
2007-12-08 19:34:15 +00:00
|
|
|
typedef enum
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
TR_NATPMP_IDLE,
|
|
|
|
TR_NATPMP_ERR,
|
2007-12-13 18:56:22 +00:00
|
|
|
TR_NATPMP_DISCOVER,
|
2007-12-08 19:34:15 +00:00
|
|
|
TR_NATPMP_RECV_PUB,
|
|
|
|
TR_NATPMP_SEND_MAP,
|
|
|
|
TR_NATPMP_RECV_MAP,
|
|
|
|
TR_NATPMP_SEND_UNMAP,
|
|
|
|
TR_NATPMP_RECV_UNMAP
|
|
|
|
}
|
|
|
|
tr_natpmp_state;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
struct tr_natpmp
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2011-03-22 15:19:54 +00:00
|
|
|
bool has_discovered;
|
|
|
|
bool is_mapped;
|
2010-07-05 21:04:17 +00:00
|
|
|
|
|
|
|
tr_port public_port;
|
|
|
|
tr_port private_port;
|
|
|
|
|
|
|
|
time_t renew_time;
|
|
|
|
time_t command_time;
|
|
|
|
tr_natpmp_state state;
|
|
|
|
natpmp_t natpmp;
|
2006-09-25 18:37:45 +00:00
|
|
|
};
|
|
|
|
|
2007-12-08 19:34:15 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
2007-02-06 03:24:55 +00:00
|
|
|
|
2007-04-15 07:36:24 +00:00
|
|
|
static void
|
2012-12-05 17:29:46 +00:00
|
|
|
logVal (const char * func,
|
|
|
|
int ret)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
if (ret == NATPMP_TRYAGAIN)
|
2008-04-14 20:06:37 +00:00
|
|
|
return;
|
2012-12-05 17:29:46 +00:00
|
|
|
if (ret >= 0)
|
2013-01-25 23:34:20 +00:00
|
|
|
tr_logAddNamedInfo (getKey (), _("%s succeeded (%d)"), func, ret);
|
2006-09-25 18:37:45 +00:00
|
|
|
else
|
2013-01-25 23:34:20 +00:00
|
|
|
tr_logAddNamedDbg (
|
2012-12-05 17:29:46 +00:00
|
|
|
getKey (),
|
2010-12-27 19:18:17 +00:00
|
|
|
"%s failed. Natpmp returned %d (%s); errno is %d (%s)",
|
2012-12-05 17:29:46 +00:00
|
|
|
func, ret, strnatpmperr (ret), errno, tr_strerror (errno));
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2007-12-08 19:34:15 +00:00
|
|
|
struct tr_natpmp*
|
2012-12-05 17:29:46 +00:00
|
|
|
tr_natpmpInit (void)
|
2007-04-15 07:36:24 +00:00
|
|
|
{
|
2007-12-13 18:56:22 +00:00
|
|
|
struct tr_natpmp * nat;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
nat = tr_new0 (struct tr_natpmp, 1);
|
2007-12-13 18:56:22 +00:00
|
|
|
nat->state = TR_NATPMP_DISCOVER;
|
2010-07-05 21:04:17 +00:00
|
|
|
nat->public_port = 0;
|
|
|
|
nat->private_port = 0;
|
2009-10-27 20:29:02 +00:00
|
|
|
nat->natpmp.s = -1; /* socket */
|
2007-12-08 19:34:15 +00:00
|
|
|
return nat;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2012-12-05 17:29:46 +00:00
|
|
|
tr_natpmpClose (tr_natpmp * nat)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat)
|
2008-04-12 02:26:45 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat->natpmp.s >= 0)
|
|
|
|
tr_netCloseSocket (nat->natpmp.s);
|
|
|
|
tr_free (nat);
|
2008-04-12 02:26:45 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2007-12-13 19:28:51 +00:00
|
|
|
static int
|
2012-12-05 17:29:46 +00:00
|
|
|
canSendCommand (const struct tr_natpmp * nat)
|
2007-12-13 19:28:51 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
return tr_time () >= nat->command_time;
|
2007-12-13 19:28:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2012-12-05 17:29:46 +00:00
|
|
|
setCommandTime (struct tr_natpmp * nat)
|
2007-12-13 19:28:51 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
nat->command_time = tr_time () + COMMAND_WAIT_SECS;
|
2007-12-13 19:28:51 +00:00
|
|
|
}
|
|
|
|
|
2007-12-08 19:34:15 +00:00
|
|
|
int
|
2012-12-05 17:29:46 +00:00
|
|
|
tr_natpmpPulse (struct tr_natpmp * nat, tr_port private_port, bool is_enabled, tr_port * public_port)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
int ret;
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (is_enabled && (nat->state == TR_NATPMP_DISCOVER))
|
2007-12-13 18:56:22 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
int val = initnatpmp (&nat->natpmp, 0, 0);
|
|
|
|
logVal ("initnatpmp", val);
|
|
|
|
val = sendpublicaddressrequest (&nat->natpmp);
|
|
|
|
logVal ("sendpublicaddressrequest", val);
|
2007-12-13 18:56:22 +00:00
|
|
|
nat->state = val < 0 ? TR_NATPMP_ERR : TR_NATPMP_RECV_PUB;
|
2011-03-22 15:19:54 +00:00
|
|
|
nat->has_discovered = true;
|
2012-12-05 17:29:46 +00:00
|
|
|
setCommandTime (nat);
|
2007-12-13 18:56:22 +00:00
|
|
|
}
|
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if ((nat->state == TR_NATPMP_RECV_PUB) && canSendCommand (nat))
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
natpmpresp_t response;
|
2012-12-05 17:29:46 +00:00
|
|
|
const int val = readnatpmpresponseorretry (&nat->natpmp, &response);
|
|
|
|
logVal ("readnatpmpresponseorretry", val);
|
|
|
|
if (val >= 0)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2011-03-24 22:57:39 +00:00
|
|
|
char str[128];
|
2012-12-05 17:29:46 +00:00
|
|
|
evutil_inet_ntop (AF_INET, &response.pnu.publicaddress.addr, str, sizeof (str));
|
2013-01-25 23:34:20 +00:00
|
|
|
tr_logAddNamedInfo (getKey (), _("Found public address \"%s\""), str);
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = TR_NATPMP_IDLE;
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
2012-12-05 17:29:46 +00:00
|
|
|
else if (val != NATPMP_TRYAGAIN)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2008-04-12 21:47:10 +00:00
|
|
|
nat->state = TR_NATPMP_ERR;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if ((nat->state == TR_NATPMP_IDLE) || (nat->state == TR_NATPMP_ERR))
|
2007-02-06 03:24:55 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat->is_mapped && (!is_enabled || (nat->private_port != private_port)))
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = TR_NATPMP_SEND_UNMAP;
|
2007-02-06 03:24:55 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if ((nat->state == TR_NATPMP_SEND_UNMAP) && canSendCommand (nat))
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
const int val = sendnewportmappingrequest (&nat->natpmp, NATPMP_PROTOCOL_TCP,
|
2010-07-05 21:04:17 +00:00
|
|
|
nat->private_port,
|
|
|
|
nat->public_port,
|
2012-12-05 17:29:46 +00:00
|
|
|
0);
|
|
|
|
logVal ("sendnewportmappingrequest", val);
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = val < 0 ? TR_NATPMP_ERR : TR_NATPMP_RECV_UNMAP;
|
2012-12-05 17:29:46 +00:00
|
|
|
setCommandTime (nat);
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat->state == TR_NATPMP_RECV_UNMAP)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
natpmpresp_t resp;
|
2012-12-05 17:29:46 +00:00
|
|
|
const int val = readnatpmpresponseorretry (&nat->natpmp, &resp);
|
|
|
|
logVal ("readnatpmpresponseorretry", val);
|
|
|
|
if (val >= 0)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2010-07-05 21:04:17 +00:00
|
|
|
const int private_port = resp.pnu.newportmapping.privateport;
|
|
|
|
|
2013-01-25 23:34:20 +00:00
|
|
|
tr_logAddNamedInfo (getKey (), _("no longer forwarding port %d"), private_port);
|
2010-07-05 21:04:17 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat->private_port == private_port)
|
2008-11-01 16:39:57 +00:00
|
|
|
{
|
2010-07-05 21:04:17 +00:00
|
|
|
nat->private_port = 0;
|
|
|
|
nat->public_port = 0;
|
2008-11-01 16:39:57 +00:00
|
|
|
nat->state = TR_NATPMP_IDLE;
|
2011-03-22 15:19:54 +00:00
|
|
|
nat->is_mapped = false;
|
2008-11-01 16:39:57 +00:00
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
2012-12-05 17:29:46 +00:00
|
|
|
else if (val != NATPMP_TRYAGAIN)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2008-04-12 21:47:10 +00:00
|
|
|
nat->state = TR_NATPMP_ERR;
|
2007-02-06 03:24:55 +00:00
|
|
|
}
|
2007-01-28 08:54:19 +00:00
|
|
|
}
|
2007-02-06 03:24:55 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat->state == TR_NATPMP_IDLE)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
if (is_enabled && !nat->is_mapped && nat->has_discovered)
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = TR_NATPMP_SEND_MAP;
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
else if (nat->is_mapped && tr_time () >= nat->renew_time)
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = TR_NATPMP_SEND_MAP;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
2007-02-06 03:24:55 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if ((nat->state == TR_NATPMP_SEND_MAP) && canSendCommand (nat))
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2012-12-05 17:29:46 +00:00
|
|
|
const int val = sendnewportmappingrequest (&nat->natpmp, NATPMP_PROTOCOL_TCP, private_port, private_port, LIFETIME_SECS);
|
|
|
|
logVal ("sendnewportmappingrequest", val);
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = val < 0 ? TR_NATPMP_ERR : TR_NATPMP_RECV_MAP;
|
2012-12-05 17:29:46 +00:00
|
|
|
setCommandTime (nat);
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (nat->state == TR_NATPMP_RECV_MAP)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
natpmpresp_t resp;
|
2012-12-05 17:29:46 +00:00
|
|
|
const int val = readnatpmpresponseorretry (&nat->natpmp, &resp);
|
|
|
|
logVal ("readnatpmpresponseorretry", val);
|
|
|
|
if (val >= 0)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
nat->state = TR_NATPMP_IDLE;
|
2011-03-22 15:19:54 +00:00
|
|
|
nat->is_mapped = true;
|
2012-12-05 17:29:46 +00:00
|
|
|
nat->renew_time = tr_time () + (resp.pnu.newportmapping.lifetime / 2);
|
2010-07-05 21:04:17 +00:00
|
|
|
nat->private_port = resp.pnu.newportmapping.privateport;
|
|
|
|
nat->public_port = resp.pnu.newportmapping.mappedpublicport;
|
2013-01-25 23:34:20 +00:00
|
|
|
tr_logAddNamedInfo (getKey (), _("Port %d forwarded successfully"), nat->private_port);
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
2012-12-05 17:29:46 +00:00
|
|
|
else if (val != NATPMP_TRYAGAIN)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2008-04-12 21:47:10 +00:00
|
|
|
nat->state = TR_NATPMP_ERR;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
switch (nat->state)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
|
|
|
case TR_NATPMP_IDLE:
|
2010-07-05 21:04:17 +00:00
|
|
|
*public_port = nat->public_port;
|
|
|
|
return nat->is_mapped ? TR_PORT_MAPPED : TR_PORT_UNMAPPED;
|
|
|
|
break;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
|
|
|
case TR_NATPMP_DISCOVER:
|
|
|
|
ret = TR_PORT_UNMAPPED; break;
|
|
|
|
|
2007-12-13 18:56:22 +00:00
|
|
|
case TR_NATPMP_RECV_PUB:
|
|
|
|
case TR_NATPMP_SEND_MAP:
|
2008-09-23 19:11:04 +00:00
|
|
|
case TR_NATPMP_RECV_MAP:
|
|
|
|
ret = TR_PORT_MAPPING; break;
|
|
|
|
|
2007-12-13 18:56:22 +00:00
|
|
|
case TR_NATPMP_SEND_UNMAP:
|
2008-09-23 19:11:04 +00:00
|
|
|
case TR_NATPMP_RECV_UNMAP:
|
|
|
|
ret = TR_PORT_UNMAPPING; break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = TR_PORT_ERROR; break;
|
2007-12-13 18:56:22 +00:00
|
|
|
}
|
2007-12-08 19:34:15 +00:00
|
|
|
return ret;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
|