| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * QEMU PC System Emulator | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |  * Copyright (c) 2003-2004 Fabrice Bellard | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
					
						
							|  |  |  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2007-11-17 17:14:51 +00:00
										 |  |  | #include "hw.h"
 | 
					
						
							|  |  |  | #include "pc.h"
 | 
					
						
							| 
									
										
										
										
											2012-10-17 09:54:19 +02:00
										 |  |  | #include "serial.h"
 | 
					
						
							| 
									
										
										
										
											2010-03-21 19:46:26 +00:00
										 |  |  | #include "apic.h"
 | 
					
						
							| 
									
										
										
										
											2007-11-17 17:14:51 +00:00
										 |  |  | #include "fdc.h"
 | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  | #include "ide.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-12 14:24:50 +02:00
										 |  |  | #include "pci/pci.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:49 +01:00
										 |  |  | #include "monitor/monitor.h"
 | 
					
						
							| 
									
										
										
										
											2008-09-18 18:27:29 +00:00
										 |  |  | #include "fw_cfg.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-17 23:28:44 +00:00
										 |  |  | #include "hpet_emul.h"
 | 
					
						
							| 
									
										
											  
											
												qemu: Add support for SMBIOS command line otions (Alex Williamson)
Create a new -smbios option (x86-only) to allow binary SMBIOS entries
to be passed through to the BIOS or modify the default values of
individual fields of type 0 and 1 entries on the command line.
Binary SMBIOS entries can be generated as follows:
dmidecode -t 1 -u | grep $'^\t\t[^"]' | xargs -n1 | \
        perl -lne 'printf "%c", hex($_)' > smbios_type_1.bin
These can then be passed to the BIOS using this switch:
 -smbios file=smbios_type_1.bin
Command line generation supports the following syntax:
 -smbios type=0[,vendor=str][,version=str][,date=str][,release=%d.%d]
 -smbios type=1[,manufacturer=str][,product=str][,version=str][,serial=str]
              [,uuid=$(uuidgen)][,sku=str][,family=str]
For instance, to add a serial number to the type 1 table:
 -smbios type=1,serial=0123456789
Interface is extensible to support more fields/tables as needed.
aliguori: remove texi formatting from help output
Signed-off-by: Alex Williamson <alex.williamson@hp.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@7163 c046a42c-6fe2-441c-8c8c-71466251a162
											
										 
											2009-04-17 18:59:56 +00:00
										 |  |  | #include "smbios.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-20 14:58:02 +00:00
										 |  |  | #include "loader.h"
 | 
					
						
							|  |  |  | #include "elf.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-26 14:13:46 +01:00
										 |  |  | #include "multiboot.h"
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:17 +09:00
										 |  |  | #include "mc146818rtc.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-01 20:31:39 +01:00
										 |  |  | #include "i8254.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-17 11:24:34 +01:00
										 |  |  | #include "pcspk.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-12 14:24:50 +02:00
										 |  |  | #include "pci/msi.h"
 | 
					
						
							| 
									
										
										
										
											2010-06-13 14:15:38 +02:00
										 |  |  | #include "sysbus.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:04 +01:00
										 |  |  | #include "sysemu/sysemu.h"
 | 
					
						
							|  |  |  | #include "sysemu/kvm.h"
 | 
					
						
							| 
									
										
										
										
											2012-07-26 15:35:13 +01:00
										 |  |  | #include "kvm_i386.h"
 | 
					
						
							| 
									
										
										
										
											2012-04-12 10:02:47 +00:00
										 |  |  | #include "xen.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:04 +01:00
										 |  |  | #include "sysemu/blockdev.h"
 | 
					
						
							| 
									
										
											  
											
												block: Geometry and translation hints are now useless, purge them
