xen.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. #!/usr/bin/python2
  2. #
  3. # The Qubes OS Project, http://www.qubes-os.org
  4. #
  5. # Copyright (C) 2013 Marek Marczykowski <marmarek@invisiblethingslab.com>
  6. #
  7. # This program is free software; you can redistribute it and/or
  8. # modify it under the terms of the GNU General Public License
  9. # as published by the Free Software Foundation; either version 2
  10. # of the License, or (at your option) any later version.
  11. #
  12. # This program is distributed in the hope that it will be useful,
  13. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. # GNU General Public License for more details.
  16. #
  17. # You should have received a copy of the GNU General Public License
  18. # along with this program; if not, write to the Free Software
  19. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
  20. # USA.
  21. #
  22. from __future__ import absolute_import
  23. import os
  24. import os.path
  25. import re
  26. import subprocess
  27. import sys
  28. from qubes.qubes import QubesException, vm_files
  29. from qubes.storage import Pool, QubesVmStorage
  30. class XenStorage(QubesVmStorage):
  31. """
  32. Class for VM storage of Xen VMs.
  33. """
  34. def __init__(self, vm, vmdir, **kwargs):
  35. """ Instantiate the storage.
  36. Args:
  37. vm: a QubesVM
  38. vmdir: the root directory of the pool
  39. """
  40. assert vm is not None
  41. assert vmdir is not None
  42. super(XenStorage, self).__init__(vm, **kwargs)
  43. self.vmdir = vmdir
  44. if self.vm.is_template():
  45. self.rootcow_img = os.path.join(self.vmdir,
  46. vm_files["rootcow_img"])
  47. else:
  48. self.rootcow_img = None
  49. self.private_img = os.path.join(vmdir, 'private.img')
  50. if self.vm.template:
  51. self.root_img = self.vm.template.root_img
  52. else:
  53. self.root_img = os.path.join(vmdir, 'root.img')
  54. self.volatile_img = os.path.join(vmdir, 'volatile.img')
  55. def _get_rootdev(self):
  56. if self.vm.is_template() and \
  57. os.path.exists(os.path.join(self.vmdir, "root-cow.img")):
  58. return self.format_disk_dev(
  59. "{dir}/root.img:{dir}/root-cow.img".format(
  60. dir=self.vmdir),
  61. "block-origin", self.root_dev, True)
  62. elif self.vm.template and not self.vm.template.storage.rootcow_img:
  63. # HVM template-based VM - template doesn't have own
  64. # root-cow.img, only one device-mapper layer
  65. return self.format_disk_dev(
  66. "{tpldir}/root.img:{vmdir}/volatile.img".format(
  67. tpldir=self.vm.template.dir_path,
  68. vmdir=self.vmdir),
  69. "block-snapshot", self.root_dev, True)
  70. elif self.vm.template:
  71. # any other template-based VM - two device-mapper layers: one
  72. # in dom0 (here) from root+root-cow, and another one from
  73. # this+volatile.img
  74. return self.format_disk_dev(
  75. "{dir}/root.img:{dir}/root-cow.img".format(
  76. dir=self.vm.template.dir_path),
  77. "block-snapshot", self.root_dev, False)
  78. else:
  79. return self.format_disk_dev(
  80. "{dir}/root.img".format(dir=self.vmdir),
  81. None, self.root_dev, True)
  82. def get_config_params(self):
  83. args = {}
  84. args['rootdev'] = self._get_rootdev()
  85. args['privatedev'] = \
  86. self.format_disk_dev(self.private_img,
  87. None, self.private_dev, True)
  88. args['volatiledev'] = \
  89. self.format_disk_dev(self.volatile_img,
  90. None, self.volatile_dev, True)
  91. if self.modules_img is not None:
  92. args['otherdevs'] = \
  93. self.format_disk_dev(self.modules_img,
  94. None, self.modules_dev, self.modules_img_rw)
  95. elif self.drive is not None:
  96. (drive_type, drive_domain, drive_path) = self.drive.split(":")
  97. if drive_type == "hd":
  98. drive_type = "disk"
  99. if drive_domain.lower() == "dom0":
  100. drive_domain = None
  101. args['otherdevs'] = self.format_disk_dev(drive_path, None,
  102. self.modules_dev,
  103. rw=True if drive_type == "disk" else False, type=drive_type,
  104. domain=drive_domain)
  105. else:
  106. args['otherdevs'] = ''
  107. return args
  108. def create_on_disk_private_img(self, verbose, source_template = None):
  109. if source_template:
  110. template_priv = source_template.private_img
  111. if verbose:
  112. print >> sys.stderr, "--> Copying the template's private image: {0}".\
  113. format(template_priv)
  114. self._copy_file(template_priv, self.private_img)
  115. else:
  116. f_private = open (self.private_img, "a+b")
  117. f_private.truncate (self.private_img_size)
  118. f_private.close ()
  119. def create_on_disk_root_img(self, verbose, source_template = None):
  120. if source_template:
  121. if not self.vm.updateable:
  122. # just use template's disk
  123. return
  124. else:
  125. template_root = source_template.root_img
  126. if verbose:
  127. print >> sys.stderr, "--> Copying the template's root image: {0}".\
  128. format(template_root)
  129. self._copy_file(template_root, self.root_img)
  130. else:
  131. f_root = open (self.root_img, "a+b")
  132. f_root.truncate (self.root_img_size)
  133. f_root.close ()
  134. if self.vm.is_template():
  135. self.commit_template_changes()
  136. def rename(self, old_name, new_name):
  137. super(XenStorage, self).rename(old_name, new_name)
  138. old_dirpath = os.path.join(os.path.dirname(self.vmdir), old_name)
  139. if self.rootcow_img:
  140. self.rootcow_img = self.rootcow_img.replace(old_dirpath,
  141. self.vmdir)
  142. def resize_private_img(self, size):
  143. f_private = open (self.private_img, "a+b")
  144. f_private.truncate (size)
  145. f_private.close ()
  146. # find loop device if any
  147. p = subprocess.Popen (["sudo", "losetup", "--associated", self.private_img],
  148. stdout=subprocess.PIPE)
  149. result = p.communicate()
  150. m = re.match(r"^(/dev/loop\d+):\s", result[0])
  151. if m is not None:
  152. loop_dev = m.group(1)
  153. # resize loop device
  154. subprocess.check_call(["sudo", "losetup", "--set-capacity", loop_dev])
  155. def commit_template_changes(self):
  156. assert self.vm.is_template()
  157. if not self.rootcow_img:
  158. return
  159. if os.path.exists (self.rootcow_img):
  160. os.rename (self.rootcow_img, self.rootcow_img + '.old')
  161. old_umask = os.umask(002)
  162. f_cow = open (self.rootcow_img, "w")
  163. f_root = open (self.root_img, "r")
  164. f_root.seek(0, os.SEEK_END)
  165. f_cow.truncate (f_root.tell()) # make empty sparse file of the same size as root.img
  166. f_cow.close ()
  167. f_root.close()
  168. os.umask(old_umask)
  169. def reset_volatile_storage(self, verbose = False, source_template = None):
  170. if source_template is None:
  171. source_template = self.vm.template
  172. if source_template is not None:
  173. # template-based VM with only one device-mapper layer -
  174. # volatile.img used as upper layer on root.img, no root-cow.img
  175. # intermediate layer
  176. if not source_template.storage.rootcow_img:
  177. if os.path.exists(self.volatile_img):
  178. if self.vm.debug:
  179. if os.path.getmtime(source_template.storage.root_img)\
  180. > os.path.getmtime(self.volatile_img):
  181. if verbose:
  182. print >>sys.stderr, "--> WARNING: template have changed, resetting root.img"
  183. else:
  184. if verbose:
  185. print >>sys.stderr, "--> Debug mode: not resetting root.img"
  186. print >>sys.stderr, "--> Debug mode: if you want to force root.img reset, either update template VM, or remove volatile.img file"
  187. return
  188. os.remove(self.volatile_img)
  189. f_volatile = open(self.volatile_img, "w")
  190. f_root = open(source_template.storage.root_img, "r")
  191. f_root.seek(0, os.SEEK_END)
  192. f_volatile.truncate(f_root.tell()) # make empty sparse file of the same size as root.img
  193. f_volatile.close()
  194. f_root.close()
  195. return
  196. super(XenStorage, self).reset_volatile_storage(
  197. verbose=verbose, source_template=source_template)
  198. def prepare_for_vm_startup(self, verbose):
  199. super(XenStorage, self).prepare_for_vm_startup(verbose=verbose)
  200. if self.drive is not None:
  201. (drive_type, drive_domain, drive_path) = self.drive.split(":")
  202. if drive_domain.lower() != "dom0":
  203. try:
  204. # FIXME: find a better way to access QubesVmCollection
  205. drive_vm = self.vm._collection.get_vm_by_name(drive_domain)
  206. # prepare for improved QubesVmCollection
  207. if drive_vm is None:
  208. raise KeyError
  209. if not drive_vm.is_running():
  210. raise QubesException(
  211. "VM '{}' holding '{}' isn't running".format(
  212. drive_domain, drive_path))
  213. except KeyError:
  214. raise QubesException(
  215. "VM '{}' holding '{}' does not exists".format(
  216. drive_domain, drive_path))
  217. class XenPool(Pool):
  218. def __init__(self, vm, dir_path):
  219. super(XenPool, self).__init__(vm, dir_path)
  220. def getStorage(self):
  221. """ Returns an instantiated ``XenStorage``. """
  222. return XenStorage(self.vm, vmdir=self.vmdir)