Commit 10c4c98ab7dc18169b37b76f6ea5e60ebe65222b

Authored by Gerd Hoffmann
Committed by Paul Brook
1 parent 0aab0d3a

qdev: replace bus_type enum with bus_info struct.

BusInfo is filled with name and size (pretty much like I did for
DeviceInfo as well).  There is also a function pointer to print
bus-specific device information to the monitor.  sysbus is hooked
up there, I've also added a print function for PCI.

Device creation is slightly modified as well:  The device type search
loop now also checks the bus type while scanning the list instead of
complaining thereafter in case of a mismatch.  This effectively gives
each bus a private namespace for device names.

Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
Signed-off-by: Paul Brook <paul@codesourcery.com>
hw/i2c.c
@@ -17,6 +17,11 @@ struct i2c_bus @@ -17,6 +17,11 @@ struct i2c_bus
17 int saved_address; 17 int saved_address;
18 }; 18 };
19 19
  20 +static struct BusInfo i2c_bus_info = {
  21 + .name = "I2C",
  22 + .size = sizeof(i2c_bus),
  23 +};
  24 +
20 static void i2c_bus_save(QEMUFile *f, void *opaque) 25 static void i2c_bus_save(QEMUFile *f, void *opaque)
21 { 26 {
22 i2c_bus *bus = (i2c_bus *)opaque; 27 i2c_bus *bus = (i2c_bus *)opaque;
@@ -44,8 +49,7 @@ i2c_bus *i2c_init_bus(DeviceState *parent, const char *name) @@ -44,8 +49,7 @@ i2c_bus *i2c_init_bus(DeviceState *parent, const char *name)
44 { 49 {
45 i2c_bus *bus; 50 i2c_bus *bus;
46 51
47 - bus = FROM_QBUS(i2c_bus, qbus_create(BUS_TYPE_I2C, sizeof(i2c_bus),  
48 - parent, name)); 52 + bus = FROM_QBUS(i2c_bus, qbus_create(&i2c_bus_info, parent, name));
49 register_savevm("i2c_bus", -1, 1, i2c_bus_save, i2c_bus_load, bus); 53 register_savevm("i2c_bus", -1, 1, i2c_bus_save, i2c_bus_load, bus);
50 return bus; 54 return bus;
51 } 55 }
@@ -156,7 +160,7 @@ void i2c_register_slave(I2CSlaveInfo *info) @@ -156,7 +160,7 @@ void i2c_register_slave(I2CSlaveInfo *info)
156 { 160 {
157 assert(info->qdev.size >= sizeof(i2c_slave)); 161 assert(info->qdev.size >= sizeof(i2c_slave));
158 info->qdev.init = i2c_slave_qdev_init; 162 info->qdev.init = i2c_slave_qdev_init;
159 - info->qdev.bus_type = BUS_TYPE_I2C; 163 + info->qdev.bus_info = &i2c_bus_info;
160 qdev_register(&info->qdev); 164 qdev_register(&info->qdev);
161 } 165 }
162 166
hw/pci.c
@@ -45,7 +45,15 @@ struct PCIBus { @@ -45,7 +45,15 @@ struct PCIBus {
45 /* The bus IRQ state is the logical OR of the connected devices. 45 /* The bus IRQ state is the logical OR of the connected devices.
46 Keep a count of the number of devices with raised IRQs. */ 46 Keep a count of the number of devices with raised IRQs. */
47 int nirq; 47 int nirq;
48 - int irq_count[]; 48 + int *irq_count;
  49 +};
  50 +
  51 +static void pcibus_dev_print(Monitor *mon, DeviceState *dev, int indent);
  52 +
  53 +static struct BusInfo pci_bus_info = {
  54 + .name = "PCI",
  55 + .size = sizeof(PCIBus),
  56 + .print_dev = pcibus_dev_print,
49 }; 57 };
50 58
51 static void pci_update_mappings(PCIDevice *d); 59 static void pci_update_mappings(PCIDevice *d);
@@ -109,14 +117,13 @@ PCIBus *pci_register_bus(DeviceState *parent, const char *name, @@ -109,14 +117,13 @@ PCIBus *pci_register_bus(DeviceState *parent, const char *name,
109 PCIBus *bus; 117 PCIBus *bus;
110 static int nbus = 0; 118 static int nbus = 0;
111 119
112 - bus = FROM_QBUS(PCIBus, qbus_create(BUS_TYPE_PCI,  
113 - sizeof(PCIBus) + (nirq * sizeof(int)),  
114 - parent, name)); 120 + bus = FROM_QBUS(PCIBus, qbus_create(&pci_bus_info, parent, name));
115 bus->set_irq = set_irq; 121 bus->set_irq = set_irq;
116 bus->map_irq = map_irq; 122 bus->map_irq = map_irq;
117 bus->irq_opaque = pic; 123 bus->irq_opaque = pic;
118 bus->devfn_min = devfn_min; 124 bus->devfn_min = devfn_min;
119 bus->nirq = nirq; 125 bus->nirq = nirq;
  126 + bus->irq_count = qemu_malloc(nirq * sizeof(bus->irq_count[0]));
120 bus->next = first_bus; 127 bus->next = first_bus;
121 first_bus = bus; 128 first_bus = bus;
122 register_savevm("PCIBUS", nbus++, 1, pcibus_save, pcibus_load, bus); 129 register_savevm("PCIBUS", nbus++, 1, pcibus_save, pcibus_load, bus);
@@ -892,7 +899,7 @@ static void pci_qdev_init(DeviceState *qdev, DeviceInfo *base) @@ -892,7 +899,7 @@ static void pci_qdev_init(DeviceState *qdev, DeviceInfo *base)
892 void pci_qdev_register(PCIDeviceInfo *info) 899 void pci_qdev_register(PCIDeviceInfo *info)
893 { 900 {
894 info->qdev.init = pci_qdev_init; 901 info->qdev.init = pci_qdev_init;
895 - info->qdev.bus_type = BUS_TYPE_PCI; 902 + info->qdev.bus_info = &pci_bus_info;
896 qdev_register(&info->qdev); 903 qdev_register(&info->qdev);
897 } 904 }
898 905
@@ -991,3 +998,39 @@ uint8_t pci_find_capability(PCIDevice *pdev, uint8_t cap_id) @@ -991,3 +998,39 @@ uint8_t pci_find_capability(PCIDevice *pdev, uint8_t cap_id)
991 { 998 {
992 return pci_find_capability_list(pdev, cap_id, NULL); 999 return pci_find_capability_list(pdev, cap_id, NULL);
993 } 1000 }
  1001 +
  1002 +static void pcibus_dev_print(Monitor *mon, DeviceState *dev, int indent)
  1003 +{
  1004 + PCIDevice *d = (PCIDevice *)dev;
  1005 + const pci_class_desc *desc;
  1006 + char ctxt[64];
  1007 + PCIIORegion *r;
  1008 + int i, class;
  1009 +
  1010 + class = le16_to_cpu(*((uint16_t *)(d->config + PCI_CLASS_DEVICE)));
  1011 + desc = pci_class_descriptions;
  1012 + while (desc->desc && class != desc->class)
  1013 + desc++;
  1014 + if (desc->desc) {
  1015 + snprintf(ctxt, sizeof(ctxt), "%s", desc->desc);
  1016 + } else {
  1017 + snprintf(ctxt, sizeof(ctxt), "Class %04x", class);
  1018 + }
  1019 +
  1020 + monitor_printf(mon, "%*sclass %s, addr %02x:%02x.%x, "
  1021 + "pci id %04x:%04x (sub %04x:%04x)\n",
  1022 + indent, "", ctxt,
  1023 + d->bus->bus_num, d->devfn >> 3, d->devfn & 7,
  1024 + le16_to_cpu(*((uint16_t *)(d->config + PCI_VENDOR_ID))),
  1025 + le16_to_cpu(*((uint16_t *)(d->config + PCI_DEVICE_ID))),
  1026 + le16_to_cpu(*((uint16_t *)(d->config + PCI_SUBSYSTEM_VENDOR_ID))),
  1027 + le16_to_cpu(*((uint16_t *)(d->config + PCI_SUBSYSTEM_ID))));
  1028 + for (i = 0; i < PCI_NUM_REGIONS; i++) {
  1029 + r = &d->io_regions[i];
  1030 + if (!r->size)
  1031 + continue;
  1032 + monitor_printf(mon, "%*sbar %d: %s at 0x%x [0x%x]\n", indent, "",
  1033 + i, r->type & PCI_ADDRESS_SPACE_IO ? "i/o" : "mem",
  1034 + r->addr, r->addr + r->size - 1);
  1035 + }
  1036 +}
hw/qdev.c
@@ -48,6 +48,7 @@ struct DeviceType { @@ -48,6 +48,7 @@ struct DeviceType {
48 48
49 /* This is a nasty hack to allow passing a NULL bus to qdev_create. */ 49 /* This is a nasty hack to allow passing a NULL bus to qdev_create. */
50 static BusState *main_system_bus; 50 static BusState *main_system_bus;
  51 +extern struct BusInfo system_bus_info;
51 52
52 static DeviceType *device_type_list; 53 static DeviceType *device_type_list;
53 54
@@ -72,31 +73,26 @@ DeviceState *qdev_create(BusState *bus, const char *name) @@ -72,31 +73,26 @@ DeviceState *qdev_create(BusState *bus, const char *name)
72 DeviceType *t; 73 DeviceType *t;
73 DeviceState *dev; 74 DeviceState *dev;
74 75
75 - for (t = device_type_list; t; t = t->next) {  
76 - if (strcmp(t->info->name, name) == 0) {  
77 - break; 76 + if (!bus) {
  77 + if (!main_system_bus) {
  78 + main_system_bus = qbus_create(&system_bus_info, NULL, "main-system-bus");
78 } 79 }
  80 + bus = main_system_bus;
  81 + }
  82 +
  83 + for (t = device_type_list; t; t = t->next) {
  84 + if (t->info->bus_info != bus->info)
  85 + continue;
  86 + if (strcmp(t->info->name, name) != 0)
  87 + continue;
  88 + break;
79 } 89 }
80 if (!t) { 90 if (!t) {
81 - hw_error("Unknown device '%s'\n", name); 91 + hw_error("Unknown device '%s' for bus '%s'\n", name, bus->info->name);
82 } 92 }
83 93
84 dev = qemu_mallocz(t->info->size); 94 dev = qemu_mallocz(t->info->size);
85 dev->type = t; 95 dev->type = t;
86 -  
87 - if (!bus) {  
88 - /* ???: This assumes system busses have no additional state. */  
89 - if (!main_system_bus) {  
90 - main_system_bus = qbus_create(BUS_TYPE_SYSTEM, sizeof(BusState),  
91 - NULL, "main-system-bus");  
92 - }  
93 - bus = main_system_bus;  
94 - }  
95 - if (t->info->bus_type != bus->type) {  
96 - /* TODO: Print bus type names. */  
97 - hw_error("Device '%s' on wrong bus type (%d/%d)", name,  
98 - t->info->bus_type, bus->type);  
99 - }  
100 dev->parent_bus = bus; 96 dev->parent_bus = bus;
101 LIST_INSERT_HEAD(&bus->children, dev, sibling); 97 LIST_INSERT_HEAD(&bus->children, dev, sibling);
102 return dev; 98 return dev;
@@ -320,13 +316,12 @@ void scsi_bus_new(DeviceState *host, SCSIAttachFn attach) @@ -320,13 +316,12 @@ void scsi_bus_new(DeviceState *host, SCSIAttachFn attach)
320 } 316 }
321 } 317 }
322 318
323 -BusState *qbus_create(BusType type, size_t size,  
324 - DeviceState *parent, const char *name) 319 +BusState *qbus_create(BusInfo *info, DeviceState *parent, const char *name)
325 { 320 {
326 BusState *bus; 321 BusState *bus;
327 322
328 - bus = qemu_mallocz(size);  
329 - bus->type = type; 323 + bus = qemu_mallocz(info->size);
  324 + bus->info = info;
330 bus->parent = parent; 325 bus->parent = parent;
331 bus->name = qemu_strdup(name); 326 bus->name = qemu_strdup(name);
332 LIST_INIT(&bus->children); 327 LIST_INIT(&bus->children);
@@ -336,14 +331,6 @@ BusState *qbus_create(BusType type, size_t size, @@ -336,14 +331,6 @@ BusState *qbus_create(BusType type, size_t size,
336 return bus; 331 return bus;
337 } 332 }
338 333
339 -static const char *bus_type_names[] = {  
340 - [ BUS_TYPE_SYSTEM ] = "System",  
341 - [ BUS_TYPE_PCI ] = "PCI",  
342 - [ BUS_TYPE_SCSI ] = "SCSI",  
343 - [ BUS_TYPE_I2C ] = "I2C",  
344 - [ BUS_TYPE_SSI ] = "SSI",  
345 -};  
346 -  
347 #define qdev_printf(fmt, ...) monitor_printf(mon, "%*s" fmt, indent, "", ## __VA_ARGS__) 334 #define qdev_printf(fmt, ...) monitor_printf(mon, "%*s" fmt, indent, "", ## __VA_ARGS__)
348 static void qbus_print(Monitor *mon, BusState *bus, int indent); 335 static void qbus_print(Monitor *mon, BusState *bus, int indent);
349 336
@@ -377,13 +364,8 @@ static void qdev_print(Monitor *mon, DeviceState *dev, int indent) @@ -377,13 +364,8 @@ static void qdev_print(Monitor *mon, DeviceState *dev, int indent)
377 break; 364 break;
378 } 365 }
379 } 366 }
380 - switch (dev->parent_bus->type) {  
381 - case BUS_TYPE_SYSTEM:  
382 - sysbus_dev_print(mon, dev, indent);  
383 - break;  
384 - default:  
385 - break;  
386 - } 367 + if (dev->parent_bus->info->print_dev)
  368 + dev->parent_bus->info->print_dev(mon, dev, indent);
