mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-23 04:55:18 +00:00
Add a mutex to serialize updates to firewall
The nwfilter conf update mutex previously serialized updates to the internal data structures for firewall rules, and updates to the firewall itself. The latter was recently turned into a read/write lock, and filter instantiation allowed to proceed in parallel. It was believed that this was ok, since each filter is created on a separate iptables/ebtables chain. It turns out that there is a subtle lock ordering problem on virNWFilterObjPtr instances. __virNWFilterInstantiateFilter will hold a lock on the virNWFilterObjPtr it is instantiating. This in turn invokes virNWFilterInstantiate which then invokes virNWFilterDetermineMissingVarsRec which then invokes virNWFilterObjFindByName. This iterates over every single virNWFilterObjPtr in the list, locking them and checking their name. So if 2 or more threads try to instantiate a filter in parallel, they'll all hold 1 lock at the top level in the __virNWFilterInstantiateFilter method which will cause the other thread to deadlock in virNWFilterObjFindByName. The fix is to add an exclusive mutex to serialize the execution of __virNWFilterInstantiateFilter. Signed-off-by: Daniel P. Berrange <berrange@redhat.com> (cherry picked from commit 925de19ed7f13e0d12d0b993496d314bab886589) Conflicts: src/nwfilter/nwfilter_gentech_driver.c
This commit is contained in:
parent
f84056cf61
commit
78c35ad5ab
@ -199,7 +199,8 @@ nwfilterStateInitialize(bool privileged,
|
|||||||
if (virNWFilterDHCPSnoopInit() < 0)
|
if (virNWFilterDHCPSnoopInit() < 0)
|
||||||
goto err_exit_learnshutdown;
|
goto err_exit_learnshutdown;
|
||||||
|
|
||||||
virNWFilterTechDriversInit(privileged);
|
if (virNWFilterTechDriversInit(privileged) < 0)
|
||||||
|
goto err_dhcpsnoop_shutdown;
|
||||||
|
|
||||||
if (virNWFilterConfLayerInit(virNWFilterDomainFWUpdateCB,
|
if (virNWFilterConfLayerInit(virNWFilterDomainFWUpdateCB,
|
||||||
driverState) < 0)
|
driverState) < 0)
|
||||||
@ -250,6 +251,7 @@ error:
|
|||||||
|
|
||||||
err_techdrivers_shutdown:
|
err_techdrivers_shutdown:
|
||||||
virNWFilterTechDriversShutdown();
|
virNWFilterTechDriversShutdown();
|
||||||
|
err_dhcpsnoop_shutdown:
|
||||||
virNWFilterDHCPSnoopShutdown();
|
virNWFilterDHCPSnoopShutdown();
|
||||||
err_exit_learnshutdown:
|
err_exit_learnshutdown:
|
||||||
virNWFilterLearnShutdown();
|
virNWFilterLearnShutdown();
|
||||||
@ -326,10 +328,10 @@ nwfilterStateCleanup(void) {
|
|||||||
|
|
||||||
if (driverState->privileged) {
|
if (driverState->privileged) {
|
||||||
virNWFilterConfLayerShutdown();
|
virNWFilterConfLayerShutdown();
|
||||||
virNWFilterTechDriversShutdown();
|
|
||||||
virNWFilterDHCPSnoopShutdown();
|
virNWFilterDHCPSnoopShutdown();
|
||||||
virNWFilterLearnShutdown();
|
virNWFilterLearnShutdown();
|
||||||
virNWFilterIPAddrMapShutdown();
|
virNWFilterIPAddrMapShutdown();
|
||||||
|
virNWFilterTechDriversShutdown();
|
||||||
|
|
||||||
nwfilterDriverLock(driverState);
|
nwfilterDriverLock(driverState);
|
||||||
|
|
||||||
|
@ -54,30 +54,53 @@ static virNWFilterTechDriverPtr filter_tech_drivers[] = {
|
|||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* Serializes instantiation of filters. This is necessary
|
||||||
|
* to avoid lock ordering deadlocks. eg __virNWFilterInstantiateFilter
|
||||||
|
* will hold a lock on a virNWFilterObjPtr. This in turn invokes
|
||||||
|
* virNWFilterInstantiate which invokes virNWFilterDetermineMissingVarsRec
|
||||||
|
* which invokes virNWFilterObjFindByName. This iterates over every single
|
||||||
|
* virNWFilterObjPtr in the list. So if 2 threads try to instantiate a
|
||||||
|
* filter in parallel, they'll both hold 1 lock at the top level in
|
||||||
|
* __virNWFilterInstantiateFilter which will cause the other thread
|
||||||
|
* to deadlock in virNWFilterObjFindByName.
|
||||||
|
*
|
||||||
|
* XXX better long term solution is to make virNWFilterObjList use a
|
||||||
|
* hash table as is done for virDomainObjList. You can then get
|
||||||
|
* lockless lookup of objects by name.
|
||||||
|
*/
|
||||||
|
static virMutex updateMutex;
|
||||||
|
|
||||||
void virNWFilterTechDriversInit(bool privileged) {
|
int virNWFilterTechDriversInit(bool privileged)
|
||||||
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
VIR_DEBUG("Initializing NWFilter technology drivers");
|
VIR_DEBUG("Initializing NWFilter technology drivers");
|
||||||
|
if (virMutexInitRecursive(&updateMutex) < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
while (filter_tech_drivers[i]) {
|
while (filter_tech_drivers[i]) {
|
||||||
if (!(filter_tech_drivers[i]->flags & TECHDRV_FLAG_INITIALIZED))
|
if (!(filter_tech_drivers[i]->flags & TECHDRV_FLAG_INITIALIZED))
|
||||||
filter_tech_drivers[i]->init(privileged);
|
filter_tech_drivers[i]->init(privileged);
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void virNWFilterTechDriversShutdown(void) {
|
void virNWFilterTechDriversShutdown(void)
|
||||||
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
while (filter_tech_drivers[i]) {
|
while (filter_tech_drivers[i]) {
|
||||||
if ((filter_tech_drivers[i]->flags & TECHDRV_FLAG_INITIALIZED))
|
if ((filter_tech_drivers[i]->flags & TECHDRV_FLAG_INITIALIZED))
|
||||||
filter_tech_drivers[i]->shutdown();
|
filter_tech_drivers[i]->shutdown();
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
virMutexDestroy(&updateMutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
virNWFilterTechDriverPtr
|
virNWFilterTechDriverPtr
|
||||||
virNWFilterTechDriverForName(const char *name) {
|
virNWFilterTechDriverForName(const char *name)
|
||||||
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
while (filter_tech_drivers[i]) {
|
while (filter_tech_drivers[i]) {
|
||||||
if (STREQ(filter_tech_drivers[i]->name, name)) {
|
if (STREQ(filter_tech_drivers[i]->name, name)) {
|
||||||
@ -948,6 +971,8 @@ _virNWFilterInstantiateFilter(virNWFilterDriverStatePtr driver,
|
|||||||
int ifindex;
|
int ifindex;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
|
virMutexLock(&updateMutex);
|
||||||
|
|
||||||
/* after grabbing the filter update lock check for the interface; if
|
/* after grabbing the filter update lock check for the interface; if
|
||||||
it's not there anymore its filters will be or are being removed
|
it's not there anymore its filters will be or are being removed
|
||||||
(while holding the lock) and we don't want to build new ones */
|
(while holding the lock) and we don't want to build new ones */
|
||||||
@ -975,6 +1000,8 @@ _virNWFilterInstantiateFilter(virNWFilterDriverStatePtr driver,
|
|||||||
foundNewFilter);
|
foundNewFilter);
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
|
virMutexUnlock(&updateMutex);
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -994,6 +1021,7 @@ virNWFilterInstantiateFilterLate(virNWFilterDriverStatePtr driver,
|
|||||||
bool foundNewFilter = false;
|
bool foundNewFilter = false;
|
||||||
|
|
||||||
virNWFilterReadLockFilterUpdates();
|
virNWFilterReadLockFilterUpdates();
|
||||||
|
virMutexLock(&updateMutex);
|
||||||
|
|
||||||
rc = __virNWFilterInstantiateFilter(driver,
|
rc = __virNWFilterInstantiateFilter(driver,
|
||||||
vmuuid,
|
vmuuid,
|
||||||
@ -1019,6 +1047,7 @@ virNWFilterInstantiateFilterLate(virNWFilterDriverStatePtr driver,
|
|||||||
}
|
}
|
||||||
|
|
||||||
virNWFilterUnlockFilterUpdates();
|
virNWFilterUnlockFilterUpdates();
|
||||||
|
virMutexUnlock(&updateMutex);
|
||||||
|
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
@ -1142,7 +1171,11 @@ _virNWFilterTeardownFilter(const char *ifname)
|
|||||||
int
|
int
|
||||||
virNWFilterTeardownFilter(const virDomainNetDefPtr net)
|
virNWFilterTeardownFilter(const virDomainNetDefPtr net)
|
||||||
{
|
{
|
||||||
return _virNWFilterTeardownFilter(net->ifname);
|
int ret;
|
||||||
|
virMutexLock(&updateMutex);
|
||||||
|
ret = _virNWFilterTeardownFilter(net->ifname);
|
||||||
|
virMutexUnlock(&updateMutex);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -30,7 +30,7 @@ virNWFilterTechDriverPtr virNWFilterTechDriverForName(const char *name);
|
|||||||
int virNWFilterRuleInstAddData(virNWFilterRuleInstPtr res,
|
int virNWFilterRuleInstAddData(virNWFilterRuleInstPtr res,
|
||||||
void *data);
|
void *data);
|
||||||
|
|
||||||
void virNWFilterTechDriversInit(bool privileged);
|
int virNWFilterTechDriversInit(bool privileged);
|
||||||
void virNWFilterTechDriversShutdown(void);
|
void virNWFilterTechDriversShutdown(void);
|
||||||
|
|
||||||
enum instCase {
|
enum instCase {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user