1f6ef71bdSMarc-André Lureau /*
2f6ef71bdSMarc-André Lureau * Copyright (C) 2021 Red Hat, Inc.
3f6ef71bdSMarc-André Lureau *
4f6ef71bdSMarc-André Lureau * This program is free software; you can redistribute it and/or
5f6ef71bdSMarc-André Lureau * modify it under the terms of the GNU General Public License as
6f6ef71bdSMarc-André Lureau * published by the Free Software Foundation; either version 2 or
7f6ef71bdSMarc-André Lureau * (at your option) version 3 of the License.
8f6ef71bdSMarc-André Lureau *
9f6ef71bdSMarc-André Lureau * This program is distributed in the hope that it will be useful,
10f6ef71bdSMarc-André Lureau * but WITHOUT ANY WARRANTY; without even the implied warranty of
11f6ef71bdSMarc-André Lureau * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12f6ef71bdSMarc-André Lureau * GNU General Public License for more details.
13f6ef71bdSMarc-André Lureau *
14f6ef71bdSMarc-André Lureau * You should have received a copy of the GNU General Public License
15f6ef71bdSMarc-André Lureau * along with this program; if not, see <http://www.gnu.org/licenses/>.
16f6ef71bdSMarc-André Lureau */
17f6ef71bdSMarc-André Lureau
18f6ef71bdSMarc-André Lureau #include "qemu/osdep.h"
19f6ef71bdSMarc-André Lureau
20*edf5ca5dSMarkus Armbruster #include "hw/pci/pci_device.h"
21f6ef71bdSMarc-André Lureau #include "hw/pci/pci_bus.h"
22f6ef71bdSMarc-André Lureau #include "qapi/error.h"
23f6ef71bdSMarc-André Lureau #include "ui/console.h"
24f6ef71bdSMarc-André Lureau
25f6ef71bdSMarc-André Lureau /*
26f6ef71bdSMarc-André Lureau * Recursively (in reverse order) appends addresses of PCI devices as it moves
27f6ef71bdSMarc-André Lureau * up in the PCI hierarchy.
28f6ef71bdSMarc-André Lureau *
29f6ef71bdSMarc-André Lureau * @returns true on success, false when the buffer wasn't large enough
30f6ef71bdSMarc-André Lureau */
append_pci_address(char * buf,size_t buf_size,const PCIDevice * pci)31f6ef71bdSMarc-André Lureau static bool append_pci_address(char *buf, size_t buf_size, const PCIDevice *pci)
32f6ef71bdSMarc-André Lureau {
33f6ef71bdSMarc-André Lureau PCIBus *bus = pci_get_bus(pci);
34f6ef71bdSMarc-André Lureau /*
35f6ef71bdSMarc-André Lureau * equivalent to if (!pci_bus_is_root(bus)), but the function is not built
36f6ef71bdSMarc-André Lureau * with PCI_CONFIG=n, avoid using an #ifdef by checking directly
37f6ef71bdSMarc-André Lureau */
38f6ef71bdSMarc-André Lureau if (bus->parent_dev != NULL) {
39f6ef71bdSMarc-André Lureau append_pci_address(buf, buf_size, bus->parent_dev);
40f6ef71bdSMarc-André Lureau }
41f6ef71bdSMarc-André Lureau
42f6ef71bdSMarc-André Lureau size_t len = strlen(buf);
43f6ef71bdSMarc-André Lureau ssize_t written = snprintf(buf + len, buf_size - len, "/%02x.%x",
44f6ef71bdSMarc-André Lureau PCI_SLOT(pci->devfn), PCI_FUNC(pci->devfn));
45f6ef71bdSMarc-André Lureau
46f6ef71bdSMarc-André Lureau return written > 0 && written < buf_size - len;
47f6ef71bdSMarc-André Lureau }
48f6ef71bdSMarc-André Lureau
qemu_console_fill_device_address(QemuConsole * con,char * device_address,size_t size,Error ** errp)49f6ef71bdSMarc-André Lureau bool qemu_console_fill_device_address(QemuConsole *con,
50f6ef71bdSMarc-André Lureau char *device_address,
51f6ef71bdSMarc-André Lureau size_t size,
52f6ef71bdSMarc-André Lureau Error **errp)
53f6ef71bdSMarc-André Lureau {
54f6ef71bdSMarc-André Lureau DeviceState *dev = DEVICE(object_property_get_link(OBJECT(con),
55f6ef71bdSMarc-André Lureau "device",
56f6ef71bdSMarc-André Lureau &error_abort));
57f6ef71bdSMarc-André Lureau PCIDevice *pci = (PCIDevice *) object_dynamic_cast(OBJECT(dev),
58f6ef71bdSMarc-André Lureau TYPE_PCI_DEVICE);
59f6ef71bdSMarc-André Lureau
60f6ef71bdSMarc-André Lureau if (pci == NULL) {
61f6ef71bdSMarc-André Lureau error_setg(errp, "Setting device address of a display device: "
62f6ef71bdSMarc-André Lureau "Not a PCI device.");
63f6ef71bdSMarc-André Lureau return false;
64f6ef71bdSMarc-André Lureau }
65f6ef71bdSMarc-André Lureau
66f6ef71bdSMarc-André Lureau strncpy(device_address, "pci/0000", size);
67f6ef71bdSMarc-André Lureau if (!append_pci_address(device_address, size, pci)) {
68f6ef71bdSMarc-André Lureau error_setg(errp, "Setting device address of a display device: "
69f6ef71bdSMarc-André Lureau "Too many PCI devices in the chain.");
70f6ef71bdSMarc-André Lureau return false;
71f6ef71bdSMarc-André Lureau }
72f6ef71bdSMarc-André Lureau
73f6ef71bdSMarc-André Lureau return true;
74f6ef71bdSMarc-André Lureau }
75