387 LIST_FOREACH(child, &dev->child_bus, sibling) { 369 LIST_FOREACH(child, &dev->child_bus, sibling) {
388 qbus_print(mon, child, indent); 370 qbus_print(mon, child, indent);
389 } 371 }
@@ -395,7 +377,7 @@ static void qbus_print(Monitor *mon, BusState *bus, int indent) @@ -395,7 +377,7 @@ static void qbus_print(Monitor *mon, BusState *bus, int indent)
395 377
396 qdev_printf("bus: %s\n", bus->name); 378 qdev_printf("bus: %s\n", bus->name);
397 indent += 2; 379 indent += 2;
398 - qdev_printf("type %s\n", bus_type_names[bus->type]); 380 + qdev_printf("type %s\n", bus->info->name);
399 LIST_FOREACH(dev, &bus->children, sibling) { 381 LIST_FOREACH(dev, &bus->children, sibling) {
400 qdev_print(mon, dev, indent); 382 qdev_print(mon, dev, indent);
401 } 383 }
hw/qdev.h
@@ -10,6 +10,8 @@ typedef struct DeviceProperty DeviceProperty; @@ -10,6 +10,8 @@ typedef struct DeviceProperty DeviceProperty;
10 10
11 typedef struct BusState BusState; 11 typedef struct BusState BusState;
12 12
  13 +typedef struct BusInfo BusInfo;
  14 +
