summaryrefslogtreecommitdiffstats
path: root/nova/virt
diff options
context:
space:
mode:
authorKei Masumoto <masumotok@nttdata.co.jp>2011-08-15 20:31:43 +0000
committerTarmac <>2011-08-15 20:31:43 +0000
commitea53d0f37a4f478ffbe18516f99ca26192117e80 (patch)
tree3a56196e11ead6af9ead0eb13a345f6c7e63dcf8 /nova/virt
parenta18d8597c2fb52d77b8b827f2c440787b1165150 (diff)
parent7393a114f421f1b54019099777fea34a09a80737 (diff)
Adding kvm-block-migration feature.
I wrote some description the below URL. I hope it may help for reviewing. <http://etherpad.openstack.org/kvm-block-migration>
Diffstat (limited to 'nova/virt')
-rw-r--r--nova/virt/fake.py2
-rw-r--r--nova/virt/libvirt/connection.py182
-rw-r--r--nova/virt/xenapi_conn.py2
3 files changed, 163 insertions, 23 deletions
diff --git a/nova/virt/fake.py b/nova/virt/fake.py
index 2ffa33d40..dc0628772 100644
--- a/nova/virt/fake.py
+++ b/nova/virt/fake.py
@@ -492,7 +492,7 @@ class FakeConnection(driver.ComputeDriver):
raise NotImplementedError('This method is supported only by libvirt.')
def live_migration(self, context, instance_ref, dest,
- post_method, recover_method):
+ post_method, recover_method, block_migration=False):
"""This method is supported only by libvirt."""
return
diff --git a/nova/virt/libvirt/connection.py b/nova/virt/libvirt/connection.py
index 71516011a..2b17e244a 100644
--- a/nova/virt/libvirt/connection.py
+++ b/nova/virt/libvirt/connection.py
@@ -117,6 +117,10 @@ flags.DEFINE_string('live_migration_uri',
flags.DEFINE_string('live_migration_flag',
"VIR_MIGRATE_UNDEFINE_SOURCE, VIR_MIGRATE_PEER2PEER",
'Define live migration behavior.')
+flags.DEFINE_string('block_migration_flag',
+ "VIR_MIGRATE_UNDEFINE_SOURCE, VIR_MIGRATE_PEER2PEER, "
+ "VIR_MIGRATE_NON_SHARED_INC",
+ 'Define block migration behavior.')
flags.DEFINE_integer('live_migration_bandwidth', 0,
'Define live migration behavior')
flags.DEFINE_string('qemu_img', 'qemu-img',
@@ -727,6 +731,7 @@ class LibvirtConnection(driver.ComputeDriver):
If cow is True, it will make a CoW image instead of a copy.
"""
+
if not os.path.exists(target):
base_dir = os.path.join(FLAGS.instances_path, '_base')
if not os.path.exists(base_dir):
@@ -1549,7 +1554,7 @@ class LibvirtConnection(driver.ComputeDriver):
time.sleep(1)
def live_migration(self, ctxt, instance_ref, dest,
- post_method, recover_method):
+ post_method, recover_method, block_migration=False):
"""Spawning live_migration operation for distributing high-load.
:params ctxt: security context
@@ -1557,20 +1562,22 @@ class LibvirtConnection(driver.ComputeDriver):
nova.db.sqlalchemy.models.Instance object
instance object that is migrated.
:params dest: destination host
+ :params block_migration: destination host
:params post_method:
post operation method.
expected nova.compute.manager.post_live_migration.
:params recover_method:
recovery method when any exception occurs.
expected nova.compute.manager.recover_live_migration.
+ :params block_migration: if true, do block migration.
"""
greenthread.spawn(self._live_migration, ctxt, instance_ref, dest,
- post_method, recover_method)
+ post_method, recover_method, block_migration)
- def _live_migration(self, ctxt, instance_ref, dest,
- post_method, recover_method):
+ def _live_migration(self, ctxt, instance_ref, dest, post_method,
+ recover_method, block_migration=False):
"""Do live migration.
:params ctxt: security context
@@ -1589,27 +1596,21 @@ class LibvirtConnection(driver.ComputeDriver):
# Do live migration.
try:
- flaglist = FLAGS.live_migration_flag.split(',')
+ if block_migration:
+ flaglist = FLAGS.block_migration_flag.split(',')
+ else:
+ flaglist = FLAGS.live_migration_flag.split(',')
flagvals = [getattr(libvirt, x.strip()) for x in flaglist]
logical_sum = reduce(lambda x, y: x | y, flagvals)
- if self.read_only:
- tmpconn = self._connect(self.libvirt_uri, False)
- dom = tmpconn.lookupByName(instance_ref.name)
- dom.migrateToURI(FLAGS.live_migration_uri % dest,
- logical_sum,
- None,
- FLAGS.live_migration_bandwidth)
- tmpconn.close()
- else:
- dom = self._conn.lookupByName(instance_ref.name)
- dom.migrateToURI(FLAGS.live_migration_uri % dest,
- logical_sum,
- None,
- FLAGS.live_migration_bandwidth)
+ dom = self._conn.lookupByName(instance_ref.name)
+ dom.migrateToURI(FLAGS.live_migration_uri % dest,
+ logical_sum,
+ None,
+ FLAGS.live_migration_bandwidth)
except Exception:
- recover_method(ctxt, instance_ref, dest=dest)
+ recover_method(ctxt, instance_ref, dest, block_migration)
raise
# Waiting for completion of live_migration.
@@ -1621,11 +1622,150 @@ class LibvirtConnection(driver.ComputeDriver):
self.get_info(instance_ref.name)['state']
except exception.NotFound:
timer.stop()
- post_method(ctxt, instance_ref, dest)
+ post_method(ctxt, instance_ref, dest, block_migration)
timer.f = wait_for_live_migration
timer.start(interval=0.5, now=True)
+ def pre_block_migration(self, ctxt, instance_ref, disk_info_json):
+ """Preparation block migration.
+
+ :params ctxt: security context
+ :params instance_ref:
+ nova.db.sqlalchemy.models.Instance object
+ instance object that is migrated.
+ :params disk_info_json:
+ json strings specified in get_instance_disk_info
+
+ """
+ disk_info = utils.loads(disk_info_json)
+
+ # make instance directory
+ instance_dir = os.path.join(FLAGS.instances_path, instance_ref['name'])
+ if os.path.exists(instance_dir):
+ raise exception.DestinationDiskExists(path=instance_dir)
+ os.mkdir(instance_dir)
+
+ for info in disk_info:
+ base = os.path.basename(info['path'])
+ # Get image type and create empty disk image.
+ instance_disk = os.path.join(instance_dir, base)
+ utils.execute('sudo', 'qemu-img', 'create', '-f', info['type'],
+ instance_disk, info['local_gb'])
+
+ # if image has kernel and ramdisk, just download
+ # following normal way.
+ if instance_ref['kernel_id']:
+ user = manager.AuthManager().get_user(instance_ref['user_id'])
+ project = manager.AuthManager().get_project(
+ instance_ref['project_id'])
+ self._fetch_image(nova_context.get_admin_context(),
+ os.path.join(instance_dir, 'kernel'),
+ instance_ref['kernel_id'],
+ user,
+ project)
+ if instance_ref['ramdisk_id']:
+ self._fetch_image(nova_context.get_admin_context(),
+ os.path.join(instance_dir, 'ramdisk'),
+ instance_ref['ramdisk_id'],
+ user,
+ project)
+
+ def post_live_migration_at_destination(self, ctxt,
+ instance_ref,
+ network_info,
+ block_migration):
+ """Post operation of live migration at destination host.
+
+ :params ctxt: security context
+ :params instance_ref:
+ nova.db.sqlalchemy.models.Instance object
+ instance object that is migrated.
+ :params network_info: instance network infomation
+ :params : block_migration: if true, post operation of block_migraiton.
+ """
+ # Define migrated instance, otherwise, suspend/destroy does not work.
+ dom_list = self._conn.listDefinedDomains()
+ if instance_ref.name not in dom_list:
+ instance_dir = os.path.join(FLAGS.instances_path,
+ instance_ref.name)
+ xml_path = os.path.join(instance_dir, 'libvirt.xml')
+ # In case of block migration, destination does not have
+ # libvirt.xml
+ if not os.path.isfile(xml_path):
+ xml = self.to_xml(instance_ref, network_info=network_info)
+ f = open(os.path.join(instance_dir, 'libvirt.xml'), 'w+')
+ f.write(xml)
+ f.close()
+ # libvirt.xml should be made by to_xml(), but libvirt
+ # does not accept to_xml() result, since uuid is not
+ # included in to_xml() result.
+ dom = self._lookup_by_name(instance_ref.name)
+ self._conn.defineXML(dom.XMLDesc(0))
+
+ def get_instance_disk_info(self, ctxt, instance_ref):
+ """Preparation block migration.
+
+ :params ctxt: security context
+ :params instance_ref:
+ nova.db.sqlalchemy.models.Instance object
+ instance object that is migrated.
+ :return:
+ json strings with below format.
+ "[{'path':'disk', 'type':'raw', 'local_gb':'10G'},...]"
+
+ """
+ disk_info = []
+
+ virt_dom = self._lookup_by_name(instance_ref.name)
+ xml = virt_dom.XMLDesc(0)
+ doc = libxml2.parseDoc(xml)
+ disk_nodes = doc.xpathEval('//devices/disk')
+ path_nodes = doc.xpathEval('//devices/disk/source')
+ driver_nodes = doc.xpathEval('//devices/disk/driver')
+
+ for cnt, path_node in enumerate(path_nodes):
+ disk_type = disk_nodes[cnt].get_properties().getContent()
+ path = path_node.get_properties().getContent()
+
+ if disk_type != 'file':
+ LOG.debug(_('skipping %(path)s since it looks like volume') %
+ locals())
+ continue
+
+ # In case of libvirt.xml, disk type can be obtained
+ # by the below statement.
+ # -> disk_type = driver_nodes[cnt].get_properties().getContent()
+ # but this xml is generated by kvm, format is slightly different.
+ disk_type = \
+ driver_nodes[cnt].get_properties().get_next().getContent()
+ if disk_type == 'raw':
+ size = int(os.path.getsize(path))
+ else:
+ out, err = utils.execute('sudo', 'qemu-img', 'info', path)
+ size = [i.split('(')[1].split()[0] for i in out.split('\n')
+ if i.strip().find('virtual size') >= 0]
+ size = int(size[0])
+
+ # block migration needs same/larger size of empty image on the
+ # destination host. since qemu-img creates bit smaller size image
+ # depending on original image size, fixed value is necessary.
+ for unit, divisor in [('G', 1024 ** 3), ('M', 1024 ** 2),
+ ('K', 1024), ('', 1)]:
+ if size / divisor == 0:
+ continue
+ if size % divisor != 0:
+ size = size / divisor + 1
+ else:
+ size = size / divisor
+ size = str(size) + unit
+ break
+
+ disk_info.append({'type': disk_type, 'path': path,
+ 'local_gb': size})
+
+ return utils.dumps(disk_info)
+
def unfilter_instance(self, instance_ref, network_info):
"""See comments of same method in firewall_driver."""
self.firewall_driver.unfilter_instance(instance_ref,
diff --git a/nova/virt/xenapi_conn.py b/nova/virt/xenapi_conn.py
index 0ec957cf3..0d23e7689 100644
--- a/nova/virt/xenapi_conn.py
+++ b/nova/virt/xenapi_conn.py
@@ -314,7 +314,7 @@ class XenAPIConnection(driver.ComputeDriver):
return
def live_migration(self, context, instance_ref, dest,
- post_method, recover_method):
+ post_method, recover_method, block_migration=False):
"""This method is supported only by libvirt."""
return