keep initrd in below 4g area.
[qemu.git] / hw / apb_pci.c
blobd6ce9f4aef51f6f8d937afc9302f1f57b65dabe5
1 /*
2 * QEMU Ultrasparc APB PCI host
4 * Copyright (c) 2006 Fabrice Bellard
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:
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
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.
25 /* XXX This file and most of its contents are somewhat misnamed. The
26 Ultrasparc PCI host is called the PCI Bus Module (PBM). The APB is
27 the secondary PCI bridge. */
29 #include "hw.h"
30 #include "pci.h"
32 /* debug APB */
33 //#define DEBUG_APB
35 #ifdef DEBUG_APB
36 #define APB_DPRINTF(fmt, ...) \
37 do { printf("APB: " fmt , ## __VA_ARGS__); } while (0)
38 #else
39 #define APB_DPRINTF(fmt, ...)
40 #endif
42 typedef target_phys_addr_t pci_addr_t;
43 #include "pci_host.h"
45 typedef PCIHostState APBState;
47 static void pci_apb_config_writel (void *opaque, target_phys_addr_t addr,
48 uint32_t val)
50 APBState *s = opaque;
52 #ifdef TARGET_WORDS_BIGENDIAN
53 val = bswap32(val);
54 #endif
55 APB_DPRINTF("config_writel addr " TARGET_FMT_plx " val %x\n", addr,
56 val);
57 s->config_reg = val;
60 static uint32_t pci_apb_config_readl (void *opaque,
61 target_phys_addr_t addr)
63 APBState *s = opaque;
64 uint32_t val;
66 val = s->config_reg;
67 #ifdef TARGET_WORDS_BIGENDIAN
68 val = bswap32(val);
69 #endif
70 APB_DPRINTF("config_readl addr " TARGET_FMT_plx " val %x\n", addr,
71 val);
72 return val;
75 static CPUWriteMemoryFunc *pci_apb_config_write[] = {
76 &pci_apb_config_writel,
77 &pci_apb_config_writel,
78 &pci_apb_config_writel,
81 static CPUReadMemoryFunc *pci_apb_config_read[] = {
82 &pci_apb_config_readl,
83 &pci_apb_config_readl,
84 &pci_apb_config_readl,
87 static void apb_config_writel (void *opaque, target_phys_addr_t addr,
88 uint32_t val)
90 //PCIBus *s = opaque;
92 switch (addr & 0x3f) {
93 case 0x00: // Control/Status
94 case 0x10: // AFSR
95 case 0x18: // AFAR
96 case 0x20: // Diagnostic
97 case 0x28: // Target address space
98 // XXX
99 default:
100 break;
104 static uint32_t apb_config_readl (void *opaque,
105 target_phys_addr_t addr)
107 //PCIBus *s = opaque;
108 uint32_t val;
110 switch (addr & 0x3f) {
111 case 0x00: // Control/Status
112 case 0x10: // AFSR
113 case 0x18: // AFAR
114 case 0x20: // Diagnostic
115 case 0x28: // Target address space
116 // XXX
117 default:
118 val = 0;
119 break;
121 return val;
124 static CPUWriteMemoryFunc *apb_config_write[] = {
125 &apb_config_writel,
126 &apb_config_writel,
127 &apb_config_writel,
130 static CPUReadMemoryFunc *apb_config_read[] = {
131 &apb_config_readl,
132 &apb_config_readl,
133 &apb_config_readl,
136 static CPUWriteMemoryFunc *pci_apb_write[] = {
137 &pci_host_data_writeb,
138 &pci_host_data_writew,
139 &pci_host_data_writel,
142 static CPUReadMemoryFunc *pci_apb_read[] = {
143 &pci_host_data_readb,
144 &pci_host_data_readw,
145 &pci_host_data_readl,
148 static void pci_apb_iowriteb (void *opaque, target_phys_addr_t addr,
149 uint32_t val)
151 cpu_outb(NULL, addr & 0xffff, val);
154 static void pci_apb_iowritew (void *opaque, target_phys_addr_t addr,
155 uint32_t val)
157 cpu_outw(NULL, addr & 0xffff, val);
160 static void pci_apb_iowritel (void *opaque, target_phys_addr_t addr,
161 uint32_t val)
163 cpu_outl(NULL, addr & 0xffff, val);
166 static uint32_t pci_apb_ioreadb (void *opaque, target_phys_addr_t addr)
168 uint32_t val;
170 val = cpu_inb(NULL, addr & 0xffff);
171 return val;
174 static uint32_t pci_apb_ioreadw (void *opaque, target_phys_addr_t addr)
176 uint32_t val;
178 val = cpu_inw(NULL, addr & 0xffff);
179 return val;
182 static uint32_t pci_apb_ioreadl (void *opaque, target_phys_addr_t addr)
184 uint32_t val;
186 val = cpu_inl(NULL, addr & 0xffff);
187 return val;
190 static CPUWriteMemoryFunc *pci_apb_iowrite[] = {
191 &pci_apb_iowriteb,
192 &pci_apb_iowritew,
193 &pci_apb_iowritel,
196 static CPUReadMemoryFunc *pci_apb_ioread[] = {
197 &pci_apb_ioreadb,
198 &pci_apb_ioreadw,
199 &pci_apb_ioreadl,
202 /* The APB host has an IRQ line for each IRQ line of each slot. */
203 static int pci_apb_map_irq(PCIDevice *pci_dev, int irq_num)
205 return ((pci_dev->devfn & 0x18) >> 1) + irq_num;
208 static int pci_pbm_map_irq(PCIDevice *pci_dev, int irq_num)
210 int bus_offset;
211 if (pci_dev->devfn & 1)
212 bus_offset = 16;
213 else
214 bus_offset = 0;
215 return bus_offset + irq_num;
218 static void pci_apb_set_irq(qemu_irq *pic, int irq_num, int level)
220 /* PCI IRQ map onto the first 32 INO. */
221 qemu_set_irq(pic[irq_num], level);
224 PCIBus *pci_apb_init(target_phys_addr_t special_base,
225 target_phys_addr_t mem_base,
226 qemu_irq *pic, PCIBus **bus2, PCIBus **bus3)
228 APBState *s;
229 PCIDevice *d;
230 int pci_mem_config, pci_mem_data, apb_config, pci_ioport;
232 s = qemu_mallocz(sizeof(APBState));
233 /* Ultrasparc PBM main bus */
234 s->bus = pci_register_bus(pci_apb_set_irq, pci_pbm_map_irq, pic, 0, 32);
236 pci_mem_config = cpu_register_io_memory(0, pci_apb_config_read,
237 pci_apb_config_write, s);
238 apb_config = cpu_register_io_memory(0, apb_config_read,
239 apb_config_write, s);
240 pci_mem_data = cpu_register_io_memory(0, pci_apb_read,
241 pci_apb_write, s);
242 pci_ioport = cpu_register_io_memory(0, pci_apb_ioread,
243 pci_apb_iowrite, s);
245 cpu_register_physical_memory(special_base + 0x2000ULL, 0x40, apb_config);
246 cpu_register_physical_memory(special_base + 0x1000000ULL, 0x10,
247 pci_mem_config);
248 cpu_register_physical_memory(special_base + 0x2000000ULL, 0x10000,
249 pci_ioport);
250 cpu_register_physical_memory(mem_base, 0x10000000,
251 pci_mem_data); // XXX size should be 4G-prom
253 d = pci_register_device(s->bus, "Advanced PCI Bus", sizeof(PCIDevice),
254 0, NULL, NULL);
255 pci_config_set_vendor_id(d->config, PCI_VENDOR_ID_SUN);
256 pci_config_set_device_id(d->config, PCI_DEVICE_ID_SUN_SABRE);
257 d->config[0x04] = 0x06; // command = bus master, pci mem
258 d->config[0x05] = 0x00;
259 d->config[0x06] = 0xa0; // status = fast back-to-back, 66MHz, no error
260 d->config[0x07] = 0x03; // status = medium devsel
261 d->config[0x08] = 0x00; // revision
262 d->config[0x09] = 0x00; // programming i/f
263 pci_config_set_class(d->config, PCI_CLASS_BRIDGE_HOST);
264 d->config[0x0D] = 0x10; // latency_timer
265 d->config[PCI_HEADER_TYPE] = PCI_HEADER_TYPE_NORMAL; // header_type
267 /* APB secondary busses */
268 *bus2 = pci_bridge_init(s->bus, 8, PCI_VENDOR_ID_SUN,
269 PCI_DEVICE_ID_SUN_SIMBA, pci_apb_map_irq,
270 "Advanced PCI Bus secondary bridge 1");
271 *bus3 = pci_bridge_init(s->bus, 9, PCI_VENDOR_ID_SUN,
272 PCI_DEVICE_ID_SUN_SIMBA, pci_apb_map_irq,
273 "Advanced PCI Bus secondary bridge 2");
274 return s->bus;