13 /* This structure should not be accessed directly. We declare it here 15 /* This structure should not be accessed directly. We declare it here
14 so that it can be embedded in individual device state structures. */ 16 so that it can be embedded in individual device state structures. */
15 struct DeviceState { 17 struct DeviceState {
@@ -25,18 +27,17 @@ struct DeviceState { @@ -25,18 +27,17 @@ struct DeviceState {
25 LIST_ENTRY(DeviceState) sibling; 27 LIST_ENTRY(DeviceState) sibling;
26 }; 28 };
27 29
28 -typedef enum {  
29 - BUS_TYPE_SYSTEM,  
30 - BUS_TYPE_PCI,  
31 - BUS_TYPE_SCSI,  
32 - BUS_TYPE_I2C,  
33 - BUS_TYPE_SSI  
34 -} BusType; 30 +typedef void (*bus_dev_printfn)(Monitor *mon, DeviceState *dev, int indent);
  31 +struct BusInfo {
  32 + const char *name;
  33 + size_t size;
  34 + bus_dev_printfn print_dev;
  35 +};
35 36
36 struct BusState { 37 struct BusState {
37 DeviceState *parent; 38 DeviceState *parent;
  39 + BusInfo *info;
38 const char *name; 40 const char *name;
39 - BusType type;  
40 LIST_HEAD(, DeviceState) children; 41 LIST_HEAD(, DeviceState) children;
41 LIST_ENTRY(BusState) sibling; 42 LIST_ENTRY(BusState) sibling;
42 }; 43 };
@@ -84,7 +85,7 @@ struct DeviceInfo { @@ -84,7 +85,7 @@ struct DeviceInfo {
84 85
85 /* Private to qdev / bus. */ 86 /* Private to qdev / bus. */
86 qdev_initfn init; 87 qdev_initfn init;
87 - BusType bus_type; 88 + BusInfo *bus_info;
88 }; 89 };
89 90
90 void qdev_register(DeviceInfo *info); 91 void qdev_register(DeviceInfo *info);
@@ -116,14 +117,12 @@ void *qdev_get_prop_ptr(DeviceState *dev, const char *name); @@ -116,14 +117,12 @@ void *qdev_get_prop_ptr(DeviceState *dev, const char *name);
116 117
117 /*** BUS API. ***/ 118 /*** BUS API. ***/
118 119
119 -BusState *qbus_create(BusType type, size_t size,  
120 - DeviceState *parent, const char *name); 120 +BusState *qbus_create(BusInfo *info, DeviceState *parent, const char *name);
121 121
122 #define FROM_QBUS(type, dev) DO_UPCAST(type, qbus, dev) 122 #define FROM_QBUS(type, dev) DO_UPCAST(type, qbus, dev)
123 123
124 /*** monitor commands ***/ 124 /*** monitor commands ***/
125 125
126 void do_info_qtree(Monitor *mon); 126 void do_info_qtree(Monitor *mon);
127 -void sysbus_dev_print(Monitor *mon, DeviceState *dev, int indent);  
128 127
129 #endif 128 #endif
hw/ssi.c
@@ -13,6 +13,11 @@ struct SSIBus { @@ -13,6 +13,11 @@ struct SSIBus {
13 BusState qbus; 13 BusState qbus;
14 }; 14 };
15 15
  16 +static struct BusInfo ssi_bus_info = {
  17 + .name = "SSI",
  18 + .size = sizeof(SSIBus),
  19 +};
  20 +
16 static void ssi_slave_init(DeviceState *dev, DeviceInfo *base_info) 21 static void ssi_slave_init(DeviceState *dev, DeviceInfo *base_info)
17 { 22 {
18 SSISlaveInfo *info = container_of(base_info, SSISlaveInfo, qdev); 23 SSISlaveInfo *info = container_of(base_info, SSISlaveInfo, qdev);
@@ -33,7 +38,7 @@ void ssi_register_slave(SSISlaveInfo *info) @@ -33,7 +38,7 @@ void ssi_register_slave(SSISlaveInfo *info)
33 { 38 {
34 assert(info->qdev.size >= sizeof(SSISlave)); 39 assert(info->qdev.size >= sizeof(SSISlave));
35 info->qdev.init = ssi_slave_init; 40 info->qdev.init = ssi_slave_init;
36 - info->qdev.bus_type = BUS_TYPE_SSI; 41 + info->qdev.bus_info = &ssi_bus_info;
37 qdev_register(&info->qdev); 42 qdev_register(&info->qdev);
38 } 43 }
39 44
@@ -48,7 +53,7 @@ DeviceState *ssi_create_slave(SSIBus *bus, const char *name) @@ -48,7 +53,7 @@ DeviceState *ssi_create_slave(SSIBus *bus, const char *name)
48 SSIBus *ssi_create_bus(DeviceState *parent, const char *name) 53 SSIBus *ssi_create_bus(DeviceState *parent, const char *name)
49 { 54 {
50 BusState *bus; 55 BusState *bus;
51 - bus = qbus_create(BUS_TYPE_SSI, sizeof(SSIBus), parent, name); 56 + bus = qbus_create(&ssi_bus_info, parent, name);
52 return FROM_QBUS(SSIBus, bus); 57 return FROM_QBUS(SSIBus, bus);
53 } 58 }
54 59
hw/sysbus.c
@@ -22,6 +22,14 @@ @@ -22,6 +22,14 @@
22 #include "sysemu.h" 22 #include "sysemu.h"
23 #include "monitor.h" 23 #include "monitor.h"
24 24
  25 +static void sysbus_dev_print(Monitor *mon, DeviceState *dev, int indent);
  26 +
  27 +struct BusInfo system_bus_info = {
  28 + .name = "System",
  29 + .size = sizeof(BusState),
  30 + .print_dev = sysbus_dev_print,
  31 +};
  32 +
25 void sysbus_connect_irq(SysBusDevice *dev, int n, qemu_irq irq) 33 void sysbus_connect_irq(SysBusDevice *dev, int n, qemu_irq irq)
26 { 34 {
27 assert(n >= 0 && n < dev->num_irq); 35 assert(n >= 0 && n < dev->num_irq);
@@ -108,7 +116,7 @@ static void sysbus_device_init(DeviceState *dev, DeviceInfo *base) @@ -108,7 +116,7 @@ static void sysbus_device_init(DeviceState *dev, DeviceInfo *base)
108 void sysbus_register_withprop(SysBusDeviceInfo *info) 116 void sysbus_register_withprop(SysBusDeviceInfo *info)
109 { 117 {
110 info->qdev.init = sysbus_device_init; 118 info->qdev.init = sysbus_device_init;
111 - info->qdev.bus_type = BUS_TYPE_SYSTEM; 119 + info->qdev.bus_info = &system_bus_info;
112 120
113 assert(info->qdev.size >= sizeof(SysBusDevice)); 121 assert(info->qdev.size >= sizeof(SysBusDevice));
114 qdev_register(&info->qdev); 122 qdev_register(&info->qdev);
@@ -153,7 +161,7 @@ DeviceState *sysbus_create_varargs(const char *name, @@ -153,7 +161,7 @@ DeviceState *sysbus_create_varargs(const char *name,
153 return dev; 161 return dev;
154 } 162 }
155 163
156 -void sysbus_dev_print(Monitor *mon, DeviceState *dev, int indent) 164 +static void sysbus_dev_print(Monitor *mon, DeviceState *dev, int indent)
157 { 165 {
158 SysBusDevice *s = sysbus_from_qdev(dev); 166 SysBusDevice *s = sysbus_from_qdev(dev);
159 int i; 167 int i;