[OVS] Add mac-table-size to be set on each ovs bridge

By default number of MAC addresses which ovs stores in memory
is quite low - 2048.

Any eviction of a MAC learning table entry triggers revalidation.
Such revalidation is very costly so it cause high CPU usage by
ovs-vswitchd process.

To workaround this problem, higher value of mac-table-size
option can be set for bridge. Then this revalidation will happen
less often and CPU usage will be lower.
This patch adds config option for neutron-openvswitch-agent to allow
users tune this setting in bridges managed by agent.
By default this value is set to 50000 which should be enough for most
systems.

Change-Id: If628f52d75c2b5fec87ad61e0219b3286423468c
Closes-Bug: #1775797
(cherry picked from commit 1f8378e0ac)
This commit is contained in:
Slawek Kaplonski 2018-06-08 15:37:39 +02:00
parent 8dc15cc053
commit 44b159aa9f
4 changed files with 36 additions and 1 deletions

View File

@ -239,6 +239,8 @@ class OVSBridge(BaseOVS):
'protocols', *protocols).execute(check_error=True) 'protocols', *protocols).execute(check_error=True)
def create(self, secure_mode=False): def create(self, secure_mode=False):
other_config = {
'mac-table-size': str(cfg.CONF.bridge_mac_table_size)}
with self.ovsdb.transaction() as txn: with self.ovsdb.transaction() as txn:
txn.add( txn.add(
self.ovsdb.add_br(self.br_name, self.ovsdb.add_br(self.br_name,
@ -250,6 +252,9 @@ class OVSBridge(BaseOVS):
txn.add( txn.add(
self.ovsdb.db_add('Bridge', self.br_name, self.ovsdb.db_add('Bridge', self.br_name,
'protocols', constants.OPENFLOW10)) 'protocols', constants.OPENFLOW10))
txn.add(
self.ovsdb.db_set('Bridge', self.br_name,
('other_config', other_config)))
if secure_mode: if secure_mode:
txn.add(self.ovsdb.set_fail_mode(self.br_name, txn.add(self.ovsdb.set_fail_mode(self.br_name,
FAILMODE_SECURE)) FAILMODE_SECURE))

View File

@ -26,6 +26,13 @@ OPTS = [
help=_('Timeout in seconds for ovs-vsctl commands. ' help=_('Timeout in seconds for ovs-vsctl commands. '
'If the timeout expires, ovs commands will fail with ' 'If the timeout expires, ovs commands will fail with '
'ALARMCLOCK error.')), 'ALARMCLOCK error.')),
cfg.IntOpt('bridge_mac_table_size',
default=50000,
help=_('The maximum number of MAC addresses to learn on '
'a bridge managed by the Neutron OVS agent. Values '
'outside a reasonable range (10 to 1,000,000) might be '
'overridden by Open vSwitch according to the '
'documentation.')),
] ]

View File

@ -18,6 +18,8 @@ import uuid
import mock import mock
from neutron_lib import constants as const from neutron_lib import constants as const
from oslo_config import cfg
import six
from neutron.agent.common import ovs_lib from neutron.agent.common import ovs_lib
from neutron.agent.linux import ip_lib from neutron.agent.linux import ip_lib
@ -572,7 +574,9 @@ class OVSLibTestCase(base.BaseOVSLinuxTestCase):
self.assertTrue(set(self.ovs.get_bridges()).issuperset(bridges)) self.assertTrue(set(self.ovs.get_bridges()).issuperset(bridges))
def test_bridge_lifecycle_ovsbridge(self): def test_bridge_lifecycle_ovsbridge(self):
name = utils.get_rand_name(prefix=net_helpers.BR_PREFIX) name = six.text_type(utils.get_rand_name(prefix=net_helpers.BR_PREFIX))
mac_table_size = 12345
cfg.CONF.set_override('bridge_mac_table_size', mac_table_size)
br = ovs_lib.OVSBridge(name) br = ovs_lib.OVSBridge(name)
self.assertEqual(br.br_name, name) self.assertEqual(br.br_name, name)
# Make sure that instantiating an OVSBridge does not actually create # Make sure that instantiating an OVSBridge does not actually create
@ -580,6 +584,11 @@ class OVSLibTestCase(base.BaseOVSLinuxTestCase):
self.addCleanup(self.ovs.delete_bridge, name) self.addCleanup(self.ovs.delete_bridge, name)
br.create() br.create()
self.assertTrue(self.ovs.bridge_exists(name)) self.assertTrue(self.ovs.bridge_exists(name))
br_other_config = self.ovs.ovsdb.db_find(
'Bridge', ('name', '=', name), columns=['other_config']
).execute()[0]['other_config']
self.assertEqual(str(mac_table_size),
br_other_config['mac-table-size'])
br.destroy() br.destroy()
self.assertFalse(self.ovs.bridge_exists(name)) self.assertFalse(self.ovs.bridge_exists(name))

View File

@ -0,0 +1,14 @@
---
features:
- |
A new config option ``bridge_mac_table_size`` has been added for
Neutron OVS agent.
This value will be set on every Open vSwitch bridge managed by the
openvswitch-neutron-agent in ``other_config:mac-table-size`` column
in ovsdb.
Default value for this new option is set to 50000 and it should be enough
for most systems.
More details about this option can be found in `Open vSwitch documentation
<http://www.openvswitch.org/support/dist-docs/ovs-vswitchd.conf.db.5.html>`_
For more information see bug
`1775797 <https://bugs.launchpad.net/neutron/+bug/1775797>`_.