d40fae9756
It may happen that one pool is inside a volume of other pool. This is the case for example for varlibqubes pool (file driver, dir_path=/var/lib/qubes) and default lvm pool (lvm_thin driver). The latter include whole root filesystem, so /var/lib/qubes too. This is relevant for proper disk space calculation - to not count some space twice. QubesOS/qubes-issues#3240 QubesOS/qubes-issues#3241
230 lines
6.4 KiB
Python
230 lines
6.4 KiB
Python
#
|
|
# 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(LinuxModules, self).__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=None, dir_path=None):
|
|
assert dir_path, 'Missing dir_path'
|
|
super(LinuxKernel, self).__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(
|
|
'LinuxModules 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)
|
|
|
|
@property
|
|
def 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)
|