Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 1 | /* |
Michael Tokarev | f1c0cff | 2023-07-14 14:27:04 +0300 | [diff] [blame] | 2 | * HP-PARISC Dino PCI chipset emulation, as in B160L and similar machines |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 3 | * |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 4 | * (C) 2017-2019 by Helge Deller <deller@gmx.de> |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 5 | * |
| 6 | * This work is licensed under the GNU GPL license version 2 or later. |
| 7 | * |
| 8 | * Documentation available at: |
| 9 | * https://parisc.wiki.kernel.org/images-parisc/9/91/Dino_ers.pdf |
| 10 | * https://parisc.wiki.kernel.org/images-parisc/7/70/Dino_3_1_Errata.pdf |
| 11 | */ |
| 12 | |
| 13 | #include "qemu/osdep.h" |
Markus Armbruster | 0b8fa32 | 2019-05-23 16:35:07 +0200 | [diff] [blame] | 14 | #include "qemu/module.h" |
Philippe Mathieu-Daudé | c108cc5 | 2018-06-25 09:42:11 -0300 | [diff] [blame] | 15 | #include "qemu/units.h" |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 16 | #include "qapi/error.h" |
Markus Armbruster | 64552b6 | 2019-08-12 07:23:42 +0200 | [diff] [blame] | 17 | #include "hw/irq.h" |
Markus Armbruster | edf5ca5 | 2022-12-22 11:03:28 +0100 | [diff] [blame] | 18 | #include "hw/pci/pci_device.h" |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 19 | #include "hw/pci/pci_bus.h" |
Mark Cave-Ayland | 270b295 | 2022-05-04 10:25:15 +0100 | [diff] [blame] | 20 | #include "hw/qdev-properties.h" |
Mark Cave-Ayland | 0db9350 | 2022-05-04 10:25:32 +0100 | [diff] [blame] | 21 | #include "hw/pci-host/dino.h" |
Markus Armbruster | d645427 | 2019-08-12 07:23:45 +0200 | [diff] [blame] | 22 | #include "migration/vmstate.h" |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 23 | #include "trace.h" |
Eduardo Habkost | db1015e | 2020-09-03 16:43:22 -0400 | [diff] [blame] | 24 | #include "qom/object.h" |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 25 | |
| 26 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 27 | /* |
| 28 | * Dino can forward memory accesses from the CPU in the range between |
| 29 | * 0xf0800000 and 0xff000000 to the PCI bus. |
| 30 | */ |
| 31 | static void gsc_to_pci_forwarding(DinoState *s) |
| 32 | { |
| 33 | uint32_t io_addr_en, tmp; |
| 34 | int enabled, i; |
| 35 | |
| 36 | tmp = extract32(s->io_control, 7, 2); |
| 37 | enabled = (tmp == 0x01); |
| 38 | io_addr_en = s->io_addr_en; |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 39 | /* Mask out first (=firmware) and last (=Dino) areas. */ |
| 40 | io_addr_en &= ~(BIT(31) | BIT(0)); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 41 | |
| 42 | memory_region_transaction_begin(); |
| 43 | for (i = 1; i < 31; i++) { |
| 44 | MemoryRegion *mem = &s->pci_mem_alias[i]; |
| 45 | if (enabled && (io_addr_en & (1U << i))) { |
| 46 | if (!memory_region_is_mapped(mem)) { |
| 47 | uint32_t addr = 0xf0000000 + i * DINO_MEM_CHUNK_SIZE; |
| 48 | memory_region_add_subregion(get_system_memory(), addr, mem); |
| 49 | } |
| 50 | } else if (memory_region_is_mapped(mem)) { |
| 51 | memory_region_del_subregion(get_system_memory(), mem); |
| 52 | } |
| 53 | } |
| 54 | memory_region_transaction_commit(); |
| 55 | } |
| 56 | |
| 57 | static bool dino_chip_mem_valid(void *opaque, hwaddr addr, |
Peter Maydell | 8372d38 | 2018-05-31 14:50:52 +0100 | [diff] [blame] | 58 | unsigned size, bool is_write, |
| 59 | MemTxAttrs attrs) |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 60 | { |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 61 | bool ret = false; |
| 62 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 63 | switch (addr) { |
| 64 | case DINO_IAR0: |
| 65 | case DINO_IAR1: |
| 66 | case DINO_IRR0: |
| 67 | case DINO_IRR1: |
| 68 | case DINO_IMR: |
| 69 | case DINO_IPR: |
| 70 | case DINO_ICR: |
| 71 | case DINO_ILR: |
| 72 | case DINO_IO_CONTROL: |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 73 | case DINO_IO_FBB_EN: |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 74 | case DINO_IO_ADDR_EN: |
| 75 | case DINO_PCI_IO_DATA: |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 76 | case DINO_TOC_ADDR: |
Philippe Mathieu-Daudé | 90e94c0 | 2020-02-18 07:33:55 +0100 | [diff] [blame] | 77 | case DINO_GMASK ... DINO_PCISTS: |
| 78 | case DINO_MLTIM ... DINO_PCIWOR: |
| 79 | case DINO_TLTIM: |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 80 | ret = true; |
| 81 | break; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 82 | case DINO_PCI_IO_DATA + 2: |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 83 | ret = (size <= 2); |
| 84 | break; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 85 | case DINO_PCI_IO_DATA + 1: |
| 86 | case DINO_PCI_IO_DATA + 3: |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 87 | ret = (size == 1); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 88 | } |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 89 | trace_dino_chip_mem_valid(addr, ret); |
| 90 | return ret; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 91 | } |
| 92 | |
| 93 | static MemTxResult dino_chip_read_with_attrs(void *opaque, hwaddr addr, |
| 94 | uint64_t *data, unsigned size, |
| 95 | MemTxAttrs attrs) |
| 96 | { |
| 97 | DinoState *s = opaque; |
Mark Cave-Ayland | ee313d5 | 2022-05-04 10:25:19 +0100 | [diff] [blame] | 98 | PCIHostState *phb = PCI_HOST_BRIDGE(s); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 99 | MemTxResult ret = MEMTX_OK; |
| 100 | AddressSpace *io; |
| 101 | uint16_t ioaddr; |
| 102 | uint32_t val; |
| 103 | |
| 104 | switch (addr) { |
| 105 | case DINO_PCI_IO_DATA ... DINO_PCI_IO_DATA + 3: |
| 106 | /* Read from PCI IO space. */ |
| 107 | io = &address_space_io; |
Mark Cave-Ayland | ee313d5 | 2022-05-04 10:25:19 +0100 | [diff] [blame] | 108 | ioaddr = phb->config_reg + (addr & 3); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 109 | switch (size) { |
| 110 | case 1: |
| 111 | val = address_space_ldub(io, ioaddr, attrs, &ret); |
| 112 | break; |
| 113 | case 2: |
| 114 | val = address_space_lduw_be(io, ioaddr, attrs, &ret); |
| 115 | break; |
| 116 | case 4: |
| 117 | val = address_space_ldl_be(io, ioaddr, attrs, &ret); |
| 118 | break; |
| 119 | default: |
| 120 | g_assert_not_reached(); |
| 121 | } |
| 122 | break; |
| 123 | |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 124 | case DINO_IO_FBB_EN: |
| 125 | val = s->io_fbb_en; |
| 126 | break; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 127 | case DINO_IO_ADDR_EN: |
| 128 | val = s->io_addr_en; |
| 129 | break; |
| 130 | case DINO_IO_CONTROL: |
| 131 | val = s->io_control; |
| 132 | break; |
| 133 | |
| 134 | case DINO_IAR0: |
| 135 | val = s->iar0; |
| 136 | break; |
| 137 | case DINO_IAR1: |
| 138 | val = s->iar1; |
| 139 | break; |
| 140 | case DINO_IMR: |
| 141 | val = s->imr; |
| 142 | break; |
| 143 | case DINO_ICR: |
| 144 | val = s->icr; |
| 145 | break; |
| 146 | case DINO_IPR: |
| 147 | val = s->ipr; |
| 148 | /* Any read to IPR clears the register. */ |
| 149 | s->ipr = 0; |
| 150 | break; |
| 151 | case DINO_ILR: |
| 152 | val = s->ilr; |
| 153 | break; |
| 154 | case DINO_IRR0: |
| 155 | val = s->ilr & s->imr & ~s->icr; |
| 156 | break; |
| 157 | case DINO_IRR1: |
| 158 | val = s->ilr & s->imr & s->icr; |
| 159 | break; |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 160 | case DINO_TOC_ADDR: |
| 161 | val = s->toc_addr; |
| 162 | break; |
| 163 | case DINO_GMASK ... DINO_TLTIM: |
| 164 | val = s->reg800[(addr - DINO_GMASK) / 4]; |
| 165 | if (addr == DINO_PAMR) { |
| 166 | val &= ~0x01; /* LSB is hardwired to 0 */ |
| 167 | } |
| 168 | if (addr == DINO_MLTIM) { |
| 169 | val &= ~0x07; /* 3 LSB are hardwired to 0 */ |
| 170 | } |
| 171 | if (addr == DINO_BRDG_FEAT) { |
| 172 | val &= ~(0x10710E0ul | 8); /* bits 5-7, 24 & 15 reserved */ |
| 173 | } |
| 174 | break; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 175 | |
| 176 | default: |
| 177 | /* Controlled by dino_chip_mem_valid above. */ |
| 178 | g_assert_not_reached(); |
| 179 | } |
| 180 | |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 181 | trace_dino_chip_read(addr, val); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 182 | *data = val; |
| 183 | return ret; |
| 184 | } |
| 185 | |
| 186 | static MemTxResult dino_chip_write_with_attrs(void *opaque, hwaddr addr, |
| 187 | uint64_t val, unsigned size, |
| 188 | MemTxAttrs attrs) |
| 189 | { |
| 190 | DinoState *s = opaque; |
Mark Cave-Ayland | ee313d5 | 2022-05-04 10:25:19 +0100 | [diff] [blame] | 191 | PCIHostState *phb = PCI_HOST_BRIDGE(s); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 192 | AddressSpace *io; |
| 193 | MemTxResult ret; |
| 194 | uint16_t ioaddr; |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 195 | int i; |
| 196 | |
| 197 | trace_dino_chip_write(addr, val); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 198 | |
| 199 | switch (addr) { |
| 200 | case DINO_IO_DATA ... DINO_PCI_IO_DATA + 3: |
| 201 | /* Write into PCI IO space. */ |
| 202 | io = &address_space_io; |
Mark Cave-Ayland | ee313d5 | 2022-05-04 10:25:19 +0100 | [diff] [blame] | 203 | ioaddr = phb->config_reg + (addr & 3); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 204 | switch (size) { |
| 205 | case 1: |
| 206 | address_space_stb(io, ioaddr, val, attrs, &ret); |
| 207 | break; |
| 208 | case 2: |
| 209 | address_space_stw_be(io, ioaddr, val, attrs, &ret); |
| 210 | break; |
| 211 | case 4: |
| 212 | address_space_stl_be(io, ioaddr, val, attrs, &ret); |
| 213 | break; |
| 214 | default: |
| 215 | g_assert_not_reached(); |
| 216 | } |
| 217 | return ret; |
| 218 | |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 219 | case DINO_IO_FBB_EN: |
| 220 | s->io_fbb_en = val & 0x03; |
| 221 | break; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 222 | case DINO_IO_ADDR_EN: |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 223 | s->io_addr_en = val; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 224 | gsc_to_pci_forwarding(s); |
| 225 | break; |
| 226 | case DINO_IO_CONTROL: |
| 227 | s->io_control = val; |
| 228 | gsc_to_pci_forwarding(s); |
| 229 | break; |
| 230 | |
| 231 | case DINO_IAR0: |
| 232 | s->iar0 = val; |
| 233 | break; |
| 234 | case DINO_IAR1: |
| 235 | s->iar1 = val; |
| 236 | break; |
| 237 | case DINO_IMR: |
| 238 | s->imr = val; |
| 239 | break; |
| 240 | case DINO_ICR: |
| 241 | s->icr = val; |
| 242 | break; |
| 243 | case DINO_IPR: |
| 244 | /* Any write to IPR clears the register. */ |
| 245 | s->ipr = 0; |
| 246 | break; |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 247 | case DINO_TOC_ADDR: |
| 248 | /* IO_COMMAND of CPU with client_id bits */ |
| 249 | s->toc_addr = 0xFFFA0030 | (val & 0x1e000); |
| 250 | break; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 251 | |
| 252 | case DINO_ILR: |
| 253 | case DINO_IRR0: |
| 254 | case DINO_IRR1: |
| 255 | /* These registers are read-only. */ |
| 256 | break; |
| 257 | |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 258 | case DINO_GMASK ... DINO_TLTIM: |
| 259 | i = (addr - DINO_GMASK) / 4; |
| 260 | val &= reg800_keep_bits[i]; |
| 261 | s->reg800[i] = val; |
| 262 | break; |
| 263 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 264 | default: |
| 265 | /* Controlled by dino_chip_mem_valid above. */ |
| 266 | g_assert_not_reached(); |
| 267 | } |
| 268 | return MEMTX_OK; |
| 269 | } |
| 270 | |
| 271 | static const MemoryRegionOps dino_chip_ops = { |
| 272 | .read_with_attrs = dino_chip_read_with_attrs, |
| 273 | .write_with_attrs = dino_chip_write_with_attrs, |
| 274 | .endianness = DEVICE_BIG_ENDIAN, |
| 275 | .valid = { |
| 276 | .min_access_size = 1, |
| 277 | .max_access_size = 4, |
| 278 | .accepts = dino_chip_mem_valid, |
| 279 | }, |
| 280 | .impl = { |
| 281 | .min_access_size = 1, |
| 282 | .max_access_size = 4, |
| 283 | }, |
| 284 | }; |
| 285 | |
| 286 | static const VMStateDescription vmstate_dino = { |
| 287 | .name = "Dino", |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 288 | .version_id = 2, |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 289 | .minimum_version_id = 1, |
| 290 | .fields = (VMStateField[]) { |
| 291 | VMSTATE_UINT32(iar0, DinoState), |
| 292 | VMSTATE_UINT32(iar1, DinoState), |
| 293 | VMSTATE_UINT32(imr, DinoState), |
| 294 | VMSTATE_UINT32(ipr, DinoState), |
| 295 | VMSTATE_UINT32(icr, DinoState), |
| 296 | VMSTATE_UINT32(ilr, DinoState), |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 297 | VMSTATE_UINT32(io_fbb_en, DinoState), |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 298 | VMSTATE_UINT32(io_addr_en, DinoState), |
| 299 | VMSTATE_UINT32(io_control, DinoState), |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 300 | VMSTATE_UINT32(toc_addr, DinoState), |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 301 | VMSTATE_END_OF_LIST() |
| 302 | } |
| 303 | }; |
| 304 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 305 | /* Unlike pci_config_data_le_ops, no check of high bit set in config_reg. */ |
| 306 | |
| 307 | static uint64_t dino_config_data_read(void *opaque, hwaddr addr, unsigned len) |
| 308 | { |
| 309 | PCIHostState *s = opaque; |
| 310 | return pci_data_read(s->bus, s->config_reg | (addr & 3), len); |
| 311 | } |
| 312 | |
| 313 | static void dino_config_data_write(void *opaque, hwaddr addr, |
| 314 | uint64_t val, unsigned len) |
| 315 | { |
| 316 | PCIHostState *s = opaque; |
| 317 | pci_data_write(s->bus, s->config_reg | (addr & 3), val, len); |
| 318 | } |
| 319 | |
| 320 | static const MemoryRegionOps dino_config_data_ops = { |
| 321 | .read = dino_config_data_read, |
| 322 | .write = dino_config_data_write, |
| 323 | .endianness = DEVICE_LITTLE_ENDIAN, |
| 324 | }; |
| 325 | |
Sven Schnelle | 368bec8 | 2019-02-18 19:33:14 +0100 | [diff] [blame] | 326 | static uint64_t dino_config_addr_read(void *opaque, hwaddr addr, unsigned len) |
| 327 | { |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 328 | DinoState *s = opaque; |
| 329 | return s->config_reg_dino; |
Sven Schnelle | 368bec8 | 2019-02-18 19:33:14 +0100 | [diff] [blame] | 330 | } |
| 331 | |
| 332 | static void dino_config_addr_write(void *opaque, hwaddr addr, |
| 333 | uint64_t val, unsigned len) |
| 334 | { |
| 335 | PCIHostState *s = opaque; |
Helge Deller | 1809259 | 2019-12-20 22:15:07 +0100 | [diff] [blame] | 336 | DinoState *ds = opaque; |
| 337 | ds->config_reg_dino = val; /* keep a copy of original value */ |
Sven Schnelle | 368bec8 | 2019-02-18 19:33:14 +0100 | [diff] [blame] | 338 | s->config_reg = val & ~3U; |
| 339 | } |
| 340 | |
| 341 | static const MemoryRegionOps dino_config_addr_ops = { |
| 342 | .read = dino_config_addr_read, |
| 343 | .write = dino_config_addr_write, |
| 344 | .valid.min_access_size = 4, |
| 345 | .valid.max_access_size = 4, |
| 346 | .endianness = DEVICE_BIG_ENDIAN, |
| 347 | }; |
| 348 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 349 | static AddressSpace *dino_pcihost_set_iommu(PCIBus *bus, void *opaque, |
| 350 | int devfn) |
| 351 | { |
| 352 | DinoState *s = opaque; |
| 353 | |
| 354 | return &s->bm_as; |
| 355 | } |
| 356 | |
| 357 | /* |
| 358 | * Dino interrupts are connected as shown on Page 78, Table 23 |
| 359 | * (Little-endian bit numbers) |
| 360 | * 0 PCI INTA |
| 361 | * 1 PCI INTB |
| 362 | * 2 PCI INTC |
| 363 | * 3 PCI INTD |
| 364 | * 4 PCI INTE |
| 365 | * 5 PCI INTF |
| 366 | * 6 GSC External Interrupt |
| 367 | * 7 Bus Error for "less than fatal" mode |
| 368 | * 8 PS2 |
| 369 | * 9 Unused |
| 370 | * 10 RS232 |
| 371 | */ |
| 372 | |
| 373 | static void dino_set_irq(void *opaque, int irq, int level) |
| 374 | { |
| 375 | DinoState *s = opaque; |
| 376 | uint32_t bit = 1u << irq; |
| 377 | uint32_t old_ilr = s->ilr; |
| 378 | |
| 379 | if (level) { |
| 380 | uint32_t ena = bit & ~old_ilr; |
| 381 | s->ipr |= ena; |
| 382 | s->ilr = old_ilr | bit; |
| 383 | if (ena & s->imr) { |
| 384 | uint32_t iar = (ena & s->icr ? s->iar1 : s->iar0); |
| 385 | stl_be_phys(&address_space_memory, iar & -32, iar & 31); |
| 386 | } |
| 387 | } else { |
| 388 | s->ilr = old_ilr & ~bit; |
| 389 | } |
| 390 | } |
| 391 | |
| 392 | static int dino_pci_map_irq(PCIDevice *d, int irq_num) |
| 393 | { |
Philippe Mathieu-Daudé | 8d40def | 2020-10-11 17:04:23 +0200 | [diff] [blame] | 394 | int slot = PCI_SLOT(d->devfn); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 395 | |
| 396 | assert(irq_num >= 0 && irq_num <= 3); |
| 397 | |
Laurent Vivier | 4a4ff4c | 2018-03-23 15:32:02 +0100 | [diff] [blame] | 398 | return slot & 0x03; |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 399 | } |
| 400 | |
Mark Cave-Ayland | 98d168f | 2022-05-04 10:25:18 +0100 | [diff] [blame] | 401 | static void dino_pcihost_reset(DeviceState *dev) |
| 402 | { |
| 403 | DinoState *s = DINO_PCI_HOST_BRIDGE(dev); |
| 404 | |
Mark Cave-Ayland | 2fb11c7 | 2022-05-04 10:25:30 +0100 | [diff] [blame] | 405 | s->iar0 = s->iar1 = 0xFFFB0000 + 3; /* CPU_HPA + 3 */ |
Mark Cave-Ayland | 98d168f | 2022-05-04 10:25:18 +0100 | [diff] [blame] | 406 | s->toc_addr = 0xFFFA0030; /* IO_COMMAND of CPU */ |
| 407 | } |
| 408 | |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 409 | static void dino_pcihost_realize(DeviceState *dev, Error **errp) |
| 410 | { |
| 411 | DinoState *s = DINO_PCI_HOST_BRIDGE(dev); |
| 412 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 413 | /* Set up PCI view of memory: Bus master address space. */ |
Philippe Mathieu-Daudé | 94c1253 | 2020-06-01 16:29:28 +0200 | [diff] [blame] | 414 | memory_region_init(&s->bm, OBJECT(s), "bm-dino", 4 * GiB); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 415 | memory_region_init_alias(&s->bm_ram_alias, OBJECT(s), |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 416 | "bm-system", s->memory_as, 0, |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 417 | 0xf0000000 + DINO_MEM_CHUNK_SIZE); |
| 418 | memory_region_init_alias(&s->bm_pci_alias, OBJECT(s), |
| 419 | "bm-pci", &s->pci_mem, |
| 420 | 0xf0000000 + DINO_MEM_CHUNK_SIZE, |
Sven Schnelle | cb82c57 | 2019-02-11 20:20:39 +0100 | [diff] [blame] | 421 | 30 * DINO_MEM_CHUNK_SIZE); |
| 422 | memory_region_init_alias(&s->bm_cpu_alias, OBJECT(s), |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 423 | "bm-cpu", s->memory_as, 0xfff00000, |
Sven Schnelle | cb82c57 | 2019-02-11 20:20:39 +0100 | [diff] [blame] | 424 | 0xfffff); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 425 | memory_region_add_subregion(&s->bm, 0, |
| 426 | &s->bm_ram_alias); |
| 427 | memory_region_add_subregion(&s->bm, |
| 428 | 0xf0000000 + DINO_MEM_CHUNK_SIZE, |
| 429 | &s->bm_pci_alias); |
Sven Schnelle | cb82c57 | 2019-02-11 20:20:39 +0100 | [diff] [blame] | 430 | memory_region_add_subregion(&s->bm, 0xfff00000, |
| 431 | &s->bm_cpu_alias); |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 432 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 433 | address_space_init(&s->bm_as, &s->bm, "pci-bm"); |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 434 | } |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 435 | |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 436 | static void dino_pcihost_unrealize(DeviceState *dev) |
| 437 | { |
| 438 | DinoState *s = DINO_PCI_HOST_BRIDGE(dev); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 439 | |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 440 | address_space_destroy(&s->bm_as); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 441 | } |
| 442 | |
Mark Cave-Ayland | 7cdfa94 | 2022-05-04 10:25:12 +0100 | [diff] [blame] | 443 | static void dino_pcihost_init(Object *obj) |
| 444 | { |
| 445 | DinoState *s = DINO_PCI_HOST_BRIDGE(obj); |
| 446 | PCIHostState *phb = PCI_HOST_BRIDGE(obj); |
| 447 | SysBusDevice *sbd = SYS_BUS_DEVICE(obj); |
Mark Cave-Ayland | cc363c4 | 2022-05-04 10:25:14 +0100 | [diff] [blame] | 448 | int i; |
Mark Cave-Ayland | 7cdfa94 | 2022-05-04 10:25:12 +0100 | [diff] [blame] | 449 | |
| 450 | /* Dino PCI access from main memory. */ |
| 451 | memory_region_init_io(&s->this_mem, OBJECT(s), &dino_chip_ops, |
| 452 | s, "dino", 4096); |
| 453 | |
| 454 | /* Dino PCI config. */ |
| 455 | memory_region_init_io(&phb->conf_mem, OBJECT(phb), |
| 456 | &dino_config_addr_ops, DEVICE(s), |
| 457 | "pci-conf-idx", 4); |
| 458 | memory_region_init_io(&phb->data_mem, OBJECT(phb), |
| 459 | &dino_config_data_ops, DEVICE(s), |
| 460 | "pci-conf-data", 4); |
| 461 | memory_region_add_subregion(&s->this_mem, DINO_PCI_CONFIG_ADDR, |
| 462 | &phb->conf_mem); |
| 463 | memory_region_add_subregion(&s->this_mem, DINO_CONFIG_DATA, |
| 464 | &phb->data_mem); |
| 465 | |
Mark Cave-Ayland | 63901b6 | 2022-05-04 10:25:13 +0100 | [diff] [blame] | 466 | /* Dino PCI bus memory. */ |
| 467 | memory_region_init(&s->pci_mem, OBJECT(s), "pci-memory", 4 * GiB); |
| 468 | |
| 469 | phb->bus = pci_register_root_bus(DEVICE(s), "pci", |
| 470 | dino_set_irq, dino_pci_map_irq, s, |
| 471 | &s->pci_mem, get_system_io(), |
| 472 | PCI_DEVFN(0, 0), 32, TYPE_PCI_BUS); |
| 473 | |
Mark Cave-Ayland | cc363c4 | 2022-05-04 10:25:14 +0100 | [diff] [blame] | 474 | /* Set up windows into PCI bus memory. */ |
| 475 | for (i = 1; i < 31; i++) { |
| 476 | uint32_t addr = 0xf0000000 + i * DINO_MEM_CHUNK_SIZE; |
| 477 | char *name = g_strdup_printf("PCI Outbound Window %d", i); |
| 478 | memory_region_init_alias(&s->pci_mem_alias[i], OBJECT(s), |
| 479 | name, &s->pci_mem, addr, |
| 480 | DINO_MEM_CHUNK_SIZE); |
| 481 | g_free(name); |
| 482 | } |
| 483 | |
Mark Cave-Ayland | 5ac6c43 | 2022-05-04 10:25:16 +0100 | [diff] [blame] | 484 | pci_setup_iommu(phb->bus, dino_pcihost_set_iommu, s); |
| 485 | |
Mark Cave-Ayland | 7cdfa94 | 2022-05-04 10:25:12 +0100 | [diff] [blame] | 486 | sysbus_init_mmio(sbd, &s->this_mem); |
Mark Cave-Ayland | 4b5faaf | 2022-05-04 10:25:26 +0100 | [diff] [blame] | 487 | |
| 488 | qdev_init_gpio_in(DEVICE(obj), dino_set_irq, DINO_IRQS); |
Mark Cave-Ayland | 7cdfa94 | 2022-05-04 10:25:12 +0100 | [diff] [blame] | 489 | } |
| 490 | |
Mark Cave-Ayland | 270b295 | 2022-05-04 10:25:15 +0100 | [diff] [blame] | 491 | static Property dino_pcihost_properties[] = { |
| 492 | DEFINE_PROP_LINK("memory-as", DinoState, memory_as, TYPE_MEMORY_REGION, |
| 493 | MemoryRegion *), |
| 494 | DEFINE_PROP_END_OF_LIST(), |
| 495 | }; |
| 496 | |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 497 | static void dino_pcihost_class_init(ObjectClass *klass, void *data) |
| 498 | { |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 499 | DeviceClass *dc = DEVICE_CLASS(klass); |
| 500 | |
Mark Cave-Ayland | 98d168f | 2022-05-04 10:25:18 +0100 | [diff] [blame] | 501 | dc->reset = dino_pcihost_reset; |
Mark Cave-Ayland | 9cf69f4 | 2022-05-04 10:25:17 +0100 | [diff] [blame] | 502 | dc->realize = dino_pcihost_realize; |
| 503 | dc->unrealize = dino_pcihost_unrealize; |
Mark Cave-Ayland | 270b295 | 2022-05-04 10:25:15 +0100 | [diff] [blame] | 504 | device_class_set_props(dc, dino_pcihost_properties); |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 505 | dc->vmsd = &vmstate_dino; |
| 506 | } |
| 507 | |
| 508 | static const TypeInfo dino_pcihost_info = { |
| 509 | .name = TYPE_DINO_PCI_HOST_BRIDGE, |
| 510 | .parent = TYPE_PCI_HOST_BRIDGE, |
Mark Cave-Ayland | 7cdfa94 | 2022-05-04 10:25:12 +0100 | [diff] [blame] | 511 | .instance_init = dino_pcihost_init, |
Helge Deller | a72bd60 | 2017-10-08 16:47:27 -0400 | [diff] [blame] | 512 | .instance_size = sizeof(DinoState), |
| 513 | .class_init = dino_pcihost_class_init, |
| 514 | }; |
| 515 | |
| 516 | static void dino_register_types(void) |
| 517 | { |
| 518 | type_register_static(&dino_pcihost_info); |
| 519 | } |
| 520 | |
| 521 | type_init(dino_register_types) |