Do not map to management partition for NPIV

NPIV based deploys attempted to put the client vFC adapters on to the
fabric by creating the WWPNs ahead of time and mapping them to the
management partition.  This created several issues and led to poor
performance.

There were issues around migration:
- Querying for a host could leave a stale adapter on the mgmt partition

There were issues with speed:
- Deploys could take over 120 seconds
- Migrations had additional delays

There were issues with some standard operations:
- Attaching a volume on a powered off VM required logging in the WWPNs

Subsequent analysis in Cinder showed that only some Cinder drivers had
this requirement.  Many could work without this.  Given the challenges
the code is removing the need to log in the WWPNs on the mgmt partition.
This resolves all of the above issues.

While it does limit some of the Cinder drivers when paired with NPIV,
ongoing work will be done to meet with the Cinder driver owners to see
if the requirements in their drivers can be relaxed.

For those that can not, the vSCSI Cinder Volume attach is also an option
and should provide the use cases that they need.

Change-Id: Id77650545ca4e0def9977f1c2572b9f13f451e0d
This commit is contained in:
Drew Thorstensen 2015-10-13 23:51:22 -04:00
parent 845342d1f8
commit 15e400c8a4
4 changed files with 119 additions and 281 deletions

View File

@ -115,11 +115,6 @@ as the libvirt driver within OpenStack.
Most operations are comparable in speed. Deployment, attach/detach volumes, Most operations are comparable in speed. Deployment, attach/detach volumes,
lifecycle, etc... are quick. lifecycle, etc... are quick.
The one exception is if the operator configures the system to use N-Port ID
Virtualization for storage (NPIV). This technology provides significant speed
increases for instance disk performance, but may increase the deployment time
by several seconds.
The driver is written to support concurrent operations. It has been tested The driver is written to support concurrent operations. It has been tested
performing 10 concurrent deploys to a given compute node. performing 10 concurrent deploys to a given compute node.

View File

@ -117,6 +117,13 @@ Volume Options
| fc_attach_strategy = vscsi | (StrOpt) The Fibre Channel Volume Strategy defines how FC | | fc_attach_strategy = vscsi | (StrOpt) The Fibre Channel Volume Strategy defines how FC |
| | Cinder volumes should be attached to the Virtual Machine. | | | Cinder volumes should be attached to the Virtual Machine. |
| | The options are: npiv or vscsi. | | | The options are: npiv or vscsi. |
| | |
| | It should be noted that if NPIV is chosen, the WWPNs will |
| | not be active on the backing fabric during the deploy. |
| | Some Cinder drivers will operate without issue. Others |
| | may query the fabric and thus will fail attachment. It is |
| | advised that if an issue occurs using NPIV, the operator |
| | fall back to vscsi based deploys. |
+--------------------------------------+------------------------------------------------------------+ +--------------------------------------+------------------------------------------------------------+
| vscsi_vios_connections_required = 1 | (IntOpt) Indicates a minimum number of Virtual I/O Servers | | vscsi_vios_connections_required = 1 | (IntOpt) Indicates a minimum number of Virtual I/O Servers |
| | that are required to support a Cinder volume attach with | | | that are required to support a Cinder volume attach with |

View File

