Commit d2b5931756fdb9f839180e33898cd1e3e4fbdc90
1 parent
e69954b9
PCI shared IRQ fix (original patch by andrzej zaborowski).
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@2165 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
8 changed files
with
95 additions
and
75 deletions
hw/apb_pci.c
| ... | ... | @@ -179,10 +179,18 @@ static CPUReadMemoryFunc *pci_apb_ioread[] = { |
| 179 | 179 | &pci_apb_ioreadl, |
| 180 | 180 | }; |
| 181 | 181 | |
| 182 | -/* ??? This is probably wrong. */ | |
| 183 | -static void pci_apb_set_irq(PCIDevice *d, void *pic, int irq_num, int level) | |
| 182 | +static int pci_apb_map_irq(PCIDevice *pci_dev, int irq_num) | |
| 184 | 183 | { |
| 185 | - pic_set_irq_new(pic, d->config[PCI_INTERRUPT_LINE], level); | |
| 184 | + /* ??? As mentioned below this is probably wrong. */ | |
| 185 | + return irq_num; | |
| 186 | +} | |
| 187 | + | |
| 188 | +static void pci_apb_set_irq(void *pic, int irq_num, int level) | |
| 189 | +{ | |
| 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. */ | |
| 193 | + pic_set_irq_new(pic, irq_num, level); | |
| 186 | 194 | } |
| 187 | 195 | |
| 188 | 196 | PCIBus *pci_apb_init(target_ulong special_base, target_ulong mem_base, |
| ... | ... | @@ -194,7 +202,7 @@ PCIBus *pci_apb_init(target_ulong special_base, target_ulong mem_base, |
| 194 | 202 | |
| 195 | 203 | s = qemu_mallocz(sizeof(APBState)); |
| 196 | 204 | /* Ultrasparc APB main bus */ |
| 197 | - s->bus = pci_register_bus(pci_apb_set_irq, pic, 0); | |
| 205 | + s->bus = pci_register_bus(pci_apb_set_irq, pci_apb_map_irq, pic, 0); | |
| 198 | 206 | |
| 199 | 207 | pci_mem_config = cpu_register_io_memory(0, pci_apb_config_read, |
| 200 | 208 | pci_apb_config_write, s); | ... | ... |
hw/grackle_pci.c
| ... | ... | @@ -74,11 +74,15 @@ static CPUReadMemoryFunc *pci_grackle_read[] = { |
| 74 | 74 | &pci_host_data_readl, |
| 75 | 75 | }; |
| 76 | 76 | |
| 77 | -/* XXX: we do not simulate the hardware - we rely on the BIOS to | |
| 78 | - set correctly for irq line field */ | |
| 79 | -static void pci_grackle_set_irq(PCIDevice *d, void *pic, int irq_num, int level) | |
| 77 | +/* Don't know if this matches real hardware, but it agrees with OHW. */ | |
| 78 | +static int pci_grackle_map_irq(PCIDevice *pci_dev, int irq_num) | |
| 80 | 79 | { |
| 81 | - heathrow_pic_set_irq(pic, d->config[PCI_INTERRUPT_LINE], level); | |
| 80 | + return (irq_num + (pci_dev->devfn >> 3)) & 3; | |
| 81 | +} | |
| 82 | + | |
| 83 | +static void pci_grackle_set_irq(void *pic, int irq_num, int level) | |
| 84 | +{ | |
| 85 | + heathrow_pic_set_irq(pic, irq_num + 8, level); | |
| 82 | 86 | } |
| 83 | 87 | |
| 84 | 88 | PCIBus *pci_grackle_init(uint32_t base, void *pic) |
| ... | ... | @@ -88,7 +92,7 @@ PCIBus *pci_grackle_init(uint32_t base, void *pic) |
| 88 | 92 | int pci_mem_config, pci_mem_data; |
| 89 | 93 | |
| 90 | 94 | s = qemu_mallocz(sizeof(GrackleState)); |
| 91 | - s->bus = pci_register_bus(pci_grackle_set_irq, pic, 0); | |
| 95 | + s->bus = pci_register_bus(pci_grackle_set_irq, pci_grackle_map_irq, pic, 0); | |
| 92 | 96 | |
| 93 | 97 | pci_mem_config = cpu_register_io_memory(0, pci_grackle_config_read, |
| 94 | 98 | pci_grackle_config_write, s); | ... | ... |
hw/pci.c
| ... | ... | @@ -29,11 +29,15 @@ struct PCIBus { |
| 29 | 29 | int bus_num; |
| 30 | 30 | int devfn_min; |
| 31 | 31 | pci_set_irq_fn set_irq; |
| 32 | + pci_map_irq_fn map_irq; | |
| 32 | 33 | uint32_t config_reg; /* XXX: suppress */ |
| 33 | 34 | /* low level pic */ |
| 34 | 35 | SetIRQFunc *low_set_irq; |
| 35 | 36 | void *irq_opaque; |
| 36 | 37 | PCIDevice *devices[256]; |
| 38 | + /* The bus IRQ state is the logical OR of the connected devices. | |
| 39 | + Keep a count of the number of devices with raised IRQs. */ | |
| 40 | + int irq_count[4]; | |
| 37 | 41 | }; |
| 38 | 42 | |
| 39 | 43 | static void pci_update_mappings(PCIDevice *d); |
| ... | ... | @@ -42,13 +46,16 @@ target_phys_addr_t pci_mem_base; |
| 42 | 46 | static int pci_irq_index; |
| 43 | 47 | static PCIBus *first_bus; |
| 44 | 48 | |
| 45 | -PCIBus *pci_register_bus(pci_set_irq_fn set_irq, void *pic, int devfn_min) | |
| 49 | +PCIBus *pci_register_bus(pci_set_irq_fn set_irq, pci_map_irq_fn map_irq, | |
| 50 | + void *pic, int devfn_min) | |
| 46 | 51 | { |
| 47 | 52 | PCIBus *bus; |
| 48 | 53 | bus = qemu_mallocz(sizeof(PCIBus)); |
| 49 | 54 | bus->set_irq = set_irq; |
| 55 | + bus->map_irq = map_irq; | |
| 50 | 56 | bus->irq_opaque = pic; |
| 51 | 57 | bus->devfn_min = devfn_min; |
| 58 | + memset(bus->irq_count, 0, sizeof(bus->irq_count)); | |
| 52 | 59 | first_bus = bus; |
| 53 | 60 | return bus; |
| 54 | 61 | } |
| ... | ... | @@ -100,6 +107,7 @@ PCIDevice *pci_register_device(PCIBus *bus, const char *name, |
| 100 | 107 | pci_dev->bus = bus; |
| 101 | 108 | pci_dev->devfn = devfn; |
| 102 | 109 | pstrcpy(pci_dev->name, sizeof(pci_dev->name), name); |
| 110 | + memset(pci_dev->irq_state, 0, sizeof(pci_dev->irq_state)); | |
| 103 | 111 | |
| 104 | 112 | if (!config_read) |
| 105 | 113 | config_read = pci_default_read_config; |
| ... | ... | @@ -404,7 +412,11 @@ uint32_t pci_data_read(void *opaque, uint32_t addr, int len) |
| 404 | 412 | void pci_set_irq(PCIDevice *pci_dev, int irq_num, int level) |
| 405 | 413 | { |
| 406 | 414 | PCIBus *bus = pci_dev->bus; |
| 407 | - bus->set_irq(pci_dev, bus->irq_opaque, irq_num, level); | |
| 415 | + | |
| 416 | + irq_num = bus->map_irq(pci_dev, irq_num); | |
| 417 | + bus->irq_count[irq_num] += level - pci_dev->irq_state[irq_num]; | |
| 418 | + pci_dev->irq_state[irq_num] = level; | |
| 419 | + bus->set_irq(bus->irq_opaque, irq_num, bus->irq_count[irq_num] != 0); | |
| 408 | 420 | } |
| 409 | 421 | |
| 410 | 422 | /***********************************************************/ | ... | ... |
hw/piix_pci.c
| ... | ... | @@ -40,7 +40,17 @@ static uint32_t i440fx_addr_readl(void* opaque, uint32_t addr) |
| 40 | 40 | return s->config_reg; |
| 41 | 41 | } |
| 42 | 42 | |
| 43 | -static void piix3_set_irq(PCIDevice *pci_dev, void *pic, int irq_num, int level); | |
| 43 | +static void piix3_set_irq(void *pic, int irq_num, int level); | |
| 44 | + | |
| 45 | +/* return the global irq number corresponding to a given device irq | |
| 46 | + pin. We could also use the bus number to have a more precise | |
| 47 | + mapping. */ | |
| 48 | +static int pci_slot_get_pirq(PCIDevice *pci_dev, int irq_num) | |
| 49 | +{ | |
| 50 | + int slot_addend; | |
| 51 | + slot_addend = (pci_dev->devfn >> 3) - 1; | |
| 52 | + return (irq_num + slot_addend) & 3; | |
| 53 | +} | |
| 44 | 54 | |
| 45 | 55 | PCIBus *i440fx_init(void) |
| 46 | 56 | { |
| ... | ... | @@ -49,7 +59,7 @@ PCIBus *i440fx_init(void) |
| 49 | 59 | I440FXState *s; |
| 50 | 60 | |
| 51 | 61 | s = qemu_mallocz(sizeof(I440FXState)); |
| 52 | - b = pci_register_bus(piix3_set_irq, NULL, 0); | |
| 62 | + b = pci_register_bus(piix3_set_irq, pci_slot_get_pirq, NULL, 0); | |
| 53 | 63 | s->bus = b; |
| 54 | 64 | |
| 55 | 65 | register_ioport_write(0xcf8, 4, 4, i440fx_addr_writel, s); |
| ... | ... | @@ -83,65 +93,25 @@ static PCIDevice *piix3_dev; |
| 83 | 93 | /* just used for simpler irq handling. */ |
| 84 | 94 | #define PCI_IRQ_WORDS ((PCI_DEVICES_MAX + 31) / 32) |
| 85 | 95 | |
| 86 | -static uint32_t pci_irq_levels[4][PCI_IRQ_WORDS]; | |
| 87 | - | |
| 88 | -/* return the global irq number corresponding to a given device irq | |
| 89 | - pin. We could also use the bus number to have a more precise | |
| 90 | - mapping. */ | |
| 91 | -static inline int pci_slot_get_pirq(PCIDevice *pci_dev, int irq_num) | |
| 92 | -{ | |
| 93 | - int slot_addend; | |
| 94 | - slot_addend = (pci_dev->devfn >> 3) - 1; | |
| 95 | - return (irq_num + slot_addend) & 3; | |
| 96 | -} | |
| 97 | - | |
| 98 | -static inline int get_pci_irq_level(int irq_num) | |
| 99 | -{ | |
| 100 | - int pic_level; | |
| 101 | -#if (PCI_IRQ_WORDS == 2) | |
| 102 | - pic_level = ((pci_irq_levels[irq_num][0] | | |
| 103 | - pci_irq_levels[irq_num][1]) != 0); | |
| 104 | -#else | |
| 105 | - { | |
| 106 | - int i; | |
| 107 | - pic_level = 0; | |
| 108 | - for(i = 0; i < PCI_IRQ_WORDS; i++) { | |
| 109 | - if (pci_irq_levels[irq_num][i]) { | |
| 110 | - pic_level = 1; | |
| 111 | - break; | |
| 112 | - } | |
| 113 | - } | |
| 114 | - } | |
| 115 | -#endif | |
| 116 | - return pic_level; | |
| 117 | -} | |
| 96 | +static int pci_irq_levels[4]; | |
| 118 | 97 | |
| 119 | -static void piix3_set_irq(PCIDevice *pci_dev, void *pic, int irq_num, int level) | |
| 98 | +static void piix3_set_irq(void *pic, int irq_num, int level) | |
| 120 | 99 | { |
| 121 | - int irq_index, shift, pic_irq, pic_level; | |
| 122 | - uint32_t *p; | |
| 100 | + int i, pic_irq, pic_level; | |
| 123 | 101 | |
| 124 | - irq_num = pci_slot_get_pirq(pci_dev, irq_num); | |
| 125 | - irq_index = pci_dev->irq_index; | |
| 126 | - p = &pci_irq_levels[irq_num][irq_index >> 5]; | |
| 127 | - shift = (irq_index & 0x1f); | |
| 128 | - *p = (*p & ~(1 << shift)) | (level << shift); | |
| 102 | + pci_irq_levels[irq_num] = level; | |
| 129 | 103 | |
| 130 | 104 | /* now we change the pic irq level according to the piix irq mappings */ |
| 131 | 105 | /* XXX: optimize */ |
| 132 | 106 | pic_irq = piix3_dev->config[0x60 + irq_num]; |
| 133 | 107 | if (pic_irq < 16) { |
| 134 | - /* the pic level is the logical OR of all the PCI irqs mapped | |
| 108 | + /* The pic level is the logical OR of all the PCI irqs mapped | |
| 135 | 109 | to it */ |
| 136 | 110 | pic_level = 0; |
| 137 | - if (pic_irq == piix3_dev->config[0x60]) | |
| 138 | - pic_level |= get_pci_irq_level(0); | |
| 139 | - if (pic_irq == piix3_dev->config[0x61]) | |
| 140 | - pic_level |= get_pci_irq_level(1); | |
| 141 | - if (pic_irq == piix3_dev->config[0x62]) | |
| 142 | - pic_level |= get_pci_irq_level(2); | |
| 143 | - if (pic_irq == piix3_dev->config[0x63]) | |
| 144 | - pic_level |= get_pci_irq_level(3); | |
| 111 | + for (i = 0; i < 4; i++) { | |
| 112 | + if (pic_irq == piix3_dev->config[0x60 + i]) | |
| 113 | + pic_level |= pci_irq_levels[i]; | |
| 114 | + } | |
| 145 | 115 | pic_set_irq(pic_irq, pic_level); |
| 146 | 116 | } |
| 147 | 117 | } | ... | ... |
hw/prep_pci.c
| ... | ... | @@ -117,11 +117,21 @@ static CPUReadMemoryFunc *PPC_PCIIO_read[] = { |
| 117 | 117 | &PPC_PCIIO_readl, |
| 118 | 118 | }; |
| 119 | 119 | |
| 120 | -static void prep_set_irq(PCIDevice *d, void *pic, int irq_num, int level) | |
| 120 | +/* Don't know if this matches real hardware, but it agrees with OHW. */ | |
| 121 | +static int prep_map_irq(PCIDevice *pci_dev, int irq_num) | |
| 121 | 122 | { |
| 122 | - /* XXX: we do not simulate the hardware - we rely on the BIOS to | |
| 123 | - set correctly for irq line field */ | |
| 124 | - pic_set_irq(d->config[PCI_INTERRUPT_LINE], level); | |
| 123 | + return (irq_num + (pci_dev->devfn >> 3)) & 3; | |
| 124 | +} | |
| 125 | + | |
| 126 | +static int prep_irq_levels[4]; | |
| 127 | + | |
| 128 | +static void prep_set_irq(void *pic, int irq_num, int level) | |
| 129 | +{ | |
| 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); | |
| 125 | 135 | } |
| 126 | 136 | |
| 127 | 137 | PCIBus *pci_prep_init(void) |
| ... | ... | @@ -131,7 +141,7 @@ PCIBus *pci_prep_init(void) |
| 131 | 141 | int PPC_io_memory; |
| 132 | 142 | |
| 133 | 143 | s = qemu_mallocz(sizeof(PREPPCIState)); |
| 134 | - s->bus = pci_register_bus(prep_set_irq, NULL, 0); | |
| 144 | + s->bus = pci_register_bus(prep_set_irq, prep_map_irq, NULL, 0); | |
| 135 | 145 | |
| 136 | 146 | register_ioport_write(0xcf8, 4, 4, pci_prep_addr_writel, s); |
| 137 | 147 | register_ioport_read(0xcf8, 4, 4, pci_prep_addr_readl, s); | ... | ... |
hw/unin_pci.c
| ... | ... | @@ -140,9 +140,15 @@ static CPUReadMemoryFunc *pci_unin_read[] = { |
| 140 | 140 | }; |
| 141 | 141 | #endif |
| 142 | 142 | |
| 143 | -static void pci_unin_set_irq(PCIDevice *d, void *pic, int irq_num, int level) | |
| 143 | +/* Don't know if this matches real hardware, but it agrees with OHW. */ | |
| 144 | +static int pci_unin_map_irq(PCIDevice *pci_dev, int irq_num) | |
| 144 | 145 | { |
| 145 | - openpic_set_irq(pic, d->config[PCI_INTERRUPT_LINE], level); | |
| 146 | + return (irq_num + (pci_dev->devfn >> 3)) & 3; | |
| 147 | +} | |
| 148 | + | |
| 149 | +static void pci_unin_set_irq(void *pic, int irq_num, int level) | |
| 150 | +{ | |
| 151 | + openpic_set_irq(pic, irq_num + 8, level); | |
| 146 | 152 | } |
| 147 | 153 | |
| 148 | 154 | PCIBus *pci_pmac_init(void *pic) |
| ... | ... | @@ -154,7 +160,8 @@ PCIBus *pci_pmac_init(void *pic) |
| 154 | 160 | /* Use values found on a real PowerMac */ |
| 155 | 161 | /* Uninorth main bus */ |
| 156 | 162 | s = qemu_mallocz(sizeof(UNINState)); |
| 157 | - s->bus = pci_register_bus(pci_unin_set_irq, NULL, 11 << 3); | |
| 163 | + s->bus = pci_register_bus(pci_unin_set_irq, pci_unin_map_irq, | |
| 164 | + pic, 11 << 3); | |
| 158 | 165 | |
| 159 | 166 | pci_mem_config = cpu_register_io_memory(0, pci_unin_main_config_read, |
| 160 | 167 | pci_unin_main_config_write, s); | ... | ... |
hw/versatile_pci.c
| ... | ... | @@ -79,7 +79,12 @@ static CPUReadMemoryFunc *pci_vpb_config_read[] = { |
| 79 | 79 | |
| 80 | 80 | static int pci_vpb_irq; |
| 81 | 81 | |
| 82 | -static void pci_vpb_set_irq(PCIDevice *d, void *pic, int irq_num, int level) | |
| 82 | +static int pci_vpb_map_irq(PCIDevice *d, int irq_num) | |
| 83 | +{ | |
| 84 | + return irq_num; | |
| 85 | +} | |
| 86 | + | |
| 87 | +static void pci_vpb_set_irq(void *pic, int irq_num, int level) | |
| 83 | 88 | { |
| 84 | 89 | pic_set_irq_new(pic, pci_vpb_irq + irq_num, level); |
| 85 | 90 | } |
| ... | ... | @@ -100,7 +105,7 @@ PCIBus *pci_vpb_init(void *pic, int irq, int realview) |
| 100 | 105 | base = 0x40000000; |
| 101 | 106 | name = "Versatile/PB PCI Controller"; |
| 102 | 107 | } |
| 103 | - s = pci_register_bus(pci_vpb_set_irq, pic, 11 << 3); | |
| 108 | + s = pci_register_bus(pci_vpb_set_irq, pci_vpb_map_irq, pic, 11 << 3); | |
| 104 | 109 | /* ??? Register memory space. */ |
| 105 | 110 | |
| 106 | 111 | mem_config = cpu_register_io_memory(0, pci_vpb_config_read, | ... | ... |
vl.h
| ... | ... | @@ -733,6 +733,9 @@ struct PCIDevice { |
| 733 | 733 | PCIConfigWriteFunc *config_write; |
| 734 | 734 | /* ??? This is a PC-specific hack, and should be removed. */ |
| 735 | 735 | int irq_index; |
| 736 | + | |
| 737 | + /* Current IRQ levels. Used internally by the generic PCI code. */ | |
| 738 | + int irq_state[4]; | |
| 736 | 739 | }; |
| 737 | 740 | |
| 738 | 741 | PCIDevice *pci_register_device(PCIBus *bus, const char *name, |
| ... | ... | @@ -753,9 +756,10 @@ void pci_default_write_config(PCIDevice *d, |
| 753 | 756 | void pci_device_save(PCIDevice *s, QEMUFile *f); |
| 754 | 757 | int pci_device_load(PCIDevice *s, QEMUFile *f); |
| 755 | 758 | |
| 756 | -typedef void (*pci_set_irq_fn)(PCIDevice *pci_dev, void *pic, | |
| 757 | - int irq_num, int level); | |
| 758 | -PCIBus *pci_register_bus(pci_set_irq_fn set_irq, void *pic, int devfn_min); | |
| 759 | +typedef void (*pci_set_irq_fn)(void *pic, int irq_num, int level); | |
| 760 | +typedef int (*pci_map_irq_fn)(PCIDevice *pci_dev, int irq_num); | |
| 761 | +PCIBus *pci_register_bus(pci_set_irq_fn set_irq, pci_map_irq_fn map_irq, | |
| 762 | + void *pic, int devfn_min); | |
| 759 | 763 | |
| 760 | 764 | void pci_nic_init(PCIBus *bus, NICInfo *nd); |
| 761 | 765 | void pci_data_write(void *opaque, uint32_t addr, uint32_t val, int len); | ... | ... |