mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-26 20:31:42 +00:00 
			
		
		
		
	 253007d147
			
		
	
	
		253007d147
		
	
	
	
	
		
			
			Assign separate address space for each device in the remote processes. Signed-off-by: Elena Ufimtseva <elena.ufimtseva@oracle.com> Signed-off-by: John G Johnson <john.g.johnson@oracle.com> Signed-off-by: Jagannathan Raman <jag.raman@oracle.com> Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com> Message-id: afe0b0a97582cdad42b5b25636a29c523265a10a.1655151679.git.jag.raman@oracle.com Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
		
			
				
	
	
		
			132 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			132 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**
 | |
|  * IOMMU for remote device
 | |
|  *
 | |
|  * Copyright © 2022 Oracle and/or its affiliates.
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | |
|  * See the COPYING file in the top-level directory.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include "qemu/osdep.h"
 | |
| 
 | |
| #include "hw/remote/iommu.h"
 | |
| #include "hw/pci/pci_bus.h"
 | |
| #include "hw/pci/pci.h"
 | |
| #include "exec/memory.h"
 | |
| #include "exec/address-spaces.h"
 | |
| #include "trace.h"
 | |
| 
 | |
| /**
 | |
|  * IOMMU for TYPE_REMOTE_MACHINE - manages DMA address space isolation
 | |
|  *     for remote machine. It is used by TYPE_VFIO_USER_SERVER.
 | |
|  *
 | |
|  * - Each TYPE_VFIO_USER_SERVER instance handles one PCIDevice on a PCIBus.
 | |
|  *   There is one RemoteIommu per PCIBus, so the RemoteIommu tracks multiple
 | |
|  *   PCIDevices by maintaining a ->elem_by_devfn mapping.
 | |
|  *
 | |
|  * - memory_region_init_iommu() is not used because vfio-user MemoryRegions
 | |
|  *   will be added to the elem->mr container instead. This is more natural
 | |
|  *   than implementing the IOMMUMemoryRegionClass APIs since vfio-user
 | |
|  *   provides something that is close to a full-fledged MemoryRegion and
 | |
|  *   not like an IOMMU mapping.
 | |
|  *
 | |
|  * - When a device is hot unplugged, the elem->mr reference is dropped so
 | |
|  *   all vfio-user MemoryRegions associated with this vfio-user server are
 | |
|  *   destroyed.
 | |
|  */
 | |
| 
 | |
| static AddressSpace *remote_iommu_find_add_as(PCIBus *pci_bus,
 | |
|                                               void *opaque, int devfn)
 | |
| {
 | |
|     RemoteIommu *iommu = opaque;
 | |
|     RemoteIommuElem *elem = NULL;
 | |
| 
 | |
|     qemu_mutex_lock(&iommu->lock);
 | |
| 
 | |
|     elem = g_hash_table_lookup(iommu->elem_by_devfn, INT2VOIDP(devfn));
 | |
| 
 | |
|     if (!elem) {
 | |
|         elem = g_malloc0(sizeof(RemoteIommuElem));
 | |
|         g_hash_table_insert(iommu->elem_by_devfn, INT2VOIDP(devfn), elem);
 | |
|     }
 | |
| 
 | |
|     if (!elem->mr) {
 | |
|         elem->mr = MEMORY_REGION(object_new(TYPE_MEMORY_REGION));
 | |
|         memory_region_set_size(elem->mr, UINT64_MAX);
 | |
|         address_space_init(&elem->as, elem->mr, NULL);
 | |
|     }
 | |
| 
 | |
|     qemu_mutex_unlock(&iommu->lock);
 | |
| 
 | |
|     return &elem->as;
 | |
| }
 | |
| 
 | |
| void remote_iommu_unplug_dev(PCIDevice *pci_dev)
 | |
| {
 | |
|     AddressSpace *as = pci_device_iommu_address_space(pci_dev);
 | |
|     RemoteIommuElem *elem = NULL;
 | |
| 
 | |
|     if (as == &address_space_memory) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     elem = container_of(as, RemoteIommuElem, as);
 | |
| 
 | |
|     address_space_destroy(&elem->as);
 | |
| 
 | |
|     object_unref(elem->mr);
 | |
| 
 | |
|     elem->mr = NULL;
 | |
| }
 | |
| 
 | |
| static void remote_iommu_init(Object *obj)
 | |
| {
 | |
|     RemoteIommu *iommu = REMOTE_IOMMU(obj);
 | |
| 
 | |
|     iommu->elem_by_devfn = g_hash_table_new_full(NULL, NULL, NULL, g_free);
 | |
| 
 | |
|     qemu_mutex_init(&iommu->lock);
 | |
| }
 | |
| 
 | |
| static void remote_iommu_finalize(Object *obj)
 | |
| {
 | |
|     RemoteIommu *iommu = REMOTE_IOMMU(obj);
 | |
| 
 | |
|     qemu_mutex_destroy(&iommu->lock);
 | |
| 
 | |
|     g_hash_table_destroy(iommu->elem_by_devfn);
 | |
| 
 | |
|     iommu->elem_by_devfn = NULL;
 | |
| }
 | |
| 
 | |
| void remote_iommu_setup(PCIBus *pci_bus)
 | |
| {
 | |
|     RemoteIommu *iommu = NULL;
 | |
| 
 | |
|     g_assert(pci_bus);
 | |
| 
 | |
|     iommu = REMOTE_IOMMU(object_new(TYPE_REMOTE_IOMMU));
 | |
| 
 | |
|     pci_setup_iommu(pci_bus, remote_iommu_find_add_as, iommu);
 | |
| 
 | |
|     object_property_add_child(OBJECT(pci_bus), "remote-iommu", OBJECT(iommu));
 | |
| 
 | |
|     object_unref(OBJECT(iommu));
 | |
| }
 | |
| 
 | |
| static const TypeInfo remote_iommu_info = {
 | |
|     .name = TYPE_REMOTE_IOMMU,
 | |
|     .parent = TYPE_OBJECT,
 | |
|     .instance_size = sizeof(RemoteIommu),
 | |
|     .instance_init = remote_iommu_init,
 | |
|     .instance_finalize = remote_iommu_finalize,
 | |
| };
 | |
| 
 | |
| static void remote_iommu_register_types(void)
 | |
| {
 | |
|     type_register_static(&remote_iommu_info);
 | |
| }
 | |
| 
 | |
| type_init(remote_iommu_register_types)
 |