@ -90,73 +90,8 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
self.mock_fabric_names_p.stop() self.mock_fabric_names_p.stop()
self.mock_fabric_ports_p.stop() self.mock_fabric_ports_p.stop()
@mock.patch('nova_powervm.virt.powervm.mgmt.get_mgmt_partition')
@mock.patch('pypowervm.tasks.vfc_mapper.add_map') @mock.patch('pypowervm.tasks.vfc_mapper.add_map')
@mock.patch('pypowervm.tasks.vfc_mapper.remove_maps') def test_connect_volume(self, mock_add_map):
def test_connect_volume(self, mock_remove_maps, mock_add_map,
mock_mgmt_lpar_id):
# Mock
self._basic_system_metadata(npiv.FS_MGMT_MAPPED)
mock_mgmt_lpar_id.return_value = mock.Mock(uuid='1')
def validate_remove_maps(vios_w, lpar_uuid, client_adpt=None,
port_map=None, **kwargs):
self.assertEqual('1', lpar_uuid)
self.assertIsInstance(vios_w, pvm_vios.VIOS)
self.assertIsNone(client_adpt)
self.assertEqual(('21000024FF649104', 'AA BB'), port_map)
return 'removed'
mock_remove_maps.side_effect = validate_remove_maps
def add_map(vios_w, host_uuid, vm_uuid, port_map, **kwargs):
self.assertIsInstance(vios_w, pvm_vios.VIOS)
self.assertEqual('host_uuid', host_uuid)
self.assertEqual('1234', vm_uuid)
self.assertEqual(('21000024FF649104', 'AA BB'), port_map)
return 'good'
mock_add_map.side_effect = add_map
# Test connect volume when the fabric is mapped with mgmt partition
self.vol_drv.connect_volume()
# Verify. Mgmt mapping should be removed
self.assertEqual(1, mock_remove_maps.call_count)
self.assertEqual(1, mock_add_map.call_count)
self.assertEqual(1, self.ft_fx.patchers['update'].mock.call_count)
self.assertEqual(npiv.FS_INST_MAPPED,
self.vol_drv._get_fabric_state('A'))
def test_connect_volume_not_valid(self):
"""Validates that a connect will fail if in a bad state."""
self.mock_inst_wrap.can_modify_io.return_value = False, 'Bleh'
self.assertRaises(exc.VolumeAttachFailed, self.vol_drv.connect_volume)
@mock.patch('pypowervm.tasks.vfc_mapper.add_map')
@mock.patch('pypowervm.tasks.vfc_mapper.remove_npiv_port_mappings')
def test_connect_volume_inst_mapped(self, mock_remove_p_maps,
mock_add_map):
"""Test if already connected to an instance, don't do anything"""
self._basic_system_metadata(npiv.FS_INST_MAPPED)
mock_add_map.return_value = None
# Test subsequent connect volume calls when the fabric is mapped with
# inst partition
self.vol_drv.connect_volume()
# Verify
# Remove mapping should not be called
self.assertEqual(0, mock_remove_p_maps.call_count)
self.assertEqual(1, mock_add_map.call_count)
self.assertEqual(0, self.ft_fx.patchers['update'].mock.call_count)
# Check the fabric state remains mapped to instance
self.assertEqual(npiv.FS_INST_MAPPED,
self.vol_drv._get_fabric_state('A'))
@mock.patch('pypowervm.tasks.vfc_mapper.add_map')
@mock.patch('pypowervm.tasks.vfc_mapper.remove_npiv_port_mappings')
def test_connect_volume_fc_unmap(self, mock_remove_p_maps,
mock_add_map):
# Mock # Mock
self._basic_system_metadata(npiv.FS_UNMAPPED) self._basic_system_metadata(npiv.FS_UNMAPPED)
@ -168,13 +103,37 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
return 'good' return 'good'
mock_add_map.side_effect = add_map mock_add_map.side_effect = add_map
# TestCase when there is no mapping # Test connect volume
self.vol_drv.connect_volume() self.vol_drv.connect_volume()
# Remove mapping should not be called # Verify that the appropriate connections were made.
self.assertEqual(0, mock_remove_p_maps.call_count)
self.assertEqual(1, mock_add_map.call_count) self.assertEqual(1, mock_add_map.call_count)
self.assertEqual(1, self.ft_fx.patchers['update'].mock.call_count) self.assertEqual(1, self.ft_fx.patchers['update'].mock.call_count)
self.assertEqual(npiv.FS_INST_MAPPED,
self.vol_drv._get_fabric_state('A'))
def test_connect_volume_not_valid(self):
"""Validates that a connect will fail if in a bad state."""
self.mock_inst_wrap.can_modify_io.return_value = False, 'Invalid I/O'
self.assertRaises(exc.VolumeAttachFailed, self.vol_drv.connect_volume)
@mock.patch('pypowervm.tasks.vfc_mapper.add_map')
def test_connect_volume_inst_mapped(self, mock_add_map):
"""Test if already connected to an instance, don't do anything"""
self._basic_system_metadata(npiv.FS_INST_MAPPED)
mock_add_map.return_value = None
# Test subsequent connect volume calls when the fabric is mapped with
# inst partition
self.vol_drv.connect_volume()
# Verify
self.assertEqual(1, mock_add_map.call_count)
self.assertEqual(0, self.ft_fx.patchers['update'].mock.call_count)
# Check the fabric state remains mapped to instance
self.assertEqual(npiv.FS_INST_MAPPED,
self.vol_drv._get_fabric_state('A'))
def _basic_system_metadata(self, fabric_state): def _basic_system_metadata(self, fabric_state):
meta_fb_key = self.vol_drv._sys_meta_fabric_key('A') meta_fb_key = self.vol_drv._sys_meta_fabric_key('A')
@ -300,19 +259,13 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
inst.host = CONF.host inst.host = CONF.host
self.assertFalse(self.vol_drv._is_migration_wwpn(npiv.FS_INST_MAPPED)) self.assertFalse(self.vol_drv._is_migration_wwpn(npiv.FS_INST_MAPPED))
@mock.patch('nova_powervm.virt.powervm.mgmt.get_mgmt_partition')
@mock.patch('pypowervm.tasks.vfc_mapper.derive_npiv_map') @mock.patch('pypowervm.tasks.vfc_mapper.derive_npiv_map')
@mock.patch('pypowervm.tasks.vfc_mapper.add_npiv_port_mappings') def test_configure_wwpns_for_migration(self, mock_derive):
def test_configure_wwpns_for_migration(
self, mock_add_npiv_port, mock_derive, mock_mgmt_lpar):
# Mock out the fabric # Mock out the fabric
meta_fb_key = self.vol_drv._sys_meta_fabric_key('A') meta_fb_key = self.vol_drv._sys_meta_fabric_key('A')
meta_fb_map = '21000024FF649104,AA,BB,21000024FF649105,CC,DD' meta_fb_map = '21000024FF649104,AA,BB,21000024FF649105,CC,DD'
self.vol_drv.instance.system_metadata = {meta_fb_key: meta_fb_map} self.vol_drv.instance.system_metadata = {meta_fb_key: meta_fb_map}
# Mock the mgmt partition
mock_mgmt_lpar.return_value = mock.Mock(uuid=0)
# Mock out what the derive returns # Mock out what the derive returns
expected_map = [('21000024FF649104', 'BB AA'), expected_map = [('21000024FF649104', 'BB AA'),
('21000024FF649105', 'DD CC')] ('21000024FF649105', 'DD CC')]
@ -321,46 +274,40 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
# Invoke # Invoke
resp_maps = self.vol_drv._configure_wwpns_for_migration('A') resp_maps = self.vol_drv._configure_wwpns_for_migration('A')
# Make sure the add port was done properly
mock_add_npiv_port.assert_called_once_with(
self.vol_drv.adapter, self.vol_drv.host_uuid, 0, expected_map)
# Make sure the updated maps are returned # Make sure the updated maps are returned
expected = [('21000024FF649104', 'BB AA'), expected = [('21000024FF649104', 'BB AA'),
('21000024FF649105', 'DD CC')] ('21000024FF649105', 'DD CC')]
self.assertEqual(expected, resp_maps) self.assertEqual(expected, resp_maps)
mock_derive.assert_called_with(
mock.ANY, ['21000024FF649104', '21000024FF649107'],
['BB', 'AA', 'DD', 'CC'])
@mock.patch('nova_powervm.virt.powervm.mgmt.get_mgmt_partition') @mock.patch('pypowervm.tasks.vfc_mapper.derive_npiv_map')
@mock.patch('pypowervm.tasks.vfc_mapper.add_npiv_port_mappings') def test_configure_wwpns_for_migration_existing(self, mock_derive):
def test_configure_wwpns_for_migration_existing( """Validates nothing is done if WWPNs are already flipped."""
self, mock_add_npiv_port, mock_mgmt_lpar):
"""Validates nothing is done if WWPNs are already mapped."""
# Mock out the fabric # Mock out the fabric
meta_fb_key = self.vol_drv._sys_meta_fabric_key('A') meta_fb_key = self.vol_drv._sys_meta_fabric_key('A')
meta_fb_map = '21000024FF649104,c05076079cff0fa0,c05076079cff0fa1' meta_fb_map = '21000024FF649104,C05076079CFF0FA0,C05076079CFF0FA1'
self.vol_drv.instance.system_metadata = {meta_fb_key: meta_fb_map} meta_fb_st_key = self.vol_drv._sys_fabric_state_key('A')
meta_fb_st_val = npiv.FS_MIGRATING
# Mock the mgmt partition self.vol_drv.instance.system_metadata = {
mock_mgmt_lpar.return_value = mock.Mock(uuid=0) meta_fb_key: meta_fb_map, meta_fb_st_key: meta_fb_st_val}
# Invoke # Invoke
resp_maps = self.vol_drv._configure_wwpns_for_migration('A') resp_maps = self.vol_drv._configure_wwpns_for_migration('A')
# Make sure invocations were not made to do any adds # Make sure that the order of the client WWPNs is not changed.
self.assertFalse(mock_add_npiv_port.called)
expected = [('21000024FF649104', 'C05076079CFF0FA0 C05076079CFF0FA1')] expected = [('21000024FF649104', 'C05076079CFF0FA0 C05076079CFF0FA1')]
self.assertEqual(expected, resp_maps) self.assertEqual(expected, resp_maps)
self.assertFalse(mock_derive.called)
@mock.patch('nova_powervm.virt.powervm.mgmt.get_mgmt_partition') @mock.patch('pypowervm.tasks.vfc_mapper.build_wwpn_pair')
@mock.patch('pypowervm.tasks.vfc_mapper.add_npiv_port_mappings') @mock.patch('pypowervm.tasks.vfc_mapper.derive_npiv_map')
def test_wwpns(self, mock_add_port, mock_mgmt_part): def test_wwpns(self, mock_derive, mock_build_pair):
"""Tests that new WWPNs get generated properly.""" """Tests that new WWPNs get generated properly."""
# Mock Data # Mock Data
mock_add_port.return_value = [('21000024FF649104', 'AA BB'), mock_derive.return_value = [('21000024FF649104', 'AA BB'),
('21000024FF649105', 'CC DD')] ('21000024FF649105', 'CC DD')]
mock_vios = mock.MagicMock()
mock_vios.uuid = '3443DB77-AED1-47ED-9AA5-3DB9C6CF7089'
mock_mgmt_part.return_value = mock_vios
self.adpt.read.return_value = self.vios_feed_resp self.adpt.read.return_value = self.vios_feed_resp
meta_key = self.vol_drv._sys_meta_fabric_key('A') meta_key = self.vol_drv._sys_meta_fabric_key('A')
@ -373,15 +320,7 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
self.assertListEqual(['AA', 'CC'], wwpns) self.assertListEqual(['AA', 'CC'], wwpns)
self.assertEqual('21000024FF649104,AA,BB,21000024FF649105,CC,DD', self.assertEqual('21000024FF649104,AA,BB,21000024FF649105,CC,DD',
self.vol_drv.instance.system_metadata[meta_key]) self.vol_drv.instance.system_metadata[meta_key])
self.assertEqual(1, mock_add_port.call_count) self.assertEqual(1, mock_derive.call_count)
# Check when mgmt_uuid is None
mock_add_port.reset_mock()
mock_vios.uuid = None
self.vol_drv.wwpns()
self.assertEqual(0, mock_add_port.call_count)
self.assertEqual('mgmt_mapped',
self.vol_drv._get_fabric_state('A'))
@mock.patch('nova_powervm.virt.powervm.volume.npiv.NPIVVolumeAdapter.' @mock.patch('nova_powervm.virt.powervm.volume.npiv.NPIVVolumeAdapter.'
'_get_fabric_state') '_get_fabric_state')
@ -473,21 +412,13 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
self.assertEqual([1, 2], mig_data.get('npiv_fabric_slots_A')) self.assertEqual([1, 2], mig_data.get('npiv_fabric_slots_A'))
self.assertEqual([3], mig_data.get('npiv_fabric_slots_B')) self.assertEqual([3], mig_data.get('npiv_fabric_slots_B'))
@mock.patch('pypowervm.tasks.vfc_mapper.remove_maps')
@mock.patch('pypowervm.tasks.vfc_mapper.find_vios_for_vfc_wwpns')
@mock.patch('pypowervm.tasks.vfc_mapper.' @mock.patch('pypowervm.tasks.vfc_mapper.'
'build_migration_mappings_for_fabric') 'build_migration_mappings_for_fabric')
@mock.patch('nova_powervm.virt.powervm.mgmt.get_mgmt_partition')
@mock.patch('nova_powervm.virt.powervm.volume.npiv.NPIVVolumeAdapter.'
'_get_fabric_meta')
@mock.patch('nova_powervm.virt.powervm.volume.npiv.NPIVVolumeAdapter.' @mock.patch('nova_powervm.virt.powervm.volume.npiv.NPIVVolumeAdapter.'
'_fabric_names') '_fabric_names')
def test_pre_live_migration_on_destination( def test_pre_live_migration_on_destination(
self, mock_fabric_names, mock_get_fabric_meta, mock_mgmt_lpar_id, self, mock_fabric_names, mock_build_mig_map):
mock_build_mig_map, mock_find_vios_for_vfc_wwpns, mock_remove_map):
mock_fabric_names.return_value = ['A', 'B'] mock_fabric_names.return_value = ['A', 'B']
mock_get_fabric_meta.side_effect = [[], []]
mock_mgmt_lpar_id.return_value = mock.Mock(uuid='1')
src_mig_data = {'npiv_fabric_slots_A': [1, 2], src_mig_data = {'npiv_fabric_slots_A': [1, 2],
'npiv_fabric_slots_B': [3]} 'npiv_fabric_slots_B': [3]}
@ -506,21 +437,6 @@ class TestNPIVAdapter(test_vol.TestVolumeAdapter):
self.assertEqual(set(['b', 'a']), self.assertEqual(set(['b', 'a']),
set(dest_mig_data.get('vfc_lpm_mappings'))) set(dest_mig_data.get('vfc_lpm_mappings')))
mock_find_vios_for_vfc_wwpns.return_value = None, None
dest_mig_data = {}
mock_fabric_names.return_value = ['A', 'B']
mock_get_fabric_meta.side_effect = [
[('11', 'AA BB'), ('22', 'CC DD')],
[('33', 'EE FF')]]
mock_build_mig_map.side_effect = [['a'], ['b']]
# Execute the test
with self.assertLogs(npiv.__name__, level='WARNING'):
self.vol_drv.pre_live_migration_on_destination(
src_mig_data, dest_mig_data)
# remove_map should not be called since vios_w is None
self.assertEqual(0, mock_remove_map.call_count)
def test_set_fabric_meta(self): def test_set_fabric_meta(self):
port_map = [('1', 'aa AA'), ('2', 'bb BB'), port_map = [('1', 'aa AA'), ('2', 'bb BB'),
('3', 'cc CC'), ('4', 'dd DD'), ('3', 'cc CC'), ('4', 'dd DD'),

View File

@ -24,7 +24,6 @@ from pypowervm.tasks import vfc_mapper as pvm_vfcm
from pypowervm.wrappers import virtual_io_server as pvm_vios from pypowervm.wrappers import virtual_io_server as pvm_vios
from nova_powervm.virt import powervm from nova_powervm.virt import powervm
from nova_powervm.virt.powervm import mgmt
from nova_powervm.virt.powervm.volume import driver as v_driver from nova_powervm.virt.powervm.volume import driver as v_driver
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
@ -33,7 +32,7 @@ CONF = cfg.CONF
WWPN_SYSTEM_METADATA_KEY = 'npiv_adpt_wwpns' WWPN_SYSTEM_METADATA_KEY = 'npiv_adpt_wwpns'
FABRIC_STATE_METADATA_KEY = 'fabric_state' FABRIC_STATE_METADATA_KEY = 'fabric_state'
FS_UNMAPPED = 'unmapped' FS_UNMAPPED = 'unmapped'
FS_MGMT_MAPPED = 'mgmt_mapped' FS_MIGRATING = 'migrating'
FS_INST_MAPPED = 'inst_mapped' FS_INST_MAPPED = 'inst_mapped'
TASK_STATES_FOR_DISCONNECT = [task_states.DELETING, task_states.SPAWNING] TASK_STATES_FOR_DISCONNECT = [task_states.DELETING, task_states.SPAWNING]
@ -135,48 +134,17 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
should be added to this dictionary. should be added to this dictionary.
""" """
vios_wraps = self.stg_ftsk.feed vios_wraps = self.stg_ftsk.feed
mgmt_uuid = mgmt.get_mgmt_partition(self.adapter).uuid
# Each mapping should attempt to remove itself from the management # Need to first derive the port mappings that can be passed back
# partition. # to the source system for the live migration call. This tells
# the source system what 'vfc mappings' to pass in on the live
# migration command.
for fabric in self._fabric_names(): for fabric in self._fabric_names():
npiv_port_maps = self._get_fabric_meta(fabric)
# Need to first derive the port mappings that can be passed back
# to the source system for the live migration call. This tells
# the source system what 'vfc mappings' to pass in on the live
# migration command.
slots = src_mig_data['npiv_fabric_slots_%s' % fabric] slots = src_mig_data['npiv_fabric_slots_%s' % fabric]
fabric_mapping = pvm_vfcm.build_migration_mappings_for_fabric( fabric_mapping = pvm_vfcm.build_migration_mappings_for_fabric(
vios_wraps, self._fabric_ports(fabric), slots) vios_wraps, self._fabric_ports(fabric), slots)
dest_mig_data['npiv_fabric_mapping_%s' % fabric] = fabric_mapping dest_mig_data['npiv_fabric_mapping_%s' % fabric] = fabric_mapping
# Next we need to remove the mappings off the mgmt partition.
for npiv_port_map in npiv_port_maps:
ls = [LOG.info, _LI("Removing mgmt NPIV mapping for instance "
"%(inst)s for fabric %(fabric)s."),
{'inst': self.instance.name, 'fabric': fabric}]
vios_w, vfc_map = pvm_vfcm.find_vios_for_vfc_wwpns(
vios_wraps, npiv_port_map[1].split())
if vios_w is not None:
# Add the subtask to remove the mapping from the management
# partition.
task_wrapper = self.stg_ftsk.wrapper_tasks[vios_w.uuid]
task_wrapper.add_functor_subtask(
pvm_vfcm.remove_maps, mgmt_uuid,
client_adpt=vfc_map.client_adapter, logspec=ls)
else:
LOG.warn(_LW("No storage connections found between the "
"Virtual I/O Servers and FC Fabric "
"%(fabric)s. The connection might be removed "
"already."), {'fabric': fabric})
# TODO(thorst) Find a better place for this execute. Works for now
# as the stg_ftsk is all local. Also won't do anything if there
# happen to be no fabric changes.
self.stg_ftsk.execute()
# Collate all of the individual fabric mappings into a single element. # Collate all of the individual fabric mappings into a single element.
full_map = [] full_map = []
for key, value in dest_mig_data.items(): for key, value in dest_mig_data.items():
@ -226,6 +194,7 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
new_map = (p_wwpn, " ".join(c_wwpns)) new_map = (p_wwpn, " ".join(c_wwpns))
new_port_maps.append(new_map) new_port_maps.append(new_map)
self._set_fabric_meta(fabric, new_port_maps) self._set_fabric_meta(fabric, new_port_maps)
self._set_fabric_state(fabric, FS_INST_MAPPED)
# Store that this fabric is now flipped. # Store that this fabric is now flipped.
mig_vol_stor[fabric_key] = True mig_vol_stor[fabric_key] = True
@ -244,9 +213,9 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
if (fc_state == FS_UNMAPPED and if (fc_state == FS_UNMAPPED and
self.instance.task_state not in [task_states.DELETING, self.instance.task_state not in [task_states.DELETING,
task_states.MIGRATING]): task_states.MIGRATING]):
LOG.info(_LI("Mapping instance %(inst)s to the mgmt partition for " LOG.info(_LI("Instance %(inst)s has not yet defined a WWPN on "
"fabric %(fabric)s because the VM does not yet have " "fabric %(fabric)s. Appropriate WWPNs will be "
"a valid vFC device."), "generated."),
{'inst': self.instance.name, 'fabric': fabric}) {'inst': self.instance.name, 'fabric': fabric})
return True return True
@ -268,71 +237,61 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
def _configure_wwpns_for_migration(self, fabric): def _configure_wwpns_for_migration(self, fabric):
"""Configures the WWPNs for a migration. """Configures the WWPNs for a migration.
During a NPIV migration, the WWPNs need to be flipped and attached to During a NPIV migration, the WWPNs need to be flipped. This is because
the management VM. This is so that the peer WWPN is brought online. the second WWPN is what will be logged in on the source system. So by
flipping them, we indicate that the 'second' wwpn is the new one to
log in.
The WWPNs will be removed from the management partition via the Another way to think of it is, this code should always return the
pre_live_migration_on_destination method. The WWPNs invocation is correct WWPNs for the system that the workload will be running on.
done prior to the migration, when the volume connector is gathered.
This WWPNs invocation is done on the target server prior to the
actual migration call. It is used to build the volume connector.
Therefore this code simply flips the ports around.
:param fabric: The fabric to configure. :param fabric: The fabric to configure.
:return: An updated port mapping. :return: An updated port mapping.
""" """
LOG.info(_LI("Mapping instance %(inst)s to the mgmt partition for " if self._get_fabric_state(fabric) == FS_MIGRATING:
"fabric %(fabric)s because the VM is migrating to " # If the fabric is migrating, just return the existing port maps.
"this host."), # They've already been flipped.
{'inst': self.instance.name, 'fabric': fabric}) return self._get_fabric_meta(fabric)
mgmt_uuid = mgmt.get_mgmt_partition(self.adapter).uuid
# When we migrate...flip the WWPNs around. This is so the other # When we migrate...flip the WWPNs around. This is so the other
# WWPN logs in on the target fabric. But we should only flip new # WWPN logs in on the target fabric. If this code is hit, the flip
# WWPNs. There may already be some on the overall fabric...and if # hasn't yet occurred (read as first volume on the instance).
# there are, we keep those 'as-is'
#
# TODO(thorst) pending API change should be able to indicate which
# wwpn is active.
port_maps = self._get_fabric_meta(fabric) port_maps = self._get_fabric_meta(fabric)
existing_wwpns = [] client_wwpns = []
new_wwpns = []
for port_map in port_maps: for port_map in port_maps:
c_wwpns = port_map[1].split() c_wwpns = port_map[1].split()
c_wwpns.reverse()
client_wwpns.extend(c_wwpns)
# Only add it as a 'new' mapping if it isn't on a VIOS already. If # Now derive the mapping to the VIOS physical ports on this system
# it is, then we know that it has already been serviced, perhaps # (the destination)
# by a previous volume.
vfc_map = pvm_vfcm.has_client_wwpns(self.stg_ftsk.feed, c_wwpns)[1]
if vfc_map is None:
c_wwpns.reverse()
new_wwpns.extend(c_wwpns)
else:
existing_wwpns.extend(c_wwpns)
# Now derive the mapping to THESE VIOSes physical ports
port_mappings = pvm_vfcm.derive_npiv_map( port_mappings = pvm_vfcm.derive_npiv_map(
self.stg_ftsk.feed, self._fabric_ports(fabric), self.stg_ftsk.feed, self._fabric_ports(fabric), client_wwpns)
new_wwpns + existing_wwpns)
# Add the port maps to the mgmt partition # This won't actually get saved by the process. The instance save will
if len(new_wwpns) > 0: # only occur after the 'post migration'. But if there are multiple
pvm_vfcm.add_npiv_port_mappings( # volumes, their WWPNs calls will subsequently see the data saved
self.adapter, self.host_uuid, mgmt_uuid, port_mappings) # temporarily here, and therefore won't "double flip" the wwpns back
# to the original.
self._set_fabric_meta(fabric, port_mappings)
self._set_fabric_state(fabric, FS_MIGRATING)
# Return the mappings
return port_mappings return port_mappings
def wwpns(self): def wwpns(self):
"""Builds the WWPNs of the adapters that will connect the ports.""" """Builds the WWPNs of the adapters that will connect the ports."""
vios_wraps, mgmt_uuid = None, None vios_wraps = None
resp_wwpns = [] resp_wwpns = []
# If this is a new mapping altogether, the WWPNs need to be logged # If this is the first time to query the WWPNs for the instance, we
# into the fabric so that Cinder can make use of them. This is a bit # need to generate a set of valid WWPNs. Loop through the configured
# of a catch-22 because the LPAR doesn't exist yet. So a mapping will # FC fabrics and determine if these are new, part of a migration, or
# be created against the mgmt partition and then upon VM creation, the # were already configured.
# mapping will be moved over to the VM.
#
# If a mapping already exists, we can instead just pull the data off
# of the system metadata from the nova instance.
for fabric in self._fabric_names(): for fabric in self._fabric_names():
fc_state = self._get_fabric_state(fabric) fc_state = self._get_fabric_state(fabric)
LOG.info(_LI("NPIV wwpns fabric state=%(st)s for " LOG.info(_LI("NPIV wwpns fabric state=%(st)s for "
@ -340,50 +299,40 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
{'st': fc_state, 'inst': self.instance.name}) {'st': fc_state, 'inst': self.instance.name})
if self._is_initial_wwpn(fc_state, fabric): if self._is_initial_wwpn(fc_state, fabric):
# At this point we've determined that we need to do a mapping. # It is a new WWPN. Need to investigate the Virtual I/O
# So we go and obtain the mgmt uuid and the VIOS wrappers. # Servers. We only do this for the first loop through so as
# We only do this for the first loop through so as to ensure # to ensure that we do not keep invoking the expensive call
# that we do not keep invoking these expensive calls
# unnecessarily. # unnecessarily.
if mgmt_uuid is None: if vios_wraps is None:
mgmt_uuid = mgmt.get_mgmt_partition(self.adapter).uuid # The VIOS wrappers are not set at this point. Seed
# them. Will get reused on subsequent loops.
# The VIOS wrappers are also not set at this point. Seed
# them as well. Will get reused on subsequent loops.
vios_wraps = self.stg_ftsk.feed vios_wraps = self.stg_ftsk.feed
# Get a set of WWPNs that are globally unique from the system.
v_wwpns = pvm_vfcm.build_wwpn_pair(
self.adapter, self.host_uuid,
pair_count=self._ports_per_fabric())
# Derive the virtual to physical port mapping # Derive the virtual to physical port mapping
port_maps = pvm_vfcm.derive_base_npiv_map( port_maps = pvm_vfcm.derive_npiv_map(
vios_wraps, self._fabric_ports(fabric), vios_wraps, self._fabric_ports(fabric), v_wwpns)
self._ports_per_fabric())
# Every loop through, we reverse the vios wrappers. This is # Every loop through, we reverse the vios wrappers. This is
# done so that if Fabric A only has 1 port, it goes on the # done so that if Fabric A only has 1 port, it goes on the
# first VIOS. Then Fabric B would put its port on a different # first VIOS. Then Fabric B would put its port on a different
# VIOS. As a form of multi pathing (so that your paths were # VIOS. This servers as a form of multi pathing (so that your
# not restricted to a single VIOS). # paths are not restricted to a single VIOS).
vios_wraps.reverse() vios_wraps.reverse()
# Check if the fabrics are unmapped then we need to map it
# temporarily with the management partition.
LOG.info(_LI("Adding NPIV Mapping with mgmt partition for "
"instance %s") % self.instance.name)
port_maps = pvm_vfcm.add_npiv_port_mappings(
self.adapter, self.host_uuid, mgmt_uuid, port_maps)
# Set the fabric meta (which indicates on the instance how # Set the fabric meta (which indicates on the instance how
# the fabric is mapped to the physical port) and the fabric # the fabric is mapped to the physical port) and the fabric
# state. # state.
self._set_fabric_meta(fabric, port_maps) self._set_fabric_meta(fabric, port_maps)
self._set_fabric_state(fabric, FS_MGMT_MAPPED) self._set_fabric_state(fabric, FS_UNMAPPED)
elif self._is_migration_wwpn(fc_state): elif self._is_migration_wwpn(fc_state):
# The migration process requires the 'second' wwpn from the
# fabric to be used.
port_maps = self._configure_wwpns_for_migration(fabric) port_maps = self._configure_wwpns_for_migration(fabric)
# This won't actually get saved by the process. The save will
# only occur after the 'post migration'. But if there are
# multiple volumes, their WWPNs calls will subsequently see
# the data saved temporarily here.
self._set_fabric_meta(fabric, port_maps)
else: else:
# This specific fabric had been previously set. Just pull # This specific fabric had been previously set. Just pull
# from the meta (as it is likely already mapped to the # from the meta (as it is likely already mapped to the
@ -404,38 +353,11 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
def _add_maps_for_fabric(self, fabric): def _add_maps_for_fabric(self, fabric):
"""Adds the vFC storage mappings to the VM for a given fabric. """Adds the vFC storage mappings to the VM for a given fabric.
Will check if the Fabric is mapped to the management partition. If it
is, then it will remove the mappings and update the fabric state. This
is because, in order for the WWPNs to be on the fabric (for Cinder)
before the VM is online, the WWPNs get mapped to the management
partition.
This method will remove from the management partition (if needed), and
then assign it to the instance itself.
:param fabric: The fabric to add the mappings to. :param fabric: The fabric to add the mappings to.
""" """
npiv_port_maps = self._get_fabric_meta(fabric) npiv_port_maps = self._get_fabric_meta(fabric)
vios_wraps = self.stg_ftsk.feed vios_wraps = self.stg_ftsk.feed
# If currently mapped to the mgmt partition, remove the mappings so
# that they can be added to the client.
if self._get_fabric_state(fabric) == FS_MGMT_MAPPED:
mgmt_uuid = mgmt.get_mgmt_partition(self.adapter).uuid
# Each port mapping should be removed from the VIOS.
for npiv_port_map in npiv_port_maps:
vios_w = pvm_vfcm.find_vios_for_port_map(vios_wraps,
npiv_port_map)
ls = [LOG.info, _LI("Removing NPIV mapping for mgmt partition "
"for instance %(inst)s on VIOS %(vios)s."),
{'inst': self.instance.name, 'vios': vios_w.name}]
# Add the subtask to remove the map from the mgmt partition
self.stg_ftsk.wrapper_tasks[vios_w.uuid].add_functor_subtask(
pvm_vfcm.remove_maps, mgmt_uuid, port_map=npiv_port_map,
logspec=ls)
# This loop adds the maps from the appropriate VIOS to the client VM # This loop adds the maps from the appropriate VIOS to the client VM
for npiv_port_map in npiv_port_maps: for npiv_port_map in npiv_port_maps:
vios_w = pvm_vfcm.find_vios_for_port_map(vios_wraps, npiv_port_map) vios_w = pvm_vfcm.find_vios_for_port_map(vios_wraps, npiv_port_map)
@ -499,7 +421,6 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
Possible Valid States: Possible Valid States:
FS_UNMAPPED: Initial state unmapped. FS_UNMAPPED: Initial state unmapped.
FS_MGMT_MAPPED: Fabric is mapped with the management partition
FS_INST_MAPPED: Fabric is mapped with the nova instance. FS_INST_MAPPED: Fabric is mapped with the nova instance.
""" """
meta_key = self._sys_fabric_state_key(fabric) meta_key = self._sys_fabric_state_key(fabric)
@ -515,7 +436,6 @@ class NPIVVolumeAdapter(v_driver.FibreChannelVolumeAdapter):
Possible Valid States: Possible Valid States:
FS_UNMAPPED: Initial state unmapped. FS_UNMAPPED: Initial state unmapped.
FS_MGMT_MAPPED: Fabric is mapped with the management partition
FS_INST_MAPPED: Fabric is mapped with the nova instance. FS_INST_MAPPED: Fabric is mapped with the nova instance.
""" """
meta_key = self._sys_fabric_state_key(fabric) meta_key = self._sys_fabric_state_key(fabric)