| /* |
| * Copyright (C) 2021 Red Hat, Inc. |
| * |
| * This program is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU General Public License as |
| * published by the Free Software Foundation; either version 2 or |
| * (at your option) version 3 of the License. |
| * |
| * This program 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 General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, see <http://www.gnu.org/licenses/>. |
| */ |
| |
| #include "qemu/osdep.h" |
| |
| #include "hw/pci/pci.h" |
| #include "hw/pci/pci_bus.h" |
| #include "qapi/error.h" |
| #include "ui/console.h" |
| |
| /* |
| * Recursively (in reverse order) appends addresses of PCI devices as it moves |
| * up in the PCI hierarchy. |
| * |
| * @returns true on success, false when the buffer wasn't large enough |
| */ |
| static bool append_pci_address(char *buf, size_t buf_size, const PCIDevice *pci) |
| { |
| PCIBus *bus = pci_get_bus(pci); |
| /* |
| * equivalent to if (!pci_bus_is_root(bus)), but the function is not built |
| * with PCI_CONFIG=n, avoid using an #ifdef by checking directly |
| */ |
| if (bus->parent_dev != NULL) { |
| append_pci_address(buf, buf_size, bus->parent_dev); |
| } |
| |
| size_t len = strlen(buf); |
| ssize_t written = snprintf(buf + len, buf_size - len, "/%02x.%x", |
| PCI_SLOT(pci->devfn), PCI_FUNC(pci->devfn)); |
| |
| return written > 0 && written < buf_size - len; |
| } |
| |
| bool qemu_console_fill_device_address(QemuConsole *con, |
| char *device_address, |
| size_t size, |
| Error **errp) |
| { |
| DeviceState *dev = DEVICE(object_property_get_link(OBJECT(con), |
| "device", |
| &error_abort)); |
| PCIDevice *pci = (PCIDevice *) object_dynamic_cast(OBJECT(dev), |
| TYPE_PCI_DEVICE); |
| |
| if (pci == NULL) { |
| error_setg(errp, "Setting device address of a display device: " |
| "Not a PCI device."); |
| return false; |
| } |
| |
| strncpy(device_address, "pci/0000", size); |
| if (!append_pci_address(device_address, size, pci)) { |
| error_setg(errp, "Setting device address of a display device: " |
| "Too many PCI devices in the chain."); |
| return false; |
| } |
| |
| return true; |
| } |