]> git.proxmox.com Git - qemu.git/blame - hw/apb_pci.c
Sparc64: use firmware configuration device for command line storage
[qemu.git] / hw / apb_pci.c
CommitLineData
502a5395
PB
1/*
2 * QEMU Ultrasparc APB PCI host
3 *
4 * Copyright (c) 2006 Fabrice Bellard
5fafdf24 5 *
502a5395
PB
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 * THE SOFTWARE.
23 */
80b3ada7 24
a94fd955 25/* XXX This file and most of its contents are somewhat misnamed. The
80b3ada7
PB
26 Ultrasparc PCI host is called the PCI Bus Module (PBM). The APB is
27 the secondary PCI bridge. */
28
72f44c8c 29#include "sysbus.h"
87ecb68b 30#include "pci.h"
4f5e19e6 31#include "pci_host.h"
18e08a55 32#include "apb_pci.h"
a94fd955
BS
33
34/* debug APB */
35//#define DEBUG_APB
36
37#ifdef DEBUG_APB
001faf32
BS
38#define APB_DPRINTF(fmt, ...) \
39do { printf("APB: " fmt , ## __VA_ARGS__); } while (0)
a94fd955 40#else
001faf32 41#define APB_DPRINTF(fmt, ...)
a94fd955
BS
42#endif
43
930f3fe1
BS
44/*
45 * Chipset docs:
46 * PBM: "UltraSPARC IIi User's Manual",
47 * http://www.sun.com/processors/manuals/805-0087.pdf
48 *
49 * APB: "Advanced PCI Bridge (APB) User's Manual",
50 * http://www.sun.com/processors/manuals/805-1251.pdf
51 */
52
72f44c8c
BS
53typedef struct APBState {
54 SysBusDevice busdev;
55 PCIHostState host_state;
56} APBState;
502a5395 57
c227f099 58static void apb_config_writel (void *opaque, target_phys_addr_t addr,
f930d07e 59 uint32_t val)
502a5395
PB
60{
61 //PCIBus *s = opaque;
62
63 switch (addr & 0x3f) {
64 case 0x00: // Control/Status
65 case 0x10: // AFSR
66 case 0x18: // AFAR
67 case 0x20: // Diagnostic
68 case 0x28: // Target address space
f930d07e 69 // XXX
502a5395 70 default:
f930d07e 71 break;
502a5395
PB
72 }
73}
74
75static uint32_t apb_config_readl (void *opaque,
c227f099 76 target_phys_addr_t addr)
502a5395
PB
77{
78 //PCIBus *s = opaque;
79 uint32_t val;
80
81 switch (addr & 0x3f) {
82 case 0x00: // Control/Status
83 case 0x10: // AFSR
84 case 0x18: // AFAR
85 case 0x20: // Diagnostic
86 case 0x28: // Target address space
f930d07e 87 // XXX
502a5395 88 default:
f930d07e
BS
89 val = 0;
90 break;
502a5395
PB
91 }
92 return val;
93}
94
d60efc6b 95static CPUWriteMemoryFunc * const apb_config_write[] = {
502a5395
PB
96 &apb_config_writel,
97 &apb_config_writel,
98 &apb_config_writel,
99};
100
d60efc6b 101static CPUReadMemoryFunc * const apb_config_read[] = {
502a5395
PB
102 &apb_config_readl,
103 &apb_config_readl,
104 &apb_config_readl,
105};
106
c227f099 107static void pci_apb_iowriteb (void *opaque, target_phys_addr_t addr,
502a5395
PB
108 uint32_t val)
109{
afcea8cb 110 cpu_outb(addr & IOPORTS_MASK, val);
502a5395
PB
111}
112
c227f099 113static void pci_apb_iowritew (void *opaque, target_phys_addr_t addr,
502a5395
PB
114 uint32_t val)
115{
afcea8cb 116 cpu_outw(addr & IOPORTS_MASK, val);
502a5395
PB
117}
118
c227f099 119static void pci_apb_iowritel (void *opaque, target_phys_addr_t addr,
502a5395
PB
120 uint32_t val)
121{
afcea8cb 122 cpu_outl(addr & IOPORTS_MASK, val);
502a5395
PB
123}
124
c227f099 125static uint32_t pci_apb_ioreadb (void *opaque, target_phys_addr_t addr)
502a5395
PB
126{
127 uint32_t val;
128
afcea8cb 129 val = cpu_inb(addr & IOPORTS_MASK);
502a5395
PB
130 return val;
131}
132
c227f099 133static uint32_t pci_apb_ioreadw (void *opaque, target_phys_addr_t addr)
502a5395
PB
134{
135 uint32_t val;
136
afcea8cb 137 val = cpu_inw(addr & IOPORTS_MASK);
502a5395
PB
138 return val;
139}
140
c227f099 141static uint32_t pci_apb_ioreadl (void *opaque, target_phys_addr_t addr)
502a5395
PB
142{
143 uint32_t val;
144
afcea8cb 145 val = cpu_inl(addr & IOPORTS_MASK);
502a5395
PB
146 return val;
147}
148
d60efc6b 149static CPUWriteMemoryFunc * const pci_apb_iowrite[] = {
502a5395
PB
150 &pci_apb_iowriteb,
151 &pci_apb_iowritew,
152 &pci_apb_iowritel,
153};
154
d60efc6b 155static CPUReadMemoryFunc * const pci_apb_ioread[] = {
502a5395
PB
156 &pci_apb_ioreadb,
157 &pci_apb_ioreadw,
158 &pci_apb_ioreadl,
159};
160
80b3ada7 161/* The APB host has an IRQ line for each IRQ line of each slot. */
d2b59317 162static int pci_apb_map_irq(PCIDevice *pci_dev, int irq_num)
502a5395 163{
80b3ada7
PB
164 return ((pci_dev->devfn & 0x18) >> 1) + irq_num;
165}
166
167static int pci_pbm_map_irq(PCIDevice *pci_dev, int irq_num)
168{
169 int bus_offset;
170 if (pci_dev->devfn & 1)
171 bus_offset = 16;
172 else
173 bus_offset = 0;
174 return bus_offset + irq_num;
d2b59317
PB
175}
176
5d4e84c8 177static void pci_apb_set_irq(void *opaque, int irq_num, int level)
d2b59317 178{
5d4e84c8
JQ
179 qemu_irq *pic = opaque;
180
80b3ada7 181 /* PCI IRQ map onto the first 32 INO. */
d537cf6c 182 qemu_set_irq(pic[irq_num], level);
502a5395
PB
183}
184
d6318738
MT
185static void apb_pci_bridge_init(PCIBus *b)
186{
187 PCIDevice *dev = pci_bridge_get_device(b);
188
189 /*
190 * command register:
191 * According to PCI bridge spec, after reset
192 * bus master bit is off
193 * memory space enable bit is off
194 * According to manual (805-1251.pdf).
195 * the reset value should be zero unless the boot pin is tied high
196 * (which is true) and thus it should be PCI_COMMAND_MEMORY.
197 */
198 pci_set_word(dev->config + PCI_COMMAND,
199 PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER);
200 dev->config[PCI_LATENCY_TIMER] = 0x10;
201 dev->config[PCI_HEADER_TYPE] |= PCI_HEADER_TYPE_MULTI_FUNCTION;
202}
203
c227f099
AL
204PCIBus *pci_apb_init(target_phys_addr_t special_base,
205 target_phys_addr_t mem_base,
c190ea07 206 qemu_irq *pic, PCIBus **bus2, PCIBus **bus3)
502a5395 207{
72f44c8c
BS
208 DeviceState *dev;
209 SysBusDevice *s;
210 APBState *d;
502a5395 211
80b3ada7 212 /* Ultrasparc PBM main bus */
72f44c8c 213 dev = qdev_create(NULL, "pbm");
e23a1b33 214 qdev_init_nofail(dev);
72f44c8c
BS
215 s = sysbus_from_qdev(dev);
216 /* apb_config */
217 sysbus_mmio_map(s, 0, special_base + 0x2000ULL);
218 /* pci_ioport */
219 sysbus_mmio_map(s, 1, special_base + 0x2000000ULL);
220 /* mem_config: XXX size should be 4G-prom */
221 sysbus_mmio_map(s, 2, special_base + 0x1000000ULL);
222 /* mem_data */
223 sysbus_mmio_map(s, 3, mem_base);
224 d = FROM_SYSBUS(APBState, s);
c5ff6d54 225 d->host_state.bus = pci_register_bus(&d->busdev.qdev, "pci",
72f44c8c
BS
226 pci_apb_set_irq, pci_pbm_map_irq, pic,
227 0, 32);
f6b6f1bc
BS
228 pci_bus_set_mem_base(d->host_state.bus, mem_base);
229
72f44c8c
BS
230 pci_create_simple(d->host_state.bus, 0, "pbm");
231 /* APB secondary busses */
2217dcff
IY
232 *bus2 = pci_bridge_init(d->host_state.bus, PCI_DEVFN(1, 0),
233 PCI_VENDOR_ID_SUN, PCI_DEVICE_ID_SUN_SIMBA,
234 pci_apb_map_irq,
72f44c8c 235 "Advanced PCI Bus secondary bridge 1");
d6318738
MT
236 apb_pci_bridge_init(*bus2);
237
2217dcff
IY
238 *bus3 = pci_bridge_init(d->host_state.bus, PCI_DEVFN(1, 1),
239 PCI_VENDOR_ID_SUN, PCI_DEVICE_ID_SUN_SIMBA,
240 pci_apb_map_irq,
72f44c8c 241 "Advanced PCI Bus secondary bridge 2");
d6318738 242 apb_pci_bridge_init(*bus3);
502a5395 243
72f44c8c
BS
244 return d->host_state.bus;
245}
246
81a322d4 247static int pci_pbm_init_device(SysBusDevice *dev)
72f44c8c
BS
248{
249
250 APBState *s;
251 int pci_mem_config, pci_mem_data, apb_config, pci_ioport;
252
253 s = FROM_SYSBUS(APBState, dev);
254 /* apb_config */
1eed09cb 255 apb_config = cpu_register_io_memory(apb_config_read,
f930d07e 256 apb_config_write, s);
72f44c8c
BS
257 sysbus_init_mmio(dev, 0x40ULL, apb_config);
258 /* pci_ioport */
1eed09cb 259 pci_ioport = cpu_register_io_memory(pci_apb_ioread,
502a5395 260 pci_apb_iowrite, s);
72f44c8c
BS
261 sysbus_init_mmio(dev, 0x10000ULL, pci_ioport);
262 /* mem_config */
f08b32fe 263 pci_mem_config = pci_host_conf_register_mmio(&s->host_state);
72f44c8c
BS
264 sysbus_init_mmio(dev, 0x10ULL, pci_mem_config);
265 /* mem_data */
f08b32fe 266 pci_mem_data = pci_host_data_register_mmio(&s->host_state);
72f44c8c 267 sysbus_init_mmio(dev, 0x10000000ULL, pci_mem_data);
81a322d4 268 return 0;
72f44c8c 269}
502a5395 270
81a322d4 271static int pbm_pci_host_init(PCIDevice *d)
72f44c8c 272{
deb54399
AL
273 pci_config_set_vendor_id(d->config, PCI_VENDOR_ID_SUN);
274 pci_config_set_device_id(d->config, PCI_DEVICE_ID_SUN_SABRE);
502a5395
PB
275 d->config[0x04] = 0x06; // command = bus master, pci mem
276 d->config[0x05] = 0x00;
277 d->config[0x06] = 0xa0; // status = fast back-to-back, 66MHz, no error
278 d->config[0x07] = 0x03; // status = medium devsel
279 d->config[0x08] = 0x00; // revision
280 d->config[0x09] = 0x00; // programming i/f
173a543b 281 pci_config_set_class(d->config, PCI_CLASS_BRIDGE_HOST);
502a5395 282 d->config[0x0D] = 0x10; // latency_timer
110c50fd 283 d->config[PCI_HEADER_TYPE] = PCI_HEADER_TYPE_NORMAL; // header_type
81a322d4 284 return 0;
72f44c8c 285}
80b3ada7 286
72f44c8c
BS
287static PCIDeviceInfo pbm_pci_host_info = {
288 .qdev.name = "pbm",
289 .qdev.size = sizeof(PCIDevice),
290 .init = pbm_pci_host_init,
291};
292
293static void pbm_register_devices(void)
294{
295 sysbus_register_dev("pbm", sizeof(APBState), pci_pbm_init_device);
296 pci_qdev_register(&pbm_pci_host_info);
502a5395 297}
72f44c8c
BS
298
299device_init(pbm_register_devices)