2 * QEMU PCI VGA Emulator.
4 * Copyright (c) 2003 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
29 #include "pixel_ops.h"
30 #include "qemu-timer.h"
33 typedef struct PCIVGAState
{
38 static const VMStateDescription vmstate_vga_pci
= {
41 .minimum_version_id
= 2,
42 .minimum_version_id_old
= 2,
43 .fields
= (VMStateField
[]) {
44 VMSTATE_PCI_DEVICE(dev
, PCIVGAState
),
45 VMSTATE_STRUCT(vga
, PCIVGAState
, 0, vmstate_vga_common
, VGACommonState
),
50 static void vga_map(PCIDevice
*pci_dev
, int region_num
,
51 pcibus_t addr
, pcibus_t size
, int type
)
53 PCIVGAState
*d
= (PCIVGAState
*)pci_dev
;
54 VGACommonState
*s
= &d
->vga
;
55 if (region_num
== PCI_ROM_SLOT
) {
56 cpu_register_physical_memory(addr
, s
->bios_size
, s
->bios_offset
);
58 cpu_register_physical_memory(addr
, s
->vram_size
, s
->vram_offset
);
60 s
->map_end
= addr
+ s
->vram_size
;
61 vga_dirty_log_start(s
);
65 static void pci_vga_write_config(PCIDevice
*d
,
66 uint32_t address
, uint32_t val
, int len
)
68 PCIVGAState
*pvs
= container_of(d
, PCIVGAState
, dev
);
69 VGACommonState
*s
= &pvs
->vga
;
71 vga_dirty_log_stop(s
);
72 pci_default_write_config(d
, address
, val
, len
);
73 if (s
->map_addr
&& pvs
->dev
.io_regions
[0].addr
== -1)
75 vga_dirty_log_start(s
);
78 static int pci_vga_initfn(PCIDevice
*dev
)
80 PCIVGAState
*d
= DO_UPCAST(PCIVGAState
, dev
, dev
);
81 VGACommonState
*s
= &d
->vga
;
82 uint8_t *pci_conf
= d
->dev
.config
;
85 vga_common_init(s
, VGA_RAM_SIZE
);
88 s
->ds
= graphic_console_init(s
->update
, s
->invalidate
,
89 s
->screen_dump
, s
->text_update
, s
);
91 // dummy VGA (same as Bochs ID)
92 pci_config_set_vendor_id(pci_conf
, PCI_VENDOR_ID_QEMU
);
93 pci_config_set_device_id(pci_conf
, PCI_DEVICE_ID_QEMU_VGA
);
94 pci_config_set_class(pci_conf
, PCI_CLASS_DISPLAY_VGA
);
95 pci_conf
[PCI_HEADER_TYPE
] = PCI_HEADER_TYPE_NORMAL
; // header_type
97 /* XXX: VGA_RAM_SIZE must be a power of two */
98 pci_register_bar(&d
->dev
, 0, VGA_RAM_SIZE
,
99 PCI_BASE_ADDRESS_MEM_PREFETCH
, vga_map
);
102 unsigned int bios_total_size
;
103 /* must be a power of two */
105 while (bios_total_size
< s
->bios_size
)
106 bios_total_size
<<= 1;
107 pci_register_bar(&d
->dev
, PCI_ROM_SLOT
, bios_total_size
,
108 PCI_BASE_ADDRESS_MEM_PREFETCH
, vga_map
);
113 rom_add_vga(VGABIOS_FILENAME
);
117 int pci_vga_init(PCIBus
*bus
,
118 unsigned long vga_bios_offset
, int vga_bios_size
)
122 dev
= pci_create(bus
, -1, "VGA");
123 qdev_prop_set_uint32(&dev
->qdev
, "bios-offset", vga_bios_offset
);
124 qdev_prop_set_uint32(&dev
->qdev
, "bios-size", vga_bios_size
);
125 qdev_init_nofail(&dev
->qdev
);
130 static PCIDeviceInfo vga_info
= {
132 .qdev
.size
= sizeof(PCIVGAState
),
133 .qdev
.vmsd
= &vmstate_vga_pci
,
134 .init
= pci_vga_initfn
,
135 .config_write
= pci_vga_write_config
,
136 .qdev
.props
= (Property
[]) {
137 DEFINE_PROP_HEX32("bios-offset", PCIVGAState
, vga
.bios_offset
, 0),
138 DEFINE_PROP_HEX32("bios-size", PCIVGAState
, vga
.bios_size
, 0),
139 DEFINE_PROP_END_OF_LIST(),
143 static void vga_register(void)
145 pci_qdev_register(&vga_info
);
147 device_init(vga_register
);