2007-11-06 16:02:50 +00:00
|
|
|
/*
|
2014-01-19 01:09:44 +00:00
|
|
|
* This file Copyright (C) 2007-2014 Mnemosyne LLC
|
2006-09-25 18:37:45 +00:00
|
|
|
*
|
2014-01-21 03:10:30 +00:00
|
|
|
* It may be used under the GNU GPL versions 2 or 3
|
2014-01-19 01:09:44 +00:00
|
|
|
* or any future license endorsed by Mnemosyne LLC.
|
2006-09-25 18:37:45 +00:00
|
|
|
*
|
2007-11-06 16:02:50 +00:00
|
|
|
*/
|
2007-07-29 18:11:21 +00:00
|
|
|
|
2021-10-17 20:17:18 +00:00
|
|
|
#include <cerrno>
|
2007-08-04 02:55:06 +00:00
|
|
|
|
2011-10-09 02:05:52 +00:00
|
|
|
#ifdef SYSTEM_MINIUPNP
|
2017-04-19 12:04:45 +00:00
|
|
|
#include <miniupnpc/miniupnpc.h>
|
|
|
|
#include <miniupnpc/upnpcommands.h>
|
2011-10-09 02:05:52 +00:00
|
|
|
#else
|
2017-04-19 12:04:45 +00:00
|
|
|
#include <miniupnp/miniupnpc.h>
|
|
|
|
#include <miniupnp/upnpcommands.h>
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
2007-07-14 16:29:21 +00:00
|
|
|
|
2006-09-25 18:37:45 +00:00
|
|
|
#include "transmission.h"
|
2013-01-25 23:34:20 +00:00
|
|
|
#include "log.h"
|
2008-04-11 17:01:13 +00:00
|
|
|
#include "port-forwarding.h"
|
|
|
|
#include "session.h"
|
2017-06-08 07:24:12 +00:00
|
|
|
#include "tr-assert.h"
|
2007-11-06 16:02:50 +00:00
|
|
|
#include "upnp.h"
|
2009-07-01 14:58:57 +00:00
|
|
|
#include "utils.h"
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static char const* getKey(void)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
|
|
|
return _("Port Forwarding (UPnP)");
|
|
|
|
}
|
2008-03-07 03:26:59 +00:00
|
|
|
|
2021-10-06 14:26:07 +00:00
|
|
|
enum tr_upnp_state
|
2007-12-08 19:34:15 +00:00
|
|
|
{
|
|
|
|
TR_UPNP_IDLE,
|
|
|
|
TR_UPNP_ERR,
|
|
|
|
TR_UPNP_DISCOVER,
|
|
|
|
TR_UPNP_MAP,
|
|
|
|
TR_UPNP_UNMAP
|
2021-10-06 14:26:07 +00:00
|
|
|
};
|
2007-12-08 19:34:15 +00:00
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
struct tr_upnp
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
bool hasDiscovered;
|
|
|
|
struct UPNPUrls urls;
|
|
|
|
struct IGDdatas data;
|
|
|
|
int port;
|
|
|
|
char lanaddr[16];
|
2017-04-30 16:25:26 +00:00
|
|
|
bool isMapped;
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_upnp_state state;
|
2006-09-25 18:37:45 +00:00
|
|
|
};
|
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
2007-06-10 22:26:59 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_upnp* tr_upnpInit(void)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_upnp* ret = tr_new0(tr_upnp, 1);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2007-12-13 18:56:22 +00:00
|
|
|
ret->state = TR_UPNP_DISCOVER;
|
2007-11-06 16:02:50 +00:00
|
|
|
ret->port = -1;
|
2006-09-25 18:37:45 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_upnpClose(tr_upnp* handle)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2017-06-08 07:24:12 +00:00
|
|
|
TR_ASSERT(!handle->isMapped);
|
|
|
|
TR_ASSERT(handle->state == TR_UPNP_IDLE || handle->state == TR_UPNP_ERR || handle->state == TR_UPNP_DISCOVER);
|
2012-12-05 17:29:46 +00:00
|
|
|
|
|
|
|
if (handle->hasDiscovered)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
|
|
|
FreeUPNPUrls(&handle->urls);
|
|
|
|
}
|
|
|
|
|
|
|
|
tr_free(handle);
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2011-10-09 02:05:52 +00:00
|
|
|
/**
|
|
|
|
*** Wrappers for miniupnpc functions
|
|
|
|
**/
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
static struct UPNPDev* tr_upnpDiscover(int msec)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
2021-10-19 16:09:38 +00:00
|
|
|
UPNPDev* ret = nullptr;
|
|
|
|
auto have_err = bool{};
|
2014-04-27 19:31:10 +00:00
|
|
|
|
|
|
|
#if (MINIUPNPC_API_VERSION >= 8) /* adds ipv6 and error args */
|
2017-04-19 12:04:45 +00:00
|
|
|
int err = UPNPDISCOVER_SUCCESS;
|
|
|
|
|
|
|
|
#if (MINIUPNPC_API_VERSION >= 14) /* adds ttl */
|
2021-09-15 00:18:09 +00:00
|
|
|
ret = upnpDiscover(msec, nullptr, nullptr, 0, 0, 2, &err);
|
2017-04-19 12:04:45 +00:00
|
|
|
#else
|
2021-09-15 00:18:09 +00:00
|
|
|
ret = upnpDiscover(msec, nullptr, nullptr, 0, 0, &err);
|
2017-04-19 12:04:45 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
have_err = err != UPNPDISCOVER_SUCCESS;
|
2014-04-27 19:31:10 +00:00
|
|
|
#else
|
2021-09-15 00:18:09 +00:00
|
|
|
ret = upnpDiscover(msec, nullptr, nullptr, 0);
|
|
|
|
have_err = ret == nullptr;
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
if (have_err)
|
|
|
|
{
|
|
|
|
tr_logAddNamedDbg(getKey(), "upnpDiscover failed (errno %d - %s)", errno, tr_strerror(errno));
|
|
|
|
}
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
return ret;
|
2011-10-09 02:05:52 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static int tr_upnpGetSpecificPortMappingEntry(tr_upnp* handle, char const* proto)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
|
|
|
char intClient[16];
|
|
|
|
char intPort[16];
|
|
|
|
char portStr[16];
|
|
|
|
|
|
|
|
*intClient = '\0';
|
|
|
|
*intPort = '\0';
|
|
|
|
|
2020-11-02 15:16:12 +00:00
|
|
|
tr_snprintf(portStr, sizeof(portStr), "%d", handle->port);
|
2014-04-27 19:31:10 +00:00
|
|
|
|
|
|
|
#if (MINIUPNPC_API_VERSION >= 10) /* adds remoteHost arg */
|
2021-10-19 16:09:38 +00:00
|
|
|
int const err = UPNP_GetSpecificPortMappingEntry(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
|
|
|
portStr,
|
|
|
|
proto,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr /*remoteHost*/,
|
2021-08-15 09:41:48 +00:00
|
|
|
intClient,
|
|
|
|
intPort,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr /*desc*/,
|
|
|
|
nullptr /*enabled*/,
|
|
|
|
nullptr /*duration*/);
|
2014-04-27 19:31:10 +00:00
|
|
|
#elif (MINIUPNPC_API_VERSION >= 8) /* adds desc, enabled and leaseDuration args */
|
2021-10-19 16:09:38 +00:00
|
|
|
int const err = UPNP_GetSpecificPortMappingEntry(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
|
|
|
portStr,
|
|
|
|
proto,
|
|
|
|
intClient,
|
|
|
|
intPort,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr /*desc*/,
|
|
|
|
nullptr /*enabled*/,
|
|
|
|
nullptr /*duration*/);
|
2011-10-09 02:05:52 +00:00
|
|
|
#else
|
2021-10-19 16:09:38 +00:00
|
|
|
int const err = UPNP_GetSpecificPortMappingEntry(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
|
|
|
portStr,
|
|
|
|
proto,
|
|
|
|
intClient,
|
2017-04-19 12:04:45 +00:00
|
|
|
intPort);
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static int tr_upnpAddPortMapping(tr_upnp const* handle, char const* proto, tr_port port, char const* desc)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
2017-04-20 16:02:19 +00:00
|
|
|
int const old_errno = errno;
|
2011-10-09 02:05:52 +00:00
|
|
|
char portStr[16];
|
|
|
|
errno = 0;
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_snprintf(portStr, sizeof(portStr), "%d", (int)port);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2014-04-27 19:31:10 +00:00
|
|
|
#if (MINIUPNPC_API_VERSION >= 8)
|
2021-10-19 16:09:38 +00:00
|
|
|
int err = UPNP_AddPortMapping(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
|
|
|
portStr,
|
|
|
|
portStr,
|
|
|
|
handle->lanaddr,
|
|
|
|
desc,
|
|
|
|
proto,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr,
|
|
|
|
nullptr);
|
2011-10-09 02:05:52 +00:00
|
|
|
#else
|
2021-10-19 16:09:38 +00:00
|
|
|
int err = UPNP_AddPortMapping(
|
2021-08-15 09:41:48 +00:00
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
|
|
|
portStr,
|
|
|
|
portStr,
|
|
|
|
handle->lanaddr,
|
|
|
|
desc,
|
|
|
|
proto,
|
2021-09-15 00:18:09 +00:00
|
|
|
nullptr);
|
2011-10-09 02:05:52 +00:00
|
|
|
#endif
|
|
|
|
|
2017-04-30 16:25:26 +00:00
|
|
|
if (err != 0)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
2021-08-15 09:41:48 +00:00
|
|
|
tr_logAddNamedDbg(
|
|
|
|
getKey(),
|
|
|
|
"%s Port forwarding failed with error %d (errno %d - %s)",
|
|
|
|
proto,
|
|
|
|
err,
|
|
|
|
errno,
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_strerror(errno));
|
|
|
|
}
|
2011-10-09 02:05:52 +00:00
|
|
|
|
|
|
|
errno = old_errno;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static void tr_upnpDeletePortMapping(tr_upnp const* handle, char const* proto, tr_port port)
|
2011-10-09 02:05:52 +00:00
|
|
|
{
|
|
|
|
char portStr[16];
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_snprintf(portStr, sizeof(portStr), "%d", (int)port);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2021-09-15 00:18:09 +00:00
|
|
|
UPNP_DeletePortMapping(handle->urls.controlURL, handle->data.first.servicetype, portStr, proto, nullptr);
|
2011-10-09 02:05:52 +00:00
|
|
|
}
|
|
|
|
|
2007-11-06 16:02:50 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
2007-04-15 07:36:24 +00:00
|
|
|
|
2010-06-27 15:29:23 +00:00
|
|
|
enum
|
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
UPNP_IGD_NONE = 0,
|
|
|
|
UPNP_IGD_VALID_CONNECTED = 1,
|
|
|
|
UPNP_IGD_VALID_NOT_CONNECTED = 2,
|
|
|
|
UPNP_IGD_INVALID = 3
|
2010-06-27 15:29:23 +00:00
|
|
|
};
|
|
|
|
|
2021-09-12 17:41:49 +00:00
|
|
|
tr_port_forwarding tr_upnpPulse(tr_upnp* handle, tr_port port, bool isEnabled, bool doPortCheck)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2017-04-30 16:25:26 +00:00
|
|
|
if (isEnabled && handle->state == TR_UPNP_DISCOVER)
|
2007-11-09 16:10:48 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
struct UPNPDev* devlist;
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
devlist = tr_upnpDiscover(2000);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2007-12-09 17:50:05 +00:00
|
|
|
errno = 0;
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2021-08-15 09:41:48 +00:00
|
|
|
if (UPNP_GetValidIGD(devlist, &handle->urls, &handle->data, handle->lanaddr, sizeof(handle->lanaddr)) ==
|
|
|
|
UPNP_IGD_VALID_CONNECTED)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_logAddNamedInfo(getKey(), _("Found Internet Gateway Device \"%s\""), handle->urls.controlURL);
|
|
|
|
tr_logAddNamedInfo(getKey(), _("Local Address is \"%s\""), handle->lanaddr);
|
2007-12-08 19:34:15 +00:00
|
|
|
handle->state = TR_UPNP_IDLE;
|
2014-11-30 19:38:47 +00:00
|
|
|
handle->hasDiscovered = true;
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-12-08 19:34:15 +00:00
|
|
|
handle->state = TR_UPNP_ERR;
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_logAddNamedDbg(getKey(), "UPNP_GetValidIGD failed (errno %d - %s)", errno, tr_strerror(errno));
|
|
|
|
tr_logAddNamedDbg(getKey(), "If your router supports UPnP, please make sure UPnP is enabled!");
|
2007-11-09 16:10:48 +00:00
|
|
|
}
|
2017-04-19 12:04:45 +00:00
|
|
|
|
|
|
|
freeUPNPDevlist(devlist);
|
2007-11-09 16:10:48 +00:00
|
|
|
}
|
|
|
|
|
2021-08-15 09:41:48 +00:00
|
|
|
if ((handle->state == TR_UPNP_IDLE) && (handle->isMapped) && (!isEnabled || handle->port != port))
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2020-11-01 21:47:57 +00:00
|
|
|
handle->state = TR_UPNP_UNMAP;
|
2007-06-10 22:26:59 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2021-08-15 09:41:48 +00:00
|
|
|
if (isEnabled && handle->isMapped && doPortCheck &&
|
2020-11-01 21:47:57 +00:00
|
|
|
((tr_upnpGetSpecificPortMappingEntry(handle, "TCP") != UPNPCOMMAND_SUCCESS) ||
|
2021-08-15 09:41:48 +00:00
|
|
|
(tr_upnpGetSpecificPortMappingEntry(handle, "UDP") != UPNPCOMMAND_SUCCESS)))
|
2009-04-22 16:00:45 +00:00
|
|
|
{
|
2020-11-01 21:47:57 +00:00
|
|
|
tr_logAddNamedInfo(getKey(), _("Port %d isn't forwarded"), handle->port);
|
|
|
|
handle->isMapped = false;
|
2009-04-22 16:00:45 +00:00
|
|
|
}
|
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (handle->state == TR_UPNP_UNMAP)
|
2006-09-25 18:37:45 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_upnpDeletePortMapping(handle, "TCP", handle->port);
|
|
|
|
tr_upnpDeletePortMapping(handle, "UDP", handle->port);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2021-08-15 09:41:48 +00:00
|
|
|
tr_logAddNamedInfo(
|
|
|
|
getKey(),
|
|
|
|
_("Stopping port forwarding through \"%s\", service \"%s\""),
|
|
|
|
handle->urls.controlURL,
|
2017-04-19 12:04:45 +00:00
|
|
|
handle->data.first.servicetype);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-30 16:25:26 +00:00
|
|
|
handle->isMapped = false;
|
2007-12-08 19:34:15 +00:00
|
|
|
handle->state = TR_UPNP_IDLE;
|
|
|
|
handle->port = -1;
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|
|
|
|
|
2020-11-01 21:47:57 +00:00
|
|
|
if ((handle->state == TR_UPNP_IDLE) && isEnabled && !handle->isMapped)
|
2007-12-08 19:34:15 +00:00
|
|
|
{
|
2020-11-01 21:47:57 +00:00
|
|
|
handle->state = TR_UPNP_MAP;
|
2007-12-08 19:34:15 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (handle->state == TR_UPNP_MAP)
|
2007-12-08 19:34:15 +00:00
|
|
|
{
|
2007-12-09 17:50:05 +00:00
|
|
|
errno = 0;
|
2007-12-24 07:02:40 +00:00
|
|
|
|
2021-09-15 00:18:09 +00:00
|
|
|
if (handle->urls.controlURL == nullptr)
|
2017-04-19 12:04:45 +00:00
|
|
|
{
|
2019-07-14 12:40:41 +00:00
|
|
|
handle->isMapped = false;
|
2017-04-19 12:04:45 +00:00
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
else
|
|
|
|
{
|
2008-12-10 06:12:04 +00:00
|
|
|
char desc[64];
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_snprintf(desc, sizeof(desc), "%s at %d", TR_NAME, port);
|
2011-01-15 17:06:11 +00:00
|
|
|
|
2020-11-01 21:47:57 +00:00
|
|
|
int const err_tcp = tr_upnpAddPortMapping(handle, "TCP", port, desc);
|
|
|
|
int const err_udp = tr_upnpAddPortMapping(handle, "UDP", port, desc);
|
2011-10-09 02:05:52 +00:00
|
|
|
|
2017-04-30 16:25:26 +00:00
|
|
|
handle->isMapped = err_tcp == 0 || err_udp == 0;
|
2007-12-24 07:02:40 +00:00
|
|
|
}
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2021-08-15 09:41:48 +00:00
|
|
|
tr_logAddNamedInfo(
|
|
|
|
getKey(),
|
|
|
|
_("Port forwarding through \"%s\", service \"%s\". (local address: %s:%d)"),
|
|
|
|
handle->urls.controlURL,
|
|
|
|
handle->data.first.servicetype,
|
|
|
|
handle->lanaddr,
|
|
|
|
port);
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
if (handle->isMapped)
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_logAddNamedInfo(getKey(), "%s", _("Port forwarding successful!"));
|
2007-12-08 19:34:15 +00:00
|
|
|
handle->port = port;
|
|
|
|
handle->state = TR_UPNP_IDLE;
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_logAddNamedDbg(getKey(), "If your router supports UPnP, please make sure UPnP is enabled!");
|
2007-12-08 19:34:15 +00:00
|
|
|
handle->port = -1;
|
|
|
|
handle->state = TR_UPNP_ERR;
|
|
|
|
}
|
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
|
2012-12-05 17:29:46 +00:00
|
|
|
switch (handle->state)
|
2007-12-13 18:56:22 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
case TR_UPNP_DISCOVER:
|
2021-10-19 16:09:38 +00:00
|
|
|
return TR_PORT_UNMAPPED;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
case TR_UPNP_MAP:
|
2021-10-19 16:09:38 +00:00
|
|
|
return TR_PORT_MAPPING;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
case TR_UPNP_UNMAP:
|
2021-10-19 16:09:38 +00:00
|
|
|
return TR_PORT_UNMAPPING;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
case TR_UPNP_IDLE:
|
2021-10-19 16:09:38 +00:00
|
|
|
return handle->isMapped ? TR_PORT_MAPPED : TR_PORT_UNMAPPED;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
default:
|
2021-10-19 16:09:38 +00:00
|
|
|
return TR_PORT_ERROR;
|
2007-12-13 18:56:22 +00:00
|
|
|
}
|
2006-09-25 18:37:45 +00:00
|
|
|
}
|