There are two producers of these hints: drive_init() on behalf of
-drive, and hd_geometry_guess().
The only consumer of the hint is hd_geometry_guess().
The callers of hd_geometry_guess() call it only when drive_init()
didn't set the hints.  Therefore, drive_init()'s hints are never used.
Thus, hd_geometry_guess() only ever sees hints it produced itself in a
prior call.  Only the first call computes something, subsequent calls
just repeat the first call's results.  However, hd_geometry_guess() is
never called more than once: the device models don't, and the block
device is destroyed on unplug.  Thus, dropping the repeat feature
doesn't break anything now.
If a block device wasn't destroyed on unplug and could be reused with
a new device, then repeating old results would be wrong.  Thus,
dropping the repeat feature prevents future breakage.
This renders the hints unused.  Purge them from the block layer.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Kevin Wolf <kwolf@redhat.com>
											
										 
											2012-07-10 11:12:50 +02:00
										 |  |  | #include "hw/block-common.h"
 | 
					
						
							| 
									
										
										
										
											2010-04-27 11:50:11 +02:00
										 |  |  | #include "ui/qemu-spice.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:49 +01:00
										 |  |  | #include "exec/memory.h"
 | 
					
						
							|  |  |  | #include "exec/address-spaces.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:04 +01:00
										 |  |  | #include "sysemu/arch_init.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:00 +01:00
										 |  |  | #include "qemu/bitmap.h"
 | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-29 20:23:49 +00:00
										 |  |  | /* debug PC/ISA interrupts */ | 
					
						
							|  |  |  | //#define DEBUG_IRQ
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_IRQ
 | 
					
						
							|  |  |  | #define DPRINTF(fmt, ...)                                       \
 | 
					
						
							|  |  |  |     do { printf("CPUIRQ: " fmt , ## __VA_ARGS__); } while (0) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define DPRINTF(fmt, ...)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-31 19:41:22 +00:00
										 |  |  | /* Leave a chunk of memory at the top of RAM for the BIOS ACPI tables.  */ | 
					
						
							|  |  |  | #define ACPI_DATA_SIZE       0x10000
 | 
					
						
							| 
									
										
										
										
											2008-09-18 18:27:29 +00:00
										 |  |  | #define BIOS_CFG_IOPORT 0x510
 | 
					
						
							| 
									
										
										
										
											2009-02-27 20:12:36 +00:00
										 |  |  | #define FW_CFG_ACPI_TABLES (FW_CFG_ARCH_LOCAL + 0)
 | 
					
						
							| 
									
										
											  
											
												qemu: Add support for SMBIOS command line otions (Alex Williamson)
Create a new -smbios option (x86-only) to allow binary SMBIOS entries
to be passed through to the BIOS or modify the default values of
individual fields of type 0 and 1 entries on the command line.
Binary SMBIOS entries can be generated as follows:
dmidecode -t 1 -u | grep $'^\t\t[^"]' | xargs -n1 | \
        perl -lne 'printf "%c", hex($_)' > smbios_type_1.bin
These can then be passed to the BIOS using this switch:
 -smbios file=smbios_type_1.bin
Command line generation supports the following syntax:
 -smbios type=0[,vendor=str][,version=str][,date=str][,release=%d.%d]
 -smbios type=1[,manufacturer=str][,product=str][,version=str][,serial=str]
              [,uuid=$(uuidgen)][,sku=str][,family=str]
For instance, to add a serial number to the type 1 table:
 -smbios type=1,serial=0123456789
Interface is extensible to support more fields/tables as needed.
aliguori: remove texi formatting from help output
Signed-off-by: Alex Williamson <alex.williamson@hp.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@7163 c046a42c-6fe2-441c-8c8c-71466251a162
											
										 
											2009-04-17 18:59:56 +00:00
										 |  |  | #define FW_CFG_SMBIOS_ENTRIES (FW_CFG_ARCH_LOCAL + 1)
 | 
					
						
							| 
									
										
										
										
											2009-08-06 16:25:50 +02:00
										 |  |  | #define FW_CFG_IRQ0_OVERRIDE (FW_CFG_ARCH_LOCAL + 2)
 | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | #define FW_CFG_E820_TABLE (FW_CFG_ARCH_LOCAL + 3)
 | 
					
						
							| 
									
										
										
										
											2010-06-14 11:29:28 +03:00
										 |  |  | #define FW_CFG_HPET (FW_CFG_ARCH_LOCAL + 4)
 | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | #define E820_NR_ENTRIES		16
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct e820_entry { | 
					
						
							|  |  |  |     uint64_t address; | 
					
						
							|  |  |  |     uint64_t length; | 
					
						
							|  |  |  |     uint32_t type; | 
					
						
							| 
									
										
										
										
											2011-08-31 12:38:01 +02:00
										 |  |  | } QEMU_PACKED __attribute((__aligned__(4))); | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct e820_table { | 
					
						
							|  |  |  |     uint32_t count; | 
					
						
							|  |  |  |     struct e820_entry entry[E820_NR_ENTRIES]; | 
					
						
							| 
									
										
										
										
											2011-08-31 12:38:01 +02:00
										 |  |  | } QEMU_PACKED __attribute((__aligned__(4))); | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct e820_table e820_table; | 
					
						
							| 
									
										
										
										
											2011-02-05 14:35:00 +00:00
										 |  |  | struct hpet_fw_config hpet_cfg = {.count = UINT8_MAX}; | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-07 09:19:35 +02:00
										 |  |  | void gsi_handler(void *opaque, int n, int level) | 
					
						
							| 
									
										
										
										
											2009-08-09 19:44:55 +03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-10-07 09:19:35 +02:00
										 |  |  |     GSIState *s = opaque; | 
					
						
							| 
									
										
										
										
											2009-08-09 19:44:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-07 09:19:35 +02:00
										 |  |  |     DPRINTF("pc: %s GSI %d\n", level ? "raising" : "lowering", n); | 
					
						
							|  |  |  |     if (n < ISA_NUM_IRQS) { | 
					
						
							|  |  |  |         qemu_set_irq(s->i8259_irq[n], level); | 
					
						
							| 
									
										
										
										
											2009-08-09 19:44:56 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-10-07 09:19:35 +02:00
										 |  |  |     qemu_set_irq(s->ioapic_irq[n], level); | 
					
						
							| 
									
										
										
										
											2011-10-07 09:19:34 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-08-09 19:44:55 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  | static void ioport80_write(void *opaque, hwaddr addr, uint64_t data, | 
					
						
							|  |  |  |                            unsigned size) | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-09 18:10:22 +00:00
										 |  |  | static uint64_t ioport80_read(void *opaque, hwaddr addr, unsigned size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-11 16:41:43 +00:00
										 |  |  |     return 0xffffffffffffffffULL; | 
					
						
							| 
									
										
										
										
											2013-01-09 18:10:22 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-05-08 21:03:41 +00:00
										 |  |  | /* MSDOS compatibility mode FPU exception support */ | 
					
						
							| 
									
										
										
										
											2007-04-07 18:14:41 +00:00
										 |  |  | static qemu_irq ferr_irq; | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:09 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | void pc_register_ferr_irq(qemu_irq irq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ferr_irq = irq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-05-08 21:03:41 +00:00
										 |  |  | /* XXX: add IGNNE support */ | 
					
						
							|  |  |  | void cpu_set_ferr(CPUX86State *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2007-04-07 18:14:41 +00:00
										 |  |  |     qemu_irq_raise(ferr_irq); | 
					
						
							| 
									
										
										
										
											2004-05-08 21:03:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  | static void ioportF0_write(void *opaque, hwaddr addr, uint64_t data, | 
					
						
							|  |  |  |                            unsigned size) | 
					
						
							| 
									
										
										
										
											2004-05-08 21:03:41 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-04-07 18:14:41 +00:00
										 |  |  |     qemu_irq_lower(ferr_irq); | 
					
						
							| 
									
										
										
										
											2004-05-08 21:03:41 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-09 18:10:22 +00:00
										 |  |  | static uint64_t ioportF0_read(void *opaque, hwaddr addr, unsigned size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-11 16:41:43 +00:00
										 |  |  |     return 0xffffffffffffffffULL; | 
					
						
							| 
									
										
										
										
											2013-01-09 18:10:22 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-05-20 14:02:14 +00:00
										 |  |  | /* TSC handling */ | 
					
						
							|  |  |  | uint64_t cpu_get_tsc(CPUX86State *env) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-08-10 17:07:24 -05:00
										 |  |  |     return cpu_get_ticks(); | 
					
						
							| 
									
										
										
										
											2004-05-20 14:02:14 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-24 18:48:00 +00:00
										 |  |  | /* SMM support */ | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:04 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | static cpu_set_smm_t smm_set; | 
					
						
							|  |  |  | static void *smm_arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void cpu_smm_register(cpu_set_smm_t callback, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(smm_set == NULL); | 
					
						
							|  |  |  |     assert(smm_arg == NULL); | 
					
						
							|  |  |  |     smm_set = callback; | 
					
						
							|  |  |  |     smm_arg = arg; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:23 +01:00
										 |  |  | void cpu_smm_update(CPUX86State *env) | 
					
						
							| 
									
										
										
										
											2006-09-24 18:48:00 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:04 +09:00
										 |  |  |     if (smm_set && smm_arg && env == first_cpu) | 
					
						
							|  |  |  |         smm_set(!!(env->hflags & HF_SMM_MASK), smm_arg); | 
					
						
							| 
									
										
										
										
											2006-09-24 18:48:00 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-02 18:11:44 +00:00
										 |  |  | /* IRQ handling */ | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:23 +01:00
										 |  |  | int cpu_get_pic_interrupt(CPUX86State *env) | 
					
						
							| 
									
										
										
										
											2005-07-02 18:11:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int intno; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-19 10:42:08 +03:00
										 |  |  |     intno = apic_get_interrupt(env->apic_state); | 
					
						
							| 
									
										
										
										
											2005-07-02 18:11:44 +00:00
										 |  |  |     if (intno >= 0) { | 
					
						
							|  |  |  |         return intno; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     /* read the irq from the PIC */ | 
					
						
							| 
									
										
										
										
											2010-06-19 10:42:08 +03:00
										 |  |  |     if (!apic_accept_pic_intr(env->apic_state)) { | 
					
						
							| 
									
										
										
										
											2007-10-09 03:08:56 +00:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2010-06-19 10:42:08 +03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-10-09 03:08:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-02 18:11:44 +00:00
										 |  |  |     intno = pic_read_irq(isa_pic); | 
					
						
							|  |  |  |     return intno; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-04-07 18:14:41 +00:00
										 |  |  | static void pic_irq_request(void *opaque, int irq, int level) | 
					
						
							| 
									
										
										
										
											2005-07-02 18:11:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:23 +01:00
										 |  |  |     CPUX86State *env = first_cpu; | 
					
						
							| 
									
										
										
										
											2008-04-13 16:08:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-29 20:23:49 +00:00
										 |  |  |     DPRINTF("pic_irqs: %s irq %d\n", level? "raise" : "lower", irq); | 
					
						
							| 
									
										
										
										
											2008-08-19 12:55:20 +00:00
										 |  |  |     if (env->apic_state) { | 
					
						
							|  |  |  |         while (env) { | 
					
						
							| 
									
										
										
										
											2010-06-19 10:42:08 +03:00
										 |  |  |             if (apic_accept_pic_intr(env->apic_state)) { | 
					
						
							|  |  |  |                 apic_deliver_pic_intr(env->apic_state, level); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-08-19 12:55:20 +00:00
										 |  |  |             env = env->next_cpu; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2008-08-21 03:14:41 +00:00
										 |  |  |         if (level) | 
					
						
							|  |  |  |             cpu_interrupt(env, CPU_INTERRUPT_HARD); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             cpu_reset_interrupt(env, CPU_INTERRUPT_HARD); | 
					
						
							| 
									
										
										
										
											2008-04-13 16:08:30 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-07-02 18:11:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  | /* PC cmos mappings */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | #define REG_EQUIPMENT_BYTE          0x14
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  | static int cmos_get_fd_drive_type(FDriveType fd0) | 
					
						
							| 
									
										
										
										
											2004-05-23 16:26:20 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int val; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (fd0) { | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  |     case FDRIVE_DRV_144: | 
					
						
							| 
									
										
										
										
											2004-05-23 16:26:20 +00:00
										 |  |  |         /* 1.44 Mb 3"5 drive */ | 
					
						
							|  |  |  |         val = 4; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  |     case FDRIVE_DRV_288: | 
					
						
							| 
									
										
										
										
											2004-05-23 16:26:20 +00:00
										 |  |  |         /* 2.88 Mb 3"5 drive */ | 
					
						
							|  |  |  |         val = 5; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  |     case FDRIVE_DRV_120: | 
					
						
							| 
									
										
										
										
											2004-05-23 16:26:20 +00:00
										 |  |  |         /* 1.2 Mb 5"5 drive */ | 
					
						
							|  |  |  |         val = 2; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  |     case FDRIVE_DRV_NONE: | 
					
						
							| 
									
										
										
										
											2004-05-23 16:26:20 +00:00
										 |  |  |     default: | 
					
						
							|  |  |  |         val = 0; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return val; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:38 +02:00
										 |  |  | static void cmos_init_hd(ISADevice *s, int type_ofs, int info_ofs, | 
					
						
							|  |  |  |                          int16_t cylinders, int8_t heads, int8_t sectors) | 
					
						
							| 
									
										
										
										
											2004-10-09 16:47:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     rtc_set_memory(s, type_ofs, 47); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs, cylinders); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 1, cylinders >> 8); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 2, heads); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 3, 0xff); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 4, 0xff); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 5, 0xc0 | ((heads > 8) << 3)); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 6, cylinders); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 7, cylinders >> 8); | 
					
						
							|  |  |  |     rtc_set_memory(s, info_ofs + 8, sectors); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-31 01:54:04 +00:00
										 |  |  | /* convert boot_device letter to something recognizable by the bios */ | 
					
						
							|  |  |  | static int boot_device2nibble(char boot_device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch(boot_device) { | 
					
						
							|  |  |  |     case 'a': | 
					
						
							|  |  |  |     case 'b': | 
					
						
							|  |  |  |         return 0x01; /* floppy boot */ | 
					
						
							|  |  |  |     case 'c': | 
					
						
							|  |  |  |         return 0x02; /* hard drive boot */ | 
					
						
							|  |  |  |     case 'd': | 
					
						
							|  |  |  |         return 0x03; /* CD-ROM boot */ | 
					
						
							|  |  |  |     case 'n': | 
					
						
							|  |  |  |         return 0x04; /* Network boot */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:17 +09:00
										 |  |  | static int set_boot_dev(ISADevice *s, const char *boot_device, int fd_bootchk) | 
					
						
							| 
									
										
										
										
											2008-05-04 20:11:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #define PC_MAX_BOOT_DEVICES 3
 | 
					
						
							|  |  |  |     int nbds, bds[3] = { 0, }; | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     nbds = strlen(boot_device); | 
					
						
							|  |  |  |     if (nbds > PC_MAX_BOOT_DEVICES) { | 
					
						
							| 
									
										
										
										
											2010-02-18 17:25:24 +01:00
										 |  |  |         error_report("Too many boot devices for PC"); | 
					
						
							| 
									
										
										
										
											2008-05-04 20:11:34 +00:00
										 |  |  |         return(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (i = 0; i < nbds; i++) { | 
					
						
							|  |  |  |         bds[i] = boot_device2nibble(boot_device[i]); | 
					
						
							|  |  |  |         if (bds[i] == 0) { | 
					
						
							| 
									
										
										
										
											2010-02-18 17:25:24 +01:00
										 |  |  |             error_report("Invalid boot device for PC: '%c'", | 
					
						
							|  |  |  |                          boot_device[i]); | 
					
						
							| 
									
										
										
										
											2008-05-04 20:11:34 +00:00
										 |  |  |             return(1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x3d, (bds[1] << 4) | bds[0]); | 
					
						
							| 
									
										
										
										
											2010-02-17 18:07:48 +01:00
										 |  |  |     rtc_set_memory(s, 0x38, (bds[2] << 4) | (fd_bootchk ? 0x0 : 0x1)); | 
					
						
							| 
									
										
										
										
											2008-05-04 20:11:34 +00:00
										 |  |  |     return(0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 18:07:48 +01:00
										 |  |  | static int pc_boot_set(void *opaque, const char *boot_device) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return set_boot_dev(opaque, boot_device, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  | typedef struct pc_cmos_init_late_arg { | 
					
						
							|  |  |  |     ISADevice *rtc_state; | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:38 +02:00
										 |  |  |     BusState *idebus[2]; | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  | } pc_cmos_init_late_arg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void pc_cmos_init_late(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     pc_cmos_init_late_arg *arg = opaque; | 
					
						
							|  |  |  |     ISADevice *s = arg->rtc_state; | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:38 +02:00
										 |  |  |     int16_t cylinders; | 
					
						
							|  |  |  |     int8_t heads, sectors; | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  |     int val; | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:53 +02:00
										 |  |  |     int i, trans; | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:38 +02:00
										 |  |  |     val = 0; | 
					
						
							|  |  |  |     if (ide_get_geometry(arg->idebus[0], 0, | 
					
						
							|  |  |  |                          &cylinders, &heads, §ors) >= 0) { | 
					
						
							|  |  |  |         cmos_init_hd(s, 0x19, 0x1b, cylinders, heads, sectors); | 
					
						
							|  |  |  |         val |= 0xf0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ide_get_geometry(arg->idebus[0], 1, | 
					
						
							|  |  |  |                          &cylinders, &heads, §ors) >= 0) { | 
					
						
							|  |  |  |         cmos_init_hd(s, 0x1a, 0x24, cylinders, heads, sectors); | 
					
						
							|  |  |  |         val |= 0x0f; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x12, val); | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     val = 0; | 
					
						
							|  |  |  |     for (i = 0; i < 4; i++) { | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:38 +02:00
										 |  |  |         /* NOTE: ide_get_geometry() returns the physical
 | 
					
						
							|  |  |  |            geometry.  It is always such that: 1 <= sects <= 63, 1 | 
					
						
							|  |  |  |            <= heads <= 16, 1 <= cylinders <= 16383. The BIOS | 
					
						
							|  |  |  |            geometry can be different if a translation is done. */ | 
					
						
							|  |  |  |         if (ide_get_geometry(arg->idebus[i / 2], i % 2, | 
					
						
							|  |  |  |                              &cylinders, &heads, §ors) >= 0) { | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:53 +02:00
										 |  |  |             trans = ide_get_bios_chs_trans(arg->idebus[i / 2], i % 2) - 1; | 
					
						
							|  |  |  |             assert((trans & ~3) == 0); | 
					
						
							|  |  |  |             val |= trans << (i * 2); | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x39, val); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_unregister_reset(pc_cmos_init_late, opaque); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:15 +09:00
										 |  |  | void pc_cmos_init(ram_addr_t ram_size, ram_addr_t above_4g_mem_size, | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  |                   const char *boot_device, | 
					
						
							| 
									
										
										
										
											2011-10-20 16:37:26 +02:00
										 |  |  |                   ISADevice *floppy, BusState *idebus0, BusState *idebus1, | 
					
						
							| 
									
										
										
										
											2011-02-05 16:32:23 +00:00
										 |  |  |                   ISADevice *s) | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:27 +02:00
										 |  |  |     int val, nb, i; | 
					
						
							| 
									
										
										
										
											2011-11-09 21:59:50 +00:00
										 |  |  |     FDriveType fd_type[2] = { FDRIVE_DRV_NONE, FDRIVE_DRV_NONE }; | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  |     static pc_cmos_init_late_arg arg; | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* various important CMOS locations needed by PC/Bochs bios */ | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* memory size */ | 
					
						
							| 
									
										
										
										
											2012-08-15 13:12:20 +02:00
										 |  |  |     /* base memory (first MiB) */ | 
					
						
							|  |  |  |     val = MIN(ram_size / 1024, 640); | 
					
						
							| 
									
										
										
										
											2004-04-07 20:51:30 +00:00
										 |  |  |     rtc_set_memory(s, 0x15, val); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x16, val >> 8); | 
					
						
							| 
									
										
										
										
											2012-08-15 13:12:20 +02:00
										 |  |  |     /* extended memory (next 64MiB) */ | 
					
						
							|  |  |  |     if (ram_size > 1024 * 1024) { | 
					
						
							|  |  |  |         val = (ram_size - 1024 * 1024) / 1024; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         val = 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |     if (val > 65535) | 
					
						
							|  |  |  |         val = 65535; | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |     rtc_set_memory(s, 0x17, val); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x18, val >> 8); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x30, val); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x31, val >> 8); | 
					
						
							| 
									
										
										
										
											2012-08-15 13:12:20 +02:00
										 |  |  |     /* memory between 16MiB and 4GiB */ | 
					
						
							|  |  |  |     if (ram_size > 16 * 1024 * 1024) { | 
					
						
							|  |  |  |         val = (ram_size - 16 * 1024 * 1024) / 65536; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2004-06-26 15:53:17 +00:00
										 |  |  |         val = 0; | 
					
						
							| 
									
										
										
										
											2012-08-15 13:12:20 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |     if (val > 65535) | 
					
						
							|  |  |  |         val = 65535; | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |     rtc_set_memory(s, 0x34, val); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x35, val >> 8); | 
					
						
							| 
									
										
										
										
											2012-08-15 13:12:20 +02:00
										 |  |  |     /* memory above 4GiB */ | 
					
						
							|  |  |  |     val = above_4g_mem_size / 65536; | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x5b, val); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x5c, val >> 8); | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x5d, val >> 16); | 
					
						
							| 
									
										
										
										
											2007-09-17 08:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-28 22:28:08 +00:00
										 |  |  |     /* set the number of CPU */ | 
					
						
							|  |  |  |     rtc_set_memory(s, 0x5f, smp_cpus - 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-31 01:54:04 +00:00
										 |  |  |     /* set boot devices, and disable floppy signature check if requested */ | 
					
						
							| 
									
										
										
										
											2010-02-17 18:07:48 +01:00
										 |  |  |     if (set_boot_dev(s, boot_device, fd_bootchk)) { | 
					
						
							| 
									
										
										
										
											2007-11-11 01:50:45 +00:00
										 |  |  |         exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-14 21:46:48 +00:00
										 |  |  |     /* floppy type */ | 
					
						
							| 
									
										
										
										
											2011-10-20 16:37:26 +02:00
										 |  |  |     if (floppy) { | 
					
						
							|  |  |  |         for (i = 0; i < 2; i++) { | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:27 +02:00
										 |  |  |             fd_type[i] = isa_fdc_get_drive_type(floppy, i); | 
					
						
							| 
									
										
										
										
											2011-02-05 16:32:23 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     val = (cmos_get_fd_drive_type(fd_type[0]) << 4) | | 
					
						
							|  |  |  |         cmos_get_fd_drive_type(fd_type[1]); | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |     rtc_set_memory(s, 0x10, val); | 
					
						
							| 
									
										
										
										
											2007-09-17 08:09:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |     val = 0; | 
					
						
							| 
									
										
										
										
											2004-03-14 21:46:48 +00:00
										 |  |  |     nb = 0; | 
					
						
							| 
									
										
										
										
											2011-02-05 16:32:23 +00:00
										 |  |  |     if (fd_type[0] < FDRIVE_DRV_NONE) { | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |         nb++; | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-05 16:32:23 +00:00
										 |  |  |     if (fd_type[1] < FDRIVE_DRV_NONE) { | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |         nb++; | 
					
						
							| 
									
										
										
										
											2011-02-12 21:23:12 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |     switch (nb) { | 
					
						
							|  |  |  |     case 0: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case 1: | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |         val |= 0x01; /* 1 drive, ready for boot */ | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case 2: | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |         val |= 0x41; /* 2 drives, ready for boot */ | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-03-31 18:58:38 +00:00
										 |  |  |     val |= 0x02; /* FPU is there */ | 
					
						
							|  |  |  |     val |= 0x04; /* PS/2 mouse installed */ | 
					
						
							|  |  |  |     rtc_set_memory(s, REG_EQUIPMENT_BYTE, val); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-10-09 16:47:59 +00:00
										 |  |  |     /* hard drives */ | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  |     arg.rtc_state = s; | 
					
						
							| 
									
										
										
										
											2012-07-10 11:12:38 +02:00
										 |  |  |     arg.idebus[0] = idebus0; | 
					
						
							|  |  |  |     arg.idebus[1] = idebus1; | 
					
						
							| 
									
										
										
										
											2010-06-24 19:58:20 +02:00
										 |  |  |     qemu_register_reset(pc_cmos_init_late, &arg); | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | /* port 92 stuff: could be split off */ | 
					
						
							|  |  |  | typedef struct Port92State { | 
					
						
							|  |  |  |     ISADevice dev; | 
					
						
							| 
									
										
										
										
											2011-08-16 08:32:44 -07:00
										 |  |  |     MemoryRegion io; | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  |     uint8_t outport; | 
					
						
							|  |  |  |     qemu_irq *a20_out; | 
					
						
							|  |  |  | } Port92State; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-08 13:24:52 +02:00
										 |  |  | static void port92_write(void *opaque, hwaddr addr, uint64_t val, | 
					
						
							|  |  |  |                          unsigned size) | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     Port92State *s = opaque; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DPRINTF("port92: write 0x%02x\n", val); | 
					
						
							|  |  |  |     s->outport = val; | 
					
						
							|  |  |  |     qemu_set_irq(*s->a20_out, (val >> 1) & 1); | 
					
						
							|  |  |  |     if (val & 1) { | 
					
						
							|  |  |  |         qemu_system_reset_request(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-08 13:24:52 +02:00
										 |  |  | static uint64_t port92_read(void *opaque, hwaddr addr, | 
					
						
							|  |  |  |                             unsigned size) | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     Port92State *s = opaque; | 
					
						
							|  |  |  |     uint32_t ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = s->outport; | 
					
						
							|  |  |  |     DPRINTF("port92: read 0x%02x\n", ret); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void port92_init(ISADevice *dev, qemu_irq *a20_out) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Port92State *s = DO_UPCAST(Port92State, dev, dev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s->a20_out = a20_out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const VMStateDescription vmstate_port92_isa = { | 
					
						
							|  |  |  |     .name = "port92", | 
					
						
							|  |  |  |     .version_id = 1, | 
					
						
							|  |  |  |     .minimum_version_id = 1, | 
					
						
							|  |  |  |     .minimum_version_id_old = 1, | 
					
						
							|  |  |  |     .fields      = (VMStateField []) { | 
					
						
							|  |  |  |         VMSTATE_UINT8(outport, Port92State), | 
					
						
							|  |  |  |         VMSTATE_END_OF_LIST() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void port92_reset(DeviceState *d) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Port92State *s = container_of(d, Port92State, dev.qdev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s->outport &= ~1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 08:32:44 -07:00
										 |  |  | static const MemoryRegionOps port92_ops = { | 
					
						
							| 
									
										
										
										
											2012-10-08 13:24:52 +02:00
										 |  |  |     .read = port92_read, | 
					
						
							|  |  |  |     .write = port92_write, | 
					
						
							|  |  |  |     .impl = { | 
					
						
							|  |  |  |         .min_access_size = 1, | 
					
						
							|  |  |  |         .max_access_size = 1, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     .endianness = DEVICE_LITTLE_ENDIAN, | 
					
						
							| 
									
										
										
										
											2011-08-16 08:32:44 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | static int port92_initfn(ISADevice *dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Port92State *s = DO_UPCAST(Port92State, dev, dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-16 08:32:44 -07:00
										 |  |  |     memory_region_init_io(&s->io, &port92_ops, s, "port92", 1); | 
					
						
							|  |  |  |     isa_register_ioport(dev, &s->io, 0x92); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  |     s->outport = 0; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-04 11:52:49 -06:00
										 |  |  | static void port92_class_initfn(ObjectClass *klass, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     DeviceClass *dc = DEVICE_CLASS(klass); | 
					
						
							| 
									
										
										
										
											2011-12-04 11:52:49 -06:00
										 |  |  |     ISADeviceClass *ic = ISA_DEVICE_CLASS(klass); | 
					
						
							|  |  |  |     ic->init = port92_initfn; | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     dc->no_user = 1; | 
					
						
							|  |  |  |     dc->reset = port92_reset; | 
					
						
							|  |  |  |     dc->vmsd = &vmstate_port92_isa; | 
					
						
							| 
									
										
										
										
											2011-12-04 11:52:49 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 16:19:07 +01:00
										 |  |  | static const TypeInfo port92_info = { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     .name          = "port92", | 
					
						
							|  |  |  |     .parent        = TYPE_ISA_DEVICE, | 
					
						
							|  |  |  |     .instance_size = sizeof(Port92State), | 
					
						
							|  |  |  |     .class_init    = port92_class_initfn, | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | static void port92_register_types(void) | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-07 21:34:16 -06:00
										 |  |  |     type_register_static(&port92_info); | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-02-09 15:20:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | type_init(port92_register_types) | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-22 07:59:01 +00:00
										 |  |  | static void handle_a20_line_change(void *opaque, int irq, int level) | 
					
						
							| 
									
										
										
										
											2005-11-21 23:34:32 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-18 15:19:06 +01:00
										 |  |  |     X86CPU *cpu = opaque; | 
					
						
							| 
									
										
										
										
											2004-04-05 20:26:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-22 07:59:01 +00:00
										 |  |  |     /* XXX: send to all CPUs ? */ | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  |     /* XXX: add logic to handle multiple A20 line sources */ | 
					
						
							| 
									
										
										
										
											2013-01-18 15:19:06 +01:00
										 |  |  |     x86_cpu_set_a20(cpu, level); | 
					
						
							| 
									
										
										
										
											2004-04-05 20:26:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | int e820_add_entry(uint64_t address, uint64_t length, uint32_t type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-11-07 20:57:00 -07:00
										 |  |  |     int index = le32_to_cpu(e820_table.count); | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  |     struct e820_entry *entry; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (index >= E820_NR_ENTRIES) | 
					
						
							|  |  |  |         return -EBUSY; | 
					
						
							| 
									
										
										
										
											2010-11-07 20:57:00 -07:00
										 |  |  |     entry = &e820_table.entry[index++]; | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-07 20:57:00 -07:00
										 |  |  |     entry->address = cpu_to_le64(address); | 
					
						
							|  |  |  |     entry->length = cpu_to_le64(length); | 
					
						
							|  |  |  |     entry->type = cpu_to_le32(type); | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-07 20:57:00 -07:00
										 |  |  |     e820_table.count = cpu_to_le32(index); | 
					
						
							|  |  |  |     return index; | 
					
						
							| 
									
										
										
										
											2010-02-15 18:33:46 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  | /* Calculates the limit to CPU APIC ID values
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function returns the limit for the APIC ID value, so that all | 
					
						
							|  |  |  |  * CPU APIC IDs are < pc_apic_id_limit(). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This is used for FW_CFG_MAX_CPUS. See comments on bochs_bios_init(). | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static unsigned int pc_apic_id_limit(unsigned int max_cpus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return x86_cpu_apic_id_from_index(max_cpus - 1) + 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:38 +02:00
										 |  |  | static void *bochs_bios_init(void) | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-09-18 18:27:29 +00:00
										 |  |  |     void *fw_cfg; | 
					
						
							| 
									
										
											  
											
												qemu: Add support for SMBIOS command line otions (Alex Williamson)
Create a new -smbios option (x86-only) to allow binary SMBIOS entries
to be passed through to the BIOS or modify the default values of
individual fields of type 0 and 1 entries on the command line.
Binary SMBIOS entries can be generated as follows:
dmidecode -t 1 -u | grep $'^\t\t[^"]' | xargs -n1 | \
        perl -lne 'printf "%c", hex($_)' > smbios_type_1.bin
These can then be passed to the BIOS using this switch:
 -smbios file=smbios_type_1.bin
Command line generation supports the following syntax:
 -smbios type=0[,vendor=str][,version=str][,date=str][,release=%d.%d]
 -smbios type=1[,manufacturer=str][,product=str][,version=str][,serial=str]
              [,uuid=$(uuidgen)][,sku=str][,family=str]
For instance, to add a serial number to the type 1 table:
 -smbios type=1,serial=0123456789
Interface is extensible to support more fields/tables as needed.
aliguori: remove texi formatting from help output
Signed-off-by: Alex Williamson <alex.williamson@hp.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@7163 c046a42c-6fe2-441c-8c8c-71466251a162
											
										 
											2009-04-17 18:59:56 +00:00
										 |  |  |     uint8_t *smbios_table; | 
					
						
							|  |  |  |     size_t smbios_len; | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  |     uint64_t *numa_fw_cfg; | 
					
						
							|  |  |  |     int i, j; | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |     unsigned int apic_id_limit = pc_apic_id_limit(max_cpus); | 
					
						
							| 
									
										
										
										
											2008-09-18 18:27:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fw_cfg = fw_cfg_init(BIOS_CFG_IOPORT, BIOS_CFG_IOPORT + 1, 0, 0); | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |     /* FW_CFG_MAX_CPUS is a bit confusing/problematic on x86:
 | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * SeaBIOS needs FW_CFG_MAX_CPUS for CPU hotplug, but the CPU hotplug | 
					
						
							|  |  |  |      * QEMU<->SeaBIOS interface is not based on the "CPU index", but on the APIC | 
					
						
							|  |  |  |      * ID of hotplugged CPUs[1]. This means that FW_CFG_MAX_CPUS is not the | 
					
						
							|  |  |  |      * "maximum number of CPUs", but the "limit to the APIC ID values SeaBIOS | 
					
						
							|  |  |  |      * may see". | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * So, this means we must not use max_cpus, here, but the maximum possible | 
					
						
							|  |  |  |      * APIC ID value, plus one. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * [1] The only kind of "CPU identifier" used between SeaBIOS and QEMU is | 
					
						
							|  |  |  |      *     the APIC ID, not the "CPU index" | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     fw_cfg_add_i16(fw_cfg, FW_CFG_MAX_CPUS, (uint16_t)apic_id_limit); | 
					
						
							| 
									
										
										
										
											2008-09-18 18:27:29 +00:00
										 |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_ID, 1); | 
					
						
							| 
									
										
										
										
											2008-09-18 18:33:18 +00:00
										 |  |  |     fw_cfg_add_i64(fw_cfg, FW_CFG_RAM_SIZE, (uint64_t)ram_size); | 
					
						
							| 
									
										
										
										
											2013-01-16 14:50:28 +01:00
										 |  |  |     fw_cfg_add_bytes(fw_cfg, FW_CFG_ACPI_TABLES, | 
					
						
							|  |  |  |                      acpi_tables, acpi_tables_len); | 
					
						
							| 
									
										
										
										
											2011-10-15 14:08:26 +02:00
										 |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_IRQ0_OVERRIDE, kvm_allows_irq0_override()); | 
					
						
							| 
									
										
											  
											
												qemu: Add support for SMBIOS command line otions (Alex Williamson)
Create a new -smbios option (x86-only) to allow binary SMBIOS entries
to be passed through to the BIOS or modify the default values of
individual fields of type 0 and 1 entries on the command line.
Binary SMBIOS entries can be generated as follows:
dmidecode -t 1 -u | grep $'^\t\t[^"]' | xargs -n1 | \
        perl -lne 'printf "%c", hex($_)' > smbios_type_1.bin
These can then be passed to the BIOS using this switch:
 -smbios file=smbios_type_1.bin
Command line generation supports the following syntax:
 -smbios type=0[,vendor=str][,version=str][,date=str][,release=%d.%d]
 -smbios type=1[,manufacturer=str][,product=str][,version=str][,serial=str]
              [,uuid=$(uuidgen)][,sku=str][,family=str]
For instance, to add a serial number to the type 1 table:
 -smbios type=1,serial=0123456789
Interface is extensible to support more fields/tables as needed.
aliguori: remove texi formatting from help output
Signed-off-by: Alex Williamson <alex.williamson@hp.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@7163 c046a42c-6fe2-441c-8c8c-71466251a162
											
										 
											2009-04-17 18:59:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     smbios_table = smbios_get_table(&smbios_len); | 
					
						
							|  |  |  |     if (smbios_table) | 
					
						
							|  |  |  |         fw_cfg_add_bytes(fw_cfg, FW_CFG_SMBIOS_ENTRIES, | 
					
						
							|  |  |  |                          smbios_table, smbios_len); | 
					
						
							| 
									
										
										
										
											2013-01-16 14:50:28 +01:00
										 |  |  |     fw_cfg_add_bytes(fw_cfg, FW_CFG_E820_TABLE, | 
					
						
							|  |  |  |                      &e820_table, sizeof(e820_table)); | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-16 14:50:28 +01:00
										 |  |  |     fw_cfg_add_bytes(fw_cfg, FW_CFG_HPET, &hpet_cfg, sizeof(hpet_cfg)); | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  |     /* allocate memory for the NUMA channel: one (64bit) word for the number
 | 
					
						
							|  |  |  |      * of nodes, one word for each VCPU->node and one word for each node to | 
					
						
							|  |  |  |      * hold the amount of memory. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |     numa_fw_cfg = g_new0(uint64_t, 1 + apic_id_limit + nb_numa_nodes); | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  |     numa_fw_cfg[0] = cpu_to_le64(nb_numa_nodes); | 
					
						
							| 
									
										
										
										
											2011-10-26 14:19:00 +02:00
										 |  |  |     for (i = 0; i < max_cpus; i++) { | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |         unsigned int apic_id = x86_cpu_apic_id_from_index(i); | 
					
						
							|  |  |  |         assert(apic_id < apic_id_limit); | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  |         for (j = 0; j < nb_numa_nodes; j++) { | 
					
						
							| 
									
										
											  
											
												Fixes related to processing of qemu's -numa option
The -numa option to qemu is used to create [fake] numa nodes
and expose them to the guest OS instance.
There are a couple of issues with the -numa option:
a) Max VCPU's that can be specified for a guest while using
   the qemu's -numa option is 64. Due to a typecasting issue
   when the number of VCPUs is > 32 the VCPUs don't show up
   under the specified [fake] numa nodes.
b) KVM currently has support for 160VCPUs per guest. The
   qemu's -numa option has only support for upto 64VCPUs
   per guest.
This patch addresses these two issues.
Below are examples of (a) and (b)
a) >32 VCPUs are specified with the -numa option:
/usr/local/bin/qemu-system-x86_64 \
-enable-kvm \
71:01:01 \
-net tap,ifname=tap0,script=no,downscript=no \
-vnc :4
...
Upstream qemu :
--------------
QEMU 1.1.50 monitor - type 'help' for more information
(qemu) info numa
6 nodes
node 0 cpus: 0 1 2 3 4 5 6 7 8 9 32 33 34 35 36 37 38 39 40 41
node 0 size: 131072 MB
node 1 cpus: 10 11 12 13 14 15 16 17 18 19 42 43 44 45 46 47 48 49 50 51
node 1 size: 131072 MB
node 2 cpus: 20 21 22 23 24 25 26 27 28 29 52 53 54 55 56 57 58 59
node 2 size: 131072 MB
node 3 cpus: 30
node 3 size: 131072 MB
node 4 cpus:
node 4 size: 131072 MB
node 5 cpus: 31
node 5 size: 131072 MB
With the patch applied :
-----------------------
QEMU 1.1.50 monitor - type 'help' for more information
(qemu) info numa
6 nodes
node 0 cpus: 0 1 2 3 4 5 6 7 8 9
node 0 size: 131072 MB
node 1 cpus: 10 11 12 13 14 15 16 17 18 19
node 1 size: 131072 MB
node 2 cpus: 20 21 22 23 24 25 26 27 28 29
node 2 size: 131072 MB
node 3 cpus: 30 31 32 33 34 35 36 37 38 39
node 3 size: 131072 MB
node 4 cpus: 40 41 42 43 44 45 46 47 48 49
node 4 size: 131072 MB
node 5 cpus: 50 51 52 53 54 55 56 57 58 59
node 5 size: 131072 MB
b) >64 VCPUs specified with -numa option:
/usr/local/bin/qemu-system-x86_64 \
-enable-kvm \
-cpu Westmere,+rdtscp,+pdpe1gb,+dca,+pdcm,+xtpr,+tm2,+est,+smx,+vmx,+ds_cpl,+monitor,+dtes64,+pclmuldq,+pbe,+tm,+ht,+ss,+acpi,+d-vnc :4
...
Upstream qemu :
--------------
only 63 CPUs in NUMA mode supported.
only 64 CPUs in NUMA mode supported.
QEMU 1.1.50 monitor - type 'help' for more information
(qemu) info numa
8 nodes
node 0 cpus: 6 7 8 9 38 39 40 41 70 71 72 73
node 0 size: 65536 MB
node 1 cpus: 10 11 12 13 14 15 16 17 18 19 42 43 44 45 46 47 48 49 50 51 74 75 76 77 78 79
node 1 size: 65536 MB
node 2 cpus: 20 21 22 23 24 25 26 27 28 29 52 53 54 55 56 57 58 59 60 61
node 2 size: 65536 MB
node 3 cpus: 30 62
node 3 size: 65536 MB
node 4 cpus:
node 4 size: 65536 MB
node 5 cpus:
node 5 size: 65536 MB
node 6 cpus: 31 63
node 6 size: 65536 MB
node 7 cpus: 0 1 2 3 4 5 32 33 34 35 36 37 64 65 66 67 68 69
node 7 size: 65536 MB
With the patch applied :
-----------------------
QEMU 1.1.50 monitor - type 'help' for more information
(qemu) info numa
8 nodes
node 0 cpus: 0 1 2 3 4 5 6 7 8 9
node 0 size: 65536 MB
node 1 cpus: 10 11 12 13 14 15 16 17 18 19
node 1 size: 65536 MB
node 2 cpus: 20 21 22 23 24 25 26 27 28 29
node 2 size: 65536 MB
node 3 cpus: 30 31 32 33 34 35 36 37 38 39
node 3 size: 65536 MB
node 4 cpus: 40 41 42 43 44 45 46 47 48 49
node 4 size: 65536 MB
node 5 cpus: 50 51 52 53 54 55 56 57 58 59
node 5 size: 65536 MB
node 6 cpus: 60 61 62 63 64 65 66 67 68 69
node 6 size: 65536 MB
node 7 cpus: 70 71 72 73 74 75 76 77 78 79
Signed-off-by: Chegu Vinod <chegu_vinod@hp.com>, Jim Hull <jim.hull@hp.com>, Craig Hada <craig.hada@hp.com>
Tested-by: Eduardo Habkost <ehabkost@redhat.com>
Reviewed-by: Eduardo Habkost <ehabkost@redhat.com>
Signed-off-by: Blue Swirl <blauwirbel@gmail.com>
											
										 
											2012-07-16 21:31:30 -07:00
										 |  |  |             if (test_bit(i, node_cpumask[j])) { | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |                 numa_fw_cfg[apic_id + 1] = cpu_to_le64(j); | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (i = 0; i < nb_numa_nodes; i++) { | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |         numa_fw_cfg[apic_id_limit + 1 + i] = cpu_to_le64(node_mem[i]); | 
					
						
							| 
									
										
										
										
											2009-04-21 22:31:41 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-01-16 14:50:28 +01:00
										 |  |  |     fw_cfg_add_bytes(fw_cfg, FW_CFG_NUMA, numa_fw_cfg, | 
					
						
							| 
									
										
										
										
											2013-01-23 15:51:18 -02:00
										 |  |  |                      (1 + apic_id_limit + nb_numa_nodes) * | 
					
						
							|  |  |  |                      sizeof(*numa_fw_cfg)); | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return fw_cfg; | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | static long get_file_size(FILE *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     long where, size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* XXX: on Unix systems, using fstat() probably makes more sense */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     where = ftell(f); | 
					
						
							|  |  |  |     fseek(f, 0, SEEK_END); | 
					
						
							|  |  |  |     size = ftell(f); | 
					
						
							|  |  |  |     fseek(f, where, SEEK_SET); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:39 +02:00
										 |  |  | static void load_linux(void *fw_cfg, | 
					
						
							| 
									
										
										
										
											2008-11-08 16:27:07 +00:00
										 |  |  |                        const char *kernel_filename, | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 		       const char *initrd_filename, | 
					
						
							| 
									
										
										
										
											2009-05-18 16:35:58 -04:00
										 |  |  | 		       const char *kernel_cmdline, | 
					
						
							| 
									
										
										
										
											2012-10-23 12:30:10 +02:00
										 |  |  |                        hwaddr max_ram_size) | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     uint16_t protocol; | 
					
						
							| 
									
										
										
										
											2009-05-30 00:52:44 +01:00
										 |  |  |     int setup_size, kernel_size, initrd_size = 0, cmdline_size; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     uint32_t initrd_max; | 
					
						
							| 
									
										
										
										
											2009-11-12 21:53:14 +01:00
										 |  |  |     uint8_t header[8192], *setup, *kernel, *initrd_data; | 
					
						
							| 
									
										
										
										
											2012-10-23 12:30:10 +02:00
										 |  |  |     hwaddr real_addr, prot_addr, cmdline_addr, initrd_addr = 0; | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  |     FILE *f; | 
					
						
							| 
									
										
										
										
											2009-07-13 17:46:42 +02:00
										 |  |  |     char *vmode; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Align to 16 bytes as a paranoia measure */ | 
					
						
							|  |  |  |     cmdline_size = (strlen(kernel_cmdline)+16) & ~15; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* load the kernel header */ | 
					
						
							|  |  |  |     f = fopen(kernel_filename, "rb"); | 
					
						
							|  |  |  |     if (!f || !(kernel_size = get_file_size(f)) || | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:39 +02:00
										 |  |  | 	fread(header, 1, MIN(ARRAY_SIZE(header), kernel_size), f) != | 
					
						
							|  |  |  | 	MIN(ARRAY_SIZE(header), kernel_size)) { | 
					
						
							| 
									
										
										
										
											2009-10-01 09:42:56 -05:00
										 |  |  | 	fprintf(stderr, "qemu: could not load kernel '%s': %s\n", | 
					
						
							|  |  |  | 		kernel_filename, strerror(errno)); | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 	exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* kernel protocol version */ | 
					
						
							| 
									
										
										
										
											2007-11-07 16:54:42 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     fprintf(stderr, "header magic: %#x\n", ldl_p(header+0x202)); | 
					
						
							| 
									
										
										
										
											2007-11-07 16:54:42 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     if (ldl_p(header+0x202) == 0x53726448) | 
					
						
							|  |  |  | 	protocol = lduw_p(header+0x206); | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:39 +02:00
										 |  |  |     else { | 
					
						
							|  |  |  | 	/* This looks like a multiboot kernel. If it is, let's stop
 | 
					
						
							|  |  |  | 	   treating it like a Linux kernel. */ | 
					
						
							| 
									
										
										
										
											2009-12-26 14:13:46 +01:00
										 |  |  |         if (load_multiboot(fw_cfg, f, kernel_filename, initrd_filename, | 
					
						
							|  |  |  |                            kernel_cmdline, kernel_size, header)) | 
					
						
							| 
									
										
										
										
											2009-09-06 16:31:58 +00:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 	protocol = 0; | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:39 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (protocol < 0x200 || !(header[0x211] & 0x01)) { | 
					
						
							|  |  |  | 	/* Low kernel */ | 
					
						
							| 
									
										
										
										
											2008-05-13 18:50:56 +00:00
										 |  |  | 	real_addr    = 0x90000; | 
					
						
							|  |  |  | 	cmdline_addr = 0x9a000 - cmdline_size; | 
					
						
							|  |  |  | 	prot_addr    = 0x10000; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     } else if (protocol < 0x202) { | 
					
						
							|  |  |  | 	/* High but ancient kernel */ | 
					
						
							| 
									
										
										
										
											2008-05-13 18:50:56 +00:00
										 |  |  | 	real_addr    = 0x90000; | 
					
						
							|  |  |  | 	cmdline_addr = 0x9a000 - cmdline_size; | 
					
						
							|  |  |  | 	prot_addr    = 0x100000; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  | 	/* High and recent kernel */ | 
					
						
							| 
									
										
										
										
											2008-05-13 18:50:56 +00:00
										 |  |  | 	real_addr    = 0x10000; | 
					
						
							|  |  |  | 	cmdline_addr = 0x20000; | 
					
						
							|  |  |  | 	prot_addr    = 0x100000; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-07 16:54:42 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     fprintf(stderr, | 
					
						
							| 
									
										
										
										
											2008-07-16 12:13:52 +00:00
										 |  |  | 	    "qemu: real_addr     = 0x" TARGET_FMT_plx "\n" | 
					
						
							|  |  |  | 	    "qemu: cmdline_addr  = 0x" TARGET_FMT_plx "\n" | 
					
						
							|  |  |  | 	    "qemu: prot_addr     = 0x" TARGET_FMT_plx "\n", | 
					
						
							| 
									
										
										
										
											2008-05-13 18:50:56 +00:00
										 |  |  | 	    real_addr, | 
					
						
							|  |  |  | 	    cmdline_addr, | 
					
						
							|  |  |  | 	    prot_addr); | 
					
						
							| 
									
										
										
										
											2007-11-07 16:54:42 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* highest address for loading the initrd */ | 
					
						
							|  |  |  |     if (protocol >= 0x203) | 
					
						
							|  |  |  | 	initrd_max = ldl_p(header+0x22c); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  | 	initrd_max = 0x37ffffff; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-18 16:35:58 -04:00
										 |  |  |     if (initrd_max >= max_ram_size-ACPI_DATA_SIZE) | 
					
						
							|  |  |  |     	initrd_max = max_ram_size-ACPI_DATA_SIZE-1; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-12 21:53:14 +01:00
										 |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_CMDLINE_ADDR, cmdline_addr); | 
					
						
							|  |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_CMDLINE_SIZE, strlen(kernel_cmdline)+1); | 
					
						
							| 
									
										
										
										
											2013-01-16 14:50:25 +01:00
										 |  |  |     fw_cfg_add_string(fw_cfg, FW_CFG_CMDLINE_DATA, kernel_cmdline); | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (protocol >= 0x202) { | 
					
						
							| 
									
										
										
										
											2008-05-13 18:50:56 +00:00
										 |  |  | 	stl_p(header+0x228, cmdline_addr); | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     } else { | 
					
						
							|  |  |  | 	stw_p(header+0x20, 0xA33F); | 
					
						
							|  |  |  | 	stw_p(header+0x22, cmdline_addr-real_addr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 17:46:42 +02:00
										 |  |  |     /* handle vga= parameter */ | 
					
						
							|  |  |  |     vmode = strstr(kernel_cmdline, "vga="); | 
					
						
							|  |  |  |     if (vmode) { | 
					
						
							|  |  |  |         unsigned int video_mode; | 
					
						
							|  |  |  |         /* skip "vga=" */ | 
					
						
							|  |  |  |         vmode += 4; | 
					
						
							|  |  |  |         if (!strncmp(vmode, "normal", 6)) { | 
					
						
							|  |  |  |             video_mode = 0xffff; | 
					
						
							|  |  |  |         } else if (!strncmp(vmode, "ext", 3)) { | 
					
						
							|  |  |  |             video_mode = 0xfffe; | 
					
						
							|  |  |  |         } else if (!strncmp(vmode, "ask", 3)) { | 
					
						
							|  |  |  |             video_mode = 0xfffd; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             video_mode = strtol(vmode, NULL, 0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         stw_p(header+0x1fa, video_mode); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     /* loader type */ | 
					
						
							| 
									
										
										
										
											2012-04-07 09:23:39 +02:00
										 |  |  |     /* High nybble = B reserved for QEMU; low nybble is revision number.
 | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |        If this code is substantially changed, you may want to consider | 
					
						
							|  |  |  |        incrementing the revision. */ | 
					
						
							|  |  |  |     if (protocol >= 0x200) | 
					
						
							|  |  |  | 	header[0x210] = 0xB0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* heap */ | 
					
						
							|  |  |  |     if (protocol >= 0x201) { | 
					
						
							|  |  |  | 	header[0x211] |= 0x80;	/* CAN_USE_HEAP */ | 
					
						
							|  |  |  | 	stw_p(header+0x224, cmdline_addr-real_addr-0x200); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* load initrd */ | 
					
						
							|  |  |  |     if (initrd_filename) { | 
					
						
							|  |  |  | 	if (protocol < 0x200) { | 
					
						
							|  |  |  | 	    fprintf(stderr, "qemu: linux kernel too old to load a ram disk\n"); | 
					
						
							|  |  |  | 	    exit(1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  | 	initrd_size = get_image_size(initrd_filename); | 
					
						
							| 
									
										
										
										
											2010-04-12 10:01:33 +05:30
										 |  |  |         if (initrd_size < 0) { | 
					
						
							|  |  |  |             fprintf(stderr, "qemu: error reading initrd %s\n", | 
					
						
							|  |  |  |                     initrd_filename); | 
					
						
							|  |  |  |             exit(1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  |         initrd_addr = (initrd_max-initrd_size) & ~4095; | 
					
						
							| 
									
										
										
										
											2009-11-12 21:53:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |         initrd_data = g_malloc(initrd_size); | 
					
						
							| 
									
										
										
										
											2009-11-12 21:53:14 +01:00
										 |  |  |         load_image(initrd_filename, initrd_data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fw_cfg_add_i32(fw_cfg, FW_CFG_INITRD_ADDR, initrd_addr); | 
					
						
							|  |  |  |         fw_cfg_add_i32(fw_cfg, FW_CFG_INITRD_SIZE, initrd_size); | 
					
						
							|  |  |  |         fw_cfg_add_bytes(fw_cfg, FW_CFG_INITRD_DATA, initrd_data, initrd_size); | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-13 18:50:56 +00:00
										 |  |  | 	stl_p(header+0x218, initrd_addr); | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 	stl_p(header+0x21c, initrd_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  |     /* load kernel and setup */ | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     setup_size = header[0x1f1]; | 
					
						
							|  |  |  |     if (setup_size == 0) | 
					
						
							|  |  |  | 	setup_size = 4; | 
					
						
							|  |  |  |     setup_size = (setup_size+1)*512; | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  |     kernel_size -= setup_size; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     setup  = g_malloc(setup_size); | 
					
						
							|  |  |  |     kernel = g_malloc(kernel_size); | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  |     fseek(f, 0, SEEK_SET); | 
					
						
							| 
									
										
										
										
											2009-12-25 18:19:17 +00:00
										 |  |  |     if (fread(setup, 1, setup_size, f) != setup_size) { | 
					
						
							|  |  |  |         fprintf(stderr, "fread() failed\n"); | 
					
						
							|  |  |  |         exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (fread(kernel, 1, kernel_size, f) != kernel_size) { | 
					
						
							|  |  |  |         fprintf(stderr, "fread() failed\n"); | 
					
						
							|  |  |  |         exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  |     fclose(f); | 
					
						
							| 
									
										
										
										
											2009-10-01 16:42:33 +02:00
										 |  |  |     memcpy(setup, header, MIN(sizeof(header), setup_size)); | 
					
						
							| 
									
										
										
										
											2009-11-12 21:53:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_ADDR, prot_addr); | 
					
						
							|  |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_KERNEL_SIZE, kernel_size); | 
					
						
							|  |  |  |     fw_cfg_add_bytes(fw_cfg, FW_CFG_KERNEL_DATA, kernel, kernel_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_SETUP_ADDR, real_addr); | 
					
						
							|  |  |  |     fw_cfg_add_i32(fw_cfg, FW_CFG_SETUP_SIZE, setup_size); | 
					
						
							|  |  |  |     fw_cfg_add_bytes(fw_cfg, FW_CFG_SETUP_DATA, setup, setup_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-08 13:35:07 +02:00
										 |  |  |     option_rom[nb_option_roms].name = "linuxboot.bin"; | 
					
						
							|  |  |  |     option_rom[nb_option_roms].bootindex = 0; | 
					
						
							| 
									
										
										
										
											2009-11-12 21:53:14 +01:00
										 |  |  |     nb_option_roms++; | 
					
						
							| 
									
										
										
										
											2007-05-19 21:04:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-14 21:46:48 +00:00
										 |  |  | #define NE2000_NB_MAX 6
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-13 08:32:37 +00:00
										 |  |  | static const int ne2000_io[NE2000_NB_MAX] = { 0x300, 0x320, 0x340, 0x360, | 
					
						
							|  |  |  |                                               0x280, 0x380 }; | 
					
						
							|  |  |  | static const int ne2000_irq[NE2000_NB_MAX] = { 9, 10, 11, 3, 4, 5 }; | 
					
						
							| 
									
										
										
										
											2004-03-14 21:46:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-13 08:32:37 +00:00
										 |  |  | static const int parallel_io[MAX_PARALLEL_PORTS] = { 0x378, 0x278, 0x3bc }; | 
					
						
							|  |  |  | static const int parallel_irq[MAX_PARALLEL_PORTS] = { 7, 7, 7 }; | 
					
						
							| 
									
										
										
										
											2005-01-15 12:02:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  | void pc_init_ne2k_isa(ISABus *bus, NICInfo *nd) | 
					
						
							| 
									
										
										
										
											2006-02-05 04:14:41 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     static int nb_ne2k = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (nb_ne2k == NE2000_NB_MAX) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |     isa_ne2000_init(bus, ne2000_io[nb_ne2k], | 
					
						
							| 
									
										
										
										
											2009-09-10 11:43:33 +02:00
										 |  |  |                     ne2000_irq[nb_ne2k], nd); | 
					
						
							| 
									
										
										
										
											2006-02-05 04:14:41 +00:00
										 |  |  |     nb_ne2k++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-19 07:47:42 +00:00
										 |  |  | DeviceState *cpu_get_current_apic(void) | 
					
						
							| 
									
										
										
										
											2010-06-19 10:42:34 +03:00
										 |  |  | { | 
					
						
							|  |  |  |     if (cpu_single_env) { | 
					
						
							|  |  |  |         return cpu_single_env->apic_state; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:15 +09:00
										 |  |  | void pc_acpi_smi_interrupt(void *opaque, int irq, int level) | 
					
						
							| 
									
										
										
										
											2010-03-29 19:23:52 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:23 +01:00
										 |  |  |     CPUX86State *s = opaque; | 
					
						
							| 
									
										
										
										
											2010-03-29 19:23:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (level) { | 
					
						
							|  |  |  |         cpu_interrupt(s, CPU_INTERRUPT_SMI); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:15 +09:00
										 |  |  | void pc_cpus_init(const char *cpu_model) | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:10 +09:00
										 |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* init CPUs */ | 
					
						
							|  |  |  |     if (cpu_model == NULL) { | 
					
						
							|  |  |  | #ifdef TARGET_X86_64
 | 
					
						
							|  |  |  |         cpu_model = "qemu64"; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |         cpu_model = "qemu32"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-13 22:35:39 +02:00
										 |  |  |     for (i = 0; i < smp_cpus; i++) { | 
					
						
							|  |  |  |         if (!cpu_x86_init(cpu_model)) { | 
					
						
							|  |  |  |             exit(1); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:10 +09:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-03 10:47:27 +01:00
										 |  |  | void pc_acpi_init(const char *default_dsdt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     char *filename = NULL, *arg = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (acpi_tables != NULL) { | 
					
						
							|  |  |  |         /* manually set via -acpitable, leave it alone */ | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     filename = qemu_find_file(QEMU_FILE_TYPE_BIOS, default_dsdt); | 
					
						
							|  |  |  |     if (filename == NULL) { | 
					
						
							|  |  |  |         fprintf(stderr, "WARNING: failed to find %s\n", default_dsdt); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     arg = g_strdup_printf("file=%s", filename); | 
					
						
							|  |  |  |     if (acpi_table_add(arg) != 0) { | 
					
						
							|  |  |  |         fprintf(stderr, "WARNING: failed to load %s\n", filename); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     g_free(arg); | 
					
						
							|  |  |  |     g_free(filename); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-04 14:31:55 +03:00
										 |  |  | void *pc_memory_init(MemoryRegion *system_memory, | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:16 +03:00
										 |  |  |                     const char *kernel_filename, | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:15 +09:00
										 |  |  |                     const char *kernel_cmdline, | 
					
						
							|  |  |  |                     const char *initrd_filename, | 
					
						
							| 
									
										
										
										
											2011-04-11 19:48:11 +01:00
										 |  |  |                     ram_addr_t below_4g_mem_size, | 
					
						
							| 
									
										
										
										
											2011-08-15 17:17:38 +03:00
										 |  |  |                     ram_addr_t above_4g_mem_size, | 
					
						
							| 
									
										
										
										
											2011-09-21 20:49:29 +02:00
										 |  |  |                     MemoryRegion *rom_memory, | 
					
						
							| 
									
										
										
										
											2011-08-15 17:17:38 +03:00
										 |  |  |                     MemoryRegion **ram_memory) | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-02-21 23:18:51 -08:00
										 |  |  |     int linux_boot, i; | 
					
						
							|  |  |  |     MemoryRegion *ram, *option_rom_mr; | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |     MemoryRegion *ram_below_4g, *ram_above_4g; | 
					
						
							| 
									
										
										
										
											2010-05-20 09:14:04 +03:00
										 |  |  |     void *fw_cfg; | 
					
						
							| 
									
										
										
										
											2005-07-23 19:05:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-14 12:20:30 +00:00
										 |  |  |     linux_boot = (kernel_filename != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |     /* Allocate RAM.  We allocate it as a single memory region and use
 | 
					
						
							| 
									
										
										
										
											2011-11-29 16:52:39 +08:00
										 |  |  |      * aliases to address portions of it, mostly for backwards compatibility | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |      * with older qemus that used qemu_ram_alloc(). | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     ram = g_malloc(sizeof(*ram)); | 
					
						
							| 
									
										
										
										
											2011-12-20 15:59:12 +02:00
										 |  |  |     memory_region_init_ram(ram, "pc.ram", | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |                            below_4g_mem_size + above_4g_mem_size); | 
					
						
							| 
									
										
										
										
											2011-12-20 15:59:12 +02:00
										 |  |  |     vmstate_register_ram_global(ram); | 
					
						
							| 
									
										
										
										
											2011-08-15 17:17:38 +03:00
										 |  |  |     *ram_memory = ram; | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     ram_below_4g = g_malloc(sizeof(*ram_below_4g)); | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |     memory_region_init_alias(ram_below_4g, "ram-below-4g", ram, | 
					
						
							|  |  |  |                              0, below_4g_mem_size); | 
					
						
							|  |  |  |     memory_region_add_subregion(system_memory, 0, ram_below_4g); | 
					
						
							| 
									
										
										
										
											2010-07-06 10:37:17 -06:00
										 |  |  |     if (above_4g_mem_size > 0) { | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |         ram_above_4g = g_malloc(sizeof(*ram_above_4g)); | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |         memory_region_init_alias(ram_above_4g, "ram-above-4g", ram, | 
					
						
							|  |  |  |                                  below_4g_mem_size, above_4g_mem_size); | 
					
						
							|  |  |  |         memory_region_add_subregion(system_memory, 0x100000000ULL, | 
					
						
							|  |  |  |                                     ram_above_4g); | 
					
						
							| 
									
										
										
										
											2010-07-06 10:37:17 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-09-15 16:01:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-21 23:18:51 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Initialize PC system firmware */ | 
					
						
							|  |  |  |     pc_system_firmware_init(rom_memory); | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-20 22:09:37 -05:00
										 |  |  |     option_rom_mr = g_malloc(sizeof(*option_rom_mr)); | 
					
						
							| 
									
										
										
										
											2011-12-20 15:59:12 +02:00
										 |  |  |     memory_region_init_ram(option_rom_mr, "pc.rom", PC_ROM_SIZE); | 
					
						
							|  |  |  |     vmstate_register_ram_global(option_rom_mr); | 
					
						
							| 
									
										
										
										
											2011-09-21 20:49:29 +02:00
										 |  |  |     memory_region_add_subregion_overlap(rom_memory, | 
					
						
							| 
									
										
										
										
											2011-07-26 14:26:17 +03:00
										 |  |  |                                         PC_ROM_MIN_VGA, | 
					
						
							|  |  |  |                                         option_rom_mr, | 
					
						
							|  |  |  |                                         1); | 
					
						
							| 
									
										
										
										
											2009-04-09 20:59:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:38 +02:00
										 |  |  |     fw_cfg = bochs_bios_init(); | 
					
						
							| 
									
										
										
										
											2010-01-08 15:25:40 +01:00
										 |  |  |     rom_set_fw(fw_cfg); | 
					
						
							| 
									
										
										
										
											2009-06-29 15:37:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-09 20:59:05 +00:00
										 |  |  |     if (linux_boot) { | 
					
						
							| 
									
										
										
										
											2010-05-20 09:14:04 +03:00
										 |  |  |         load_linux(fw_cfg, kernel_filename, initrd_filename, kernel_cmdline, below_4g_mem_size); | 
					
						
							| 
									
										
										
										
											2009-04-09 20:59:05 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < nb_option_roms; i++) { | 
					
						
							| 
									
										
										
										
											2010-12-08 13:35:07 +02:00
										 |  |  |         rom_add_option(option_rom[i].name, option_rom[i].bootindex); | 
					
						
							| 
									
										
										
										
											2009-06-17 09:05:30 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-06-04 14:31:55 +03:00
										 |  |  |     return fw_cfg; | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:11 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:15 +09:00
										 |  |  | qemu_irq *pc_allocate_cpu_irq(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return qemu_allocate_irqs(pic_irq_request, NULL, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  | DeviceState *pc_vga_init(ISABus *isa_bus, PCIBus *pci_bus) | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:12 +09:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-12 14:29:41 -06:00
										 |  |  |     DeviceState *dev = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-08 12:47:45 +02:00
										 |  |  |     if (pci_bus) { | 
					
						
							|  |  |  |         PCIDevice *pcidev = pci_vga_init(pci_bus); | 
					
						
							|  |  |  |         dev = pcidev ? &pcidev->qdev : NULL; | 
					
						
							|  |  |  |     } else if (isa_bus) { | 
					
						
							|  |  |  |         ISADevice *isadev = isa_vga_init(isa_bus); | 
					
						
							|  |  |  |         dev = isadev ? &isadev->qdev : NULL; | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:12 +09:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-12-12 14:29:41 -06:00
										 |  |  |     return dev; | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:12 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-22 08:00:52 +00:00
										 |  |  | static void cpu_request_exit(void *opaque, int irq, int level) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:23 +01:00
										 |  |  |     CPUX86State *env = cpu_single_env; | 
					
						
							| 
									
										
										
										
											2010-05-22 08:00:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (env && level) { | 
					
						
							|  |  |  |         cpu_exit(env); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  | static const MemoryRegionOps ioport80_io_ops = { | 
					
						
							|  |  |  |     .write = ioport80_write, | 
					
						
							| 
									
										
										
										
											2013-01-09 18:10:22 +00:00
										 |  |  |     .read = ioport80_read, | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  |     .endianness = DEVICE_NATIVE_ENDIAN, | 
					
						
							|  |  |  |     .impl = { | 
					
						
							|  |  |  |         .min_access_size = 1, | 
					
						
							|  |  |  |         .max_access_size = 1, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const MemoryRegionOps ioportF0_io_ops = { | 
					
						
							|  |  |  |     .write = ioportF0_write, | 
					
						
							| 
									
										
										
										
											2013-01-09 18:10:22 +00:00
										 |  |  |     .read = ioportF0_read, | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  |     .endianness = DEVICE_NATIVE_ENDIAN, | 
					
						
							|  |  |  |     .impl = { | 
					
						
							|  |  |  |         .min_access_size = 1, | 
					
						
							|  |  |  |         .max_access_size = 1, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  | void pc_basic_device_init(ISABus *isa_bus, qemu_irq *gsi, | 
					
						
							| 
									
										
										
										
											2011-05-03 17:06:54 +01:00
										 |  |  |                           ISADevice **rtc_state, | 
					
						
							| 
									
										
										
										
											2011-10-20 16:37:26 +02:00
										 |  |  |                           ISADevice **floppy, | 
					
						
							| 
									
										
										
										
											2011-05-03 17:06:54 +01:00
										 |  |  |                           bool no_vmport) | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     DriveInfo *fd[MAX_FD]; | 
					
						
							| 
									
										
										
										
											2012-02-01 20:31:41 +01:00
										 |  |  |     DeviceState *hpet = NULL; | 
					
						
							|  |  |  |     int pit_isa_irq = 0; | 
					
						
							|  |  |  |     qemu_irq pit_alt_irq = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-13 14:15:40 +02:00
										 |  |  |     qemu_irq rtc_irq = NULL; | 
					
						
							| 
									
										
										
										
											2010-05-22 07:59:01 +00:00
										 |  |  |     qemu_irq *a20_line; | 
					
						
							| 
									
										
										
										
											2011-11-14 15:07:01 +00:00
										 |  |  |     ISADevice *i8042, *port92, *vmmouse, *pit = NULL; | 
					
						
							| 
									
										
										
										
											2010-05-22 08:00:52 +00:00
										 |  |  |     qemu_irq *cpu_exit_irq; | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  |     MemoryRegion *ioport80_io = g_new(MemoryRegion, 1); | 
					
						
							|  |  |  |     MemoryRegion *ioportF0_io = g_new(MemoryRegion, 1); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  |     memory_region_init_io(ioport80_io, &ioport80_io_ops, NULL, "ioport80", 1); | 
					
						
							|  |  |  |     memory_region_add_subregion(isa_bus->address_space_io, 0x80, ioport80_io); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-19 12:50:08 +01:00
										 |  |  |     memory_region_init_io(ioportF0_io, &ioportF0_io_ops, NULL, "ioportF0", 1); | 
					
						
							|  |  |  |     memory_region_add_subregion(isa_bus->address_space_io, 0xf0, ioportF0_io); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 20:28:49 +01:00
										 |  |  |     /*
 | 
					
						
							|  |  |  |      * Check if an HPET shall be created. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * Without KVM_CAP_PIT_STATE2, we cannot switch off the in-kernel PIT | 
					
						
							|  |  |  |      * when the HPET wants to take over. Thus we have to disable the latter. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     if (!no_hpet && (!kvm_irqchip_in_kernel() || kvm_has_pit_state2())) { | 
					
						
							| 
									
										
										
										
											2012-02-01 20:31:41 +01:00
										 |  |  |         hpet = sysbus_try_create_simple("hpet", HPET_BASE, NULL); | 
					
						
							| 
									
										
										
										
											2010-06-13 14:15:38 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-05 14:35:00 +00:00
										 |  |  |         if (hpet) { | 
					
						
							| 
									
										
										
										
											2011-10-07 09:19:35 +02:00
										 |  |  |             for (i = 0; i < GSI_NUM_PINS; i++) { | 
					
						
							| 
									
										
										
										
											2013-01-20 02:47:33 +01:00
										 |  |  |                 sysbus_connect_irq(SYS_BUS_DEVICE(hpet), i, gsi[i]); | 
					
						
							| 
									
										
										
										
											2011-02-05 14:35:00 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2012-02-01 20:31:41 +01:00
										 |  |  |             pit_isa_irq = -1; | 
					
						
							|  |  |  |             pit_alt_irq = qdev_get_gpio_in(hpet, HPET_LEGACY_PIT_INT); | 
					
						
							|  |  |  |             rtc_irq = qdev_get_gpio_in(hpet, HPET_LEGACY_RTC_INT); | 
					
						
							| 
									
										
										
										
											2010-06-13 14:15:38 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |     *rtc_state = rtc_init(isa_bus, 2000, rtc_irq); | 
					
						
							| 
									
										
										
										
											2010-06-13 14:15:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     qemu_register_boot_set(pc_boot_set, *rtc_state); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-14 15:07:01 +00:00
										 |  |  |     if (!xen_enabled()) { | 
					
						
							|  |  |  |         if (kvm_irqchip_in_kernel()) { | 
					
						
							|  |  |  |             pit = kvm_pit_init(isa_bus, 0x40); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             pit = pit_init(isa_bus, 0x40, pit_isa_irq, pit_alt_irq); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (hpet) { | 
					
						
							|  |  |  |             /* connect PIT to output control line of the HPET */ | 
					
						
							|  |  |  |             qdev_connect_gpio_out(hpet, 0, qdev_get_gpio_in(&pit->qdev, 0)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         pcspk_init(isa_bus, pit); | 
					
						
							| 
									
										
										
										
											2012-02-01 20:31:41 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for(i = 0; i < MAX_SERIAL_PORTS; i++) { | 
					
						
							|  |  |  |         if (serial_hds[i]) { | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |             serial_isa_init(isa_bus, i, serial_hds[i]); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for(i = 0; i < MAX_PARALLEL_PORTS; i++) { | 
					
						
							|  |  |  |         if (parallel_hds[i]) { | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |             parallel_init(isa_bus, i, parallel_hds[i]); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-18 15:19:06 +01:00
										 |  |  |     a20_line = qemu_allocate_irqs(handle_a20_line_change, | 
					
						
							|  |  |  |                                   x86_env_get_cpu(first_cpu), 2); | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |     i8042 = isa_create_simple(isa_bus, "i8042"); | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  |     i8042_setup_a20_line(i8042, &a20_line[0]); | 
					
						
							| 
									
										
										
										
											2011-05-03 17:06:54 +01:00
										 |  |  |     if (!no_vmport) { | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |         vmport_init(isa_bus); | 
					
						
							|  |  |  |         vmmouse = isa_try_create(isa_bus, "vmmouse"); | 
					
						
							| 
									
										
										
										
											2011-05-03 17:06:54 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         vmmouse = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-02-05 14:34:52 +00:00
										 |  |  |     if (vmmouse) { | 
					
						
							|  |  |  |         qdev_prop_set_ptr(&vmmouse->qdev, "ps2_mouse", i8042); | 
					
						
							| 
									
										
										
										
											2011-03-21 10:52:24 +01:00
										 |  |  |         qdev_init_nofail(&vmmouse->qdev); | 
					
						
							| 
									
										
										
										
											2011-02-05 14:34:52 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |     port92 = isa_create_simple(isa_bus, "port92"); | 
					
						
							| 
									
										
										
										
											2011-01-06 18:24:35 +00:00
										 |  |  |     port92_init(port92, &a20_line[1]); | 
					
						
							| 
									
										
										
										
											2010-05-22 07:59:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-22 08:00:52 +00:00
										 |  |  |     cpu_exit_irq = qemu_allocate_irqs(cpu_request_exit, NULL, 1); | 
					
						
							|  |  |  |     DMA_init(0, cpu_exit_irq); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for(i = 0; i < MAX_FD; i++) { | 
					
						
							|  |  |  |         fd[i] = drive_get(IF_FLOPPY, 0, i); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-12-15 22:09:51 +01:00
										 |  |  |     *floppy = fdctrl_init_isa(isa_bus, fd); | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:13 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-14 15:54:01 -05:00
										 |  |  | void pc_nic_init(ISABus *isa_bus, PCIBus *pci_bus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < nb_nics; i++) { | 
					
						
							|  |  |  |         NICInfo *nd = &nd_table[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!pci_bus || (nd->model && strcmp(nd->model, "ne2k_isa") == 0)) { | 
					
						
							|  |  |  |             pc_init_ne2k_isa(isa_bus, nd); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             pci_nic_init_nofail(nd, "e1000", NULL); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:15 +09:00
										 |  |  | void pc_pci_device_init(PCIBus *pci_bus) | 
					
						
							| 
									
										
										
										
											2010-05-14 16:29:14 +09:00
										 |  |  | { | 
					
						
							|  |  |  |     int max_bus; | 
					
						
							|  |  |  |     int bus; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     max_bus = drive_get_max_bus(IF_SCSI); | 
					
						
							|  |  |  |     for (bus = 0; bus <= max_bus; bus++) { | 
					
						
							|  |  |  |         pci_create_simple(pci_bus, -1, "lsi53c895a"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-11-14 15:54:01 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | void ioapic_init_gsi(GSIState *gsi_state, const char *parent_name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DeviceState *dev; | 
					
						
							|  |  |  |     SysBusDevice *d; | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (kvm_irqchip_in_kernel()) { | 
					
						
							|  |  |  |         dev = qdev_create(NULL, "kvm-ioapic"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         dev = qdev_create(NULL, "ioapic"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (parent_name) { | 
					
						
							|  |  |  |         object_property_add_child(object_resolve_path(parent_name, NULL), | 
					
						
							|  |  |  |                                   "ioapic", OBJECT(dev), NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     qdev_init_nofail(dev); | 
					
						
							| 
									
										
										
										
											2013-01-20 02:47:33 +01:00
										 |  |  |     d = SYS_BUS_DEVICE(dev); | 
					
						
							| 
									
										
										
										
											2012-11-14 15:54:01 -05:00
										 |  |  |     sysbus_mmio_map(d, 0, 0xfec00000); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < IOAPIC_NUM_PINS; i++) { | 
					
						
							|  |  |  |         gsi_state->ioapic_irq[i] = qdev_get_gpio_in(dev, i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |