Cinder RSD Driver

Add a new Cinder Driver for Intel RSD with NVMeoF support:
- report stats
- create volume
- delete volume
- attach volume
- detach volume
- create snapshot
- delete snapshot
- create volume from snapshot
- clone volume
- extend volume
- copy volume to image (reuse the base-class impl)
- copy image to volume (reuse the base-class impl)
- host-assisted volume migration (reuse the manager code)
- unit tests
- release-note
- documentation

Change-Id: Ie83dab4858729353865268231352033934dafdc2
Implements: blueprint nvme-volume-driver
This commit is contained in:
Yingxin 2018-12-03 16:51:13 +08:00 committed by Tejdeep Kautharam
parent 3c29c3846e
commit b92b80241c
5 changed files with 2058 additions and 0 deletions

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,723 @@
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
"""
Driver for RackScale Design.
"""
import json
from oslo_config import cfg
from oslo_log import log as logging
from oslo_utils import units
from cinder import exception
from cinder.i18n import _
from cinder import interface
from cinder import utils
from cinder.volume import driver
from distutils import version
try:
from rsd_lib import RSDLib
from sushy import exceptions as sushy_exceptions
except ImportError:
# Used for tests, when no rsd-lib is installed
RSDLib = None
sushy_exceptions = None
LOG = logging.getLogger(__name__)
RSD_OPTS = [
cfg.StrOpt('podm_url',
default='',
help='URL of PODM service'),
cfg.StrOpt('podm_username',
default='',
help='Username of PODM service'),
cfg.StrOpt('podm_password',
default='',
help='Password of PODM service',
secret=True),
]
class RSDRetryableException(exception.VolumeDriverException):
message = _("RSD retryable exception: %(reason)s")
def get_volume_metadata(volume):
metadata = volume.get('volume_metadata')
if metadata:
ret = {data['key']: data['value'] for data in metadata}
else:
ret = volume.get('metadata', {})
return ret
class RSDClient(object):
def __init__(self, rsdlib):
self.rsdlib = rsdlib
@classmethod
def initialize(cls, url, username, password, verify):
if not RSDLib:
raise exception.VolumeBackendAPIException(
data=(_("RSDLib is not available, please install rsd-lib.")))
try:
rsdlib = RSDLib(url, username, password, verify=verify).factory()
except Exception:
# error credentials may throw unexpected exception
LOG.exception("Cannot connect to RSD PODM")
raise exception.VolumeBackendAPIException(
data=_("initialize: Cannot connect to RSD PODM."))
rsd_lib_version = version.LooseVersion(rsdlib._rsd_api_version)
if rsd_lib_version < version.LooseVersion("2.4.0"):
raise exception.VolumeBackendAPIException(
data=(_("initialize: Unsupported rsd_lib version: "
"%(current)s < %(expected)s.")
% {'current': rsdlib._rsd_api_version,
'expected': "2.4.0"}))
LOG.info("initialize: Connected to %s at version %s.",
url, rsdlib._rsd_api_version)
return cls(rsdlib)
def _get_storage(self, storage_url):
ss_url = "/".join(storage_url.split("/", 5)[:5])
storage_service = self.rsdlib.get_storage_service(ss_url)
return storage_service
def _get_storages(self, filter_nvme=True):
ret = []
for storage in (self.rsdlib
.get_storage_service_collection().get_members()):
if filter_nvme:
drives = storage.drives.get_members()
if drives and (any(map(lambda drive:
False if not drive.protocol
else 'nvme' in drive.protocol.lower(),
drives))):
ret.append(storage)
else:
ret.append(storage)
return ret
def _get_node(self, node_url):
return self.rsdlib.get_node(node_url)
def _get_volume(self, volume_url):
ss = self._get_storage(volume_url)
volume = ss.volumes.get_member(volume_url)
return volume
def _get_providing_pool(self, volume):
len_cs = len(volume.capacity_sources)
if len_cs != 1:
raise exception.ValidationError(
detail=(_("Volume %(vol)s has %(len_cs)d capacity_sources!")
% {'vol': volume.path,
'len_cs': len_cs}))
len_pp = len(volume.capacity_sources[0].providing_pools)
if len_pp != 1:
raise exception.ValidationError(
detail=(_("Volume %(vol)s has %(len_pp)d providing_pools!")
% {'vol': volume.path,
'len_pp': len_pp}))
return volume.capacity_sources[0].providing_pools[0]
def _create_vol_or_snap(self,
storage,
size_in_bytes,
pool_url=None,
source_snap=None,
source_vol=None):
capacity_sources = None
if pool_url:
capacity_sources = [{
"ProvidingPools": [{
"@odata.id": pool_url
}]
}]
replica_infos = None
if source_snap:
replica_infos = [{
"ReplicaType": "Clone",
"Replica": {"@odata.id": source_snap}
}]
if source_vol:
raise exception.InvalidInput(
reason=(_("Cannot specify both source_snap=%(snap)s and "
"source_vol=%(vol)s!")
% {'snap': source_snap,
'vol': source_vol}))
elif source_vol:
replica_infos = [{
"ReplicaType": "Snapshot",
"Replica": {"@odata.id": source_vol}
}]
LOG.debug("Creating... with size_byte=%s, "
"capacity_sources=%s, replica_infos=%s",
size_in_bytes, capacity_sources, replica_infos)
volume_url = storage.volumes.create_volume(
size_in_bytes,
capacity_sources=capacity_sources,
replica_infos=replica_infos)
LOG.debug("Created volume_url=%s", volume_url)
return volume_url
def create_volume(self, size_in_gb):
size_in_bytes = size_in_gb * units.Gi
try:
for storage in self._get_storages():
try:
volume_url = self._create_vol_or_snap(
storage, size_in_bytes)
LOG.info("RSD volume %s created, with size %s GiB",
volume_url, size_in_gb)
return volume_url
# NOTE(Yingxin): Currently, we capture sushy_exception to
# identify that volume creation is failed at RSD backend.
except (sushy_exceptions.HTTPError,
sushy_exceptions.ConnectionError) as e:
LOG.warning("skipped storage %s for creation error %s",
storage.path, e)
except Exception:
LOG.exception("Create volume failed")
raise exception.VolumeBackendAPIException(
data=(_('Unable to create new volume with %d GiB') % size_in_gb))
def create_snap(self, volume_url):
try:
ss = self._get_storage(volume_url)
volume = self._get_volume(volume_url)
pool_url = self._get_providing_pool(volume)
snap_url = self._create_vol_or_snap(
ss, volume.capacity_bytes,
pool_url=pool_url,
source_vol=volume_url)
LOG.info("RSD snapshot %s created, from volume %s",
snap_url, volume_url)
return snap_url
except Exception:
LOG.exception("Create snapshot failed")
raise exception.VolumeBackendAPIException(
data=(_('Unable to create snapshot from volume %s')
% volume_url))
def create_volume_from_snap(self, snap_url, size_in_gb=None):
try:
ss = self._get_storage(snap_url)
snap = self._get_volume(snap_url)
if not size_in_gb:
size_in_bytes = snap.capacity_bytes
else:
size_in_bytes = size_in_gb * units.Gi
pool_url = self._get_providing_pool(snap)
volume_url = self._create_vol_or_snap(
ss, size_in_bytes,
pool_url=pool_url,
source_snap=snap_url)
LOG.info("RSD volume %s created, from snap %s, "
"with size %s GiB.",
volume_url, snap_url,
size_in_bytes / units.Gi)
return volume_url
except Exception:
LOG.exception("Create volume from snapshot failed")
raise exception.VolumeBackendAPIException(
data=(_('Unable to create volume from snapshot %s')
% snap_url))
def clone_volume(self, volume_url, size_in_gb=None):
try:
ss = self._get_storage(volume_url)
origin_volume = self._get_volume(volume_url)
pool_url = self._get_providing_pool(origin_volume)
snap_url = self._create_vol_or_snap(
ss, origin_volume.capacity_bytes,
pool_url=pool_url,
source_vol=volume_url)
except Exception:
LOG.exception("Clone volume failed (create snapshot phase)")
raise exception.VolumeBackendAPIException(
data=(_('Unable to create volume from volume %s, snapshot '
'creation failed.')
% volume_url))
try:
if not size_in_gb:
size_in_bytes = origin_volume.capacity_bytes
else:
size_in_bytes = size_in_gb * units.Gi
new_vol_url = self._create_vol_or_snap(
ss, size_in_bytes,
pool_url=pool_url,
source_snap=snap_url)
LOG.info("RSD volume %s created, from volume %s and snap %s, "
"with size %s GiB.",
new_vol_url, volume_url, snap_url,
size_in_bytes / units.Gi)
return new_vol_url, snap_url
except Exception:
LOG.exception("Clone volume failed (clone volume phase)")
try:
self.delete_vol_or_snap(snap_url)
except Exception:
LOG.exception("Clone volume failed (undo snapshot)")
raise exception.VolumeBackendAPIException(
data=(_('Unable to delete the temp snapshot %(snap)s, '
'during a failure to clone volume %(vol)s.')
% {'snap': snap_url,
'vol': volume_url}))
raise exception.VolumeBackendAPIException(
data=(_('Unable to create volume from volume %s, volume '
'creation failed.')
% volume_url))
def extend_volume(self, volume_url, size_in_gb):
size_in_bytes = size_in_gb * units.Gi
try:
volume = self._get_volume(volume_url)
volume.resize(size_in_bytes)
LOG.info("RSD volume %s resized to %s Bytes",
volume.path, size_in_bytes)
except Exception:
LOG.exception("Extend volume failed")
raise exception.VolumeBackendAPIException(
data=(_('Unable to extend volume %s.') % volume_url))
def delete_vol_or_snap(self, volume_url,
volume_name='', ignore_non_exist=False):
try:
try:
volume = self._get_volume(volume_url)
except sushy_exceptions.ResourceNotFoundError:
if ignore_non_exist:
LOG.warning("Deleted non existent vol/snap %s", volume_url)
else:
raise
if volume.links.endpoints:
LOG.warning("Delete vol/snap failed, attached: %s", volume_url)
raise exception.VolumeIsBusy(_("Volume is already attached"),
volume_name=volume_name)
volume.delete()
except sushy_exceptions.BadRequestError as e:
try:
msg = e.body['@Message.ExtendedInfo'][0]['Message']
if (msg == "Cannot delete source snapshot volume when "
"other clone volumes are based on this snapshot."):
LOG.warning("Delete vol/snap failed, has-deps: %s",
volume_url)
raise exception.SnapshotIsBusy(snapshot_name=volume_name)
except Exception:
LOG.exception("Delete vol/snap failed")
raise exception.VolumeBackendAPIException(
data=(_('Unable to delete volume %s.') % volume_url))
except Exception:
LOG.exception("Delete vol/snap failed")
raise exception.VolumeBackendAPIException(
data=(_('Unable to delete volume %s.') % volume_url))
LOG.info("RSD volume deleted: %s", volume_url)
def get_node_url_by_uuid(self, uuid):
uuid = uuid.upper()
try:
nodes = self.rsdlib.get_node_collection().get_members()
for node in nodes:
node_system = None
if node:
node_system = self.rsdlib.get_system(
node.links.computer_system)
if (node and
node_system and
node_system.uuid and
node_system.uuid.upper() == uuid):
return node.path
except Exception:
LOG.exception("Get node url failed")
return ""
def get_stats(self):
free_capacity_gb = 0
total_capacity_gb = 0
allocated_capacity_gb = 0
total_volumes = 0
try:
storages = self._get_storages()
for storage in storages:
for pool in storage.storage_pools.get_members():
total_capacity_gb += (
float(pool.capacity.allocated_bytes or 0) / units.Gi)
allocated_capacity_gb += (
float(pool.capacity.consumed_bytes or 0) / units.Gi)
total_volumes += len(storage.volumes.members_identities)
free_capacity_gb = total_capacity_gb - allocated_capacity_gb
LOG.info("Got RSD stats: free_gb:%s, total_gb:%s, "
"allocated_gb:%s, volumes:%s",
free_capacity_gb,
total_capacity_gb,
allocated_capacity_gb,
total_volumes)
except Exception:
LOG.exception("Get stats failed")
return (free_capacity_gb,
total_capacity_gb,
allocated_capacity_gb,
total_volumes)
def _get_nqn_endpoints(self, endpoint_urls):
ret = []
for endpoint_url in endpoint_urls:
endpoint_json = (
json.loads(self.rsdlib._conn.get(endpoint_url).text))
for ident in endpoint_json["Identifiers"]:
if ident["DurableNameFormat"] == "NQN":
nqn = ident["DurableName"]
ret.append((nqn, endpoint_json))
break
return ret
@utils.retry(RSDRetryableException,
interval=4,
retries=5,
backoff_rate=2)
def attach_volume_to_node(self, volume_url, node_url):
LOG.info('Trying attach from node %s to volume %s',
node_url, volume_url)
try:
volume = self._get_volume(volume_url)
node = self._get_node(node_url)
if len(volume.links.endpoints) > 0:
raise exception.ValidationError(
detail=(_("Volume %s already attached") % volume_url))
node.attach_endpoint(volume.path)
except sushy_exceptions.InvalidParameterValueError as e:
LOG.exception("Attach volume failed (not allowable)")
raise RSDRetryableException(
reason=(_("Not allowed to attach from "
"%(node)s to %(volume)s.")
% {'node': node_url,
'volume': volume_url}))
except Exception:
LOG.exception("Attach volume failed (attach phase)")
raise exception.VolumeBackendAPIException(
data=(_("Attach failed from %(node)s to %(volume)s.")
% {'node': node_url,
'volume': volume_url}))
try:
volume.refresh()
node.refresh()
v_endpoints = volume.links.endpoints
v_endpoints = self._get_nqn_endpoints(v_endpoints)
if len(v_endpoints) != 1:
raise exception.ValidationError(
detail=(_("Attach volume error: %d target nqns")
% len(v_endpoints)))
target_nqn, v_endpoint = v_endpoints[0]
ip_transports = v_endpoint["IPTransportDetails"]
if len(ip_transports) != 1:
raise exception.ValidationError(
detail=(_("Attach volume error: %d target ips")
% len(ip_transports)))
ip_transport = ip_transports[0]
target_ip = ip_transport["IPv4Address"]["Address"]
target_port = ip_transport["Port"]
node_system = self.rsdlib.get_system(node.links.computer_system)
n_endpoints = tuple(
val["@odata.id"]
for val in node_system.json["Links"]["Endpoints"])
n_endpoints = self._get_nqn_endpoints(n_endpoints)
if len(n_endpoints) == 0:
raise exception.ValidationError(
detail=(_("Attach volume error: %d host nqns")
% len(n_endpoints)))
host_nqn, v_endpoint = n_endpoints[0]
LOG.info('Attachment successful: Retrieved target IP %s, '
'target Port %s, target NQN %s and initiator NQN %s',
target_ip, target_port, target_nqn, host_nqn)
return (target_ip, target_port, target_nqn, host_nqn)
except Exception as e:
LOG.exception("Attach volume failed (post-attach)")
try:
node.refresh()
node.detach_endpoint(volume.path)
LOG.info('Detached from node %s to volume %s',
node_url, volume_url)
except Exception:
LOG.exception("Attach volume failed (undo attach)")
raise exception.VolumeBackendAPIException(
data=(_("Undo-attach failed from %(node)s to %(volume)s.")
% {'node': node_url,
'volume': volume_url}))
if isinstance(e, exception.ValidationError):
raise RSDRetryableException(
reason=(_("Validation error during post-attach from "
"%(node)s to %(volume)s.")
% {'node': node_url,
'volume': volume_url}))
else:
raise exception.VolumeBackendAPIException(
data=(_("Post-attach failed from %(node)s to %(volume)s.")
% {'node': node_url,
'volume': volume_url}))
def detach_volume_from_node(self, volume_url, node_url):
LOG.info('Trying detach from node %s for volume %s',
node_url, volume_url)
try:
volume = self._get_volume(volume_url)
node = self._get_node(node_url)
node.detach_endpoint(volume.path)
except Exception:
LOG.exception("Detach volume failed")
raise exception.VolumeBackendAPIException(
data=(_("Detach failed from %(node)s for %(volume)s.")
% {'node': node_url,
'volume': volume_url}))
def detach_all_node_connections_for_volume(self, volume_url):
try:
volume = self._get_volume(volume_url)
nodes = self.rsdlib.get_node_collection().get_members()
for node in nodes:
if node:
if volume.path in node.get_allowed_detach_endpoints():
node.detach_endpoint(volume.path)
except Exception:
LOG.exception("Detach failed for volume from all host "
"connections")
raise exception.VolumeBackendAPIException(
data=(_("Detach failed for %(volume)s from all host "
"connections.")
% {'volume': volume_url}))
@interface.volumedriver
class RSDDriver(driver.VolumeDriver):
"""Openstack driver to perform NVMe-oF volume management in RSD Solution
.. code-block:: none
Version History:
1.0.0: Initial driver
"""
VERSION = '1.0.0'
CI_WIKI_NAME = 'INTEL-RSD-CI'
def __init__(self, *args, **kwargs):
super(RSDDriver, self).__init__(*args, **kwargs)
self.configuration.append_config_values(RSD_OPTS)
self.rsdClient = None
@staticmethod
def get_driver_options():
return RSD_OPTS
@utils.trace
def do_setup(self, context):
self.rsdClient = RSDClient.initialize(
self.configuration.podm_url,
self.configuration.podm_username,
self.configuration.podm_password,
self.configuration.suppress_requests_ssl_warnings)
def check_for_setup_error(self):
pass
@utils.trace
def create_volume(self, volume):
size_in_gb = int(volume['size'])
volume_url = self.rsdClient.create_volume(size_in_gb)
return {'provider_location': volume_url}
@utils.trace
def delete_volume(self, volume):
volume_url = volume['provider_location']
if not volume_url:
return
self.rsdClient.delete_vol_or_snap(volume_url,
volume_name=volume.name,
ignore_non_exist=True)
provider_snap_url = volume.metadata.get("rsd_provider_snap")
if provider_snap_url:
self.rsdClient.delete_vol_or_snap(provider_snap_url,
volume_name=volume.name,
ignore_non_exist=True)
def _update_volume_stats(self):
backend_name = (
self.configuration.safe_get('volume_backend_name') or 'RSD')
ret = self.rsdClient.get_stats()
(free_capacity_gb,
total_capacity_gb,
allocated_capacity_gb,
total_volumes) = ret
spool = {}
spool['pool_name'] = backend_name
spool['total_capacity_gb'] = total_capacity_gb
spool['free_capacity_gb'] = free_capacity_gb
spool['allocated_capacity_gb'] = allocated_capacity_gb
spool['thin_provisioning_support'] = True
spool['thick_provisioning_support'] = True
spool['multiattach'] = False
self._stats['volume_backend_name'] = backend_name
self._stats['vendor_name'] = 'Intel'
self._stats['driver_version'] = self.VERSION
self._stats['storage_protocol'] = 'nvmeof'
# SinglePool
self._stats['pools'] = [spool]
@utils.trace
def get_volume_stats(self, refresh=False):
if refresh:
self._update_volume_stats()
return self._stats
@utils.trace
def initialize_connection(self, volume, connector, **kwargs):
uuid = connector.get("system uuid")
if not uuid:
msg = _("initialize_connection error: no uuid available!")
LOG.exception(msg)
raise exception.VolumeBackendAPIException(msg)
node_url = self.rsdClient.get_node_url_by_uuid(uuid)
if not node_url:
msg = (_("initialize_connection error: no node_url from uuid %s!")
% uuid)
LOG.exception(msg)
raise exception.VolumeBackendAPIException(msg)
volume_url = volume['provider_location']
target_ip, target_port, target_nqn, initiator_nqn = (
self.rsdClient.attach_volume_to_node(volume_url, node_url))
conn_info = {
'driver_volume_type': 'nvmeof',
'data': {
'transport_type': 'rdma',
'host_nqn': initiator_nqn,
'nqn': target_nqn,
'target_port': target_port,
'target_portal': target_ip,
}
}
return conn_info
@utils.trace
def terminate_connection(self, volume, connector, **kwargs):
if connector is None:
# None connector means force-detach
volume_url = volume['provider_location']
self.rsdClient.detach_all_node_connections_for_volume(volume_url)
return
uuid = connector.get("system uuid")
if not uuid:
msg = _("terminate_connection error: no uuid available!")
LOG.exception(msg)
raise exception.VolumeBackendAPIException(msg)
node_url = self.rsdClient.get_node_url_by_uuid(uuid)
if not node_url:
msg = (_("terminate_connection error: no node_url from uuid %s!")
% uuid)
LOG.exception(msg)
raise exception.VolumeBackendAPIException(msg)
volume_url = volume['provider_location']
self.rsdClient.detach_volume_from_node(volume_url, node_url)
def ensure_export(self, context, volume):
pass
def create_export(self, context, volume, connector):
pass
def remove_export(self, context, volume):
pass
@utils.trace
def create_volume_from_snapshot(self, volume, snapshot):
snap_url = snapshot.provider_location
old_size_in_gb = snapshot.volume_size
size_in_gb = volume.size
volume_url = self.rsdClient.create_volume_from_snap(snap_url)
if size_in_gb != old_size_in_gb:
try:
self.rsdClient.extend_volume(volume_url, size_in_gb)
except Exception:
self.rsdClient.delete_vol_or_snap(volume_url,
volume_name=volume.name)
raise
return {'provider_location': volume_url}
@utils.trace
def create_snapshot(self, snapshot):
volume_url = snapshot.volume.provider_location
snap_url = self.rsdClient.create_snap(volume_url)
snapshot.provider_location = snap_url
snapshot.save()
@utils.trace
def delete_snapshot(self, snapshot):
snap_url = snapshot.provider_location
if not snap_url:
return
self.rsdClient.delete_vol_or_snap(snap_url,
volume_name=snapshot.name,
ignore_non_exist=True)
@utils.trace
def extend_volume(self, volume, new_size):
volume_url = volume.provider_location
self.rsdClient.extend_volume(volume_url, new_size)
def clone_image(self, context, volume,
image_location, image_meta,
image_service):
return None, False
@utils.trace
def create_cloned_volume(self, volume, src_vref):
volume_url = src_vref.provider_location
old_size_in_gb = src_vref.size
size_in_gb = volume.size
new_vol_url, provider_snap_url = self.rsdClient.clone_volume(
volume_url)
metadata = get_volume_metadata(volume)
metadata["rsd_provider_snap"] = provider_snap_url
if size_in_gb != old_size_in_gb:
try:
self.rsdClient.extend_volume(new_vol_url, size_in_gb)
except Exception:
self.rsdClient.delete_vol_or_snap(new_vol_url,
volume_name=volume.name)
self.rsdClient.delete_vol_or_snap(provider_snap_url,
volume_name=volume.name)
raise
return {'provider_location': new_vol_url,
'metadata': metadata}

