123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- #
- # The Qubes OS Project, http://www.qubes-os.org
- #
- # Copyright (C) 2010-2015 Joanna Rutkowska <joanna@invisiblethingslab.com>
- # Copyright (C) 2015 Wojtek Porczyk <woju@invisiblethingslab.com>
- # Copyright (C) 2016 Bahtiar `kalkin-` Gadimov <bahtiar@gadimov.de>
- #
- # This library is free software; you can redistribute it and/or
- # modify it under the terms of the GNU Lesser General Public
- # License as published by the Free Software Foundation; either
- # version 2.1 of the License, or (at your option) any later version.
- #
- # This library is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- # Lesser General Public License for more details.
- #
- # You should have received a copy of the GNU Lesser General Public
- # License along with this library; if not, see <https://www.gnu.org/licenses/>.
- #
- ''' This module contains pool implementations for different OS kernels. '''
- import os
- import qubes.exc
- import qubes.storage
- from qubes.storage import Pool, StoragePoolException, Volume
- class LinuxModules(Volume):
- ''' A volume representing a ro linux kernel '''
- def __init__(self, target_dir, kernel_version, **kwargs):
- kwargs['vid'] = ''
- super().__init__(**kwargs)
- self._kernel_version = kernel_version
- self.target_dir = target_dir
- assert self.revisions_to_keep == 0
- assert self.rw is False
- @property
- def vid(self):
- if callable(self._kernel_version):
- return self._kernel_version()
- return self._kernel_version
- @vid.setter
- def vid(self, value):
- # ignore
- pass
- @property
- def kernels_dir(self):
- kernel_version = self.vid
- if not kernel_version:
- return None
- return os.path.join(self.target_dir, kernel_version)
- @property
- def path(self):
- kernels_dir = self.kernels_dir
- if not kernels_dir:
- return None
- return os.path.join(kernels_dir, 'modules.img')
- @property
- def vmlinuz(self):
- kernels_dir = self.kernels_dir
- if not kernels_dir:
- return None
- return os.path.join(kernels_dir, 'vmlinuz')
- @property
- def initramfs(self):
- kernels_dir = self.kernels_dir
- if not kernels_dir:
- return None
- return os.path.join(kernels_dir, 'initramfs')
- @property
- def revisions(self):
- return {}
- def is_dirty(self):
- return False
- def import_volume(self, src_volume):
- if isinstance(src_volume, LinuxModules):
- # do nothing
- return self
- raise StoragePoolException('clone of LinuxModules volume from '
- 'different volume type is not supported')
- def create(self):
- return self
- def remove(self):
- pass
- def commit(self):
- return self
- def export(self):
- return self.path
- def is_outdated(self):
- return False
- @property
- def revisions_to_keep(self):
- return 0
- @revisions_to_keep.setter
- def revisions_to_keep(self, value):
- # pylint: disable=no-self-use
- if value:
- raise qubes.exc.QubesValueError(
- 'LinuxModules supports only revisions_to_keep=0')
- @property
- def rw(self):
- return False
- @rw.setter
- def rw(self, value):
- # pylint: disable=no-self-use
- if value:
- raise qubes.exc.QubesValueError(
- 'LinuxModules supports only read-only volumes')
- def start(self):
- return self
- def stop(self):
- pass
- def verify(self):
- if self.vid:
- _check_path(self.vmlinuz)
- _check_path(self.initramfs)
- def block_device(self):
- path = self.path
- # create block device for modules.img only if:
- # - there is kernel set for the VM
- # - that kernel directory contains modules.img file
- if path and os.path.exists(path):
- return super().block_device()
- return None
- class LinuxKernel(Pool):
- ''' Provides linux kernels '''
- driver = 'linux-kernel'
- def __init__(self, *, name, dir_path):
- super().__init__(name=name, revisions_to_keep=0)
- self.dir_path = dir_path
- def init_volume(self, vm, volume_config):
- assert not volume_config['rw']
- # migrate old config
- if volume_config.get('snap_on_start', False) and not \
- volume_config.get('source', None):
- volume_config['snap_on_start'] = False
- if volume_config.get('save_on_stop', False):
- raise NotImplementedError(
- 'LinuxKernel pool does not support save_on_stop=True')
- volume_config['pool'] = self
- volume = LinuxModules(self.dir_path, lambda: vm.kernel, **volume_config)
- return volume
- @property
- def config(self):
- return {
- 'name': self.name,
- 'dir_path': self.dir_path,
- 'driver': LinuxKernel.driver,
- }
- def destroy(self):
- pass
- def import_volume(self, dst_pool, dst_volume, src_pool, src_volume):
- pass
- def setup(self):
- pass
- @property
- def revisions_to_keep(self):
- return 0
- @revisions_to_keep.setter
- def revisions_to_keep(self, value):
- # pylint: disable=no-self-use
- if value:
- raise qubes.exc.QubesValueError(
- 'LinuxKernel supports only revisions_to_keep=0')
- def included_in(self, app):
- ''' Check if there is pool containing /var/lib/qubes/vm-kernels '''
- return qubes.storage.search_pool_containing_dir(
- [pool for pool in app.pools.values() if pool is not self],
- self.dir_path)
- def list_volumes(self):
- ''' Return all known kernel volumes '''
- return [LinuxModules(self.dir_path,
- kernel_version,
- pool=self,
- name=kernel_version,
- rw=False
- )
- for kernel_version in os.listdir(self.dir_path)]
- def _check_path(path):
- ''' Raise an :py:class:`qubes.storage.StoragePoolException` if ``path`` does
- not exist.
- '''
- if not os.path.exists(path):
- raise StoragePoolException('Missing file: %s' % path)
|