/* * This file Copyright (C) 2007-2014 Mnemosyne LLC * * It may be used under the GNU GPL versions 2 or 3 * or any future license endorsed by Mnemosyne LLC. * * $Id$ */ #include #include #ifdef SYSTEM_MINIUPNP #include #include #else #include #include #endif #include "transmission.h" #include "log.h" #include "port-forwarding.h" #include "session.h" #include "upnp.h" #include "utils.h" static const char * getKey (void) { return _("Port Forwarding (UPnP)"); } typedef enum { TR_UPNP_IDLE, TR_UPNP_ERR, TR_UPNP_DISCOVER, TR_UPNP_MAP, TR_UPNP_UNMAP } tr_upnp_state; struct tr_upnp { bool hasDiscovered; struct UPNPUrls urls; struct IGDdatas data; int port; char lanaddr[16]; unsigned int isMapped; tr_upnp_state state; }; /** *** **/ tr_upnp* tr_upnpInit (void) { tr_upnp * ret = tr_new0 (tr_upnp, 1); ret->state = TR_UPNP_DISCOVER; ret->port = -1; return ret; } void tr_upnpClose (tr_upnp * handle) { assert (!handle->isMapped); assert ((handle->state == TR_UPNP_IDLE) || (handle->state == TR_UPNP_ERR) || (handle->state == TR_UPNP_DISCOVER)); if (handle->hasDiscovered) FreeUPNPUrls (&handle->urls); tr_free (handle); } /** *** Wrappers for miniupnpc functions **/ static struct UPNPDev * tr_upnpDiscover (int msec) { struct UPNPDev * ret; bool have_err; #if (MINIUPNPC_API_VERSION >= 8) /* adds ipv6 and error args */ int err = UPNPDISCOVER_SUCCESS; ret = upnpDiscover (msec, NULL, NULL, 0, 0, &err); have_err = err != UPNPDISCOVER_SUCCESS; #else ret = upnpDiscover (msec, NULL, NULL, 0); have_err = ret == NULL; #endif if (have_err) tr_logAddNamedDbg (getKey (), "upnpDiscover failed (errno %d - %s)", errno, tr_strerror (errno)); return ret; } static int tr_upnpGetSpecificPortMappingEntry (tr_upnp * handle, const char * proto) { int err; char intClient[16]; char intPort[16]; char portStr[16]; *intClient = '\0'; *intPort = '\0'; tr_snprintf (portStr, sizeof(portStr), "%d", (int)handle->port); #if (MINIUPNPC_API_VERSION >= 10) /* adds remoteHost arg */ err = UPNP_GetSpecificPortMappingEntry (handle->urls.controlURL, handle->data.first.servicetype, portStr, proto, NULL /*remoteHost*/, intClient, intPort, NULL /*desc*/, NULL /*enabled*/, NULL /*duration*/); #elif (MINIUPNPC_API_VERSION >= 8) /* adds desc, enabled and leaseDuration args */ err = UPNP_GetSpecificPortMappingEntry (handle->urls.controlURL, handle->data.first.servicetype, portStr, proto, intClient, intPort, NULL /*desc*/, NULL /*enabled*/, NULL /*duration*/); #else err = UPNP_GetSpecificPortMappingEntry (handle->urls.controlURL, handle->data.first.servicetype, portStr, proto, intClient, intPort); #endif return err; } static int tr_upnpAddPortMapping (const tr_upnp * handle, const char * proto, tr_port port, const char * desc) { int err; const int old_errno = errno; char portStr[16]; errno = 0; tr_snprintf (portStr, sizeof (portStr), "%d", (int)port); #if (MINIUPNPC_API_VERSION >= 8) err = UPNP_AddPortMapping (handle->urls.controlURL, handle->data.first.servicetype, portStr, portStr, handle->lanaddr, desc, proto, NULL, NULL); #else err = UPNP_AddPortMapping (handle->urls.controlURL, handle->data.first.servicetype, portStr, portStr, handle->lanaddr, desc, proto, NULL); #endif if (err) tr_logAddNamedDbg (getKey (), "%s Port forwarding failed with error %d (errno %d - %s)", proto, err, errno, tr_strerror (errno)); errno = old_errno; return err; } static void tr_upnpDeletePortMapping (const tr_upnp * handle, const char * proto, tr_port port) { char portStr[16]; tr_snprintf (portStr, sizeof (portStr), "%d", (int)port); UPNP_DeletePortMapping (handle->urls.controlURL, handle->data.first.servicetype, portStr, proto, NULL); } /** *** **/ enum { UPNP_IGD_NONE = 0, UPNP_IGD_VALID_CONNECTED = 1, UPNP_IGD_VALID_NOT_CONNECTED = 2, UPNP_IGD_INVALID = 3 }; int tr_upnpPulse (tr_upnp * handle, int port, int isEnabled, int doPortCheck) { int ret; if (isEnabled && (handle->state == TR_UPNP_DISCOVER)) { struct UPNPDev * devlist; devlist = tr_upnpDiscover (2000); errno = 0; if (UPNP_GetValidIGD (devlist, &handle->urls, &handle->data, handle->lanaddr, sizeof (handle->lanaddr)) == UPNP_IGD_VALID_CONNECTED) { tr_logAddNamedInfo (getKey (), _( "Found Internet Gateway Device \"%s\""), handle->urls.controlURL); tr_logAddNamedInfo (getKey (), _( "Local Address is \"%s\""), handle->lanaddr); handle->state = TR_UPNP_IDLE; handle->hasDiscovered = true; } else { handle->state = TR_UPNP_ERR; 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!"); } freeUPNPDevlist (devlist); } if (handle->state == TR_UPNP_IDLE) { if (handle->isMapped && (!isEnabled || (handle->port != port))) handle->state = TR_UPNP_UNMAP; } if (isEnabled && handle->isMapped && doPortCheck) { if ((tr_upnpGetSpecificPortMappingEntry (handle, "TCP") != UPNPCOMMAND_SUCCESS) || (tr_upnpGetSpecificPortMappingEntry (handle, "UDP") != UPNPCOMMAND_SUCCESS)) { tr_logAddNamedInfo (getKey (), _("Port %d isn't forwarded"), handle->port); handle->isMapped = false; } } if (handle->state == TR_UPNP_UNMAP) { tr_upnpDeletePortMapping (handle, "TCP", handle->port); tr_upnpDeletePortMapping (handle, "UDP", handle->port); tr_logAddNamedInfo (getKey (), _("Stopping port forwarding through \"%s\", service \"%s\""), handle->urls.controlURL, handle->data.first.servicetype); handle->isMapped = 0; handle->state = TR_UPNP_IDLE; handle->port = -1; } if (handle->state == TR_UPNP_IDLE) { if (isEnabled && !handle->isMapped) handle->state = TR_UPNP_MAP; } if (handle->state == TR_UPNP_MAP) { int err_tcp = -1; int err_udp = -1; errno = 0; if (!handle->urls.controlURL) handle->isMapped = 0; else { char desc[64]; tr_snprintf (desc, sizeof (desc), "%s at %d", TR_NAME, port); err_tcp = tr_upnpAddPortMapping (handle, "TCP", port, desc); err_udp = tr_upnpAddPortMapping (handle, "UDP", port, desc); handle->isMapped = !err_tcp | !err_udp; } tr_logAddNamedInfo (getKey (), _("Port forwarding through \"%s\", service \"%s\". (local address: %s:%d)"), handle->urls.controlURL, handle->data.first.servicetype, handle->lanaddr, port); if (handle->isMapped) { tr_logAddNamedInfo (getKey (), "%s", _("Port forwarding successful!")); handle->port = port; handle->state = TR_UPNP_IDLE; } else { tr_logAddNamedDbg (getKey (), "If your router supports UPnP, please make sure UPnP is enabled!"); handle->port = -1; handle->state = TR_UPNP_ERR; } } switch (handle->state) { case TR_UPNP_DISCOVER: ret = TR_PORT_UNMAPPED; break; case TR_UPNP_MAP: ret = TR_PORT_MAPPING; break; case TR_UPNP_UNMAP: ret = TR_PORT_UNMAPPING; break; case TR_UPNP_IDLE: ret = handle->isMapped ? TR_PORT_MAPPED : TR_PORT_UNMAPPED; break; default: ret = TR_PORT_ERROR; break; } return ret; }