20 #include "C4Version.h"
23 #include <miniupnpc.h>
24 #include <upnpcommands.h>
25 #include <upnperrors.h>
29 static const char *description =
"OpenClonk";
44 uint16_t external_port;
45 uint16_t internal_port;
49 void AddPortMapping(
const PortMapping& mapping);
50 void RemovePortMapping(
const PortMapping& mapping);
52 std::vector<PortMapping> added_mappings;
55 std::future<void> action;
57 bool initialized =
false;
59 UPNPDev *devlist =
nullptr;
66 action = std::async(&C4Network2UPnPP::Init,
this);
69 void C4Network2UPnPP::Init()
73 #if MINIUPNPC_API_VERSION == 10
75 if ((devlist = upnpDiscover(2000,
nullptr,
nullptr, 0, 0, &error)))
77 if ((devlist = upnpDiscover(2000,
nullptr,
nullptr, UPNP_LOCAL_PORT_ANY, 0, 2, &error)))
80 #if MINIUPNPC_API_VERSION >= 18
82 if ((status = UPNP_GetValidIGD(devlist, &upnp_urls, &igd_data, lanaddr,
sizeof(lanaddr), wanaddr,
sizeof(wanaddr))))
84 if ((status = UPNP_GetValidIGD(devlist, &upnp_urls, &igd_data, lanaddr,
sizeof(lanaddr))))
87 ThreadLogS(
"UPnP: Found IGD %s (status %d)", upnp_urls.controlURL, status);
93 freeUPNPDevlist(devlist);
98 ThreadLog(
"UPnP: No UPnP device found on the network.");
110 FreeUPNPUrls(&upnp_urls);
111 freeUPNPDevlist(devlist);
118 mapping.external_port = extport;
119 mapping.internal_port = intport;
120 mapping.protocol = (protocol ==
P_TCP ?
"TCP" :
"UDP");
122 added_mappings.push_back(mapping);
124 action = std::async([
this, action = std::move(action), mapping]() {
126 AddPortMapping(mapping);
132 action = std::async([
this, action = std::move(action)]() {
135 for (
auto mapping : added_mappings)
136 RemovePortMapping(mapping);
138 added_mappings.clear();
142 void C4Network2UPnPP::AddPortMapping(
const PortMapping& mapping)
144 if (!initialized)
return;
146 auto eport = std::to_string(mapping.external_port);
147 auto iport = std::to_string(mapping.internal_port);
149 int r = UPNP_AddPortMapping(upnp_urls.controlURL, igd_data.first.servicetype,
150 eport.c_str(), iport.c_str(), lanaddr, description,
151 mapping.protocol.c_str(), 0, 0);
152 if (r == UPNPCOMMAND_SUCCESS)
153 ThreadLogS(
"UPnP: Added mapping %s %s -> %s:%s", mapping.protocol.c_str(), eport.c_str(), lanaddr, iport.c_str());
155 ThreadLog(
"UPnP: AddPortMapping failed with code %d (%s)", r, strupnperror(r));
158 void C4Network2UPnPP::RemovePortMapping(
const PortMapping& mapping)
160 if(!initialized)
return;
162 auto eport = std::to_string(mapping.external_port);
164 int r = UPNP_DeletePortMapping(upnp_urls.controlURL, igd_data.first.servicetype,
165 eport.c_str(), mapping.protocol.c_str(), 0);
166 if (r == UPNPCOMMAND_SUCCESS)
167 ThreadLogS(
"UPnP: Removed mapping %s %s", mapping.protocol.c_str(), eport.c_str());
169 ThreadLog(
"UPnP: DeletePortMapping failed with code %d (%s)", r, strupnperror(r));
bool ThreadLog(const char *szMessage,...) GNUC_FORMAT_ATTRIBUTE_O
bool ThreadLogS(const char *szMessage,...) GNUC_FORMAT_ATTRIBUTE_O
void AddMapping(enum C4Network2IOProtocol protocol, uint16_t intport, uint16_t extport)
void AddMapping(C4Network2IOProtocol protocol, uint16_t intport, uint16_t extport)
virtual ~C4Network2UPnPP()