Commit 80b3ada7dd56088613a446934d144a747e740fa1
1 parent
d2b59317
Implement sun4u PCI IRQ routing.
Allow multiple PCI busses and PCI-PCI bridges. Fix bugs in Versatile PCI implementation. git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@2166 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
8 changed files
with
146 additions
and
39 deletions
hw/apb_pci.c
... | ... | @@ -21,6 +21,11 @@ |
21 | 21 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
22 | 22 | * THE SOFTWARE. |
23 | 23 | */ |
24 | + | |
25 | +/* XXX This file and most of its contests are somewhat misnamed. The | |
26 | + Ultrasparc PCI host is called the PCI Bus Module (PBM). The APB is | |
27 | + the secondary PCI bridge. */ | |
28 | + | |
24 | 29 | #include "vl.h" |
25 | 30 | typedef target_phys_addr_t pci_addr_t; |
26 | 31 | #include "pci_host.h" |
... | ... | @@ -179,17 +184,25 @@ static CPUReadMemoryFunc *pci_apb_ioread[] = { |
179 | 184 | &pci_apb_ioreadl, |
180 | 185 | }; |
181 | 186 | |
187 | +/* The APB host has an IRQ line for each IRQ line of each slot. */ | |
182 | 188 | static int pci_apb_map_irq(PCIDevice *pci_dev, int irq_num) |
183 | 189 | { |
184 | - /* ??? As mentioned below this is probably wrong. */ | |
185 | - return irq_num; | |
190 | + return ((pci_dev->devfn & 0x18) >> 1) + irq_num; | |
191 | +} | |
192 | + | |
193 | +static int pci_pbm_map_irq(PCIDevice *pci_dev, int irq_num) | |
194 | +{ | |
195 | + int bus_offset; | |
196 | + if (pci_dev->devfn & 1) | |
197 | + bus_offset = 16; | |
198 | + else | |
199 | + bus_offset = 0; | |
200 | + return bus_offset + irq_num; | |
186 | 201 | } |
187 | 202 | |
188 | 203 | static void pci_apb_set_irq(void *pic, int irq_num, int level) |
189 | 204 | { |
190 | - /* ??? This is almost certainly wrong. However the rest of the sun4u | |
191 | - IRQ handling is missing, as is OpenBIOS support, so it wouldn't work | |
192 | - anyway. */ | |
205 | + /* PCI IRQ map onto the first 32 INO. */ | |
193 | 206 | pic_set_irq_new(pic, irq_num, level); |
194 | 207 | } |
195 | 208 | |
... | ... | @@ -199,10 +212,12 @@ PCIBus *pci_apb_init(target_ulong special_base, target_ulong mem_base, |
199 | 212 | APBState *s; |
200 | 213 | PCIDevice *d; |
201 | 214 | int pci_mem_config, pci_mem_data, apb_config, pci_ioport; |
215 | + PCIDevice *apb; | |
216 | + PCIBus *secondary; | |
202 | 217 | |
203 | 218 | s = qemu_mallocz(sizeof(APBState)); |
204 | - /* Ultrasparc APB main bus */ | |
205 | - s->bus = pci_register_bus(pci_apb_set_irq, pci_apb_map_irq, pic, 0); | |
219 | + /* Ultrasparc PBM main bus */ | |
220 | + s->bus = pci_register_bus(pci_apb_set_irq, pci_pbm_map_irq, pic, 0, 32); | |
206 | 221 | |
207 | 222 | pci_mem_config = cpu_register_io_memory(0, pci_apb_config_read, |
208 | 223 | pci_apb_config_write, s); |
... | ... | @@ -219,7 +234,7 @@ PCIBus *pci_apb_init(target_ulong special_base, target_ulong mem_base, |
219 | 234 | cpu_register_physical_memory(mem_base, 0x10000000, pci_mem_data); // XXX size should be 4G-prom |
220 | 235 | |
221 | 236 | d = pci_register_device(s->bus, "Advanced PCI Bus", sizeof(PCIDevice), |
222 | - -1, NULL, NULL); | |
237 | + 0, NULL, NULL); | |
223 | 238 | d->config[0x00] = 0x8e; // vendor_id : Sun |
224 | 239 | d->config[0x01] = 0x10; |
225 | 240 | d->config[0x02] = 0x00; // device_id |
... | ... | @@ -234,7 +249,11 @@ PCIBus *pci_apb_init(target_ulong special_base, target_ulong mem_base, |
234 | 249 | d->config[0x0B] = 0x06; // class_base = PCI_bridge |
235 | 250 | d->config[0x0D] = 0x10; // latency_timer |
236 | 251 | d->config[0x0E] = 0x00; // header_type |
237 | - return s->bus; | |
252 | + | |
253 | + /* APB secondary busses */ | |
254 | + secondary = pci_bridge_init(s->bus, 8, 0x108e5000, pci_apb_map_irq); | |
255 | + pci_bridge_init(s->bus, 9, 0x108e5000, pci_apb_map_irq); | |
256 | + return secondary; | |
238 | 257 | } |
239 | 258 | |
240 | 259 | ... | ... |
hw/grackle_pci.c
... | ... | @@ -92,7 +92,8 @@ PCIBus *pci_grackle_init(uint32_t base, void *pic) |
92 | 92 | int pci_mem_config, pci_mem_data; |
93 | 93 | |
94 | 94 | s = qemu_mallocz(sizeof(GrackleState)); |
95 | - s->bus = pci_register_bus(pci_grackle_set_irq, pci_grackle_map_irq, pic, 0); | |
95 | + s->bus = pci_register_bus(pci_grackle_set_irq, pci_grackle_map_irq, | |
96 | + pic, 0, 0); | |
96 | 97 | |
97 | 98 | pci_mem_config = cpu_register_io_memory(0, pci_grackle_config_read, |
98 | 99 | pci_grackle_config_write, s); | ... | ... |
hw/pci.c
... | ... | @@ -35,9 +35,11 @@ struct PCIBus { |
35 | 35 | SetIRQFunc *low_set_irq; |
36 | 36 | void *irq_opaque; |
37 | 37 | PCIDevice *devices[256]; |
38 | + PCIDevice *parent_dev; | |
39 | + PCIBus *next; | |
38 | 40 | /* The bus IRQ state is the logical OR of the connected devices. |
39 | 41 | Keep a count of the number of devices with raised IRQs. */ |
40 | - int irq_count[4]; | |
42 | + int irq_count[]; | |
41 | 43 | }; |
42 | 44 | |
43 | 45 | static void pci_update_mappings(PCIDevice *d); |
... | ... | @@ -47,19 +49,29 @@ static int pci_irq_index; |
47 | 49 | static PCIBus *first_bus; |
48 | 50 | |
49 | 51 | PCIBus *pci_register_bus(pci_set_irq_fn set_irq, pci_map_irq_fn map_irq, |
50 | - void *pic, int devfn_min) | |
52 | + void *pic, int devfn_min, int nirq) | |
51 | 53 | { |
52 | 54 | PCIBus *bus; |
53 | - bus = qemu_mallocz(sizeof(PCIBus)); | |
55 | + bus = qemu_mallocz(sizeof(PCIBus) + (nirq * sizeof(int))); | |
54 | 56 | bus->set_irq = set_irq; |
55 | 57 | bus->map_irq = map_irq; |
56 | 58 | bus->irq_opaque = pic; |
57 | 59 | bus->devfn_min = devfn_min; |
58 | - memset(bus->irq_count, 0, sizeof(bus->irq_count)); | |
59 | 60 | first_bus = bus; |
60 | 61 | return bus; |
61 | 62 | } |
62 | 63 | |
64 | +PCIBus *pci_register_secondary_bus(PCIDevice *dev, pci_map_irq_fn map_irq) | |
65 | +{ | |
66 | + PCIBus *bus; | |
67 | + bus = qemu_mallocz(sizeof(PCIBus)); | |
68 | + bus->map_irq = map_irq; | |
69 | + bus->parent_dev = dev; | |
70 | + bus->next = dev->bus->next; | |
71 | + dev->bus->next = bus; | |
72 | + return bus; | |
73 | +} | |
74 | + | |
63 | 75 | int pci_bus_num(PCIBus *s) |
64 | 76 | { |
65 | 77 | return s->bus_num; |
... | ... | @@ -351,7 +363,9 @@ void pci_data_write(void *opaque, uint32_t addr, uint32_t val, int len) |
351 | 363 | addr, val, len); |
352 | 364 | #endif |
353 | 365 | bus_num = (addr >> 16) & 0xff; |
354 | - if (bus_num != 0) | |
366 | + while (s && s->bus_num != bus_num) | |
367 | + s = s->next; | |
368 | + if (!s) | |
355 | 369 | return; |
356 | 370 | pci_dev = s->devices[(addr >> 8) & 0xff]; |
357 | 371 | if (!pci_dev) |
... | ... | @@ -372,7 +386,9 @@ uint32_t pci_data_read(void *opaque, uint32_t addr, int len) |
372 | 386 | uint32_t val; |
373 | 387 | |
374 | 388 | bus_num = (addr >> 16) & 0xff; |
375 | - if (bus_num != 0) | |
389 | + while (s && s->bus_num != bus_num) | |
390 | + s= s->next; | |
391 | + if (!s) | |
376 | 392 | goto fail; |
377 | 393 | pci_dev = s->devices[(addr >> 8) & 0xff]; |
378 | 394 | if (!pci_dev) { |
... | ... | @@ -411,11 +427,21 @@ uint32_t pci_data_read(void *opaque, uint32_t addr, int len) |
411 | 427 | /* 0 <= irq_num <= 3. level must be 0 or 1 */ |
412 | 428 | void pci_set_irq(PCIDevice *pci_dev, int irq_num, int level) |
413 | 429 | { |
414 | - PCIBus *bus = pci_dev->bus; | |
430 | + PCIBus *bus; | |
431 | + int change; | |
432 | + | |
433 | + change = level - pci_dev->irq_state[irq_num]; | |
434 | + if (!change) | |
435 | + return; | |
415 | 436 | |
416 | - irq_num = bus->map_irq(pci_dev, irq_num); | |
417 | - bus->irq_count[irq_num] += level - pci_dev->irq_state[irq_num]; | |
418 | 437 | pci_dev->irq_state[irq_num] = level; |
438 | + bus = pci_dev->bus; | |
439 | + while (!bus->set_irq) { | |
440 | + irq_num = bus->map_irq(pci_dev, irq_num); | |
441 | + pci_dev = bus->parent_dev; | |
442 | + bus = pci_dev->bus; | |
443 | + } | |
444 | + bus->irq_count[irq_num] += change; | |
419 | 445 | bus->set_irq(bus->irq_opaque, irq_num, bus->irq_count[irq_num] != 0); |
420 | 446 | } |
421 | 447 | |
... | ... | @@ -465,6 +491,9 @@ static void pci_info_device(PCIDevice *d) |
465 | 491 | if (d->config[PCI_INTERRUPT_PIN] != 0) { |
466 | 492 | term_printf(" IRQ %d.\n", d->config[PCI_INTERRUPT_LINE]); |
467 | 493 | } |
494 | + if (class == 0x0604) { | |
495 | + term_printf(" BUS %d.\n", d->config[0x19]); | |
496 | + } | |
468 | 497 | for(i = 0;i < PCI_NUM_REGIONS; i++) { |
469 | 498 | r = &d->io_regions[i]; |
470 | 499 | if (r->size != 0) { |
... | ... | @@ -478,14 +507,19 @@ static void pci_info_device(PCIDevice *d) |
478 | 507 | } |
479 | 508 | } |
480 | 509 | } |
510 | + if (class == 0x0604 && d->config[0x19] != 0) { | |
511 | + pci_for_each_device(d->config[0x19], pci_info_device); | |
512 | + } | |
481 | 513 | } |
482 | 514 | |
483 | -void pci_for_each_device(void (*fn)(PCIDevice *d)) | |
515 | +void pci_for_each_device(int bus_num, void (*fn)(PCIDevice *d)) | |
484 | 516 | { |
485 | 517 | PCIBus *bus = first_bus; |
486 | 518 | PCIDevice *d; |
487 | 519 | int devfn; |
488 | 520 | |
521 | + while (bus && bus->bus_num != bus_num) | |
522 | + bus = bus->next; | |
489 | 523 | if (bus) { |
490 | 524 | for(devfn = 0; devfn < 256; devfn++) { |
491 | 525 | d = bus->devices[devfn]; |
... | ... | @@ -497,7 +531,7 @@ void pci_for_each_device(void (*fn)(PCIDevice *d)) |
497 | 531 | |
498 | 532 | void pci_info(void) |
499 | 533 | { |
500 | - pci_for_each_device(pci_info_device); | |
534 | + pci_for_each_device(0, pci_info_device); | |
501 | 535 | } |
502 | 536 | |
503 | 537 | /* Initialize a PCI NIC. */ |
... | ... | @@ -515,3 +549,50 @@ void pci_nic_init(PCIBus *bus, NICInfo *nd) |
515 | 549 | } |
516 | 550 | } |
517 | 551 | |
552 | +typedef struct { | |
553 | + PCIDevice dev; | |
554 | + PCIBus *bus; | |
555 | +} PCIBridge; | |
556 | + | |
557 | +void pci_bridge_write_config(PCIDevice *d, | |
558 | + uint32_t address, uint32_t val, int len) | |
559 | +{ | |
560 | + PCIBridge *s = (PCIBridge *)d; | |
561 | + | |
562 | + if (address == 0x19 || (address == 0x18 && len > 1)) { | |
563 | + if (address == 0x19) | |
564 | + s->bus->bus_num = val & 0xff; | |
565 | + else | |
566 | + s->bus->bus_num = (val >> 8) & 0xff; | |
567 | +#if defined(DEBUG_PCI) | |
568 | + printf ("pci-bridge: %s: Assigned bus %d\n", d->name, s->bus->bus_num); | |
569 | +#endif | |
570 | + } | |
571 | + pci_default_write_config(d, address, val, len); | |
572 | +} | |
573 | + | |
574 | +PCIBus *pci_bridge_init(PCIBus *bus, int devfn, uint32_t id, | |
575 | + pci_map_irq_fn map_irq, const char *name) | |
576 | +{ | |
577 | + PCIBridge *s; | |
578 | + s = (PCIBridge *)pci_register_device(bus, name, sizeof(PCIBridge), | |
579 | + devfn, NULL, pci_bridge_write_config); | |
580 | + s->dev.config[0x00] = id >> 16; | |
581 | + s->dev.config[0x01] = id > 24; | |
582 | + s->dev.config[0x02] = id; // device_id | |
583 | + s->dev.config[0x03] = id >> 8; | |
584 | + s->dev.config[0x04] = 0x06; // command = bus master, pci mem | |
585 | + s->dev.config[0x05] = 0x00; | |
586 | + s->dev.config[0x06] = 0xa0; // status = fast back-to-back, 66MHz, no error | |
587 | + s->dev.config[0x07] = 0x00; // status = fast devsel | |
588 | + s->dev.config[0x08] = 0x00; // revision | |
589 | + s->dev.config[0x09] = 0x00; // programming i/f | |
590 | + s->dev.config[0x0A] = 0x04; // class_sub = PCI to PCI bridge | |
591 | + s->dev.config[0x0B] = 0x06; // class_base = PCI_bridge | |
592 | + s->dev.config[0x0D] = 0x10; // latency_timer | |
593 | + s->dev.config[0x0E] = 0x81; // header_type | |
594 | + s->dev.config[0x1E] = 0xa0; // secondary status | |
595 | + | |
596 | + s->bus = pci_register_secondary_bus(&s->dev, map_irq); | |
597 | + return s->bus; | |
598 | +} | ... | ... |
hw/piix_pci.c
... | ... | @@ -59,7 +59,7 @@ PCIBus *i440fx_init(void) |
59 | 59 | I440FXState *s; |
60 | 60 | |
61 | 61 | s = qemu_mallocz(sizeof(I440FXState)); |
62 | - b = pci_register_bus(piix3_set_irq, pci_slot_get_pirq, NULL, 0); | |
62 | + b = pci_register_bus(piix3_set_irq, pci_slot_get_pirq, NULL, 0, 4); | |
63 | 63 | s->bus = b; |
64 | 64 | |
65 | 65 | register_ioport_write(0xcf8, 4, 4, i440fx_addr_writel, s); |
... | ... | @@ -226,6 +226,7 @@ static uint32_t pci_bios_io_addr; |
226 | 226 | static uint32_t pci_bios_mem_addr; |
227 | 227 | /* host irqs corresponding to PCI irqs A-D */ |
228 | 228 | static uint8_t pci_irqs[4] = { 11, 9, 11, 9 }; |
229 | +static int pci_bios_next_bus; | |
229 | 230 | |
230 | 231 | static void pci_config_writel(PCIDevice *d, uint32_t addr, uint32_t val) |
231 | 232 | { |
... | ... | @@ -320,6 +321,14 @@ static void pci_bios_init_device(PCIDevice *d) |
320 | 321 | pci_set_io_region_addr(d, 3, 0x374); |
321 | 322 | } |
322 | 323 | break; |
324 | + case 0x0604: | |
325 | + /* PCI to PCI bridge. Assign bus ID and recurse to configure | |
326 | + devices on the secondary bus. */ | |
327 | + i = pci_bios_next_bus++; | |
328 | + pci_config_writeb(d, 0x18, pci_bus_num(d->bus)); | |
329 | + pci_config_writeb(d, 0x19, i); | |
330 | + pci_for_each_device(i, pci_bios_init_device); | |
331 | + break; | |
323 | 332 | case 0x0300: |
324 | 333 | if (vendor_id != 0x1234) |
325 | 334 | goto default_map; |
... | ... | @@ -398,6 +407,7 @@ void pci_bios_init(void) |
398 | 407 | isa_outb(elcr[0], 0x4d0); |
399 | 408 | isa_outb(elcr[1], 0x4d1); |
400 | 409 | |
401 | - pci_for_each_device(pci_bios_init_device); | |
410 | + pci_bios_next_bus = 1; | |
411 | + pci_for_each_device(0, pci_bios_init_device); | |
402 | 412 | } |
403 | 413 | ... | ... |
hw/prep_pci.c
... | ... | @@ -120,18 +120,12 @@ static CPUReadMemoryFunc *PPC_PCIIO_read[] = { |
120 | 120 | /* Don't know if this matches real hardware, but it agrees with OHW. */ |
121 | 121 | static int prep_map_irq(PCIDevice *pci_dev, int irq_num) |
122 | 122 | { |
123 | - return (irq_num + (pci_dev->devfn >> 3)) & 3; | |
123 | + return (irq_num + (pci_dev->devfn >> 3)) & 1; | |
124 | 124 | } |
125 | 125 | |
126 | -static int prep_irq_levels[4]; | |
127 | - | |
128 | 126 | static void prep_set_irq(void *pic, int irq_num, int level) |
129 | 127 | { |
130 | - int pic_irq_num; | |
131 | - prep_irq_levels[irq_num] = level; | |
132 | - level |= prep_irq_levels[irq_num ^ 2]; | |
133 | - pic_irq_num = (irq_num == 0 || irq_num == 2) ? 9 : 11; | |
134 | - pic_set_irq(pic_irq_num, level); | |
128 | + pic_set_irq(irq_num ? 11 : 9, level); | |
135 | 129 | } |
136 | 130 | |
137 | 131 | PCIBus *pci_prep_init(void) |
... | ... | @@ -141,7 +135,7 @@ PCIBus *pci_prep_init(void) |
141 | 135 | int PPC_io_memory; |
142 | 136 | |
143 | 137 | s = qemu_mallocz(sizeof(PREPPCIState)); |
144 | - s->bus = pci_register_bus(prep_set_irq, prep_map_irq, NULL, 0); | |
138 | + s->bus = pci_register_bus(prep_set_irq, prep_map_irq, NULL, 0, 2); | |
145 | 139 | |
146 | 140 | register_ioport_write(0xcf8, 4, 4, pci_prep_addr_writel, s); |
147 | 141 | register_ioport_read(0xcf8, 4, 4, pci_prep_addr_readl, s); | ... | ... |
hw/unin_pci.c
... | ... | @@ -161,7 +161,7 @@ PCIBus *pci_pmac_init(void *pic) |
161 | 161 | /* Uninorth main bus */ |
162 | 162 | s = qemu_mallocz(sizeof(UNINState)); |
163 | 163 | s->bus = pci_register_bus(pci_unin_set_irq, pci_unin_map_irq, |
164 | - pic, 11 << 3); | |
164 | + pic, 11 << 3, 4); | |
165 | 165 | |
166 | 166 | pci_mem_config = cpu_register_io_memory(0, pci_unin_main_config_read, |
167 | 167 | pci_unin_main_config_write, s); | ... | ... |
hw/versatile_pci.c
... | ... | @@ -11,7 +11,7 @@ |
11 | 11 | |
12 | 12 | static inline uint32_t vpb_pci_config_addr(target_phys_addr_t addr) |
13 | 13 | { |
14 | - return addr & 0xf8ff; | |
14 | + return addr & 0xffffff; | |
15 | 15 | } |
16 | 16 | |
17 | 17 | static void pci_vpb_config_writeb (void *opaque, target_phys_addr_t addr, |
... | ... | @@ -105,15 +105,15 @@ PCIBus *pci_vpb_init(void *pic, int irq, int realview) |
105 | 105 | base = 0x40000000; |
106 | 106 | name = "Versatile/PB PCI Controller"; |
107 | 107 | } |
108 | - s = pci_register_bus(pci_vpb_set_irq, pci_vpb_map_irq, pic, 11 << 3); | |
108 | + s = pci_register_bus(pci_vpb_set_irq, pci_vpb_map_irq, pic, 11 << 3, 4); | |
109 | 109 | /* ??? Register memory space. */ |
110 | 110 | |
111 | 111 | mem_config = cpu_register_io_memory(0, pci_vpb_config_read, |
112 | 112 | pci_vpb_config_write, s); |
113 | 113 | /* Selfconfig area. */ |
114 | - cpu_register_physical_memory(base + 0x01000000, 0x10000, mem_config); | |
114 | + cpu_register_physical_memory(base + 0x01000000, 0x1000000, mem_config); | |
115 | 115 | /* Normal config area. */ |
116 | - cpu_register_physical_memory(base + 0x02000000, 0x10000, mem_config); | |
116 | + cpu_register_physical_memory(base + 0x02000000, 0x1000000, mem_config); | |
117 | 117 | |
118 | 118 | d = pci_register_device(s, name, sizeof(PCIDevice), -1, NULL, NULL); |
119 | 119 | ... | ... |
vl.h
... | ... | @@ -759,15 +759,17 @@ int pci_device_load(PCIDevice *s, QEMUFile *f); |
759 | 759 | typedef void (*pci_set_irq_fn)(void *pic, int irq_num, int level); |
760 | 760 | typedef int (*pci_map_irq_fn)(PCIDevice *pci_dev, int irq_num); |
761 | 761 | PCIBus *pci_register_bus(pci_set_irq_fn set_irq, pci_map_irq_fn map_irq, |
762 | - void *pic, int devfn_min); | |
762 | + void *pic, int devfn_min, int nirq); | |
763 | 763 | |
764 | 764 | void pci_nic_init(PCIBus *bus, NICInfo *nd); |
765 | 765 | void pci_data_write(void *opaque, uint32_t addr, uint32_t val, int len); |
766 | 766 | uint32_t pci_data_read(void *opaque, uint32_t addr, int len); |
767 | 767 | int pci_bus_num(PCIBus *s); |
768 | -void pci_for_each_device(void (*fn)(PCIDevice *d)); | |
768 | +void pci_for_each_device(int bus_num, void (*fn)(PCIDevice *d)); | |
769 | 769 | |
770 | 770 | void pci_info(void); |
771 | +PCIBus *pci_bridge_init(PCIBus *bus, int devfn, uint32_t id, | |
772 | + pci_map_irq_fn map_irq, const char *name); | |
771 | 773 | |
772 | 774 | /* prep_pci.c */ |
773 | 775 | PCIBus *pci_prep_init(void); | ... | ... |