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
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. */
31 typedef target_phys_addr_t pci_addr_t
;
34 typedef PCIHostState APBState
;
36 static void pci_apb_config_writel (void *opaque
, target_phys_addr_t addr
,
42 for (i
= 11; i
< 32; i
++) {
43 if ((val
& (1 << i
)) != 0)
46 s
->config_reg
= (1 << 16) | (val
& 0x7FC) | (i
<< 11);
49 static uint32_t pci_apb_config_readl (void *opaque
,
50 target_phys_addr_t addr
)
56 devfn
= (s
->config_reg
>> 8) & 0xFF;
57 val
= (1 << (devfn
>> 3)) | ((devfn
& 0x07) << 8) | (s
->config_reg
& 0xFC);
61 static CPUWriteMemoryFunc
*pci_apb_config_write
[] = {
62 &pci_apb_config_writel
,
63 &pci_apb_config_writel
,
64 &pci_apb_config_writel
,
67 static CPUReadMemoryFunc
*pci_apb_config_read
[] = {
68 &pci_apb_config_readl
,
69 &pci_apb_config_readl
,
70 &pci_apb_config_readl
,
73 static void apb_config_writel (void *opaque
, target_phys_addr_t addr
,
78 switch (addr
& 0x3f) {
79 case 0x00: // Control/Status
82 case 0x20: // Diagnostic
83 case 0x28: // Target address space
90 static uint32_t apb_config_readl (void *opaque
,
91 target_phys_addr_t addr
)
96 switch (addr
& 0x3f) {
97 case 0x00: // Control/Status
100 case 0x20: // Diagnostic
101 case 0x28: // Target address space
110 static CPUWriteMemoryFunc
*apb_config_write
[] = {
116 static CPUReadMemoryFunc
*apb_config_read
[] = {
122 static CPUWriteMemoryFunc
*pci_apb_write
[] = {
123 &pci_host_data_writeb
,
124 &pci_host_data_writew
,
125 &pci_host_data_writel
,
128 static CPUReadMemoryFunc
*pci_apb_read
[] = {
129 &pci_host_data_readb
,
130 &pci_host_data_readw
,
131 &pci_host_data_readl
,
134 static void pci_apb_iowriteb (void *opaque
, target_phys_addr_t addr
,
137 cpu_outb(NULL
, addr
& 0xffff, val
);
140 static void pci_apb_iowritew (void *opaque
, target_phys_addr_t addr
,
143 cpu_outw(NULL
, addr
& 0xffff, val
);
146 static void pci_apb_iowritel (void *opaque
, target_phys_addr_t addr
,
149 cpu_outl(NULL
, addr
& 0xffff, val
);
152 static uint32_t pci_apb_ioreadb (void *opaque
, target_phys_addr_t addr
)
156 val
= cpu_inb(NULL
, addr
& 0xffff);
160 static uint32_t pci_apb_ioreadw (void *opaque
, target_phys_addr_t addr
)
164 val
= cpu_inw(NULL
, addr
& 0xffff);
168 static uint32_t pci_apb_ioreadl (void *opaque
, target_phys_addr_t addr
)
172 val
= cpu_inl(NULL
, addr
& 0xffff);
176 static CPUWriteMemoryFunc
*pci_apb_iowrite
[] = {
182 static CPUReadMemoryFunc
*pci_apb_ioread
[] = {
188 /* The APB host has an IRQ line for each IRQ line of each slot. */
189 static int pci_apb_map_irq(PCIDevice
*pci_dev
, int irq_num
)
191 return ((pci_dev
->devfn
& 0x18) >> 1) + irq_num
;
194 static int pci_pbm_map_irq(PCIDevice
*pci_dev
, int irq_num
)
197 if (pci_dev
->devfn
& 1)
201 return bus_offset
+ irq_num
;
204 static void pci_apb_set_irq(qemu_irq
*pic
, int irq_num
, int level
)
206 /* PCI IRQ map onto the first 32 INO. */
207 qemu_set_irq(pic
[irq_num
], level
);
210 PCIBus
*pci_apb_init(target_phys_addr_t special_base
,
211 target_phys_addr_t mem_base
,
216 int pci_mem_config
, pci_mem_data
, apb_config
, pci_ioport
;
219 s
= qemu_mallocz(sizeof(APBState
));
220 /* Ultrasparc PBM main bus */
221 s
->bus
= pci_register_bus(pci_apb_set_irq
, pci_pbm_map_irq
, pic
, 0, 32);
223 pci_mem_config
= cpu_register_io_memory(0, pci_apb_config_read
,
224 pci_apb_config_write
, s
);
225 apb_config
= cpu_register_io_memory(0, apb_config_read
,
226 apb_config_write
, s
);
227 pci_mem_data
= cpu_register_io_memory(0, pci_apb_read
,
229 pci_ioport
= cpu_register_io_memory(0, pci_apb_ioread
,
232 cpu_register_physical_memory(special_base
+ 0x2000ULL
, 0x40, apb_config
);
233 cpu_register_physical_memory(special_base
+ 0x1000000ULL
, 0x10, pci_mem_config
);
234 cpu_register_physical_memory(special_base
+ 0x2000000ULL
, 0x10000, pci_ioport
);
235 cpu_register_physical_memory(mem_base
, 0x10000000, pci_mem_data
); // XXX size should be 4G-prom
237 d
= pci_register_device(s
->bus
, "Advanced PCI Bus", sizeof(PCIDevice
),
239 d
->config
[0x00] = 0x8e; // vendor_id : Sun
240 d
->config
[0x01] = 0x10;
241 d
->config
[0x02] = 0x00; // device_id
242 d
->config
[0x03] = 0xa0;
243 d
->config
[0x04] = 0x06; // command = bus master, pci mem
244 d
->config
[0x05] = 0x00;
245 d
->config
[0x06] = 0xa0; // status = fast back-to-back, 66MHz, no error
246 d
->config
[0x07] = 0x03; // status = medium devsel
247 d
->config
[0x08] = 0x00; // revision
248 d
->config
[0x09] = 0x00; // programming i/f
249 d
->config
[0x0A] = 0x00; // class_sub = pci host
250 d
->config
[0x0B] = 0x06; // class_base = PCI_bridge
251 d
->config
[0x0D] = 0x10; // latency_timer
252 d
->config
[0x0E] = 0x00; // header_type
254 /* APB secondary busses */
255 secondary
= pci_bridge_init(s
->bus
, 8, 0x108e5000, pci_apb_map_irq
, "Advanced PCI Bus secondary bridge 1");
256 pci_bridge_init(s
->bus
, 9, 0x108e5000, pci_apb_map_irq
, "Advanced PCI Bus secondary bridge 2");