View File

@ -0,0 +1,52 @@
====================================
Intel Rack Scale Design (RSD) driver
====================================
The Intel Rack Scale Design volume driver is a block storage driver providing
NVMe-oF support for RSD storage.
System requirements
~~~~~~~~~~~~~~~~~~~
To use the RSD driver, the following requirements are needed:
* The driver only supports RSD API at version 2.4 or later.
* The driver requires rsd-lib.
* ``cinder-volume`` should be running on one of the composed node in RSD, and
have access to the PODM url.
* All the ``nova-compute`` services should be running on the composed nodes in
RSD.
* All the ``cinder-volume`` and ``nova-compute`` nodes should have installed
``dmidecode`` and the latest ``nvme-cli`` with connect/disconnect
subcommands.
Supported operations
~~~~~~~~~~~~~~~~~~~~
* Create, delete volumes.
* Attach, detach volumes.
* Copy an image to a volume.
* Copy a volume to an image.
* Create, delete snapshots.
* Create a volume from a snapshot.
* Clone a volume.
* Extend a volume.
* Get volume statistics.
Configuration
~~~~~~~~~~~~~
On ``cinder-volume`` nodes, using the following configurations in your
``/etc/cinder/cinder.conf``:
.. code-block:: ini
volume_driver = cinder.volume.drivers.rsd.RSDDriver
The following table contains the configuration options supported by the
RSD driver:
.. config-table::
:config-target: RSD
cinder.volume.drivers.rsd

View File

@ -44,3 +44,6 @@ infi.dtypes.iqn # PSF
# Storpool # Storpool
storpool # Apache-2.0 storpool # Apache-2.0
# RSD Driver
rsd-lib # Apache-2.0

View File

@ -0,0 +1,4 @@
---
features:
- |
Added a new Cinder driver for RackScale Design NVMe-oF storage solution.