From 3564250298240c176d6a9c6b4c865aab6b0d683a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marek=20Marczykowski-G=C3=B3recki?= Date: Mon, 12 Jun 2017 09:46:26 +0200 Subject: [PATCH 1/4] devices: drop 'data' and 'frontend_domain' fields, rename 'devclass' to 'bus' Drop DeviceInfo.data - device extension should provide a subclass with proper individual fields. Drop DeviceAssignment.frontend_domain - this information is redundant - frontend domain is defined by where DeviceAssignment is attached. Rename DeviceCollection.devclass to bus - devclass if confusing here, because this term is also used for DeviceInfo subclass. --- qubes/api/admin.py | 3 +- qubes/devices.py | 74 ++++++++++++++++------------------------ qubes/tests/api_admin.py | 3 +- 3 files changed, 32 insertions(+), 48 deletions(-) diff --git a/qubes/api/admin.py b/qubes/api/admin.py index 712b75a7..be596696 100644 --- a/qubes/api/admin.py +++ b/qubes/api/admin.py @@ -817,12 +817,11 @@ class QubesAdminAPI(qubes.api.AbstractQubesAPI): non_default_attrs = set(attr for attr in dir(dev) if not attr.startswith('_')).difference(( 'backend_domain', 'ident', 'frontend_domain', - 'description', 'options', 'data')) + 'description', 'options')) properties_txt = ' '.join( '{}={!s}'.format(prop, value) for prop, value in itertools.chain( ((key, getattr(dev, key)) for key in non_default_attrs), - dev.data.items(), # keep description as the last one, according to API # specification (('description', dev.description),) diff --git a/qubes/devices.py b/qubes/devices.py index 4540f19d..4bcb63a1 100644 --- a/qubes/devices.py +++ b/qubes/devices.py @@ -60,13 +60,12 @@ class DeviceAssignment(object): # pylint: disable=too-few-public-methods ''' Maps a device to a frontend_domain. ''' def __init__(self, backend_domain, ident, options=None, persistent=False, - frontend_domain=None, devclass=None): + bus=None): self.backend_domain = backend_domain self.ident = ident self.options = options or {} self.persistent = persistent - self.frontend_domain = frontend_domain - self.devclass = devclass + self.bus = bus def __repr__(self): return "[%s]:%s" % (self.backend_domain, self.ident) @@ -89,14 +88,13 @@ class DeviceAssignment(object): # pylint: disable=too-few-public-methods self.ident, self.options, self.persistent, - self.frontend_domain, - self.devclass, + self.bus, ) @property def device(self): '''Get DeviceInfo object corresponding to this DeviceAssignment''' - return self.backend_domain.devices[self.devclass][self.ident] + return self.backend_domain.devices[self.bus][self.ident] class DeviceCollection(object): @@ -105,7 +103,7 @@ class DeviceCollection(object): Used as default value for :py:meth:`DeviceManager.__missing__` factory. :param vm: VM for which we manage devices - :param class_: device class + :param bus: device bus This class emits following events on VM object: @@ -154,13 +152,13 @@ class DeviceCollection(object): ''' - def __init__(self, vm, class_): + def __init__(self, vm, bus): self._vm = vm - self._class = class_ + self._bus = bus self._set = PersistentCollection() self.devclass = qubes.utils.get_entry_point_one( - 'qubes.devices', self._class) + 'qubes.devices', self._bus) def attach(self, device_assignment: DeviceAssignment): '''Attach (add) device to domain. @@ -168,16 +166,10 @@ class DeviceCollection(object): :param DeviceInfo device: device object ''' - if not device_assignment.frontend_domain: - device_assignment.frontend_domain = self._vm + if device_assignment.bus is None: + device_assignment.bus = self._bus else: - assert device_assignment.frontend_domain == self._vm, \ - "Trying to attach DeviceAssignment belonging to other domain" - - if device_assignment.devclass is None: - device_assignment.devclass = self._class - else: - assert device_assignment.devclass == self._class, \ + assert device_assignment.bus == self._bus, \ "Trying to attach DeviceAssignment of a different device class" if not device_assignment.persistent and self._vm.is_halted(): @@ -187,12 +179,12 @@ class DeviceCollection(object): if device in self.assignments(): raise DeviceAlreadyAttached( 'device {!s} of class {} already attached to {!s}'.format( - device, self._class, self._vm)) - self._vm.fire_event_pre('device-pre-attach:'+self._class, + device, self._bus, self._vm)) + self._vm.fire_event_pre('device-pre-attach:'+self._bus, device=device, options=device_assignment.options) if device_assignment.persistent: self._set.add(device_assignment) - self._vm.fire_event('device-attach:' + self._class, + self._vm.fire_event('device-attach:' + self._bus, device=device, options=device_assignment.options) def detach(self, device_assignment: DeviceAssignment): @@ -201,13 +193,10 @@ class DeviceCollection(object): :param DeviceInfo device: device object ''' - if not device_assignment.frontend_domain: - device_assignment.frontend_domain = self._vm - - if device_assignment.devclass is None: - device_assignment.devclass = self._class + if device_assignment.bus is None: + device_assignment.bus = self._bus else: - assert device_assignment.devclass == self._class, \ + assert device_assignment.bus == self._bus, \ "Trying to attach DeviceAssignment of a different device class" if device_assignment in self._set and not self._vm.is_halted(): @@ -216,19 +205,19 @@ class DeviceCollection(object): if device_assignment not in self.assignments(): raise DeviceNotAttached( 'device {!s} of class {} not attached to {!s}'.format( - device_assignment.ident, self._class, self._vm)) + device_assignment.ident, self._bus, self._vm)) device = device_assignment.device - self._vm.fire_event_pre('device-pre-detach:'+self._class, device=device) + self._vm.fire_event_pre('device-pre-detach:'+self._bus, device=device) if device in self._set: device_assignment.persistent = True self._set.discard(device_assignment) - self._vm.fire_event('device-detach:' + self._class, device=device) + self._vm.fire_event('device-detach:' + self._bus, device=device) def attached(self): '''List devices which are (or may be) attached to this vm ''' - attached = self._vm.fire_event('device-list-attached:' + self._class) + attached = self._vm.fire_event('device-list-attached:' + self._bus) if attached: return [dev for dev, _ in attached] @@ -252,7 +241,7 @@ class DeviceCollection(object): attached persistently. ''' - devices = self._vm.fire_event('device-list-attached:' + self._class, + devices = self._vm.fire_event('device-list-attached:' + self._bus, persistent=persistent) result = set() for dev, options in devices: @@ -267,15 +256,14 @@ class DeviceCollection(object): DeviceAssignment( backend_domain=dev.backend_domain, ident=dev.ident, options=options, - devclass=self._class, - frontend_domain=self._vm)) + bus=self._bus)) if persistent is not False: result.update(self._set) return result def available(self): '''List devices exposed by this vm''' - devices = self._vm.fire_event('device-list:' + self._class) + devices = self._vm.fire_event('device-list:' + self._bus) return devices def __iter__(self): @@ -294,7 +282,7 @@ class DeviceCollection(object): :raises AssertionError: when multiple devices with the same ident are found ''' - dev = self._vm.fire_event('device-get:' + self._class, ident=ident) + dev = self._vm.fire_event('device-get:' + self._bus, ident=ident) if dev: assert len(dev) == 1 return dev[0] @@ -321,7 +309,7 @@ class DeviceInfo(object): ''' Holds all information about a device ''' # pylint: disable=too-few-public-methods def __init__(self, backend_domain, ident, description=None, - frontend_domain=None, options=None, **kwargs): + frontend_domain=None): #: domain providing this device self.backend_domain = backend_domain #: device identifier (unique for given domain and device type) @@ -337,8 +325,6 @@ class DeviceInfo(object): self.frontend_domain = frontend_domain except AttributeError: pass - self.options = options or dict() - self.data = kwargs if hasattr(self, 'regex'): # pylint: disable=no-member @@ -368,11 +354,11 @@ class UnknownDevice(DeviceInfo): '''Unknown device - for example exposed by domain not running currently''' def __init__(self, backend_domain, ident, description=None, - frontend_domain=None, **kwargs): + frontend_domain=None): if description is None: description = "Unknown device" super(UnknownDevice, self).__init__(backend_domain, ident, description, - frontend_domain, **kwargs) + frontend_domain) class PersistentCollection(object): @@ -385,7 +371,7 @@ class PersistentCollection(object): def add(self, assignment: DeviceAssignment): ''' Add assignment to collection ''' - assert assignment.persistent and assignment.frontend_domain + assert assignment.persistent vm = assignment.backend_domain ident = assignment.ident key = (vm, ident) @@ -395,7 +381,7 @@ class PersistentCollection(object): def discard(self, assignment): ''' Discard assignment from collection ''' - assert assignment.persistent and assignment.frontend_domain + assert assignment.persistent vm = assignment.backend_domain ident = assignment.ident key = (vm, ident) diff --git a/qubes/tests/api_admin.py b/qubes/tests/api_admin.py index c10ea375..36c3d8ac 100644 --- a/qubes/tests/api_admin.py +++ b/qubes/tests/api_admin.py @@ -1349,7 +1349,6 @@ class TC_00_VMs(AdminAPITestCase): return dev = qubes.devices.DeviceInfo(self.vm, '1234') dev.description = 'Some device' - dev.data = {'other_property': 'property-value'} dev.extra_prop = 'xx' yield dev dev = qubes.devices.DeviceInfo(self.vm, '4321') @@ -1361,7 +1360,7 @@ class TC_00_VMs(AdminAPITestCase): value = self.call_mgmt_func(b'admin.vm.device.testclass.Available', b'test-vm1') self.assertEqual(value, - '1234 extra_prop=xx other_property=property-value description=Some ' + '1234 extra_prop=xx description=Some ' 'device\n' '4321 description=Some other device\n') self.assertFalse(self.app.save.called) From 9015414119defa30b87b8dd7b785dd11e803449a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marek=20Marczykowski-G=C3=B3recki?= Date: Mon, 12 Jun 2017 09:52:05 +0200 Subject: [PATCH 2/4] qmemman: fix meminfo parsing for python 3 One more place not converted to python 3. --- qubes/qmemman/algo.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/qubes/qmemman/algo.py b/qubes/qmemman/algo.py index 3beef4b0..9dc766c3 100644 --- a/qubes/qmemman/algo.py +++ b/qubes/qmemman/algo.py @@ -49,12 +49,12 @@ def sanitize_and_parse_meminfo(untrusted_meminfo): # not new syntax - try the old one untrusted_dict = {} # split meminfo contents into lines - untrusted_lines = string.split(untrusted_meminfo, "\n") + untrusted_lines = untrusted_meminfo.split("\n") for untrusted_lines_iterator in untrusted_lines: # split a single meminfo line into words - untrusted_words = string.split(untrusted_lines_iterator) + untrusted_words = untrusted_lines_iterator.split() if len(untrusted_words) >= 2: - untrusted_dict[string.rstrip(untrusted_words[0], ":")] = \ + untrusted_dict[untrusted_words[0].rstrip(":")] = \ untrusted_words[1] # sanitize start From 206b7c02d53133c2aa2dbf7a4462ae13ca8cf590 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marek=20Marczykowski-G=C3=B3recki?= Date: Mon, 12 Jun 2017 09:52:44 +0200 Subject: [PATCH 3/4] qmemman: make sure to release lock Even when handling updated meminfo or domain list something goes wrong, make sure to release the lock - otherwise the whole qmemman will be blocked. --- qubes/tools/qmemmand.py | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/qubes/tools/qmemmand.py b/qubes/tools/qmemmand.py index 44376bfb..b0ae55f0 100644 --- a/qubes/tools/qmemmand.py +++ b/qubes/tools/qmemmand.py @@ -143,13 +143,14 @@ class XS_Watcher(object): self.log.debug('acquiring global_lock') global_lock.acquire() self.log.debug('global_lock acquired') - if force_refresh_domain_list: - self.domain_list_changed(refresh_only=True) + try: + if force_refresh_domain_list: + self.domain_list_changed(refresh_only=True) - system_state.refresh_meminfo(domain_id, untrusted_meminfo_key) - - global_lock.release() - self.log.debug('global_lock released') + system_state.refresh_meminfo(domain_id, untrusted_meminfo_key) + finally: + global_lock.release() + self.log.debug('global_lock released') def watch_loop(self): From 9d99232515642de5f67c937d7c36236abc78654a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marek=20Marczykowski-G=C3=B3recki?= Date: Mon, 12 Jun 2017 09:54:24 +0200 Subject: [PATCH 4/4] vm: drop 'internal' property It isn't used anywhere in the code right now. And when it will be needed, it should be a "feature" not "property". --- qubes/tests/vm/qubesvm.py | 4 ---- qubes/vm/qubesvm.py | 7 +------ 2 files changed, 1 insertion(+), 10 deletions(-) diff --git a/qubes/tests/vm/qubesvm.py b/qubes/tests/vm/qubesvm.py index f7a08160..1315e01f 100644 --- a/qubes/tests/vm/qubesvm.py +++ b/qubes/tests/vm/qubesvm.py @@ -310,10 +310,6 @@ class TC_90_QubesVM(QubesVMTestsMixin,qubes.tests.QubesTestCase): # TODO: lower than memory # TODO: human readable setter (500M, 4G)? - def test_180_internal(self): - vm = self.get_vm(label='red') - self._test_generic_bool_property(vm, 'internal', False) - def test_190_vcpus(self): vm = self.get_vm() self.assertPropertyDefaultValue(vm, 'vcpus', self.app.host.no_cpus) diff --git a/qubes/vm/qubesvm.py b/qubes/vm/qubesvm.py index 1d30b77f..a3cfbf25 100644 --- a/qubes/vm/qubesvm.py +++ b/qubes/vm/qubesvm.py @@ -430,11 +430,6 @@ class QubesVM(qubes.vm.mix.net.NetVMMixin, qubes.vm.BaseVM): default=None, doc='Memory ammount allocated for the stubdom') - internal = qubes.property('internal', default=False, - type=bool, setter=qubes.property.bool, - doc='''Internal VM (not shown in qubes-manager, don't create appmenus - entries.''') - vcpus = qubes.property('vcpus', type=int, setter=_setter_positive_int, @@ -492,7 +487,7 @@ class QubesVM(qubes.vm.mix.net.NetVMMixin, qubes.vm.BaseVM): dom0 boot.''') include_in_backups = qubes.property('include_in_backups', - default=(lambda self: not self.internal), + default=True, type=bool, setter=qubes.property.bool, doc='If this domain is to be included in default backup.')