network: delay global firewall setup if no networks are running

Creating firewall rules for the virtual networks causes the kernel to
load the conntrack module. This imposes a significant performance
penalty on Linux network traffic. Thus we want to only take that hit if
we actually have virtual networks running.

We need to create global firewall rules during startup in order to
"upgrade" rules for any running networks created by older libvirt.
If no running networks are present though, we can safely delay setup
until the time we actually start a network.

Reviewed-by: Jim Fehlig <jfehlig@suse.com>
Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
This commit is contained in:
Daniel P. Berrangé 2019-05-21 12:40:13 +01:00
parent 3b66bd9aa1
commit c6cbe18771
6 changed files with 198 additions and 13 deletions

View File

@ -2122,7 +2122,7 @@ networkReloadFirewallRules(virNetworkDriverStatePtr driver, bool startup)
* but until we untangle the virt driver that's not viable */ * but until we untangle the virt driver that's not viable */
if (!driver->privileged) if (!driver->privileged)
return; return;
networkPreReloadFirewallRules(startup); networkPreReloadFirewallRules(driver, startup);
virNetworkObjListForEach(driver->networks, virNetworkObjListForEach(driver->networks,
networkReloadFirewallRulesHelper, networkReloadFirewallRulesHelper,
NULL); NULL);

View File

