::arg().set("smtpredirector","Our smtpredir MX host")="a.misconfigured.powerdns.smtp.server";
::arg().set("local-address","Local IP addresses to which we bind")="0.0.0.0";
::arg().set("local-ipv6","Local IP address to which we bind")="";
+ ::arg().setSwitch("reuseport","Enable higher performance on compliant kernels by using SO_REUSEPORT allowing each receiver thread to open its own socket")="no";
::arg().set("query-local-address","Source IP address for sending queries")="0.0.0.0";
::arg().set("query-local-address6","Source IPv6 address for sending queries")="::";
::arg().set("overload-queue-length","Maximum queuelength moving to packetcache only")="0";
bool logDNSQueries = ::arg().mustDo("log-dns-queries");
bool skipfirst=true;
unsigned int maintcount = 0;
+ UDPNameserver *NS = N;
+
+ // If we have SO_REUSEPORT then create a new port for all receiver threads
+ // other than the first one.
+ if( number > 0 && NS->canReusePort() ) {
+ L<<Logger::Notice<<"Starting new listen thread on the same IPs/ports using SO_REUSEPORT"<<endl;
+ NS = new UDPNameserver( true );
+ }
+
for(;;) {
if (skipfirst)
skipfirst=false;
}
}
- if(!(P=N->receive(&question))) { // receive a packet inline
+
+ if(!(P=NS->receive(&question))) { // receive a packet inline
continue; // packet was broken, try again
}
cached.d.id=P->d.id;
cached.commitD(); // commit d to the packet inlined
- N->send(&cached); // answer it then inlined
+ NS->send(&cached); // answer it then inlined
diff=P->d_dt.udiff();
avg_latency=(int)(0.999*avg_latency+0.001*diff); // 'EWMA'
<listitem><para>
If this many packets are waiting for database attention, answer any new questions strictly from the packet cache.
</para></listitem></varlistentry>
+ <varlistentry><term>reuseport=[yes|no]</term>
+ <listitem><para>
+ On Linux 3.9 and some BSD kernels the SO_REUSEPORT option allows each
+ receiver-thread to open a new socket on the same port which allows
+ for much higher performance on multi-core boxes. Setting this option
+ will enable use of SO_REUSEPORT when available and seamlessly fall
+ back to a single socket when it is not available. A side-effect is
+ that you can start multiple servers on the same IP/port combination
+ which may or may not be a good idea. You could use this to enable
+ transparent restarts, but it may also mask configuration issues and
+ for this reason it is disabled by default.
+ </para></listitem></varlistentry>
<varlistentry>
<term>server-id</term>
<listitem>
{
vector<string>locals;
stringtok(locals,::arg()["local-address"]," ,");
+ int one = 1;
if(locals.empty())
throw PDNSException("No local address specified");
memset(&locala,0,sizeof(locala));
locala.sin4.sin_family=AF_INET;
- if(localname=="0.0.0.0") {
- int val=1;
- setsockopt(s, IPPROTO_IP, GEN_IP_PKTINFO, &val, sizeof(val));
- }
+ if(localname=="0.0.0.0")
+ setsockopt(s, IPPROTO_IP, GEN_IP_PKTINFO, &one, sizeof(one));
+
+#ifdef SO_REUSEPORT
+ if( d_can_reuseport )
+ if( setsockopt(s, SOL_SOCKET, SO_REUSEPORT, &one, sizeof(one)) )
+ d_can_reuseport = false;
+#endif
+
locala=ComboAddress(localname, ::arg().asNum("local-port"));
if(locala.sin4.sin_family != AF_INET)
throw PDNSException("Attempting to bind IPv4 socket to IPv6 address");
- g_localaddresses.push_back(locala);
+ if( !d_additional_socket )
+ g_localaddresses.push_back(locala);
+
if(::bind(s, (sockaddr*)&locala, locala.getSocklen()) < 0) {
L<<Logger::Error<<"binding UDP socket to '"+locala.toStringWithPort()+": "<<strerror(errno)<<endl;
throw PDNSException("Unable to bind to UDP socket");
#if HAVE_IPV6
vector<string> locals;
stringtok(locals,::arg()["local-ipv6"]," ,");
+ int one=1;
if(locals.empty())
return;
ComboAddress locala(localname, ::arg().asNum("local-port"));
if(IsAnyAddress(locala)) {
- int val=1;
- setsockopt(s, IPPROTO_IP, GEN_IP_PKTINFO, &val, sizeof(val)); // linux supports this, so why not - might fail on other systems
- setsockopt(s, IPPROTO_IPV6, IPV6_RECVPKTINFO, &val, sizeof(val));
- setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &val, sizeof(val)); // if this fails, we report an error in tcpreceiver too
+ setsockopt(s, IPPROTO_IP, GEN_IP_PKTINFO, &one, sizeof(one)); // linux supports this, so why not - might fail on other systems
+ setsockopt(s, IPPROTO_IPV6, IPV6_RECVPKTINFO, &one, sizeof(one));
+ setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &one, sizeof(one)); // if this fails, we report an error in tcpreceiver too
}
- g_localaddresses.push_back(locala);
+
+#ifdef SO_REUSEPORT
+ if( d_can_reuseport )
+ if( setsockopt(s, SOL_SOCKET, SO_REUSEPORT, &one, sizeof(one)) )
+ d_can_reuseport = false;
+#endif
+
+ if( !d_additional_socket )
+ g_localaddresses.push_back(locala);
if(::bind(s, (sockaddr*)&locala, sizeof(locala))<0) {
L<<Logger::Error<<"binding to UDP ipv6 socket: "<<strerror(errno)<<endl;
throw PDNSException("Unable to bind to UDP ipv6 socket");
#endif
}
-UDPNameserver::UDPNameserver()
+UDPNameserver::UDPNameserver( bool additional_socket )
{
+#ifdef SO_REUSEPORT
+ d_can_reuseport = ::arg().mustDo("reuseport");
+#endif
+ // Are we the main socket (false) or a rebinding using SO_REUSEPORT ?
+ d_additional_socket = additional_socket;
+
if(!::arg()["local-address"].empty())
bindIPv4();
if(!::arg()["local-ipv6"].empty())