/* * ACPI Virtual I/O Translation table implementation * * SPDX-License-Identifier: GPL-2.0-or-later */ #include "qemu/osdep.h" #include "hw/acpi/acpi.h" #include "hw/acpi/aml-build.h" #include "hw/acpi/viot.h" #include "hw/pci/pci.h" #include "hw/pci/pci_host.h" struct viot_pci_ranges { GArray *blob; size_t count; uint16_t output_node; }; static void build_pci_host_range(GArray *table_data, int min_bus, int max_bus, uint16_t output_node) { /* Type */ build_append_int_noprefix(table_data, 1 /* PCI range */, 1); /* Reserved */ build_append_int_noprefix(table_data, 0, 1); /* Length */ build_append_int_noprefix(table_data, 24, 2); /* Endpoint start */ build_append_int_noprefix(table_data, PCI_BUILD_BDF(min_bus, 0), 4); /* PCI Segment start */ build_append_int_noprefix(table_data, 0, 2); /* PCI Segment end */ build_append_int_noprefix(table_data, 0, 2); /* PCI BDF start */ build_append_int_noprefix(table_data, PCI_BUILD_BDF(min_bus, 0), 2); /* PCI BDF end */ build_append_int_noprefix(table_data, PCI_BUILD_BDF(max_bus, 0xff), 2); /* Output node */ build_append_int_noprefix(table_data, output_node, 2); /* Reserved */ build_append_int_noprefix(table_data, 0, 6); } /* Build PCI range for a given PCI host bridge */ static int enumerate_pci_host_bridges(Object *obj, void *opaque) { struct viot_pci_ranges *pci_ranges = opaque; GArray *blob = pci_ranges->blob; if (object_dynamic_cast(obj, TYPE_PCI_HOST_BRIDGE)) { PCIBus *bus = PCI_HOST_BRIDGE(obj)->bus; if (bus && !pci_bus_bypass_iommu(bus)) { int min_bus, max_bus; pci_bus_range(bus, &min_bus, &max_bus); build_pci_host_range(blob, min_bus, max_bus, pci_ranges->output_node); pci_ranges->count++; } } return 0; } /* * Generate a VIOT table with one PCI-based virtio-iommu that manages PCI * endpoints. * * Defined in the ACPI Specification (Version TBD) */ void build_viot(MachineState *ms, GArray *table_data, BIOSLinker *linker, uint16_t virtio_iommu_bdf, const char *oem_id, const char *oem_table_id) { /* The virtio-iommu node follows the 48-bytes header */ int viommu_off = 48; AcpiTable table = { .sig = "VIOT", .rev = 0, .oem_id = oem_id, .oem_table_id = oem_table_id }; struct viot_pci_ranges pci_ranges = { .output_node = viommu_off, .blob = g_array_new(false, true /* clear */, 1), }; /* Build the list of PCI ranges that this viommu manages */ object_child_foreach_recursive(OBJECT(ms), enumerate_pci_host_bridges, &pci_ranges); /* ACPI table header */ acpi_table_begin(&table, table_data); /* Node count */ build_append_int_noprefix(table_data, pci_ranges.count + 1, 2); /* Node offset */ build_append_int_noprefix(table_data, viommu_off, 2); /* Reserved */ build_append_int_noprefix(table_data, 0, 8); /* Virtio-iommu node */ /* Type */ build_append_int_noprefix(table_data, 3 /* virtio-pci IOMMU */, 1); /* Reserved */ build_append_int_noprefix(table_data, 0, 1); /* Length */ build_append_int_noprefix(table_data, 16, 2); /* PCI Segment */ build_append_int_noprefix(table_data, 0, 2); /* PCI BDF number */ build_append_int_noprefix(table_data, virtio_iommu_bdf, 2); /* Reserved */ build_append_int_noprefix(table_data, 0, 8); /* PCI ranges found above */ g_array_append_vals(table_data, pci_ranges.blob->data, pci_ranges.blob->len); g_array_free(pci_ranges.blob, true); acpi_table_end(linker, &table); }