@ -84,16 +84,57 @@ static void networkSetupPrivateChains(void)
} }
} }
void networkPreReloadFirewallRules(bool startup)
static int
networkHasRunningNetworksHelper(virNetworkObjPtr obj,
void *opaque)
{ {
/* We create global rules upfront as we don't want bool *running = opaque;
* the perf hit of conditionally figuring out whether
* to create them each time a network is started. virObjectLock(obj);
if (virNetworkObjIsActive(obj))
*running = true;
virObjectUnlock(obj);
return 0;
}
static bool
networkHasRunningNetworks(virNetworkDriverStatePtr driver)
{
bool running = false;
virNetworkObjListForEach(driver->networks,
networkHasRunningNetworksHelper,
&running);
return running;
}
void networkPreReloadFirewallRules(virNetworkDriverStatePtr driver, bool startup)
{
/*
* If there are any running networks, we need to
* create the global rules upfront. This allows us
* convert rules created by old libvirt into the new
* format.
*
* If there are not any running networks, then we
* must not create rules, because the rules will
* cause the conntrack kernel module to be loaded.
* This imposes a significant performance hit on
* the networking stack. Thus we will only create
* rules if a network is later startup.
* *
* Any errors here are saved to be reported at time * Any errors here are saved to be reported at time
* of starting the network though as that makes them * of starting the network though as that makes them
* more likely to be seen by a human * more likely to be seen by a human
*/ */
if (!networkHasRunningNetworks(driver)) {
VIR_DEBUG("Delayed global rule setup as no networks are running");
return;
}
ignore_value(virOnce(&createdOnce, networkSetupPrivateChains)); ignore_value(virOnce(&createdOnce, networkSetupPrivateChains));
/* /*
@ -726,6 +767,9 @@ int networkAddFirewallRules(virNetworkDefPtr def)
virFirewallPtr fw = NULL; virFirewallPtr fw = NULL;
int ret = -1; int ret = -1;
if (virOnce(&createdOnce, networkSetupPrivateChains) < 0)
return -1;
if (errInitV4 && if (errInitV4 &&
(virNetworkDefGetIPByIndex(def, AF_INET, 0) || (virNetworkDefGetIPByIndex(def, AF_INET, 0) ||
virNetworkDefGetRouteByIndex(def, AF_INET, 0))) { virNetworkDefGetRouteByIndex(def, AF_INET, 0))) {

View File

@ -19,7 +19,8 @@
#include <config.h> #include <config.h>
void networkPreReloadFirewallRules(bool startup ATTRIBUTE_UNUSED) void networkPreReloadFirewallRules(virNetworkDriverStatePtr driver ATTRIBUTE_UNUSED,
bool startup ATTRIBUTE_UNUSED)
{ {
} }

View File

@ -58,7 +58,7 @@ struct _virNetworkDriverState {
typedef struct _virNetworkDriverState virNetworkDriverState; typedef struct _virNetworkDriverState virNetworkDriverState;
typedef virNetworkDriverState *virNetworkDriverStatePtr; typedef virNetworkDriverState *virNetworkDriverStatePtr;
void networkPreReloadFirewallRules(bool startup); void networkPreReloadFirewallRules(virNetworkDriverStatePtr driver, bool startup);
void networkPostReloadFirewallRules(bool startup); void networkPostReloadFirewallRules(bool startup);
int networkCheckRouteCollision(virNetworkDefPtr def); int networkCheckRouteCollision(virNetworkDefPtr def);

View File

@ -0,0 +1,116 @@
iptables \
--table filter \
--list-rules
iptables \
--table nat \
--list-rules
iptables \
--table mangle \
--list-rules
iptables \
--table filter \
--new-chain LIBVIRT_INP
iptables \
--table filter \
--insert INPUT \
--jump LIBVIRT_INP
iptables \
--table filter \
--new-chain LIBVIRT_OUT
iptables \
--table filter \
--insert OUTPUT \
--jump LIBVIRT_OUT
iptables \
--table filter \
--new-chain LIBVIRT_FWO
iptables \
--table filter \
--insert FORWARD \
--jump LIBVIRT_FWO
iptables \
--table filter \
--new-chain LIBVIRT_FWI
iptables \
--table filter \
--insert FORWARD \
--jump LIBVIRT_FWI
iptables \
--table filter \
--new-chain LIBVIRT_FWX
iptables \
--table filter \
--insert FORWARD \
--jump LIBVIRT_FWX
iptables \
--table nat \
--new-chain LIBVIRT_PRT
iptables \
--table nat \
--insert POSTROUTING \
--jump LIBVIRT_PRT
iptables \
--table mangle \
--new-chain LIBVIRT_PRT
iptables \
--table mangle \
--insert POSTROUTING \
--jump LIBVIRT_PRT
ip6tables \
--table filter \
--list-rules
ip6tables \
--table nat \
--list-rules
ip6tables \
--table mangle \
--list-rules
ip6tables \
--table filter \
--new-chain LIBVIRT_INP
ip6tables \
--table filter \
--insert INPUT \
--jump LIBVIRT_INP
ip6tables \
--table filter \
--new-chain LIBVIRT_OUT
ip6tables \
--table filter \
--insert OUTPUT \
--jump LIBVIRT_OUT
ip6tables \
--table filter \
--new-chain LIBVIRT_FWO
ip6tables \
--table filter \
--insert FORWARD \
--jump LIBVIRT_FWO
ip6tables \
--table filter \
--new-chain LIBVIRT_FWI
ip6tables \
--table filter \
--insert FORWARD \
--jump LIBVIRT_FWI
ip6tables \
--table filter \
--new-chain LIBVIRT_FWX
ip6tables \
--table filter \
--insert FORWARD \
--jump LIBVIRT_FWX
ip6tables \
--table nat \
--new-chain LIBVIRT_PRT
ip6tables \
--table nat \
--insert POSTROUTING \
--jump LIBVIRT_PRT
ip6tables \
--table mangle \
--new-chain LIBVIRT_PRT
ip6tables \
--table mangle \
--insert POSTROUTING \
--jump LIBVIRT_PRT

View File

@ -22,6 +22,7 @@
#include <config.h> #include <config.h>
#include "testutils.h" #include "testutils.h"
#include "viralloc.h"
#if defined (__linux__) #if defined (__linux__)
@ -57,13 +58,15 @@ testCommandDryRun(const char *const*args ATTRIBUTE_UNUSED,
} }
static int testCompareXMLToArgvFiles(const char *xml, static int testCompareXMLToArgvFiles(const char *xml,
const char *cmdline) const char *cmdline,
const char *baseargs)
{ {
char *expectargv = NULL; char *expectargv = NULL;
char *actualargv = NULL; char *actualargv = NULL;
virBuffer buf = VIR_BUFFER_INITIALIZER; virBuffer buf = VIR_BUFFER_INITIALIZER;
virNetworkDefPtr def = NULL; virNetworkDefPtr def = NULL;
int ret = -1; int ret = -1;
char *actual;
virCommandSetDryRun(&buf, testCommandDryRun, NULL); virCommandSetDryRun(&buf, testCommandDryRun, NULL);
@ -76,11 +79,18 @@ static int testCompareXMLToArgvFiles(const char *xml,
if (virBufferError(&buf)) if (virBufferError(&buf))
goto cleanup; goto cleanup;
actualargv = virBufferContentAndReset(&buf); actual = actualargv = virBufferContentAndReset(&buf);
virTestClearCommandPath(actualargv); virTestClearCommandPath(actualargv);
virCommandSetDryRun(NULL, NULL, NULL); virCommandSetDryRun(NULL, NULL, NULL);
if (virTestCompareToFile(actualargv, cmdline) < 0) /* The first network to be created populates the
* libvirt global chains. We must skip args for
* that if present
*/
if (STRPREFIX(actual, baseargs))
actual += strlen(baseargs);
if (virTestCompareToFile(actual, cmdline) < 0)
goto cleanup; goto cleanup;
ret = 0; ret = 0;
@ -95,6 +105,7 @@ static int testCompareXMLToArgvFiles(const char *xml,
struct testInfo { struct testInfo {
const char *name; const char *name;
const char *baseargs;
}; };
@ -112,7 +123,7 @@ testCompareXMLToIPTablesHelper(const void *data)
abs_srcdir, info->name, RULESTYPE) < 0) abs_srcdir, info->name, RULESTYPE) < 0)
goto cleanup; goto cleanup;
result = testCompareXMLToArgvFiles(xml, args); result = testCompareXMLToArgvFiles(xml, args, info->baseargs);
cleanup: cleanup:
VIR_FREE(xml); VIR_FREE(xml);
@ -133,11 +144,13 @@ static int
mymain(void) mymain(void)
{ {
int ret = 0; int ret = 0;
VIR_AUTOFREE(char *)basefile = NULL;
VIR_AUTOFREE(char *)baseargs = NULL;
# define DO_TEST(name) \ # define DO_TEST(name) \
do { \ do { \
static struct testInfo info = { \ struct testInfo info = { \
name, \ name, baseargs, \
}; \ }; \
if (virTestRun("Network XML-2-iptables " name, \ if (virTestRun("Network XML-2-iptables " name, \
testCompareXMLToIPTablesHelper, &info) < 0) \ testCompareXMLToIPTablesHelper, &info) < 0) \
@ -156,6 +169,17 @@ mymain(void)
goto cleanup; goto cleanup;
} }
if (virAsprintf(&basefile, "%s/networkxml2firewalldata/base.args",
abs_srcdir) < 0) {
ret = -1;
goto cleanup;
}
if (virTestLoadFile(basefile, &baseargs) < 0) {
ret = -1;
goto cleanup;
}
DO_TEST("nat-default"); DO_TEST("nat-default");
DO_TEST("nat-tftp"); DO_TEST("nat-tftp");
DO_TEST("nat-many-ips"); DO_TEST("nat-many-ips");