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 ((status = UPNP_GetValidIGD(devlist, &upnp_urls, &igd_data, lanaddr,
sizeof(lanaddr))))
82 ThreadLogS(
"UPnP: Found IGD %s (status %d)", upnp_urls.controlURL, status);
88 freeUPNPDevlist(devlist);
93 ThreadLog(
"UPnP: No UPnP device found on the network.");
105 FreeUPNPUrls(&upnp_urls);
106 freeUPNPDevlist(devlist);
113 mapping.external_port = extport;
114 mapping.internal_port = intport;
115 mapping.protocol = (protocol ==
P_TCP ?
"TCP" :
"UDP");
117 added_mappings.push_back(mapping);
119 action = std::async([
this, action = std::move(action), mapping]() {
121 AddPortMapping(mapping);
127 action = std::async([
this, action = std::move(action)]() {
130 for (
auto mapping : added_mappings)
131 RemovePortMapping(mapping);
133 added_mappings.clear();
137 void C4Network2UPnPP::AddPortMapping(
const PortMapping& mapping)
139 if (!initialized)
return;
141 auto eport = std::to_string(mapping.external_port);
142 auto iport = std::to_string(mapping.internal_port);
144 int r = UPNP_AddPortMapping(upnp_urls.controlURL, igd_data.first.servicetype,
145 eport.c_str(), iport.c_str(), lanaddr, description,
146 mapping.protocol.c_str(), 0, 0);
147 if (r == UPNPCOMMAND_SUCCESS)
148 ThreadLogS(
"UPnP: Added mapping %s %s -> %s:%s", mapping.protocol.c_str(), eport.c_str(), lanaddr, iport.c_str());
150 ThreadLog(
"UPnP: AddPortMapping failed with code %d (%s)", r, strupnperror(r));
153 void C4Network2UPnPP::RemovePortMapping(
const PortMapping& mapping)
155 if(!initialized)
return;
157 auto eport = std::to_string(mapping.external_port);
159 int r = UPNP_DeletePortMapping(upnp_urls.controlURL, igd_data.first.servicetype,
160 eport.c_str(), mapping.protocol.c_str(), 0);
161 if (r == UPNPCOMMAND_SUCCESS)
162 ThreadLogS(
"UPnP: Removed mapping %s %s", mapping.protocol.c_str(), eport.c_str());
164 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()