2003-06-24 21:42:40 +08:00
|
|
|
/*
|
2004-03-14 20:20:30 +08:00
|
|
|
* QEMU System Emulator
|
2007-09-17 05:08:06 +08:00
|
|
|
*
|
2008-01-07 01:21:48 +08:00
|
|
|
* Copyright (c) 2003-2008 Fabrice Bellard
|
2007-09-17 05:08:06 +08:00
|
|
|
*
|
2003-06-26 00:20:35 +08: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.
|
2003-06-24 21:42:40 +08:00
|
|
|
*/
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <errno.h>
|
2004-04-01 07:37:16 +08:00
|
|
|
#include <sys/time.h>
|
2006-08-06 21:36:11 +08:00
|
|
|
#include <zlib.h>
|
2004-04-01 07:37:16 +08:00
|
|
|
|
2009-07-27 22:12:56 +08:00
|
|
|
/* Needed early for CONFIG_BSD etc. */
|
2009-03-08 00:52:02 +08:00
|
|
|
#include "config-host.h"
|
|
|
|
|
2004-04-01 07:37:16 +08:00
|
|
|
#ifndef _WIN32
|
2009-05-30 07:52:44 +08:00
|
|
|
#include <libgen.h>
|
2009-02-28 06:09:45 +08:00
|
|
|
#include <pwd.h>
|
2004-04-01 07:37:16 +08:00
|
|
|
#include <sys/times.h>
|
2003-06-25 08:07:40 +08:00
|
|
|
#include <sys/wait.h>
|
2004-04-01 07:37:16 +08:00
|
|
|
#include <termios.h>
|
|
|
|
#include <sys/mman.h>
|
2003-06-25 08:07:40 +08:00
|
|
|
#include <sys/ioctl.h>
|
2008-11-08 00:55:48 +08:00
|
|
|
#include <sys/resource.h>
|
2003-06-25 08:07:40 +08:00
|
|
|
#include <sys/socket.h>
|
2004-09-14 05:37:34 +08:00
|
|
|
#include <netinet/in.h>
|
2008-11-08 00:55:48 +08:00
|
|
|
#include <net/if.h>
|
|
|
|
#include <arpa/inet.h>
|
2004-09-06 07:09:03 +08:00
|
|
|
#include <dirent.h>
|
2005-11-16 06:16:05 +08:00
|
|
|
#include <netdb.h>
|
2007-09-13 20:39:35 +08:00
|
|
|
#include <sys/select.h>
|
2009-07-27 22:12:56 +08:00
|
|
|
#ifdef CONFIG_BSD
|
2004-05-13 03:32:15 +08:00
|
|
|
#include <sys/stat.h>
|
2009-11-30 01:00:41 +08:00
|
|
|
#if defined(__FreeBSD__) || defined(__FreeBSD_kernel__) || defined(__DragonFly__)
|
2004-05-13 03:32:15 +08:00
|
|
|
#include <libutil.h>
|
2008-11-08 00:55:48 +08:00
|
|
|
#else
|
|
|
|
#include <util.h>
|
2008-08-16 02:33:42 +08:00
|
|
|
#endif
|
2009-11-30 22:42:59 +08:00
|
|
|
#else
|
2008-10-01 02:12:18 +08:00
|
|
|
#ifdef __linux__
|
2004-05-13 03:32:15 +08:00
|
|
|
#include <pty.h>
|
|
|
|
#include <malloc.h>
|
2004-05-13 03:11:15 +08:00
|
|
|
#include <linux/rtc.h>
|
2009-07-02 15:34:17 +08:00
|
|
|
#include <sys/prctl.h>
|
2007-09-17 04:03:23 +08:00
|
|
|
|
|
|
|
/* For the benefit of older linux systems which don't supply it,
|
|
|
|
we use a local copy of hpet.h. */
|
|
|
|
/* #include <linux/hpet.h> */
|
|
|
|
#include "hpet.h"
|
|
|
|
|
2005-11-11 07:58:52 +08:00
|
|
|
#include <linux/ppdev.h>
|
2007-02-18 07:44:43 +08:00
|
|
|
#include <linux/parport.h>
|
2008-10-01 02:12:18 +08:00
|
|
|
#endif
|
|
|
|
#ifdef __sun__
|
2007-02-18 06:54:49 +08:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/ethernet.h>
|
|
|
|
#include <sys/sockio.h>
|
|
|
|
#include <netinet/arp.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <netinet/in_systm.h>
|
|
|
|
#include <netinet/ip.h>
|
|
|
|
#include <netinet/ip_icmp.h> // must come after ip.h
|
|
|
|
#include <netinet/udp.h>
|
|
|
|
#include <netinet/tcp.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <syslog.h>
|
|
|
|
#include <stropts.h>
|
2009-10-03 03:32:12 +08:00
|
|
|
/* See MySQL bug #7156 (http://bugs.mysql.com/bug.php?id=7156) for
|
|
|
|
discussion about Solaris header problems */
|
|
|
|
extern int madvise(caddr_t, size_t, int);
|
2004-04-01 07:37:16 +08:00
|
|
|
#endif
|
2004-05-13 03:32:15 +08:00
|
|
|
#endif
|
2006-04-26 06:36:06 +08:00
|
|
|
#endif
|
2004-04-01 07:37:16 +08:00
|
|
|
|
2008-08-24 18:34:20 +08:00
|
|
|
#if defined(__OpenBSD__)
|
|
|
|
#include <util.h>
|
|
|
|
#endif
|
|
|
|
|
2008-07-19 17:56:24 +08:00
|
|
|
#if defined(CONFIG_VDE)
|
|
|
|
#include <libvdeplug.h>
|
|
|
|
#endif
|
|
|
|
|
2004-04-01 07:37:16 +08:00
|
|
|
#ifdef _WIN32
|
2009-03-09 00:26:59 +08:00
|
|
|
#include <windows.h>
|
2007-12-17 12:42:29 +08:00
|
|
|
#include <mmsystem.h>
|
2004-04-01 07:37:16 +08:00
|
|
|
#endif
|
|
|
|
|
2004-04-05 04:22:28 +08:00
|
|
|
#ifdef CONFIG_SDL
|
2009-06-19 02:11:03 +08:00
|
|
|
#if defined(__APPLE__) || defined(main)
|
2009-06-13 19:19:11 +08:00
|
|
|
#include <SDL.h>
|
2009-02-16 04:18:41 +08:00
|
|
|
int qemu_main(int argc, char **argv, char **envp);
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
2009-06-19 02:11:03 +08:00
|
|
|
return qemu_main(argc, argv, NULL);
|
2009-02-16 04:18:41 +08:00
|
|
|
}
|
|
|
|
#undef main
|
|
|
|
#define main qemu_main
|
2004-07-11 00:26:15 +08:00
|
|
|
#endif
|
2004-04-05 04:22:28 +08:00
|
|
|
#endif /* CONFIG_SDL */
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2005-03-02 05:37:28 +08:00
|
|
|
#ifdef CONFIG_COCOA
|
|
|
|
#undef main
|
|
|
|
#define main qemu_main
|
|
|
|
#endif /* CONFIG_COCOA */
|
|
|
|
|
2009-03-07 23:32:56 +08:00
|
|
|
#include "hw/hw.h"
|
|
|
|
#include "hw/boards.h"
|
|
|
|
#include "hw/usb.h"
|
|
|
|
#include "hw/pcmcia.h"
|
|
|
|
#include "hw/pc.h"
|
|
|
|
#include "hw/audiodev.h"
|
|
|
|
#include "hw/isa.h"
|
|
|
|
#include "hw/baum.h"
|
|
|
|
#include "hw/bt.h"
|
2009-04-25 20:56:19 +08:00
|
|
|
#include "hw/watchdog.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-18 02:59:56 +08:00
|
|
|
#include "hw/smbios.h"
|
2009-04-22 23:19:10 +08:00
|
|
|
#include "hw/xen.h"
|
2009-07-15 19:59:26 +08:00
|
|
|
#include "hw/qdev.h"
|
2009-10-01 22:42:33 +08:00
|
|
|
#include "hw/loader.h"
|
2009-03-11 05:43:35 +08:00
|
|
|
#include "bt-host.h"
|
2009-03-07 23:32:56 +08:00
|
|
|
#include "net.h"
|
2009-11-26 02:48:54 +08:00
|
|
|
#include "net/slirp.h"
|
2009-03-07 23:32:56 +08:00
|
|
|
#include "monitor.h"
|
|
|
|
#include "console.h"
|
|
|
|
#include "sysemu.h"
|
|
|
|
#include "gdbstub.h"
|
|
|
|
#include "qemu-timer.h"
|
|
|
|
#include "qemu-char.h"
|
|
|
|
#include "cache-utils.h"
|
|
|
|
#include "block.h"
|
2009-11-02 21:40:58 +08:00
|
|
|
#include "block_int.h"
|
|
|
|
#include "block-migration.h"
|
2009-03-28 16:24:44 +08:00
|
|
|
#include "dma.h"
|
2009-03-07 23:32:56 +08:00
|
|
|
#include "audio/audio.h"
|
|
|
|
#include "migration.h"
|
|
|
|
#include "kvm.h"
|
|
|
|
#include "balloon.h"
|
2009-05-18 22:42:09 +08:00
|
|
|
#include "qemu-option.h"
|
2009-07-31 18:25:35 +08:00
|
|
|
#include "qemu-config.h"
|
2009-12-11 03:16:04 +08:00
|
|
|
#include "qemu-objects.h"
|
2009-03-07 23:32:56 +08:00
|
|
|
|
2003-06-24 21:42:40 +08:00
|
|
|
#include "disas.h"
|
2003-06-30 18:03:06 +08:00
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
#include "exec-all.h"
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2009-03-07 23:32:56 +08:00
|
|
|
#include "qemu_socket.h"
|
|
|
|
|
2009-06-24 20:42:30 +08:00
|
|
|
#include "slirp/libslirp.h"
|
2009-03-07 23:32:56 +08:00
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
#include "qemu-queue.h"
|
|
|
|
|
2008-10-04 15:25:46 +08:00
|
|
|
//#define DEBUG_NET
|
|
|
|
//#define DEBUG_SLIRP
|
2003-07-27 02:11:40 +08:00
|
|
|
|
2004-07-09 05:17:50 +08:00
|
|
|
#define DEFAULT_RAM_SIZE 128
|
2003-08-11 05:52:11 +08:00
|
|
|
|
virtio-console: qdev conversion, new virtio-serial-bus
This commit converts the virtio-console device to create a new
virtio-serial bus that can host console and generic serial ports. The
file hosting this code is now called virtio-serial-bus.c.
The virtio console is now a very simple qdev device that sits on the
virtio-serial-bus and communicates between the bus and qemu's chardevs.
This commit also includes a few changes to the virtio backing code for
pci and s390 to spawn the virtio-serial bus.
As a result of the qdev conversion, we get rid of a lot of legacy code.
The old-style way of instantiating a virtio console using
-virtioconsole ...
is maintained, but the new, preferred way is to use
-device virtio-serial -device virtconsole,chardev=...
With this commit, multiple devices as well as multiple ports with a
single device can be supported.
For multiple ports support, each port gets an IO vq pair. Since the
guest needs to know in advance how many vqs a particular device will
need, we have to set this number as a property of the virtio-serial
device and also as a config option.
In addition, we also spawn a pair of control IO vqs. This is an internal
channel meant for guest-host communication for things like port
open/close, sending port properties over to the guest, etc.
This commit is a part of a series of other commits to get the full
implementation of multiport support. Future commits will add other
support as well as ride on the savevm version that we bump up here.
Signed-off-by: Amit Shah <amit.shah@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2010-01-20 03:06:52 +08:00
|
|
|
#define MAX_VIRTIO_CONSOLES 1
|
|
|
|
|
2009-05-30 07:52:44 +08:00
|
|
|
static const char *data_dir;
|
2007-10-05 21:08:35 +08:00
|
|
|
const char *bios_name = NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
/* Note: drives_table[MAX_DRIVES] is a dummy block driver if none available
|
2006-08-06 05:31:00 +08:00
|
|
|
to store the VM snapshots */
|
2009-09-12 15:36:22 +08:00
|
|
|
struct drivelist drives = QTAILQ_HEAD_INITIALIZER(drives);
|
|
|
|
struct driveoptlist driveopts = QTAILQ_HEAD_INITIALIZER(driveopts);
|
2008-09-28 08:42:12 +08:00
|
|
|
enum vga_retrace_method vga_retrace_method = VGA_RETRACE_DUMB;
|
2009-05-22 05:54:00 +08:00
|
|
|
DisplayType display_type = DT_DEFAULT;
|
2004-12-13 00:56:30 +08:00
|
|
|
const char* keyboard_layout = NULL;
|
2009-10-02 05:12:16 +08:00
|
|
|
ram_addr_t ram_size;
|
2010-03-02 07:25:08 +08:00
|
|
|
const char *mem_path = NULL;
|
|
|
|
#ifdef MAP_POPULATE
|
|
|
|
int mem_prealloc = 0; /* force preallocation of physical target memory */
|
|
|
|
#endif
|
2004-03-15 05:44:30 +08:00
|
|
|
int nb_nics;
|
2005-11-16 06:16:05 +08:00
|
|
|
NICInfo nd_table[MAX_NICS];
|
2004-04-01 03:00:16 +08:00
|
|
|
int vm_running;
|
2009-07-28 05:17:51 +08:00
|
|
|
int autostart;
|
2008-02-17 19:42:19 +08:00
|
|
|
static int rtc_utc = 1;
|
|
|
|
static int rtc_date_offset = -1; /* -1 means no change */
|
2009-09-15 19:36:04 +08:00
|
|
|
QEMUClock *rtc_clock;
|
2009-12-08 20:11:45 +08:00
|
|
|
int vga_interface_type = VGA_NONE;
|
2005-04-07 04:32:23 +08:00
|
|
|
#ifdef TARGET_SPARC
|
|
|
|
int graphic_width = 1024;
|
|
|
|
int graphic_height = 768;
|
2007-04-22 03:45:49 +08:00
|
|
|
int graphic_depth = 8;
|
2005-04-07 04:32:23 +08:00
|
|
|
#else
|
2004-07-09 05:17:50 +08:00
|
|
|
int graphic_width = 800;
|
|
|
|
int graphic_height = 600;
|
2004-06-22 00:46:10 +08:00
|
|
|
int graphic_depth = 15;
|
2007-04-22 03:45:49 +08:00
|
|
|
#endif
|
2008-10-02 03:38:09 +08:00
|
|
|
static int full_screen = 0;
|
2008-11-16 19:34:07 +08:00
|
|
|
#ifdef CONFIG_SDL
|
2008-10-02 03:38:09 +08:00
|
|
|
static int no_frame = 0;
|
2008-11-16 19:34:07 +08:00
|
|
|
#endif
|
2006-12-11 10:08:05 +08:00
|
|
|
int no_quit = 0;
|
2004-08-25 05:13:40 +08:00
|
|
|
CharDriverState *serial_hds[MAX_SERIAL_PORTS];
|
2005-01-15 20:02:56 +08:00
|
|
|
CharDriverState *parallel_hds[MAX_PARALLEL_PORTS];
|
2009-01-16 04:05:25 +08:00
|
|
|
CharDriverState *virtcon_hds[MAX_VIRTIO_CONSOLES];
|
2005-05-01 00:10:35 +08:00
|
|
|
#ifdef TARGET_I386
|
|
|
|
int win2k_install_hack = 0;
|
2009-01-16 04:11:34 +08:00
|
|
|
int rtc_td_hack = 0;
|
2005-05-01 00:10:35 +08:00
|
|
|
#endif
|
2005-11-05 22:22:28 +08:00
|
|
|
int usb_enabled = 0;
|
2009-04-06 04:08:59 +08:00
|
|
|
int singlestep = 0;
|
2005-11-22 07:25:50 +08:00
|
|
|
int smp_cpus = 1;
|
2009-07-23 23:03:42 +08:00
|
|
|
int max_cpus = 0;
|
extend -smp parsing to include cores= and threads= options
For injecting multi-core and multi-threading CPU topology into guests
extend the -smp syntax to accommodate cores and threads specification.
Syntax: -smp smp_value[,cores=nr_cores][,threads=nr_threads]\
[,socket=nr_sockets][,maxcpus=max_cpus]
smp_value is the legacy value specifying the total number of vCPUs for
the guest. If you specify one of cores, threads or sockets this value
can be omitted. Missing values will be computed to fulfill:
smp_value = nr_cores * nr_threads * nr_sockets
where it will favour sockets over cores over threads (to mimic the
current behavior, which will only inject multiple sockets.)
So -smp 4,threads=2 will inject two sockets with 2 threads each,
-smp cores=4 is an abbreviation for -smp 4,cores=4,threads=1,sockets=1.
If max_cpus (the number of hotpluggable CPUs) is omitted, it will
be set to smp_value.
Signed-off-by: Andre Przywara <andre.przywara@amd.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-08-19 21:42:40 +08:00
|
|
|
int smp_cores = 1;
|
|
|
|
int smp_threads = 1;
|
2006-12-22 10:09:07 +08:00
|
|
|
const char *vnc_display;
|
2006-05-04 06:02:44 +08:00
|
|
|
int acpi_enabled = 1;
|
2008-12-18 07:28:44 +08:00
|
|
|
int no_hpet = 0;
|
2006-06-15 00:03:05 +08:00
|
|
|
int fd_bootchk = 1;
|
2006-10-03 03:44:22 +08:00
|
|
|
int no_reboot = 0;
|
2008-04-12 05:35:52 +08:00
|
|
|
int no_shutdown = 0;
|
2007-05-01 09:34:14 +08:00
|
|
|
int cursor_hide = 1;
|
2007-04-30 09:48:07 +08:00
|
|
|
int graphic_rotate = 0;
|
2009-08-06 22:25:50 +08:00
|
|
|
uint8_t irq0override = 1;
|
2009-04-06 02:03:31 +08:00
|
|
|
#ifndef _WIN32
|
2006-12-22 10:11:31 +08:00
|
|
|
int daemonize = 0;
|
2009-04-06 02:03:31 +08:00
|
|
|
#endif
|
2009-08-21 16:31:34 +08:00
|
|
|
const char *watchdog;
|
2007-01-06 01:39:04 +08:00
|
|
|
const char *option_rom[MAX_OPTION_ROMS];
|
|
|
|
int nb_option_roms;
|
2007-01-21 01:12:09 +08:00
|
|
|
int semihosting_enabled = 0;
|
2007-07-28 06:08:46 +08:00
|
|
|
#ifdef TARGET_ARM
|
|
|
|
int old_param = 0;
|
|
|
|
#endif
|
2007-03-19 23:17:08 +08:00
|
|
|
const char *qemu_name;
|
2007-06-22 05:08:02 +08:00
|
|
|
int alt_grab = 0;
|
2009-09-18 04:48:04 +08:00
|
|
|
int ctrl_grab = 0;
|
2008-12-25 04:26:14 +08:00
|
|
|
#if defined(TARGET_SPARC) || defined(TARGET_PPC)
|
2007-05-01 22:16:52 +08:00
|
|
|
unsigned int nb_prom_envs = 0;
|
|
|
|
const char *prom_envs[MAX_PROM_ENVS];
|
|
|
|
#endif
|
2009-07-02 06:19:02 +08:00
|
|
|
int boot_menu;
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2009-04-22 06:30:27 +08:00
|
|
|
int nb_numa_nodes;
|
|
|
|
uint64_t node_mem[MAX_NODES];
|
|
|
|
uint64_t node_cpumask[MAX_NODES];
|
|
|
|
|
2007-12-03 11:01:40 +08:00
|
|
|
static CPUState *cur_cpu;
|
|
|
|
static CPUState *next_cpu;
|
2008-07-01 01:22:19 +08:00
|
|
|
/* Conversion factor from emulated instructions to virtual clock ticks. */
|
2008-06-29 09:03:05 +08:00
|
|
|
static int icount_time_shift;
|
2008-07-01 01:22:19 +08:00
|
|
|
/* Arbitrarily pick 1MIPS as the minimum allowable speed. */
|
2008-06-29 09:03:05 +08:00
|
|
|
#define MAX_ICOUNT_SHIFT 10
|
|
|
|
/* Compensate for varying guest execution speed. */
|
|
|
|
static int64_t qemu_icount_bias;
|
2008-10-02 03:38:09 +08:00
|
|
|
static QEMUTimer *icount_rt_timer;
|
|
|
|
static QEMUTimer *icount_vm_timer;
|
2009-01-22 03:28:13 +08:00
|
|
|
static QEMUTimer *nographic_timer;
|
2007-12-03 11:01:40 +08:00
|
|
|
|
2008-09-19 02:29:08 +08:00
|
|
|
uint8_t qemu_uuid[16];
|
|
|
|
|
2009-07-02 06:19:02 +08:00
|
|
|
static QEMUBootSetHandler *boot_set_handler;
|
|
|
|
static void *boot_set_opaque;
|
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
#ifdef SIGRTMIN
|
|
|
|
#define SIG_IPI (SIGRTMIN+4)
|
|
|
|
#else
|
|
|
|
#define SIG_IPI SIGUSR1
|
|
|
|
#endif
|
|
|
|
|
2009-12-08 20:11:41 +08:00
|
|
|
static int default_serial = 1;
|
2009-12-08 20:11:42 +08:00
|
|
|
static int default_parallel = 1;
|
2009-12-08 20:11:54 +08:00
|
|
|
static int default_virtcon = 1;
|
2009-12-08 20:11:43 +08:00
|
|
|
static int default_monitor = 1;
|
2009-12-08 20:11:45 +08:00
|
|
|
static int default_vga = 1;
|
2009-12-16 21:25:39 +08:00
|
|
|
static int default_floppy = 1;
|
|
|
|
static int default_cdrom = 1;
|
|
|
|
static int default_sdcard = 1;
|
2009-12-08 20:11:41 +08:00
|
|
|
|
|
|
|
static struct {
|
|
|
|
const char *driver;
|
|
|
|
int *flag;
|
|
|
|
} default_list[] = {
|
2009-12-08 20:11:42 +08:00
|
|
|
{ .driver = "isa-serial", .flag = &default_serial },
|
|
|
|
{ .driver = "isa-parallel", .flag = &default_parallel },
|
2009-12-16 21:25:40 +08:00
|
|
|
{ .driver = "isa-fdc", .flag = &default_floppy },
|
|
|
|
{ .driver = "ide-drive", .flag = &default_cdrom },
|
2010-01-21 18:49:23 +08:00
|
|
|
{ .driver = "virtio-serial-pci", .flag = &default_virtcon },
|
|
|
|
{ .driver = "virtio-serial-s390", .flag = &default_virtcon },
|
|
|
|
{ .driver = "virtio-serial", .flag = &default_virtcon },
|
2009-12-08 20:11:45 +08:00
|
|
|
{ .driver = "VGA", .flag = &default_vga },
|
2009-12-16 20:35:19 +08:00
|
|
|
{ .driver = "cirrus-vga", .flag = &default_vga },
|
|
|
|
{ .driver = "vmware-svga", .flag = &default_vga },
|
2009-12-08 20:11:41 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static int default_driver_check(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
const char *driver = qemu_opt_get(opts, "driver");
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!driver)
|
|
|
|
return 0;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(default_list); i++) {
|
|
|
|
if (strcmp(default_list[i].driver, driver) != 0)
|
|
|
|
continue;
|
|
|
|
*(default_list[i].flag) = 0;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-06-24 21:42:40 +08:00
|
|
|
/***********************************************************/
|
2004-04-29 06:26:05 +08:00
|
|
|
/* x86 ISA bus support */
|
|
|
|
|
2009-10-02 05:12:16 +08:00
|
|
|
target_phys_addr_t isa_mem_base = 0;
|
2005-07-03 02:11:44 +08:00
|
|
|
PicState2 *isa_pic;
|
2003-06-24 21:42:40 +08:00
|
|
|
|
|
|
|
/***********************************************************/
|
|
|
|
void hw_error(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
2005-11-22 07:25:50 +08:00
|
|
|
CPUState *env;
|
2003-06-24 21:42:40 +08:00
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
fprintf(stderr, "qemu: hardware error: ");
|
|
|
|
vfprintf(stderr, fmt, ap);
|
|
|
|
fprintf(stderr, "\n");
|
2005-11-22 07:25:50 +08:00
|
|
|
for(env = first_cpu; env != NULL; env = env->next_cpu) {
|
|
|
|
fprintf(stderr, "CPU #%d:\n", env->cpu_index);
|
2003-06-24 21:42:40 +08:00
|
|
|
#ifdef TARGET_I386
|
2005-11-22 07:25:50 +08:00
|
|
|
cpu_dump_state(env, stderr, fprintf, X86_DUMP_FPU);
|
2004-01-05 08:02:06 +08:00
|
|
|
#else
|
2005-11-22 07:25:50 +08:00
|
|
|
cpu_dump_state(env, stderr, fprintf, 0);
|
2003-06-24 21:42:40 +08:00
|
|
|
#endif
|
2005-11-22 07:25:50 +08:00
|
|
|
}
|
2003-06-24 21:42:40 +08:00
|
|
|
va_end(ap);
|
|
|
|
abort();
|
|
|
|
}
|
2009-07-02 15:34:17 +08:00
|
|
|
|
|
|
|
static void set_proc_name(const char *s)
|
|
|
|
{
|
2009-08-03 22:32:12 +08:00
|
|
|
#if defined(__linux__) && defined(PR_SET_NAME)
|
2009-07-02 15:34:17 +08:00
|
|
|
char name[16];
|
|
|
|
if (!s)
|
|
|
|
return;
|
|
|
|
name[sizeof(name) - 1] = 0;
|
|
|
|
strncpy(name, s, sizeof(name));
|
|
|
|
/* Could rewrite argv[0] too, but that's a bit more complicated.
|
|
|
|
This simple way is enough for `top'. */
|
|
|
|
prctl(PR_SET_NAME, name);
|
|
|
|
#endif
|
|
|
|
}
|
2008-12-05 04:19:35 +08:00
|
|
|
|
|
|
|
/***************/
|
|
|
|
/* ballooning */
|
|
|
|
|
|
|
|
static QEMUBalloonEvent *qemu_balloon_event;
|
|
|
|
void *qemu_balloon_event_opaque;
|
|
|
|
|
|
|
|
void qemu_add_balloon_handler(QEMUBalloonEvent *func, void *opaque)
|
|
|
|
{
|
|
|
|
qemu_balloon_event = func;
|
|
|
|
qemu_balloon_event_opaque = opaque;
|
|
|
|
}
|
|
|
|
|
2010-01-27 04:17:35 +08:00
|
|
|
int qemu_balloon(ram_addr_t target, MonitorCompletion cb, void *opaque)
|
2008-12-05 04:19:35 +08:00
|
|
|
{
|
2010-01-27 04:17:35 +08:00
|
|
|
if (qemu_balloon_event) {
|
|
|
|
qemu_balloon_event(qemu_balloon_event_opaque, target, cb, opaque);
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2008-12-05 04:19:35 +08:00
|
|
|
}
|
|
|
|
|
2010-01-27 04:17:35 +08:00
|
|
|
int qemu_balloon_status(MonitorCompletion cb, void *opaque)
|
2008-12-05 04:19:35 +08:00
|
|
|
{
|
2010-01-27 04:17:35 +08:00
|
|
|
if (qemu_balloon_event) {
|
|
|
|
qemu_balloon_event(qemu_balloon_event_opaque, 0, cb, opaque);
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2008-12-05 04:19:35 +08:00
|
|
|
}
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2007-01-06 00:42:13 +08:00
|
|
|
|
2010-01-13 21:05:34 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* real time host monotonic timer */
|
2006-04-13 05:09:08 +08:00
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
/* compute with 96 bit intermediate result: (a*b)/c */
|
|
|
|
uint64_t muldiv64(uint64_t a, uint32_t b, uint32_t c)
|
2003-06-24 21:42:40 +08:00
|
|
|
{
|
2006-07-14 07:20:22 +08:00
|
|
|
union {
|
|
|
|
uint64_t ll;
|
|
|
|
struct {
|
2009-07-27 22:13:06 +08:00
|
|
|
#ifdef HOST_WORDS_BIGENDIAN
|
2006-07-14 07:20:22 +08:00
|
|
|
uint32_t high, low;
|
|
|
|
#else
|
|
|
|
uint32_t low, high;
|
2007-09-17 16:09:54 +08:00
|
|
|
#endif
|
2006-07-14 07:20:22 +08:00
|
|
|
} l;
|
|
|
|
} u, res;
|
|
|
|
uint64_t rl, rh;
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
u.ll = a;
|
|
|
|
rl = (uint64_t)u.l.low * (uint64_t)b;
|
|
|
|
rh = (uint64_t)u.l.high * (uint64_t)b;
|
|
|
|
rh += (rl >> 32);
|
|
|
|
res.l.high = rh / c;
|
|
|
|
res.l.low = (((rh % c) << 32) + (rl & 0xffffffff)) / c;
|
|
|
|
return res.ll;
|
2003-10-05 22:28:56 +08:00
|
|
|
}
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
static int64_t get_clock_realtime(void)
|
|
|
|
{
|
|
|
|
struct timeval tv;
|
|
|
|
|
|
|
|
gettimeofday(&tv, NULL);
|
|
|
|
return tv.tv_sec * 1000000000LL + (tv.tv_usec * 1000);
|
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
#ifdef WIN32
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
static int64_t clock_freq;
|
2004-04-26 02:57:49 +08:00
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
static void init_get_clock(void)
|
2004-04-26 02:57:49 +08:00
|
|
|
{
|
2006-07-14 17:36:13 +08:00
|
|
|
LARGE_INTEGER freq;
|
|
|
|
int ret;
|
2006-07-14 07:20:22 +08:00
|
|
|
ret = QueryPerformanceFrequency(&freq);
|
|
|
|
if (ret == 0) {
|
|
|
|
fprintf(stderr, "Could not calibrate ticks\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
clock_freq = freq.QuadPart;
|
2004-04-26 02:57:49 +08:00
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
static int64_t get_clock(void)
|
2005-04-08 06:20:31 +08:00
|
|
|
{
|
2006-07-14 07:20:22 +08:00
|
|
|
LARGE_INTEGER ti;
|
|
|
|
QueryPerformanceCounter(&ti);
|
2009-09-11 23:28:26 +08:00
|
|
|
return muldiv64(ti.QuadPart, get_ticks_per_sec(), clock_freq);
|
2005-04-08 06:20:31 +08:00
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
#else
|
2005-07-24 23:11:38 +08:00
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
static int use_rt_clock;
|
|
|
|
|
|
|
|
static void init_get_clock(void)
|
2005-07-24 23:11:38 +08:00
|
|
|
{
|
2006-07-14 07:20:22 +08:00
|
|
|
use_rt_clock = 0;
|
2009-03-08 04:06:23 +08:00
|
|
|
#if defined(__linux__) || (defined(__FreeBSD__) && __FreeBSD_version >= 500000) \
|
2009-11-30 01:00:41 +08:00
|
|
|
|| defined(__DragonFly__) || defined(__FreeBSD_kernel__)
|
2006-07-14 07:20:22 +08:00
|
|
|
{
|
|
|
|
struct timespec ts;
|
|
|
|
if (clock_gettime(CLOCK_MONOTONIC, &ts) == 0) {
|
|
|
|
use_rt_clock = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2005-07-24 23:11:38 +08:00
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
static int64_t get_clock(void)
|
2006-06-15 01:32:25 +08:00
|
|
|
{
|
2009-03-08 04:06:23 +08:00
|
|
|
#if defined(__linux__) || (defined(__FreeBSD__) && __FreeBSD_version >= 500000) \
|
2009-11-30 01:00:41 +08:00
|
|
|
|| defined(__DragonFly__) || defined(__FreeBSD_kernel__)
|
2006-07-14 07:20:22 +08:00
|
|
|
if (use_rt_clock) {
|
|
|
|
struct timespec ts;
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &ts);
|
|
|
|
return ts.tv_sec * 1000000000LL + ts.tv_nsec;
|
2007-09-17 05:08:06 +08:00
|
|
|
} else
|
2006-06-15 01:32:25 +08:00
|
|
|
#endif
|
2006-07-14 07:20:22 +08:00
|
|
|
{
|
|
|
|
/* XXX: using gettimeofday leads to problems if the date
|
|
|
|
changes, so it should be avoided. */
|
2009-09-15 19:36:04 +08:00
|
|
|
return get_clock_realtime();
|
2006-07-14 07:20:22 +08:00
|
|
|
}
|
2006-06-15 01:32:25 +08:00
|
|
|
}
|
2003-10-05 22:28:56 +08:00
|
|
|
#endif
|
|
|
|
|
2008-06-29 09:03:05 +08:00
|
|
|
/* Return the virtual CPU time, based on the instruction counter. */
|
|
|
|
static int64_t cpu_get_icount(void)
|
|
|
|
{
|
|
|
|
int64_t icount;
|
|
|
|
CPUState *env = cpu_single_env;;
|
|
|
|
icount = qemu_icount;
|
|
|
|
if (env) {
|
|
|
|
if (!can_do_io(env))
|
|
|
|
fprintf(stderr, "Bad clock read\n");
|
|
|
|
icount -= (env->icount_decr.u16.low + env->icount_extra);
|
|
|
|
}
|
|
|
|
return qemu_icount_bias + (icount << icount_time_shift);
|
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* guest cycle counter */
|
|
|
|
|
2009-09-10 09:04:27 +08:00
|
|
|
typedef struct TimersState {
|
|
|
|
int64_t cpu_ticks_prev;
|
|
|
|
int64_t cpu_ticks_offset;
|
|
|
|
int64_t cpu_clock_offset;
|
|
|
|
int32_t cpu_ticks_enabled;
|
2009-09-11 23:28:26 +08:00
|
|
|
int64_t dummy;
|
2009-09-10 09:04:27 +08:00
|
|
|
} TimersState;
|
|
|
|
|
|
|
|
TimersState timers_state;
|
2003-10-05 22:28:56 +08:00
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
/* return the host CPU cycle counter and handle stop/restart */
|
|
|
|
int64_t cpu_get_ticks(void)
|
2003-10-05 22:28:56 +08:00
|
|
|
{
|
2008-06-29 09:03:05 +08:00
|
|
|
if (use_icount) {
|
|
|
|
return cpu_get_icount();
|
|
|
|
}
|
2009-09-10 09:04:27 +08:00
|
|
|
if (!timers_state.cpu_ticks_enabled) {
|
|
|
|
return timers_state.cpu_ticks_offset;
|
2004-04-01 03:00:16 +08:00
|
|
|
} else {
|
2006-05-01 20:43:29 +08:00
|
|
|
int64_t ticks;
|
|
|
|
ticks = cpu_get_real_ticks();
|
2009-09-10 09:04:27 +08:00
|
|
|
if (timers_state.cpu_ticks_prev > ticks) {
|
2006-05-01 20:43:29 +08:00
|
|
|
/* Note: non increasing ticks may happen if the host uses
|
|
|
|
software suspend */
|
2009-09-10 09:04:27 +08:00
|
|
|
timers_state.cpu_ticks_offset += timers_state.cpu_ticks_prev - ticks;
|
2006-05-01 20:43:29 +08:00
|
|
|
}
|
2009-09-10 09:04:27 +08:00
|
|
|
timers_state.cpu_ticks_prev = ticks;
|
|
|
|
return ticks + timers_state.cpu_ticks_offset;
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
2003-10-05 22:28:56 +08:00
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
/* return the host CPU monotonic timer and handle stop/restart */
|
|
|
|
static int64_t cpu_get_clock(void)
|
|
|
|
{
|
|
|
|
int64_t ti;
|
2009-09-10 09:04:27 +08:00
|
|
|
if (!timers_state.cpu_ticks_enabled) {
|
|
|
|
return timers_state.cpu_clock_offset;
|
2006-07-14 07:20:22 +08:00
|
|
|
} else {
|
|
|
|
ti = get_clock();
|
2009-09-10 09:04:27 +08:00
|
|
|
return ti + timers_state.cpu_clock_offset;
|
2006-07-14 07:20:22 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-10 18:38:51 +08:00
|
|
|
#ifndef CONFIG_IOTHREAD
|
|
|
|
static int64_t qemu_icount_delta(void)
|
|
|
|
{
|
|
|
|
if (!use_icount) {
|
|
|
|
return 5000 * (int64_t) 1000000;
|
|
|
|
} else if (use_icount == 1) {
|
|
|
|
/* When not using an adaptive execution frequency
|
|
|
|
we tend to get badly out of sync with real time,
|
|
|
|
so just delay for a reasonable amount of time. */
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return cpu_get_icount() - cpu_get_clock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-10-05 22:28:56 +08:00
|
|
|
/* enable cpu_get_ticks() */
|
|
|
|
void cpu_enable_ticks(void)
|
|
|
|
{
|
2009-09-10 09:04:27 +08:00
|
|
|
if (!timers_state.cpu_ticks_enabled) {
|
|
|
|
timers_state.cpu_ticks_offset -= cpu_get_real_ticks();
|
|
|
|
timers_state.cpu_clock_offset -= get_clock();
|
|
|
|
timers_state.cpu_ticks_enabled = 1;
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
2003-10-05 22:28:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* disable cpu_get_ticks() : the clock is stopped. You must not call
|
|
|
|
cpu_get_ticks() after that. */
|
|
|
|
void cpu_disable_ticks(void)
|
|
|
|
{
|
2009-09-10 09:04:27 +08:00
|
|
|
if (timers_state.cpu_ticks_enabled) {
|
|
|
|
timers_state.cpu_ticks_offset = cpu_get_ticks();
|
|
|
|
timers_state.cpu_clock_offset = cpu_get_clock();
|
|
|
|
timers_state.cpu_ticks_enabled = 0;
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
2003-10-05 22:28:56 +08:00
|
|
|
}
|
|
|
|
|
2006-07-14 07:20:22 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* timers */
|
2007-09-17 05:08:06 +08:00
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
#define QEMU_CLOCK_REALTIME 0
|
|
|
|
#define QEMU_CLOCK_VIRTUAL 1
|
2009-09-15 19:36:04 +08:00
|
|
|
#define QEMU_CLOCK_HOST 2
|
2004-04-01 03:00:16 +08:00
|
|
|
|
|
|
|
struct QEMUClock {
|
|
|
|
int type;
|
2010-03-10 18:38:47 +08:00
|
|
|
int enabled;
|
2004-04-01 03:00:16 +08:00
|
|
|
/* XXX: add frequency */
|
|
|
|
};
|
|
|
|
|
|
|
|
struct QEMUTimer {
|
|
|
|
QEMUClock *clock;
|
|
|
|
int64_t expire_time;
|
|
|
|
QEMUTimerCB *cb;
|
|
|
|
void *opaque;
|
|
|
|
struct QEMUTimer *next;
|
|
|
|
};
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
struct qemu_alarm_timer {
|
|
|
|
char const *name;
|
|
|
|
int (*start)(struct qemu_alarm_timer *t);
|
|
|
|
void (*stop)(struct qemu_alarm_timer *t);
|
2007-08-24 09:36:32 +08:00
|
|
|
void (*rearm)(struct qemu_alarm_timer *t);
|
2007-08-20 05:56:03 +08:00
|
|
|
void *priv;
|
|
|
|
|
2010-03-10 18:38:41 +08:00
|
|
|
char expired;
|
|
|
|
char pending;
|
2010-03-10 18:38:40 +08:00
|
|
|
};
|
2007-08-24 09:36:32 +08:00
|
|
|
|
2010-03-10 18:38:50 +08:00
|
|
|
static struct qemu_alarm_timer *alarm_timer;
|
|
|
|
|
|
|
|
static inline int qemu_alarm_pending(void)
|
|
|
|
{
|
|
|
|
return alarm_timer->pending;
|
|
|
|
}
|
|
|
|
|
2007-08-24 09:36:32 +08:00
|
|
|
static inline int alarm_has_dynticks(struct qemu_alarm_timer *t)
|
|
|
|
{
|
2010-03-10 18:38:41 +08:00
|
|
|
return !!t->rearm;
|
2007-08-24 09:36:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_rearm_alarm_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
if (!alarm_has_dynticks(t))
|
|
|
|
return;
|
|
|
|
|
|
|
|
t->rearm(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: MIN_TIMER_REARM_US should be optimized */
|
|
|
|
#define MIN_TIMER_REARM_US 250
|
|
|
|
|
2004-04-04 20:56:28 +08:00
|
|
|
#ifdef _WIN32
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
struct qemu_alarm_win32 {
|
|
|
|
MMRESULT timerId;
|
|
|
|
unsigned int period;
|
2010-03-10 18:38:38 +08:00
|
|
|
} alarm_win32_data = {0, 0};
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
static int win32_start_timer(struct qemu_alarm_timer *t);
|
|
|
|
static void win32_stop_timer(struct qemu_alarm_timer *t);
|
2007-08-24 09:36:32 +08:00
|
|
|
static void win32_rearm_timer(struct qemu_alarm_timer *t);
|
2007-08-20 05:56:03 +08:00
|
|
|
|
2004-04-04 20:56:28 +08:00
|
|
|
#else
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
static int unix_start_timer(struct qemu_alarm_timer *t);
|
|
|
|
static void unix_stop_timer(struct qemu_alarm_timer *t);
|
|
|
|
|
2007-08-27 01:29:15 +08:00
|
|
|
#ifdef __linux__
|
|
|
|
|
2007-08-24 09:36:32 +08:00
|
|
|
static int dynticks_start_timer(struct qemu_alarm_timer *t);
|
|
|
|
static void dynticks_stop_timer(struct qemu_alarm_timer *t);
|
|
|
|
static void dynticks_rearm_timer(struct qemu_alarm_timer *t);
|
|
|
|
|
2007-08-20 06:09:40 +08:00
|
|
|
static int hpet_start_timer(struct qemu_alarm_timer *t);
|
|
|
|
static void hpet_stop_timer(struct qemu_alarm_timer *t);
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
static int rtc_start_timer(struct qemu_alarm_timer *t);
|
|
|
|
static void rtc_stop_timer(struct qemu_alarm_timer *t);
|
|
|
|
|
2007-08-24 09:36:32 +08:00
|
|
|
#endif /* __linux__ */
|
2004-04-01 03:00:16 +08:00
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
#endif /* _WIN32 */
|
|
|
|
|
2008-06-29 09:03:05 +08:00
|
|
|
/* Correlation between real and virtual time is always going to be
|
2008-07-01 01:22:19 +08:00
|
|
|
fairly approximate, so ignore small variation.
|
2008-06-29 09:03:05 +08:00
|
|
|
When the guest is idle real and virtual time will be aligned in
|
|
|
|
the IO wait loop. */
|
2009-09-11 23:28:26 +08:00
|
|
|
#define ICOUNT_WOBBLE (get_ticks_per_sec() / 10)
|
2008-06-29 09:03:05 +08:00
|
|
|
|
|
|
|
static void icount_adjust(void)
|
|
|
|
{
|
|
|
|
int64_t cur_time;
|
|
|
|
int64_t cur_icount;
|
|
|
|
int64_t delta;
|
|
|
|
static int64_t last_delta;
|
|
|
|
/* If the VM is not running, then do nothing. */
|
|
|
|
if (!vm_running)
|
|
|
|
return;
|
|
|
|
|
|
|
|
cur_time = cpu_get_clock();
|
|
|
|
cur_icount = qemu_get_clock(vm_clock);
|
|
|
|
delta = cur_icount - cur_time;
|
|
|
|
/* FIXME: This is a very crude algorithm, somewhat prone to oscillation. */
|
|
|
|
if (delta > 0
|
|
|
|
&& last_delta + ICOUNT_WOBBLE < delta * 2
|
|
|
|
&& icount_time_shift > 0) {
|
|
|
|
/* The guest is getting too far ahead. Slow time down. */
|
|
|
|
icount_time_shift--;
|
|
|
|
}
|
|
|
|
if (delta < 0
|
|
|
|
&& last_delta - ICOUNT_WOBBLE > delta * 2
|
|
|
|
&& icount_time_shift < MAX_ICOUNT_SHIFT) {
|
|
|
|
/* The guest is getting too far behind. Speed time up. */
|
|
|
|
icount_time_shift++;
|
|
|
|
}
|
|
|
|
last_delta = delta;
|
|
|
|
qemu_icount_bias = cur_icount - (qemu_icount << icount_time_shift);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void icount_adjust_rt(void * opaque)
|
|
|
|
{
|
|
|
|
qemu_mod_timer(icount_rt_timer,
|
|
|
|
qemu_get_clock(rt_clock) + 1000);
|
|
|
|
icount_adjust();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void icount_adjust_vm(void * opaque)
|
|
|
|
{
|
|
|
|
qemu_mod_timer(icount_vm_timer,
|
2009-09-11 23:28:26 +08:00
|
|
|
qemu_get_clock(vm_clock) + get_ticks_per_sec() / 10);
|
2008-06-29 09:03:05 +08:00
|
|
|
icount_adjust();
|
|
|
|
}
|
|
|
|
|
2010-03-10 18:38:49 +08:00
|
|
|
static int64_t qemu_icount_round(int64_t count)
|
|
|
|
{
|
|
|
|
return (count + (1 << icount_time_shift) - 1) >> icount_time_shift;
|
|
|
|
}
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
static struct qemu_alarm_timer alarm_timers[] = {
|
2007-08-24 09:36:32 +08:00
|
|
|
#ifndef _WIN32
|
2007-08-27 01:29:15 +08:00
|
|
|
#ifdef __linux__
|
2010-03-10 18:38:40 +08:00
|
|
|
{"dynticks", dynticks_start_timer,
|
2007-08-24 09:36:32 +08:00
|
|
|
dynticks_stop_timer, dynticks_rearm_timer, NULL},
|
2007-08-20 06:09:40 +08:00
|
|
|
/* HPET - if available - is preferred */
|
2010-03-10 18:38:40 +08:00
|
|
|
{"hpet", hpet_start_timer, hpet_stop_timer, NULL, NULL},
|
2007-08-20 06:09:40 +08:00
|
|
|
/* ...otherwise try RTC */
|
2010-03-10 18:38:40 +08:00
|
|
|
{"rtc", rtc_start_timer, rtc_stop_timer, NULL, NULL},
|
2007-08-20 05:56:03 +08:00
|
|
|
#endif
|
2010-03-10 18:38:40 +08:00
|
|
|
{"unix", unix_start_timer, unix_stop_timer, NULL, NULL},
|
2007-08-20 05:56:03 +08:00
|
|
|
#else
|
2010-03-10 18:38:40 +08:00
|
|
|
{"dynticks", win32_start_timer,
|
2007-08-24 09:36:32 +08:00
|
|
|
win32_stop_timer, win32_rearm_timer, &alarm_win32_data},
|
2010-03-10 18:38:40 +08:00
|
|
|
{"win32", win32_start_timer,
|
2007-08-24 09:36:32 +08:00
|
|
|
win32_stop_timer, NULL, &alarm_win32_data},
|
2007-08-20 05:56:03 +08:00
|
|
|
#endif
|
|
|
|
{NULL, }
|
|
|
|
};
|
|
|
|
|
2008-03-09 14:59:01 +08:00
|
|
|
static void show_available_alarms(void)
|
2007-08-24 09:26:02 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
printf("Available alarm timers, in order of precedence:\n");
|
|
|
|
for (i = 0; alarm_timers[i].name; i++)
|
|
|
|
printf("%s\n", alarm_timers[i].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void configure_alarms(char const *opt)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int cur = 0;
|
2008-12-23 04:33:55 +08:00
|
|
|
int count = ARRAY_SIZE(alarm_timers) - 1;
|
2007-08-24 09:26:02 +08:00
|
|
|
char *arg;
|
|
|
|
char *name;
|
2008-06-29 09:03:05 +08:00
|
|
|
struct qemu_alarm_timer tmp;
|
2007-08-24 09:26:02 +08:00
|
|
|
|
2008-03-10 07:43:49 +08:00
|
|
|
if (!strcmp(opt, "?")) {
|
2007-08-24 09:26:02 +08:00
|
|
|
show_available_alarms();
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
2009-09-03 05:59:06 +08:00
|
|
|
arg = qemu_strdup(opt);
|
2007-08-24 09:26:02 +08:00
|
|
|
|
|
|
|
/* Reorder the array */
|
|
|
|
name = strtok(arg, ",");
|
|
|
|
while (name) {
|
2007-09-18 05:25:20 +08:00
|
|
|
for (i = 0; i < count && alarm_timers[i].name; i++) {
|
2007-08-24 09:26:02 +08:00
|
|
|
if (!strcmp(alarm_timers[i].name, name))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == count) {
|
|
|
|
fprintf(stderr, "Unknown clock %s\n", name);
|
|
|
|
goto next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i < cur)
|
|
|
|
/* Ignore */
|
|
|
|
goto next;
|
|
|
|
|
|
|
|
/* Swap */
|
|
|
|
tmp = alarm_timers[i];
|
|
|
|
alarm_timers[i] = alarm_timers[cur];
|
|
|
|
alarm_timers[cur] = tmp;
|
|
|
|
|
|
|
|
cur++;
|
|
|
|
next:
|
|
|
|
name = strtok(NULL, ",");
|
|
|
|
}
|
|
|
|
|
2009-09-03 05:59:06 +08:00
|
|
|
qemu_free(arg);
|
2007-08-24 09:26:02 +08:00
|
|
|
|
|
|
|
if (cur) {
|
2008-06-29 09:03:05 +08:00
|
|
|
/* Disable remaining timers */
|
2007-08-24 09:26:02 +08:00
|
|
|
for (i = cur; i < count; i++)
|
|
|
|
alarm_timers[i].name = NULL;
|
2008-03-10 07:43:49 +08:00
|
|
|
} else {
|
|
|
|
show_available_alarms();
|
|
|
|
exit(1);
|
2007-08-24 09:26:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
#define QEMU_NUM_CLOCKS 3
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
QEMUClock *rt_clock;
|
|
|
|
QEMUClock *vm_clock;
|
2009-09-15 19:36:04 +08:00
|
|
|
QEMUClock *host_clock;
|
2007-08-20 05:56:03 +08:00
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
static QEMUTimer *active_timers[QEMU_NUM_CLOCKS];
|
2007-08-20 05:56:03 +08:00
|
|
|
|
2007-11-18 09:44:38 +08:00
|
|
|
static QEMUClock *qemu_new_clock(int type)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
|
|
|
QEMUClock *clock;
|
|
|
|
clock = qemu_mallocz(sizeof(QEMUClock));
|
|
|
|
clock->type = type;
|
2010-03-10 18:38:47 +08:00
|
|
|
clock->enabled = 1;
|
2004-04-01 03:00:16 +08:00
|
|
|
return clock;
|
|
|
|
}
|
|
|
|
|
2010-03-10 18:38:47 +08:00
|
|
|
static void qemu_clock_enable(QEMUClock *clock, int enabled)
|
|
|
|
{
|
|
|
|
clock->enabled = enabled;
|
|
|
|
}
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
QEMUTimer *qemu_new_timer(QEMUClock *clock, QEMUTimerCB *cb, void *opaque)
|
|
|
|
{
|
|
|
|
QEMUTimer *ts;
|
|
|
|
|
|
|
|
ts = qemu_mallocz(sizeof(QEMUTimer));
|
|
|
|
ts->clock = clock;
|
|
|
|
ts->cb = cb;
|
|
|
|
ts->opaque = opaque;
|
|
|
|
return ts;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_free_timer(QEMUTimer *ts)
|
|
|
|
{
|
|
|
|
qemu_free(ts);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stop a timer, but do not dealloc it */
|
|
|
|
void qemu_del_timer(QEMUTimer *ts)
|
|
|
|
{
|
|
|
|
QEMUTimer **pt, *t;
|
|
|
|
|
|
|
|
/* NOTE: this code must be signal safe because
|
|
|
|
qemu_timer_expired() can be called from a signal. */
|
|
|
|
pt = &active_timers[ts->clock->type];
|
|
|
|
for(;;) {
|
|
|
|
t = *pt;
|
|
|
|
if (!t)
|
|
|
|
break;
|
|
|
|
if (t == ts) {
|
|
|
|
*pt = t->next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pt = &t->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* modify the current timer so that it will be fired when current_time
|
|
|
|
>= expire_time. The corresponding callback will be called. */
|
|
|
|
void qemu_mod_timer(QEMUTimer *ts, int64_t expire_time)
|
|
|
|
{
|
|
|
|
QEMUTimer **pt, *t;
|
|
|
|
|
|
|
|
qemu_del_timer(ts);
|
|
|
|
|
|
|
|
/* add the timer in the sorted list */
|
|
|
|
/* NOTE: this code must be signal safe because
|
|
|
|
qemu_timer_expired() can be called from a signal. */
|
|
|
|
pt = &active_timers[ts->clock->type];
|
|
|
|
for(;;) {
|
|
|
|
t = *pt;
|
|
|
|
if (!t)
|
|
|
|
break;
|
2007-09-17 05:08:06 +08:00
|
|
|
if (t->expire_time > expire_time)
|
2004-04-01 03:00:16 +08:00
|
|
|
break;
|
|
|
|
pt = &t->next;
|
|
|
|
}
|
|
|
|
ts->expire_time = expire_time;
|
|
|
|
ts->next = *pt;
|
|
|
|
*pt = ts;
|
2008-01-06 03:41:47 +08:00
|
|
|
|
|
|
|
/* Rearm if necessary */
|
2008-06-29 09:03:05 +08:00
|
|
|
if (pt == &active_timers[ts->clock->type]) {
|
2010-03-10 18:38:41 +08:00
|
|
|
if (!alarm_timer->pending) {
|
2008-06-29 09:03:05 +08:00
|
|
|
qemu_rearm_alarm_timer(alarm_timer);
|
|
|
|
}
|
|
|
|
/* Interrupt execution to force deadline recalculation. */
|
2009-04-25 02:03:11 +08:00
|
|
|
if (use_icount)
|
|
|
|
qemu_notify_event();
|
2008-06-29 09:03:05 +08:00
|
|
|
}
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int qemu_timer_pending(QEMUTimer *ts)
|
|
|
|
{
|
|
|
|
QEMUTimer *t;
|
|
|
|
for(t = active_timers[ts->clock->type]; t != NULL; t = t->next) {
|
|
|
|
if (t == ts)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-03 17:56:01 +08:00
|
|
|
int qemu_timer_expired(QEMUTimer *timer_head, int64_t current_time)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
|
|
|
if (!timer_head)
|
|
|
|
return 0;
|
|
|
|
return (timer_head->expire_time <= current_time);
|
|
|
|
}
|
|
|
|
|
2010-03-10 18:38:46 +08:00
|
|
|
static void qemu_run_timers(QEMUClock *clock)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
2010-03-10 18:38:46 +08:00
|
|
|
QEMUTimer **ptimer_head, *ts;
|
|
|
|
int64_t current_time;
|
2010-03-10 18:38:47 +08:00
|
|
|
|
|
|
|
if (!clock->enabled)
|
|
|
|
return;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2010-03-10 18:38:46 +08:00
|
|
|
current_time = qemu_get_clock (clock);
|
|
|
|
ptimer_head = &active_timers[clock->type];
|
2004-04-01 03:00:16 +08:00
|
|
|
for(;;) {
|
|
|
|
ts = *ptimer_head;
|
2004-10-01 06:22:08 +08:00
|
|
|
if (!ts || ts->expire_time > current_time)
|
2004-04-01 03:00:16 +08:00
|
|
|
break;
|
|
|
|
/* remove timer from the list before calling the callback */
|
|
|
|
*ptimer_head = ts->next;
|
|
|
|
ts->next = NULL;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
/* run the callback (the timer list can be modified) */
|
|
|
|
ts->cb(ts->opaque);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t qemu_get_clock(QEMUClock *clock)
|
|
|
|
{
|
|
|
|
switch(clock->type) {
|
2009-09-15 19:36:04 +08:00
|
|
|
case QEMU_CLOCK_REALTIME:
|
2006-07-14 07:20:22 +08:00
|
|
|
return get_clock() / 1000000;
|
2004-04-01 03:00:16 +08:00
|
|
|
default:
|
2009-09-15 19:36:04 +08:00
|
|
|
case QEMU_CLOCK_VIRTUAL:
|
2008-06-29 09:03:05 +08:00
|
|
|
if (use_icount) {
|
|
|
|
return cpu_get_icount();
|
|
|
|
} else {
|
|
|
|
return cpu_get_clock();
|
|
|
|
}
|
2009-09-15 19:36:04 +08:00
|
|
|
case QEMU_CLOCK_HOST:
|
|
|
|
return get_clock_realtime();
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-26 16:31:46 +08:00
|
|
|
int64_t qemu_get_clock_ns(QEMUClock *clock)
|
|
|
|
{
|
|
|
|
switch(clock->type) {
|
|
|
|
case QEMU_CLOCK_REALTIME:
|
|
|
|
return get_clock();
|
|
|
|
default:
|
|
|
|
case QEMU_CLOCK_VIRTUAL:
|
|
|
|
if (use_icount) {
|
|
|
|
return cpu_get_icount();
|
|
|
|
} else {
|
|
|
|
return cpu_get_clock();
|
|
|
|
}
|
|
|
|
case QEMU_CLOCK_HOST:
|
|
|
|
return get_clock_realtime();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
static void init_clocks(void)
|
2006-07-14 07:20:22 +08:00
|
|
|
{
|
|
|
|
init_get_clock();
|
2009-09-15 19:36:04 +08:00
|
|
|
rt_clock = qemu_new_clock(QEMU_CLOCK_REALTIME);
|
|
|
|
vm_clock = qemu_new_clock(QEMU_CLOCK_VIRTUAL);
|
2009-09-15 19:36:04 +08:00
|
|
|
host_clock = qemu_new_clock(QEMU_CLOCK_HOST);
|
2009-09-15 19:36:04 +08:00
|
|
|
|
|
|
|
rtc_clock = host_clock;
|
2006-07-14 07:20:22 +08:00
|
|
|
}
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
/* save a timer */
|
|
|
|
void qemu_put_timer(QEMUFile *f, QEMUTimer *ts)
|
|
|
|
{
|
|
|
|
uint64_t expire_time;
|
|
|
|
|
|
|
|
if (qemu_timer_pending(ts)) {
|
|
|
|
expire_time = ts->expire_time;
|
|
|
|
} else {
|
|
|
|
expire_time = -1;
|
|
|
|
}
|
|
|
|
qemu_put_be64(f, expire_time);
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_get_timer(QEMUFile *f, QEMUTimer *ts)
|
|
|
|
{
|
|
|
|
uint64_t expire_time;
|
|
|
|
|
|
|
|
expire_time = qemu_get_be64(f);
|
|
|
|
if (expire_time != -1) {
|
|
|
|
qemu_mod_timer(ts, expire_time);
|
|
|
|
} else {
|
|
|
|
qemu_del_timer(ts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-10 09:04:28 +08:00
|
|
|
static const VMStateDescription vmstate_timers = {
|
|
|
|
.name = "timer",
|
|
|
|
.version_id = 2,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.minimum_version_id_old = 1,
|
|
|
|
.fields = (VMStateField []) {
|
|
|
|
VMSTATE_INT64(cpu_ticks_offset, TimersState),
|
2009-09-11 23:28:26 +08:00
|
|
|
VMSTATE_INT64(dummy, TimersState),
|
2009-09-10 09:04:28 +08:00
|
|
|
VMSTATE_INT64_V(cpu_clock_offset, TimersState, 2),
|
|
|
|
VMSTATE_END_OF_LIST()
|
2006-08-06 21:36:11 +08:00
|
|
|
}
|
2009-09-10 09:04:28 +08:00
|
|
|
};
|
2004-04-01 03:00:16 +08:00
|
|
|
|
2010-03-10 18:38:52 +08:00
|
|
|
static void configure_icount(const char *option)
|
|
|
|
{
|
|
|
|
vmstate_register(0, &vmstate_timers, &timers_state);
|
|
|
|
if (!option)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (strcmp(option, "auto") != 0) {
|
|
|
|
icount_time_shift = strtol(option, NULL, 0);
|
|
|
|
use_icount = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
use_icount = 2;
|
|
|
|
|
|
|
|
/* 125MIPS seems a reasonable initial guess at the guest speed.
|
|
|
|
It will be corrected fairly quickly anyway. */
|
|
|
|
icount_time_shift = 3;
|
|
|
|
|
|
|
|
/* Have both realtime and virtual time triggers for speed adjustment.
|
|
|
|
The realtime trigger catches emulated time passing too slowly,
|
|
|
|
the virtual time trigger catches emulated time passing too fast.
|
|
|
|
Realtime triggers occur even when idle, so use them less frequently
|
|
|
|
than VM triggers. */
|
|
|
|
icount_rt_timer = qemu_new_timer(rt_clock, icount_adjust_rt, NULL);
|
|
|
|
qemu_mod_timer(icount_rt_timer,
|
|
|
|
qemu_get_clock(rt_clock) + 1000);
|
|
|
|
icount_vm_timer = qemu_new_timer(vm_clock, icount_adjust_vm, NULL);
|
|
|
|
qemu_mod_timer(icount_vm_timer,
|
|
|
|
qemu_get_clock(vm_clock) + get_ticks_per_sec() / 10);
|
|
|
|
}
|
|
|
|
|
2010-03-10 18:38:45 +08:00
|
|
|
static void qemu_run_all_timers(void)
|
|
|
|
{
|
|
|
|
/* rearm timer, if not periodic */
|
|
|
|
if (alarm_timer->expired) {
|
|
|
|
alarm_timer->expired = 0;
|
|
|
|
qemu_rearm_alarm_timer(alarm_timer);
|
|
|
|
}
|
|
|
|
|
|
|
|
alarm_timer->pending = 0;
|
|
|
|
|
|
|
|
/* vm time timers */
|
|
|
|
if (vm_running) {
|
2010-03-10 18:38:47 +08:00
|
|
|
qemu_run_timers(vm_clock);
|
2010-03-10 18:38:45 +08:00
|
|
|
}
|
|
|
|
|
2010-03-10 18:38:46 +08:00
|
|
|
qemu_run_timers(rt_clock);
|
|
|
|
qemu_run_timers(host_clock);
|
2010-03-10 18:38:45 +08:00
|
|
|
}
|
2009-04-25 02:03:29 +08:00
|
|
|
|
2004-04-01 07:37:16 +08:00
|
|
|
#ifdef _WIN32
|
2009-04-06 02:03:31 +08:00
|
|
|
static void CALLBACK host_alarm_handler(UINT uTimerID, UINT uMsg,
|
|
|
|
DWORD_PTR dwUser, DWORD_PTR dw1,
|
|
|
|
DWORD_PTR dw2)
|
2004-04-01 07:37:16 +08:00
|
|
|
#else
|
2004-04-01 03:00:16 +08:00
|
|
|
static void host_alarm_handler(int host_signum)
|
2004-04-01 07:37:16 +08:00
|
|
|
#endif
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
2010-03-10 18:38:41 +08:00
|
|
|
struct qemu_alarm_timer *t = alarm_timer;
|
|
|
|
if (!t)
|
|
|
|
return;
|
|
|
|
|
2004-06-25 22:46:23 +08:00
|
|
|
#if 0
|
|
|
|
#define DISP_FREQ 1000
|
|
|
|
{
|
|
|
|
static int64_t delta_min = INT64_MAX;
|
|
|
|
static int64_t delta_max, delta_cum, last_clock, delta, ti;
|
|
|
|
static int count;
|
|
|
|
ti = qemu_get_clock(vm_clock);
|
|
|
|
if (last_clock != 0) {
|
|
|
|
delta = ti - last_clock;
|
|
|
|
if (delta < delta_min)
|
|
|
|
delta_min = delta;
|
|
|
|
if (delta > delta_max)
|
|
|
|
delta_max = delta;
|
|
|
|
delta_cum += delta;
|
|
|
|
if (++count == DISP_FREQ) {
|
2006-06-26 02:15:32 +08:00
|
|
|
printf("timer: min=%" PRId64 " us max=%" PRId64 " us avg=%" PRId64 " us avg_freq=%0.3f Hz\n",
|
2009-09-10 09:04:26 +08:00
|
|
|
muldiv64(delta_min, 1000000, get_ticks_per_sec()),
|
|
|
|
muldiv64(delta_max, 1000000, get_ticks_per_sec()),
|
|
|
|
muldiv64(delta_cum, 1000000 / DISP_FREQ, get_ticks_per_sec()),
|
|
|
|
(double)get_ticks_per_sec() / ((double)delta_cum / DISP_FREQ));
|
2004-06-25 22:46:23 +08:00
|
|
|
count = 0;
|
|
|
|
delta_min = INT64_MAX;
|
|
|
|
delta_max = 0;
|
|
|
|
delta_cum = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
last_clock = ti;
|
|
|
|
}
|
|
|
|
#endif
|
2010-03-10 18:38:41 +08:00
|
|
|
if (alarm_has_dynticks(t) ||
|
2008-06-29 09:03:05 +08:00
|
|
|
(!use_icount &&
|
2009-09-15 19:36:04 +08:00
|
|
|
qemu_timer_expired(active_timers[QEMU_CLOCK_VIRTUAL],
|
2008-06-29 09:03:05 +08:00
|
|
|
qemu_get_clock(vm_clock))) ||
|
2009-09-15 19:36:04 +08:00
|
|
|
qemu_timer_expired(active_timers[QEMU_CLOCK_REALTIME],
|
2009-09-15 19:36:04 +08:00
|
|
|
qemu_get_clock(rt_clock)) ||
|
|
|
|
qemu_timer_expired(active_timers[QEMU_CLOCK_HOST],
|
|
|
|
qemu_get_clock(host_clock))) {
|
2010-03-10 18:38:42 +08:00
|
|
|
|
2010-03-10 18:38:41 +08:00
|
|
|
t->expired = alarm_has_dynticks(t);
|
|
|
|
t->pending = 1;
|
2009-04-25 02:03:11 +08:00
|
|
|
qemu_notify_event();
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-29 09:03:05 +08:00
|
|
|
static int64_t qemu_next_deadline(void)
|
2007-08-24 09:36:32 +08:00
|
|
|
{
|
2009-09-15 19:36:04 +08:00
|
|
|
/* To avoid problems with overflow limit this to 2^32. */
|
|
|
|
int64_t delta = INT32_MAX;
|
2007-08-24 09:36:32 +08:00
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
if (active_timers[QEMU_CLOCK_VIRTUAL]) {
|
|
|
|
delta = active_timers[QEMU_CLOCK_VIRTUAL]->expire_time -
|
2008-06-29 09:03:05 +08:00
|
|
|
qemu_get_clock(vm_clock);
|
2009-09-15 19:36:04 +08:00
|
|
|
}
|
|
|
|
if (active_timers[QEMU_CLOCK_HOST]) {
|
|
|
|
int64_t hdelta = active_timers[QEMU_CLOCK_HOST]->expire_time -
|
|
|
|
qemu_get_clock(host_clock);
|
|
|
|
if (hdelta < delta)
|
|
|
|
delta = hdelta;
|
2007-08-24 09:36:32 +08:00
|
|
|
}
|
|
|
|
|
2008-06-29 09:03:05 +08:00
|
|
|
if (delta < 0)
|
|
|
|
delta = 0;
|
2007-08-24 09:36:32 +08:00
|
|
|
|
2008-06-29 09:03:05 +08:00
|
|
|
return delta;
|
|
|
|
}
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
#if defined(__linux__)
|
2008-06-29 09:03:05 +08:00
|
|
|
static uint64_t qemu_next_deadline_dyntick(void)
|
|
|
|
{
|
|
|
|
int64_t delta;
|
|
|
|
int64_t rtdelta;
|
|
|
|
|
|
|
|
if (use_icount)
|
|
|
|
delta = INT32_MAX;
|
|
|
|
else
|
|
|
|
delta = (qemu_next_deadline() + 999) / 1000;
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
if (active_timers[QEMU_CLOCK_REALTIME]) {
|
|
|
|
rtdelta = (active_timers[QEMU_CLOCK_REALTIME]->expire_time -
|
2008-06-29 09:03:05 +08:00
|
|
|
qemu_get_clock(rt_clock))*1000;
|
|
|
|
if (rtdelta < delta)
|
|
|
|
delta = rtdelta;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (delta < MIN_TIMER_REARM_US)
|
|
|
|
delta = MIN_TIMER_REARM_US;
|
|
|
|
|
|
|
|
return delta;
|
2007-08-24 09:36:32 +08:00
|
|
|
}
|
2008-09-14 21:59:34 +08:00
|
|
|
#endif
|
2007-08-24 09:36:32 +08:00
|
|
|
|
2004-05-13 03:11:15 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
|
2008-11-06 04:40:18 +08:00
|
|
|
/* Sets a specific flag */
|
|
|
|
static int fcntl_setfl(int fd, int flag)
|
|
|
|
{
|
|
|
|
int flags;
|
|
|
|
|
|
|
|
flags = fcntl(fd, F_GETFL);
|
|
|
|
if (flags == -1)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
if (fcntl(fd, F_SETFL, flags | flag) == -1)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-05-20 21:20:12 +08:00
|
|
|
#if defined(__linux__)
|
|
|
|
|
2004-05-13 03:11:15 +08:00
|
|
|
#define RTC_FREQ 1024
|
|
|
|
|
2008-11-11 21:41:01 +08:00
|
|
|
static void enable_sigio_timer(int fd)
|
2007-08-20 05:56:03 +08:00
|
|
|
{
|
|
|
|
struct sigaction act;
|
|
|
|
|
|
|
|
/* timer signal */
|
|
|
|
sigfillset(&act.sa_mask);
|
|
|
|
act.sa_flags = 0;
|
|
|
|
act.sa_handler = host_alarm_handler;
|
|
|
|
|
|
|
|
sigaction(SIGIO, &act, NULL);
|
2008-11-06 04:40:18 +08:00
|
|
|
fcntl_setfl(fd, O_ASYNC);
|
2007-08-20 05:56:03 +08:00
|
|
|
fcntl(fd, F_SETOWN, getpid());
|
|
|
|
}
|
2004-05-20 21:20:12 +08:00
|
|
|
|
2007-08-20 06:09:40 +08:00
|
|
|
static int hpet_start_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
struct hpet_info info;
|
|
|
|
int r, fd;
|
|
|
|
|
2009-12-02 19:24:42 +08:00
|
|
|
fd = qemu_open("/dev/hpet", O_RDONLY);
|
2007-08-20 06:09:40 +08:00
|
|
|
if (fd < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* Set frequency */
|
|
|
|
r = ioctl(fd, HPET_IRQFREQ, RTC_FREQ);
|
|
|
|
if (r < 0) {
|
|
|
|
fprintf(stderr, "Could not configure '/dev/hpet' to have a 1024Hz timer. This is not a fatal\n"
|
|
|
|
"error, but for better emulation accuracy type:\n"
|
|
|
|
"'echo 1024 > /proc/sys/dev/hpet/max-user-freq' as root.\n");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check capabilities */
|
|
|
|
r = ioctl(fd, HPET_INFO, &info);
|
|
|
|
if (r < 0)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
/* Enable periodic mode */
|
|
|
|
r = ioctl(fd, HPET_EPI, 0);
|
|
|
|
if (info.hi_flags && (r < 0))
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
/* Enable interrupt */
|
|
|
|
r = ioctl(fd, HPET_IE_ON, 0);
|
|
|
|
if (r < 0)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
enable_sigio_timer(fd);
|
2007-08-24 04:22:22 +08:00
|
|
|
t->priv = (void *)(long)fd;
|
2007-08-20 06:09:40 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
fail:
|
|
|
|
close(fd);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hpet_stop_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
2007-08-24 04:22:22 +08:00
|
|
|
int fd = (long)t->priv;
|
2007-08-20 06:09:40 +08:00
|
|
|
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
static int rtc_start_timer(struct qemu_alarm_timer *t)
|
2004-05-13 03:11:15 +08:00
|
|
|
{
|
2007-08-20 05:56:03 +08:00
|
|
|
int rtc_fd;
|
2008-02-03 11:45:47 +08:00
|
|
|
unsigned long current_rtc_freq = 0;
|
2007-08-20 05:56:03 +08:00
|
|
|
|
2009-12-02 19:24:42 +08:00
|
|
|
TFR(rtc_fd = qemu_open("/dev/rtc", O_RDONLY));
|
2004-05-13 03:11:15 +08:00
|
|
|
if (rtc_fd < 0)
|
|
|
|
return -1;
|
2008-02-03 11:45:47 +08:00
|
|
|
ioctl(rtc_fd, RTC_IRQP_READ, ¤t_rtc_freq);
|
|
|
|
if (current_rtc_freq != RTC_FREQ &&
|
|
|
|
ioctl(rtc_fd, RTC_IRQP_SET, RTC_FREQ) < 0) {
|
2004-05-13 03:11:15 +08:00
|
|
|
fprintf(stderr, "Could not configure '/dev/rtc' to have a 1024 Hz timer. This is not a fatal\n"
|
|
|
|
"error, but for better emulation accuracy either use a 2.6 host Linux kernel or\n"
|
|
|
|
"type 'echo 1024 > /proc/sys/dev/rtc/max-user-freq' as root.\n");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
if (ioctl(rtc_fd, RTC_PIE_ON, 0) < 0) {
|
|
|
|
fail:
|
|
|
|
close(rtc_fd);
|
|
|
|
return -1;
|
|
|
|
}
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
enable_sigio_timer(rtc_fd);
|
|
|
|
|
2007-08-24 04:22:22 +08:00
|
|
|
t->priv = (void *)(long)rtc_fd;
|
2007-08-20 05:56:03 +08:00
|
|
|
|
2004-05-13 03:11:15 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
static void rtc_stop_timer(struct qemu_alarm_timer *t)
|
2004-05-20 21:20:12 +08:00
|
|
|
{
|
2007-08-24 04:22:22 +08:00
|
|
|
int rtc_fd = (long)t->priv;
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
close(rtc_fd);
|
2004-05-20 21:20:12 +08:00
|
|
|
}
|
|
|
|
|
2007-08-24 09:36:32 +08:00
|
|
|
static int dynticks_start_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
struct sigevent ev;
|
|
|
|
timer_t host_timer;
|
|
|
|
struct sigaction act;
|
|
|
|
|
|
|
|
sigfillset(&act.sa_mask);
|
|
|
|
act.sa_flags = 0;
|
|
|
|
act.sa_handler = host_alarm_handler;
|
|
|
|
|
|
|
|
sigaction(SIGALRM, &act, NULL);
|
|
|
|
|
2009-05-18 00:41:16 +08:00
|
|
|
/*
|
|
|
|
* Initialize ev struct to 0 to avoid valgrind complaining
|
|
|
|
* about uninitialized data in timer_create call
|
|
|
|
*/
|
|
|
|
memset(&ev, 0, sizeof(ev));
|
2007-08-24 09:36:32 +08:00
|
|
|
ev.sigev_value.sival_int = 0;
|
|
|
|
ev.sigev_notify = SIGEV_SIGNAL;
|
|
|
|
ev.sigev_signo = SIGALRM;
|
|
|
|
|
|
|
|
if (timer_create(CLOCK_REALTIME, &ev, &host_timer)) {
|
|
|
|
perror("timer_create");
|
|
|
|
|
|
|
|
/* disable dynticks */
|
|
|
|
fprintf(stderr, "Dynamic Ticks disabled\n");
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-11-16 19:37:18 +08:00
|
|
|
t->priv = (void *)(long)host_timer;
|
2007-08-24 09:36:32 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dynticks_stop_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
2008-11-16 19:37:18 +08:00
|
|
|
timer_t host_timer = (timer_t)(long)t->priv;
|
2007-08-24 09:36:32 +08:00
|
|
|
|
|
|
|
timer_delete(host_timer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dynticks_rearm_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
2008-11-16 19:37:18 +08:00
|
|
|
timer_t host_timer = (timer_t)(long)t->priv;
|
2007-08-24 09:36:32 +08:00
|
|
|
struct itimerspec timeout;
|
|
|
|
int64_t nearest_delta_us = INT64_MAX;
|
|
|
|
int64_t current_us;
|
|
|
|
|
2010-03-10 18:38:40 +08:00
|
|
|
assert(alarm_has_dynticks(t));
|
2009-09-15 19:36:04 +08:00
|
|
|
if (!active_timers[QEMU_CLOCK_REALTIME] &&
|
2009-09-15 19:36:04 +08:00
|
|
|
!active_timers[QEMU_CLOCK_VIRTUAL] &&
|
|
|
|
!active_timers[QEMU_CLOCK_HOST])
|
2008-01-06 03:41:47 +08:00
|
|
|
return;
|
2007-08-24 09:36:32 +08:00
|
|
|
|
2008-06-29 09:03:05 +08:00
|
|
|
nearest_delta_us = qemu_next_deadline_dyntick();
|
2007-08-24 09:36:32 +08:00
|
|
|
|
|
|
|
/* check whether a timer is already running */
|
|
|
|
if (timer_gettime(host_timer, &timeout)) {
|
|
|
|
perror("gettime");
|
|
|
|
fprintf(stderr, "Internal timer error: aborting\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
current_us = timeout.it_value.tv_sec * 1000000 + timeout.it_value.tv_nsec/1000;
|
|
|
|
if (current_us && current_us <= nearest_delta_us)
|
|
|
|
return;
|
|
|
|
|
|
|
|
timeout.it_interval.tv_sec = 0;
|
|
|
|
timeout.it_interval.tv_nsec = 0; /* 0 for one-shot timer */
|
|
|
|
timeout.it_value.tv_sec = nearest_delta_us / 1000000;
|
|
|
|
timeout.it_value.tv_nsec = (nearest_delta_us % 1000000) * 1000;
|
|
|
|
if (timer_settime(host_timer, 0 /* RELATIVE */, &timeout, NULL)) {
|
|
|
|
perror("settime");
|
|
|
|
fprintf(stderr, "Internal timer error: aborting\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-27 01:31:30 +08:00
|
|
|
#endif /* defined(__linux__) */
|
2007-08-27 01:29:15 +08:00
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
static int unix_start_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
struct sigaction act;
|
|
|
|
struct itimerval itv;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* timer signal */
|
|
|
|
sigfillset(&act.sa_mask);
|
|
|
|
act.sa_flags = 0;
|
|
|
|
act.sa_handler = host_alarm_handler;
|
|
|
|
|
|
|
|
sigaction(SIGALRM, &act, NULL);
|
|
|
|
|
|
|
|
itv.it_interval.tv_sec = 0;
|
|
|
|
/* for i386 kernel 2.6 to get 1 ms */
|
|
|
|
itv.it_interval.tv_usec = 999;
|
|
|
|
itv.it_value.tv_sec = 0;
|
|
|
|
itv.it_value.tv_usec = 10 * 1000;
|
|
|
|
|
|
|
|
err = setitimer(ITIMER_REAL, &itv, NULL);
|
|
|
|
if (err)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void unix_stop_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
struct itimerval itv;
|
|
|
|
|
|
|
|
memset(&itv, 0, sizeof(itv));
|
|
|
|
setitimer(ITIMER_REAL, &itv, NULL);
|
|
|
|
}
|
|
|
|
|
2004-05-20 21:20:12 +08:00
|
|
|
#endif /* !defined(_WIN32) */
|
2004-05-13 03:11:15 +08:00
|
|
|
|
2008-11-06 05:22:34 +08:00
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
#ifdef _WIN32
|
|
|
|
|
|
|
|
static int win32_start_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
TIMECAPS tc;
|
|
|
|
struct qemu_alarm_win32 *data = t->priv;
|
2007-08-24 09:36:32 +08:00
|
|
|
UINT flags;
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
memset(&tc, 0, sizeof(tc));
|
|
|
|
timeGetDevCaps(&tc, sizeof(tc));
|
|
|
|
|
2010-03-10 18:38:38 +08:00
|
|
|
data->period = tc.wPeriodMin;
|
2007-08-20 05:56:03 +08:00
|
|
|
timeBeginPeriod(data->period);
|
|
|
|
|
2007-08-24 09:36:32 +08:00
|
|
|
flags = TIME_CALLBACK_FUNCTION;
|
|
|
|
if (alarm_has_dynticks(t))
|
|
|
|
flags |= TIME_ONESHOT;
|
|
|
|
else
|
|
|
|
flags |= TIME_PERIODIC;
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
data->timerId = timeSetEvent(1, // interval (ms)
|
|
|
|
data->period, // resolution
|
|
|
|
host_alarm_handler, // function
|
|
|
|
(DWORD)t, // parameter
|
2007-08-24 09:36:32 +08:00
|
|
|
flags);
|
2007-08-20 05:56:03 +08:00
|
|
|
|
|
|
|
if (!data->timerId) {
|
2009-09-28 04:03:56 +08:00
|
|
|
fprintf(stderr, "Failed to initialize win32 alarm timer: %ld\n",
|
2009-09-27 18:30:33 +08:00
|
|
|
GetLastError());
|
2007-08-20 05:56:03 +08:00
|
|
|
timeEndPeriod(data->period);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void win32_stop_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
struct qemu_alarm_win32 *data = t->priv;
|
|
|
|
|
|
|
|
timeKillEvent(data->timerId);
|
|
|
|
timeEndPeriod(data->period);
|
|
|
|
}
|
|
|
|
|
2007-08-24 09:36:32 +08:00
|
|
|
static void win32_rearm_timer(struct qemu_alarm_timer *t)
|
|
|
|
{
|
|
|
|
struct qemu_alarm_win32 *data = t->priv;
|
|
|
|
|
2010-03-10 18:38:40 +08:00
|
|
|
assert(alarm_has_dynticks(t));
|
2009-09-15 19:36:04 +08:00
|
|
|
if (!active_timers[QEMU_CLOCK_REALTIME] &&
|
2009-09-15 19:36:04 +08:00
|
|
|
!active_timers[QEMU_CLOCK_VIRTUAL] &&
|
|
|
|
!active_timers[QEMU_CLOCK_HOST])
|
2008-01-06 03:41:47 +08:00
|
|
|
return;
|
2007-08-24 09:36:32 +08:00
|
|
|
|
|
|
|
timeKillEvent(data->timerId);
|
|
|
|
|
|
|
|
data->timerId = timeSetEvent(1,
|
|
|
|
data->period,
|
|
|
|
host_alarm_handler,
|
|
|
|
(DWORD)t,
|
2010-03-10 18:38:39 +08:00
|
|
|
TIME_ONESHOT | TIME_CALLBACK_FUNCTION);
|
2007-08-24 09:36:32 +08:00
|
|
|
|
|
|
|
if (!data->timerId) {
|
2009-09-28 04:03:56 +08:00
|
|
|
fprintf(stderr, "Failed to re-arm win32 alarm timer %ld\n",
|
2009-09-27 18:30:33 +08:00
|
|
|
GetLastError());
|
2007-08-24 09:36:32 +08:00
|
|
|
|
|
|
|
timeEndPeriod(data->period);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
#endif /* _WIN32 */
|
|
|
|
|
2010-03-10 18:38:44 +08:00
|
|
|
static void alarm_timer_on_change_state_rearm(void *opaque, int running, int reason)
|
|
|
|
{
|
|
|
|
if (running)
|
|
|
|
qemu_rearm_alarm_timer((struct qemu_alarm_timer *) opaque);
|
|
|
|
}
|
|
|
|
|
2008-11-06 04:40:18 +08:00
|
|
|
static int init_timer_alarm(void)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
2008-10-01 02:12:18 +08:00
|
|
|
struct qemu_alarm_timer *t = NULL;
|
2007-08-20 05:56:03 +08:00
|
|
|
int i, err = -1;
|
2008-11-06 05:22:34 +08:00
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
for (i = 0; alarm_timers[i].name; i++) {
|
|
|
|
t = &alarm_timers[i];
|
|
|
|
|
|
|
|
err = t->start(t);
|
|
|
|
if (!err)
|
|
|
|
break;
|
2004-04-01 07:37:16 +08:00
|
|
|
}
|
2004-05-13 03:11:15 +08:00
|
|
|
|
2007-08-20 05:56:03 +08:00
|
|
|
if (err) {
|
2008-11-06 04:40:18 +08:00
|
|
|
err = -ENOENT;
|
|
|
|
goto fail;
|
2004-04-01 07:37:16 +08:00
|
|
|
}
|
2007-08-20 05:56:03 +08:00
|
|
|
|
2010-03-10 18:38:41 +08:00
|
|
|
/* first event is at time 0 */
|
|
|
|
t->pending = 1;
|
2007-08-20 05:56:03 +08:00
|
|
|
alarm_timer = t;
|
2010-03-10 18:38:44 +08:00
|
|
|
qemu_add_vm_change_state_handler(alarm_timer_on_change_state_rearm, t);
|
2008-11-06 04:40:18 +08:00
|
|
|
|
2008-11-06 04:49:37 +08:00
|
|
|
return 0;
|
2008-11-06 04:40:18 +08:00
|
|
|
|
|
|
|
fail:
|
|
|
|
return err;
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
|
2007-11-18 09:44:38 +08:00
|
|
|
static void quit_timers(void)
|
2004-04-04 20:56:28 +08:00
|
|
|
{
|
2010-03-10 18:38:41 +08:00
|
|
|
struct qemu_alarm_timer *t = alarm_timer;
|
2007-08-20 05:56:03 +08:00
|
|
|
alarm_timer = NULL;
|
2010-03-10 18:38:41 +08:00
|
|
|
t->stop(t);
|
2004-04-04 20:56:28 +08:00
|
|
|
}
|
|
|
|
|
2008-02-17 19:42:19 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* host time/date access */
|
|
|
|
void qemu_get_timedate(struct tm *tm, int offset)
|
|
|
|
{
|
|
|
|
time_t ti;
|
|
|
|
struct tm *ret;
|
|
|
|
|
|
|
|
time(&ti);
|
|
|
|
ti += offset;
|
|
|
|
if (rtc_date_offset == -1) {
|
|
|
|
if (rtc_utc)
|
|
|
|
ret = gmtime(&ti);
|
|
|
|
else
|
|
|
|
ret = localtime(&ti);
|
|
|
|
} else {
|
|
|
|
ti -= rtc_date_offset;
|
|
|
|
ret = gmtime(&ti);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(tm, ret, sizeof(struct tm));
|
|
|
|
}
|
|
|
|
|
|
|
|
int qemu_timedate_diff(struct tm *tm)
|
|
|
|
{
|
|
|
|
time_t seconds;
|
|
|
|
|
|
|
|
if (rtc_date_offset == -1)
|
|
|
|
if (rtc_utc)
|
|
|
|
seconds = mktimegm(tm);
|
|
|
|
else
|
|
|
|
seconds = mktime(tm);
|
|
|
|
else
|
|
|
|
seconds = mktimegm(tm) + rtc_date_offset;
|
|
|
|
|
|
|
|
return seconds - time(NULL);
|
|
|
|
}
|
|
|
|
|
2010-02-25 23:11:44 +08:00
|
|
|
void rtc_change_mon_event(struct tm *tm)
|
|
|
|
{
|
|
|
|
QObject *data;
|
|
|
|
|
|
|
|
data = qobject_from_jsonf("{ 'offset': %d }", qemu_timedate_diff(tm));
|
|
|
|
monitor_protocol_event(QEVENT_RTC_CHANGE, data);
|
|
|
|
qobject_decref(data);
|
|
|
|
}
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
static void configure_rtc_date_offset(const char *startdate, int legacy)
|
|
|
|
{
|
|
|
|
time_t rtc_start_date;
|
|
|
|
struct tm tm;
|
|
|
|
|
|
|
|
if (!strcmp(startdate, "now") && legacy) {
|
|
|
|
rtc_date_offset = -1;
|
|
|
|
} else {
|
|
|
|
if (sscanf(startdate, "%d-%d-%dT%d:%d:%d",
|
|
|
|
&tm.tm_year,
|
|
|
|
&tm.tm_mon,
|
|
|
|
&tm.tm_mday,
|
|
|
|
&tm.tm_hour,
|
|
|
|
&tm.tm_min,
|
|
|
|
&tm.tm_sec) == 6) {
|
|
|
|
/* OK */
|
|
|
|
} else if (sscanf(startdate, "%d-%d-%d",
|
|
|
|
&tm.tm_year,
|
|
|
|
&tm.tm_mon,
|
|
|
|
&tm.tm_mday) == 3) {
|
|
|
|
tm.tm_hour = 0;
|
|
|
|
tm.tm_min = 0;
|
|
|
|
tm.tm_sec = 0;
|
|
|
|
} else {
|
|
|
|
goto date_fail;
|
|
|
|
}
|
|
|
|
tm.tm_year -= 1900;
|
|
|
|
tm.tm_mon--;
|
|
|
|
rtc_start_date = mktimegm(&tm);
|
|
|
|
if (rtc_start_date == -1) {
|
|
|
|
date_fail:
|
|
|
|
fprintf(stderr, "Invalid date format. Valid formats are:\n"
|
|
|
|
"'2006-06-17T16:01:21' or '2006-06-17'\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
rtc_date_offset = time(NULL) - rtc_start_date;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void configure_rtc(QemuOpts *opts)
|
|
|
|
{
|
|
|
|
const char *value;
|
|
|
|
|
|
|
|
value = qemu_opt_get(opts, "base");
|
|
|
|
if (value) {
|
|
|
|
if (!strcmp(value, "utc")) {
|
|
|
|
rtc_utc = 1;
|
|
|
|
} else if (!strcmp(value, "localtime")) {
|
|
|
|
rtc_utc = 0;
|
|
|
|
} else {
|
|
|
|
configure_rtc_date_offset(value, 0);
|
|
|
|
}
|
|
|
|
}
|
2009-09-15 19:36:04 +08:00
|
|
|
value = qemu_opt_get(opts, "clock");
|
|
|
|
if (value) {
|
|
|
|
if (!strcmp(value, "host")) {
|
|
|
|
rtc_clock = host_clock;
|
|
|
|
} else if (!strcmp(value, "vm")) {
|
|
|
|
rtc_clock = vm_clock;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "qemu: invalid option value '%s'\n", value);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
2009-09-15 19:36:04 +08:00
|
|
|
#ifdef CONFIG_TARGET_I386
|
|
|
|
value = qemu_opt_get(opts, "driftfix");
|
|
|
|
if (value) {
|
|
|
|
if (!strcmp(buf, "slew")) {
|
|
|
|
rtc_td_hack = 1;
|
|
|
|
} else if (!strcmp(buf, "none")) {
|
|
|
|
rtc_td_hack = 0;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "qemu: invalid option value '%s'\n", value);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2006-02-02 05:29:26 +08:00
|
|
|
#ifdef _WIN32
|
|
|
|
static void socket_cleanup(void)
|
|
|
|
{
|
|
|
|
WSACleanup();
|
|
|
|
}
|
2004-07-15 01:28:13 +08:00
|
|
|
|
2006-02-02 05:29:26 +08:00
|
|
|
static int socket_init(void)
|
|
|
|
{
|
|
|
|
WSADATA Data;
|
|
|
|
int ret, err;
|
|
|
|
|
|
|
|
ret = WSAStartup(MAKEWORD(2,2), &Data);
|
|
|
|
if (ret != 0) {
|
|
|
|
err = WSAGetLastError();
|
|
|
|
fprintf(stderr, "WSAStartup: %d\n", err);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
atexit(socket_cleanup);
|
|
|
|
return 0;
|
|
|
|
}
|
2008-05-05 18:05:31 +08:00
|
|
|
#endif
|
|
|
|
|
2008-09-29 07:19:47 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* Bluetooth support */
|
|
|
|
static int nb_hcis;
|
|
|
|
static int cur_hci;
|
|
|
|
static struct HCIInfo *hci_table[MAX_NICS];
|
2008-11-09 08:04:26 +08:00
|
|
|
|
2008-09-29 07:19:47 +08:00
|
|
|
static struct bt_vlan_s {
|
|
|
|
struct bt_scatternet_s net;
|
|
|
|
int id;
|
|
|
|
struct bt_vlan_s *next;
|
|
|
|
} *first_bt_vlan;
|
|
|
|
|
|
|
|
/* find or alloc a new bluetooth "VLAN" */
|
2008-10-01 02:18:27 +08:00
|
|
|
static struct bt_scatternet_s *qemu_find_bt_vlan(int id)
|
2008-09-29 07:19:47 +08:00
|
|
|
{
|
|
|
|
struct bt_vlan_s **pvlan, *vlan;
|
|
|
|
for (vlan = first_bt_vlan; vlan != NULL; vlan = vlan->next) {
|
|
|
|
if (vlan->id == id)
|
|
|
|
return &vlan->net;
|
|
|
|
}
|
|
|
|
vlan = qemu_mallocz(sizeof(struct bt_vlan_s));
|
|
|
|
vlan->id = id;
|
|
|
|
pvlan = &first_bt_vlan;
|
|
|
|
while (*pvlan != NULL)
|
|
|
|
pvlan = &(*pvlan)->next;
|
|
|
|
*pvlan = vlan;
|
|
|
|
return &vlan->net;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_hci_send(struct HCIInfo *hci, const uint8_t *data, int len)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static int null_hci_addr_set(struct HCIInfo *hci, const uint8_t *bd_addr)
|
|
|
|
{
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct HCIInfo null_hci = {
|
|
|
|
.cmd_send = null_hci_send,
|
|
|
|
.sco_send = null_hci_send,
|
|
|
|
.acl_send = null_hci_send,
|
|
|
|
.bdaddr_set = null_hci_addr_set,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct HCIInfo *qemu_next_hci(void)
|
|
|
|
{
|
|
|
|
if (cur_hci == nb_hcis)
|
|
|
|
return &null_hci;
|
|
|
|
|
|
|
|
return hci_table[cur_hci++];
|
|
|
|
}
|
|
|
|
|
2008-11-09 08:04:26 +08:00
|
|
|
static struct HCIInfo *hci_init(const char *str)
|
|
|
|
{
|
|
|
|
char *endp;
|
|
|
|
struct bt_scatternet_s *vlan = 0;
|
|
|
|
|
|
|
|
if (!strcmp(str, "null"))
|
|
|
|
/* null */
|
|
|
|
return &null_hci;
|
|
|
|
else if (!strncmp(str, "host", 4) && (str[4] == '\0' || str[4] == ':'))
|
|
|
|
/* host[:hciN] */
|
|
|
|
return bt_host_hci(str[4] ? str + 5 : "hci0");
|
|
|
|
else if (!strncmp(str, "hci", 3)) {
|
|
|
|
/* hci[,vlan=n] */
|
|
|
|
if (str[3]) {
|
|
|
|
if (!strncmp(str + 3, ",vlan=", 6)) {
|
|
|
|
vlan = qemu_find_bt_vlan(strtol(str + 9, &endp, 0));
|
|
|
|
if (*endp)
|
|
|
|
vlan = 0;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
vlan = qemu_find_bt_vlan(0);
|
|
|
|
if (vlan)
|
|
|
|
return bt_new_hci(vlan);
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "qemu: Unknown bluetooth HCI `%s'.\n", str);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bt_hci_parse(const char *str)
|
|
|
|
{
|
|
|
|
struct HCIInfo *hci;
|
2009-10-02 05:12:16 +08:00
|
|
|
bdaddr_t bdaddr;
|
2008-11-09 08:04:26 +08:00
|
|
|
|
|
|
|
if (nb_hcis >= MAX_NICS) {
|
|
|
|
fprintf(stderr, "qemu: Too many bluetooth HCIs (max %i).\n", MAX_NICS);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
hci = hci_init(str);
|
|
|
|
if (!hci)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
bdaddr.b[0] = 0x52;
|
|
|
|
bdaddr.b[1] = 0x54;
|
|
|
|
bdaddr.b[2] = 0x00;
|
|
|
|
bdaddr.b[3] = 0x12;
|
|
|
|
bdaddr.b[4] = 0x34;
|
|
|
|
bdaddr.b[5] = 0x56 + nb_hcis;
|
|
|
|
hci->bdaddr_set(hci, bdaddr.b);
|
|
|
|
|
|
|
|
hci_table[nb_hcis++] = hci;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void bt_vhci_add(int vlan_id)
|
|
|
|
{
|
|
|
|
struct bt_scatternet_s *vlan = qemu_find_bt_vlan(vlan_id);
|
|
|
|
|
|
|
|
if (!vlan->slave)
|
|
|
|
fprintf(stderr, "qemu: warning: adding a VHCI to "
|
|
|
|
"an empty scatternet %i\n", vlan_id);
|
|
|
|
|
|
|
|
bt_vhci_init(bt_new_hci(vlan));
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct bt_device_s *bt_device_add(const char *opt)
|
|
|
|
{
|
|
|
|
struct bt_scatternet_s *vlan;
|
|
|
|
int vlan_id = 0;
|
|
|
|
char *endp = strstr(opt, ",vlan=");
|
|
|
|
int len = (endp ? endp - opt : strlen(opt)) + 1;
|
|
|
|
char devname[10];
|
|
|
|
|
|
|
|
pstrcpy(devname, MIN(sizeof(devname), len), opt);
|
|
|
|
|
|
|
|
if (endp) {
|
|
|
|
vlan_id = strtol(endp + 6, &endp, 0);
|
|
|
|
if (*endp) {
|
|
|
|
fprintf(stderr, "qemu: unrecognised bluetooth vlan Id\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vlan = qemu_find_bt_vlan(vlan_id);
|
|
|
|
|
|
|
|
if (!vlan->slave)
|
|
|
|
fprintf(stderr, "qemu: warning: adding a slave device to "
|
|
|
|
"an empty scatternet %i\n", vlan_id);
|
|
|
|
|
|
|
|
if (!strcmp(devname, "keyboard"))
|
|
|
|
return bt_keyboard_init(vlan);
|
|
|
|
|
|
|
|
fprintf(stderr, "qemu: unsupported bluetooth device `%s'\n", devname);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bt_parse(const char *opt)
|
|
|
|
{
|
|
|
|
const char *endp, *p;
|
|
|
|
int vlan;
|
|
|
|
|
|
|
|
if (strstart(opt, "hci", &endp)) {
|
|
|
|
if (!*endp || *endp == ',') {
|
|
|
|
if (*endp)
|
|
|
|
if (!strstart(endp, ",vlan=", 0))
|
|
|
|
opt = endp + 1;
|
|
|
|
|
|
|
|
return bt_hci_parse(opt);
|
|
|
|
}
|
|
|
|
} else if (strstart(opt, "vhci", &endp)) {
|
|
|
|
if (!*endp || *endp == ',') {
|
|
|
|
if (*endp) {
|
|
|
|
if (strstart(endp, ",vlan=", &p)) {
|
|
|
|
vlan = strtol(p, (char **) &endp, 0);
|
|
|
|
if (*endp) {
|
|
|
|
fprintf(stderr, "qemu: bad scatternet '%s'\n", p);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "qemu: bad parameter '%s'\n", endp + 1);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
vlan = 0;
|
|
|
|
|
|
|
|
bt_vhci_add(vlan);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else if (strstart(opt, "device:", &endp))
|
|
|
|
return !bt_device_add(endp);
|
|
|
|
|
|
|
|
fprintf(stderr, "qemu: bad bluetooth parameter '%s'\n", opt);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2008-09-29 07:19:47 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* QEMU Block devices */
|
|
|
|
|
2008-01-14 10:56:53 +08:00
|
|
|
#define HD_ALIAS "index=%d,media=disk"
|
2007-12-02 12:51:10 +08:00
|
|
|
#define CDROM_ALIAS "index=2,media=cdrom"
|
|
|
|
#define FD_ALIAS "index=%d,if=floppy"
|
2008-01-14 10:56:53 +08:00
|
|
|
#define PFLASH_ALIAS "if=pflash"
|
|
|
|
#define MTD_ALIAS "if=mtd"
|
2007-12-04 08:10:34 +08:00
|
|
|
#define SD_ALIAS "index=0,if=sd"
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
QemuOpts *drive_add(const char *file, const char *fmt, ...)
|
2007-12-02 12:51:10 +08:00
|
|
|
{
|
|
|
|
va_list ap;
|
2009-07-22 22:43:04 +08:00
|
|
|
char optstr[1024];
|
|
|
|
QemuOpts *opts;
|
2007-12-02 12:51:10 +08:00
|
|
|
|
|
|
|
va_start(ap, fmt);
|
2009-07-22 22:43:04 +08:00
|
|
|
vsnprintf(optstr, sizeof(optstr), fmt, ap);
|
2007-12-02 12:51:10 +08:00
|
|
|
va_end(ap);
|
|
|
|
|
2010-02-11 02:52:18 +08:00
|
|
|
opts = qemu_opts_parse(&qemu_drive_opts, optstr, 0);
|
2009-07-22 22:43:04 +08:00
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "%s: huh? duplicate? (%s)\n",
|
|
|
|
__FUNCTION__, optstr);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (file)
|
|
|
|
qemu_opt_set(opts, "file", file);
|
|
|
|
return opts;
|
2009-02-11 23:20:20 +08:00
|
|
|
}
|
|
|
|
|
2009-07-22 22:42:57 +08:00
|
|
|
DriveInfo *drive_get(BlockInterfaceType type, int bus, int unit)
|
2007-12-02 12:51:10 +08:00
|
|
|
{
|
2009-07-22 22:42:57 +08:00
|
|
|
DriveInfo *dinfo;
|
2007-12-02 12:51:10 +08:00
|
|
|
|
|
|
|
/* seek interface, bus and unit */
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
2009-07-22 22:42:57 +08:00
|
|
|
if (dinfo->type == type &&
|
|
|
|
dinfo->bus == bus &&
|
|
|
|
dinfo->unit == unit)
|
|
|
|
return dinfo;
|
|
|
|
}
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
|
2009-07-31 18:25:38 +08:00
|
|
|
DriveInfo *drive_get_by_id(const char *id)
|
2009-07-22 22:42:58 +08:00
|
|
|
{
|
|
|
|
DriveInfo *dinfo;
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
2009-07-22 22:42:58 +08:00
|
|
|
if (strcmp(id, dinfo->id))
|
|
|
|
continue;
|
|
|
|
return dinfo;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-12-17 11:55:57 +08:00
|
|
|
int drive_get_max_bus(BlockInterfaceType type)
|
2007-12-02 12:51:10 +08:00
|
|
|
{
|
|
|
|
int max_bus;
|
2009-07-22 22:42:57 +08:00
|
|
|
DriveInfo *dinfo;
|
2007-12-02 12:51:10 +08:00
|
|
|
|
|
|
|
max_bus = -1;
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
2009-07-22 22:42:57 +08:00
|
|
|
if(dinfo->type == type &&
|
|
|
|
dinfo->bus > max_bus)
|
|
|
|
max_bus = dinfo->bus;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
return max_bus;
|
|
|
|
}
|
|
|
|
|
2009-01-08 01:32:33 +08:00
|
|
|
const char *drive_get_serial(BlockDriverState *bdrv)
|
|
|
|
{
|
2009-07-22 22:42:57 +08:00
|
|
|
DriveInfo *dinfo;
|
2009-01-08 01:32:33 +08:00
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
2009-07-22 22:42:57 +08:00
|
|
|
if (dinfo->bdrv == bdrv)
|
|
|
|
return dinfo->serial;
|
|
|
|
}
|
2009-01-08 01:32:33 +08:00
|
|
|
|
|
|
|
return "\0";
|
|
|
|
}
|
|
|
|
|
2009-11-27 20:25:36 +08:00
|
|
|
BlockInterfaceErrorAction drive_get_on_error(
|
|
|
|
BlockDriverState *bdrv, int is_read)
|
2009-01-22 02:59:04 +08:00
|
|
|
{
|
2009-07-22 22:42:57 +08:00
|
|
|
DriveInfo *dinfo;
|
2009-01-22 02:59:04 +08:00
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(dinfo, &drives, next) {
|
2009-07-22 22:42:57 +08:00
|
|
|
if (dinfo->bdrv == bdrv)
|
2009-11-27 20:25:37 +08:00
|
|
|
return is_read ? dinfo->on_read_error : dinfo->on_write_error;
|
2009-07-22 22:42:57 +08:00
|
|
|
}
|
2009-01-22 02:59:04 +08:00
|
|
|
|
2009-11-27 20:25:37 +08:00
|
|
|
return is_read ? BLOCK_ERR_REPORT : BLOCK_ERR_STOP_ENOSPC;
|
2009-01-22 02:59:04 +08:00
|
|
|
}
|
|
|
|
|
2008-04-29 13:58:01 +08:00
|
|
|
static void bdrv_format_print(void *opaque, const char *name)
|
|
|
|
{
|
|
|
|
fprintf(stderr, " %s", name);
|
|
|
|
}
|
|
|
|
|
2009-09-26 03:42:46 +08:00
|
|
|
void drive_uninit(DriveInfo *dinfo)
|
2009-02-11 23:20:20 +08:00
|
|
|
{
|
2009-09-26 03:42:46 +08:00
|
|
|
qemu_opts_del(dinfo->opts);
|
|
|
|
bdrv_delete(dinfo->bdrv);
|
|
|
|
QTAILQ_REMOVE(&drives, dinfo, next);
|
|
|
|
qemu_free(dinfo);
|
2009-02-11 23:20:20 +08:00
|
|
|
}
|
|
|
|
|
2009-11-27 20:25:37 +08:00
|
|
|
static int parse_block_error_action(const char *buf, int is_read)
|
|
|
|
{
|
|
|
|
if (!strcmp(buf, "ignore")) {
|
|
|
|
return BLOCK_ERR_IGNORE;
|
|
|
|
} else if (!is_read && !strcmp(buf, "enospc")) {
|
|
|
|
return BLOCK_ERR_STOP_ENOSPC;
|
|
|
|
} else if (!strcmp(buf, "stop")) {
|
|
|
|
return BLOCK_ERR_STOP_ANY;
|
|
|
|
} else if (!strcmp(buf, "report")) {
|
|
|
|
return BLOCK_ERR_REPORT;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "qemu: '%s' invalid %s error action\n",
|
|
|
|
buf, is_read ? "read" : "write");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
DriveInfo *drive_init(QemuOpts *opts, void *opaque,
|
2009-07-22 22:42:57 +08:00
|
|
|
int *fatal_error)
|
2007-12-02 12:51:10 +08:00
|
|
|
{
|
2009-07-22 22:43:04 +08:00
|
|
|
const char *buf;
|
|
|
|
const char *file = NULL;
|
2007-12-07 06:11:20 +08:00
|
|
|
char devname[128];
|
2009-07-22 22:43:04 +08:00
|
|
|
const char *serial;
|
2007-12-07 06:11:20 +08:00
|
|
|
const char *mediastr = "";
|
2007-12-17 11:55:57 +08:00
|
|
|
BlockInterfaceType type;
|
2007-12-02 12:51:10 +08:00
|
|
|
enum { MEDIA_DISK, MEDIA_CDROM } media;
|
|
|
|
int bus_id, unit_id;
|
|
|
|
int cyls, heads, secs, translation;
|
2008-04-29 04:26:45 +08:00
|
|
|
BlockDriver *drv = NULL;
|
2009-02-11 23:20:46 +08:00
|
|
|
QEMUMachine *machine = opaque;
|
2007-12-02 12:51:10 +08:00
|
|
|
int max_devs;
|
|
|
|
int index;
|
2007-12-24 22:33:24 +08:00
|
|
|
int cache;
|
2009-08-20 22:58:35 +08:00
|
|
|
int aio = 0;
|
2009-10-26 22:25:16 +08:00
|
|
|
int ro = 0;
|
2009-11-27 20:25:37 +08:00
|
|
|
int bdrv_flags;
|
|
|
|
int on_read_error, on_write_error;
|
2009-06-18 21:14:10 +08:00
|
|
|
const char *devaddr;
|
2009-07-22 22:42:57 +08:00
|
|
|
DriveInfo *dinfo;
|
2009-07-22 22:43:04 +08:00
|
|
|
int snapshot = 0;
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
*fatal_error = 1;
|
2007-12-02 12:51:10 +08:00
|
|
|
|
|
|
|
translation = BIOS_ATA_TRANSLATION_AUTO;
|
2009-06-30 19:06:04 +08:00
|
|
|
cache = 1;
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-08-31 20:23:57 +08:00
|
|
|
if (machine && machine->use_scsi) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_SCSI;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = MAX_SCSI_DEVS;
|
2008-08-22 01:58:08 +08:00
|
|
|
pstrcpy(devname, sizeof(devname), "scsi");
|
2007-12-02 12:51:10 +08:00
|
|
|
} else {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_IDE;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = MAX_IDE_DEVS;
|
2008-08-22 01:58:08 +08:00
|
|
|
pstrcpy(devname, sizeof(devname), "ide");
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
media = MEDIA_DISK;
|
|
|
|
|
|
|
|
/* extract parameters */
|
2009-07-22 22:43:04 +08:00
|
|
|
bus_id = qemu_opt_get_number(opts, "bus", 0);
|
|
|
|
unit_id = qemu_opt_get_number(opts, "unit", -1);
|
|
|
|
index = qemu_opt_get_number(opts, "index", -1);
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
cyls = qemu_opt_get_number(opts, "cyls", 0);
|
|
|
|
heads = qemu_opt_get_number(opts, "heads", 0);
|
|
|
|
secs = qemu_opt_get_number(opts, "secs", 0);
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
snapshot = qemu_opt_get_bool(opts, "snapshot", 0);
|
2009-10-26 22:25:16 +08:00
|
|
|
ro = qemu_opt_get_bool(opts, "readonly", 0);
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
file = qemu_opt_get(opts, "file");
|
|
|
|
serial = qemu_opt_get(opts, "serial");
|
|
|
|
|
|
|
|
if ((buf = qemu_opt_get(opts, "if")) != NULL) {
|
2008-06-11 17:44:44 +08:00
|
|
|
pstrcpy(devname, sizeof(devname), buf);
|
2007-12-02 12:51:10 +08:00
|
|
|
if (!strcmp(buf, "ide")) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_IDE;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = MAX_IDE_DEVS;
|
|
|
|
} else if (!strcmp(buf, "scsi")) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_SCSI;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = MAX_SCSI_DEVS;
|
|
|
|
} else if (!strcmp(buf, "floppy")) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_FLOPPY;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = 0;
|
|
|
|
} else if (!strcmp(buf, "pflash")) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_PFLASH;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = 0;
|
|
|
|
} else if (!strcmp(buf, "mtd")) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_MTD;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = 0;
|
|
|
|
} else if (!strcmp(buf, "sd")) {
|
2007-12-17 11:55:57 +08:00
|
|
|
type = IF_SD;
|
2007-12-02 12:51:10 +08:00
|
|
|
max_devs = 0;
|
2008-12-05 03:52:44 +08:00
|
|
|
} else if (!strcmp(buf, "virtio")) {
|
|
|
|
type = IF_VIRTIO;
|
|
|
|
max_devs = 0;
|
2009-04-22 23:19:30 +08:00
|
|
|
} else if (!strcmp(buf, "xen")) {
|
|
|
|
type = IF_XEN;
|
|
|
|
max_devs = 0;
|
2009-07-31 18:25:39 +08:00
|
|
|
} else if (!strcmp(buf, "none")) {
|
|
|
|
type = IF_NONE;
|
|
|
|
max_devs = 0;
|
2009-04-22 23:19:30 +08:00
|
|
|
} else {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: unsupported bus type '%s'\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cyls || heads || secs) {
|
2009-10-17 17:08:47 +08:00
|
|
|
if (cyls < 1 || (type == IF_IDE && cyls > 16383)) {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: '%s' invalid physical cyls number\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
2009-10-17 17:08:47 +08:00
|
|
|
if (heads < 1 || (type == IF_IDE && heads > 16)) {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: '%s' invalid physical heads number\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
2009-10-17 17:08:47 +08:00
|
|
|
if (secs < 1 || (type == IF_IDE && secs > 63)) {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: '%s' invalid physical secs number\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
if ((buf = qemu_opt_get(opts, "trans")) != NULL) {
|
2007-12-02 12:51:10 +08:00
|
|
|
if (!cyls) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"qemu: '%s' trans must be used with cyls,heads and secs\n",
|
2009-07-22 22:43:04 +08:00
|
|
|
buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
if (!strcmp(buf, "none"))
|
|
|
|
translation = BIOS_ATA_TRANSLATION_NONE;
|
|
|
|
else if (!strcmp(buf, "lba"))
|
|
|
|
translation = BIOS_ATA_TRANSLATION_LBA;
|
|
|
|
else if (!strcmp(buf, "auto"))
|
|
|
|
translation = BIOS_ATA_TRANSLATION_AUTO;
|
|
|
|
else {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: '%s' invalid translation type\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
if ((buf = qemu_opt_get(opts, "media")) != NULL) {
|
2007-12-02 12:51:10 +08:00
|
|
|
if (!strcmp(buf, "disk")) {
|
|
|
|
media = MEDIA_DISK;
|
|
|
|
} else if (!strcmp(buf, "cdrom")) {
|
|
|
|
if (cyls || secs || heads) {
|
|
|
|
fprintf(stderr,
|
2009-07-22 22:43:04 +08:00
|
|
|
"qemu: '%s' invalid physical CHS format\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
media = MEDIA_CDROM;
|
|
|
|
} else {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: '%s' invalid media\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
if ((buf = qemu_opt_get(opts, "cache")) != NULL) {
|
2008-10-14 22:42:54 +08:00
|
|
|
if (!strcmp(buf, "off") || !strcmp(buf, "none"))
|
2007-12-24 22:33:24 +08:00
|
|
|
cache = 0;
|
2008-10-14 22:42:54 +08:00
|
|
|
else if (!strcmp(buf, "writethrough"))
|
2007-12-24 22:33:24 +08:00
|
|
|
cache = 1;
|
2008-10-14 22:42:54 +08:00
|
|
|
else if (!strcmp(buf, "writeback"))
|
|
|
|
cache = 2;
|
2007-12-24 22:33:24 +08:00
|
|
|
else {
|
|
|
|
fprintf(stderr, "qemu: invalid cache option\n");
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-24 22:33:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-20 22:58:35 +08:00
|
|
|
#ifdef CONFIG_LINUX_AIO
|
|
|
|
if ((buf = qemu_opt_get(opts, "aio")) != NULL) {
|
|
|
|
if (!strcmp(buf, "threads"))
|
|
|
|
aio = 0;
|
|
|
|
else if (!strcmp(buf, "native"))
|
|
|
|
aio = 1;
|
|
|
|
else {
|
|
|
|
fprintf(stderr, "qemu: invalid aio option\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
if ((buf = qemu_opt_get(opts, "format")) != NULL) {
|
2008-04-29 13:58:01 +08:00
|
|
|
if (strcmp(buf, "?") == 0) {
|
|
|
|
fprintf(stderr, "qemu: Supported formats:");
|
|
|
|
bdrv_iterate_format(bdrv_format_print, NULL);
|
|
|
|
fprintf(stderr, "\n");
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2008-04-29 13:58:01 +08:00
|
|
|
}
|
2009-10-28 01:41:44 +08:00
|
|
|
drv = bdrv_find_whitelisted_format(buf);
|
2008-04-29 04:26:45 +08:00
|
|
|
if (!drv) {
|
|
|
|
fprintf(stderr, "qemu: '%s' invalid format\n", buf);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2008-04-29 04:26:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-27 20:25:37 +08:00
|
|
|
on_write_error = BLOCK_ERR_STOP_ENOSPC;
|
2009-07-22 22:43:04 +08:00
|
|
|
if ((buf = qemu_opt_get(opts, "werror")) != NULL) {
|
2009-01-23 03:52:25 +08:00
|
|
|
if (type != IF_IDE && type != IF_SCSI && type != IF_VIRTIO) {
|
2009-01-23 03:52:21 +08:00
|
|
|
fprintf(stderr, "werror is no supported by this format\n");
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2009-01-22 02:59:04 +08:00
|
|
|
}
|
2009-11-27 20:25:37 +08:00
|
|
|
|
|
|
|
on_write_error = parse_block_error_action(buf, 0);
|
|
|
|
if (on_write_error < 0) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
on_read_error = BLOCK_ERR_REPORT;
|
|
|
|
if ((buf = qemu_opt_get(opts, "rerror")) != NULL) {
|
2009-11-27 20:25:39 +08:00
|
|
|
if (type != IF_IDE && type != IF_VIRTIO) {
|
2009-11-27 20:25:37 +08:00
|
|
|
fprintf(stderr, "rerror is no supported by this format\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
on_read_error = parse_block_error_action(buf, 1);
|
|
|
|
if (on_read_error < 0) {
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2009-01-22 02:59:04 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
if ((devaddr = qemu_opt_get(opts, "addr")) != NULL) {
|
2009-06-18 21:14:10 +08:00
|
|
|
if (type != IF_VIRTIO) {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "addr is not supported\n");
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2009-06-18 21:14:10 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-12-02 12:51:10 +08:00
|
|
|
/* compute bus and unit according index */
|
|
|
|
|
|
|
|
if (index != -1) {
|
|
|
|
if (bus_id != 0 || unit_id != -1) {
|
|
|
|
fprintf(stderr,
|
2009-07-22 22:43:04 +08:00
|
|
|
"qemu: index cannot be used with bus and unit\n");
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
if (max_devs == 0)
|
|
|
|
{
|
|
|
|
unit_id = index;
|
|
|
|
bus_id = 0;
|
|
|
|
} else {
|
|
|
|
unit_id = index % max_devs;
|
|
|
|
bus_id = index / max_devs;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if user doesn't specify a unit_id,
|
|
|
|
* try to find the first free
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (unit_id == -1) {
|
|
|
|
unit_id = 0;
|
2009-07-22 22:42:57 +08:00
|
|
|
while (drive_get(type, bus_id, unit_id) != NULL) {
|
2007-12-02 12:51:10 +08:00
|
|
|
unit_id++;
|
|
|
|
if (max_devs && unit_id >= max_devs) {
|
|
|
|
unit_id -= max_devs;
|
|
|
|
bus_id++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check unit id */
|
|
|
|
|
|
|
|
if (max_devs && unit_id >= max_devs) {
|
2009-07-22 22:43:04 +08:00
|
|
|
fprintf(stderr, "qemu: unit %d too big (max is %d)\n",
|
|
|
|
unit_id, max_devs - 1);
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ignore multiple definitions
|
|
|
|
*/
|
|
|
|
|
2009-07-22 22:42:57 +08:00
|
|
|
if (drive_get(type, bus_id, unit_id) != NULL) {
|
|
|
|
*fatal_error = 0;
|
|
|
|
return NULL;
|
|
|
|
}
|
2007-12-02 12:51:10 +08:00
|
|
|
|
|
|
|
/* init */
|
|
|
|
|
2009-07-22 22:42:57 +08:00
|
|
|
dinfo = qemu_mallocz(sizeof(*dinfo));
|
2009-07-31 18:25:34 +08:00
|
|
|
if ((buf = qemu_opts_id(opts)) != NULL) {
|
2009-07-22 22:43:04 +08:00
|
|
|
dinfo->id = qemu_strdup(buf);
|
|
|
|
} else {
|
2009-07-22 22:42:58 +08:00
|
|
|
/* no id supplied -> create one */
|
2009-07-22 22:43:04 +08:00
|
|
|
dinfo->id = qemu_mallocz(32);
|
2009-07-22 22:42:58 +08:00
|
|
|
if (type == IF_IDE || type == IF_SCSI)
|
|
|
|
mediastr = (media == MEDIA_CDROM) ? "-cd" : "-hd";
|
|
|
|
if (max_devs)
|
2009-07-22 22:43:04 +08:00
|
|
|
snprintf(dinfo->id, 32, "%s%i%s%i",
|
2009-07-22 22:42:58 +08:00
|
|
|
devname, bus_id, mediastr, unit_id);
|
|
|
|
else
|
2009-07-22 22:43:04 +08:00
|
|
|
snprintf(dinfo->id, 32, "%s%s%i",
|
2009-07-22 22:42:58 +08:00
|
|
|
devname, mediastr, unit_id);
|
|
|
|
}
|
|
|
|
dinfo->bdrv = bdrv_new(dinfo->id);
|
2009-07-22 22:42:57 +08:00
|
|
|
dinfo->devaddr = devaddr;
|
|
|
|
dinfo->type = type;
|
|
|
|
dinfo->bus = bus_id;
|
|
|
|
dinfo->unit = unit_id;
|
2009-11-27 20:25:37 +08:00
|
|
|
dinfo->on_read_error = on_read_error;
|
|
|
|
dinfo->on_write_error = on_write_error;
|
2009-07-22 22:43:04 +08:00
|
|
|
dinfo->opts = opts;
|
|
|
|
if (serial)
|
|
|
|
strncpy(dinfo->serial, serial, sizeof(serial));
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_INSERT_TAIL(&drives, dinfo, next);
|
2007-12-02 12:51:10 +08:00
|
|
|
|
2007-12-17 11:55:57 +08:00
|
|
|
switch(type) {
|
2007-12-02 12:51:10 +08:00
|
|
|
case IF_IDE:
|
|
|
|
case IF_SCSI:
|
2009-04-22 23:19:30 +08:00
|
|
|
case IF_XEN:
|
2009-09-16 03:23:28 +08:00
|
|
|
case IF_NONE:
|
2007-12-02 12:51:10 +08:00
|
|
|
switch(media) {
|
|
|
|
case MEDIA_DISK:
|
|
|
|
if (cyls != 0) {
|
2009-07-22 22:42:58 +08:00
|
|
|
bdrv_set_geometry_hint(dinfo->bdrv, cyls, heads, secs);
|
|
|
|
bdrv_set_translation_hint(dinfo->bdrv, translation);
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MEDIA_CDROM:
|
2009-07-22 22:42:58 +08:00
|
|
|
bdrv_set_type_hint(dinfo->bdrv, BDRV_TYPE_CDROM);
|
2007-12-02 12:51:10 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IF_SD:
|
|
|
|
/* FIXME: This isn't really a floppy, but it's a reasonable
|
|
|
|
approximation. */
|
|
|
|
case IF_FLOPPY:
|
2009-07-22 22:42:58 +08:00
|
|
|
bdrv_set_type_hint(dinfo->bdrv, BDRV_TYPE_FLOPPY);
|
2007-12-02 12:51:10 +08:00
|
|
|
break;
|
|
|
|
case IF_PFLASH:
|
|
|
|
case IF_MTD:
|
|
|
|
break;
|
qdev-ify virtio-blk.
First user of the new drive property. With this patch applied host
and guest config can be specified separately, like this:
-drive if=none,id=disk1,file=/path/to/disk.img
-device virtio-blk-pci,drive=disk1
You can set any property for virtio-blk-pci now. You can set the pci
address via addr=. You can switch the device into 0.10 compat mode
using class=0x0180. As this is per device you can have one 0.10 and one
0.11 virtio block device in a single virtual machine.
Old syntax continues to work. Internally it does the same as the two
lines above though. One side effect this has is a different
initialization order, which might result in a different pci address
being assigned by default.
Long term plan here is to have this working for all block devices, i.e.
once all scsi is properly qdev-ified you will be able to do something
like this:
-drive if=none,id=sda,file=/path/to/disk.img
-device lsi,id=lsi,addr=<pciaddr>
-device scsi-disk,drive=sda,bus=lsi.0,lun=<n>
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
Message-Id:
2009-07-31 18:25:41 +08:00
|
|
|
case IF_VIRTIO:
|
|
|
|
/* add virtio block device */
|
|
|
|
opts = qemu_opts_create(&qemu_device_opts, NULL, 0);
|
|
|
|
qemu_opt_set(opts, "driver", "virtio-blk-pci");
|
|
|
|
qemu_opt_set(opts, "drive", dinfo->id);
|
|
|
|
if (devaddr)
|
|
|
|
qemu_opt_set(opts, "addr", devaddr);
|
|
|
|
break;
|
2009-05-15 05:35:06 +08:00
|
|
|
case IF_COUNT:
|
|
|
|
abort();
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
2009-07-22 22:43:04 +08:00
|
|
|
if (!file) {
|
2009-07-22 22:42:57 +08:00
|
|
|
*fatal_error = 0;
|
|
|
|
return NULL;
|
|
|
|
}
|
2007-12-24 22:33:24 +08:00
|
|
|
bdrv_flags = 0;
|
2008-10-14 22:42:54 +08:00
|
|
|
if (snapshot) {
|
2007-12-24 22:33:24 +08:00
|
|
|
bdrv_flags |= BDRV_O_SNAPSHOT;
|
2008-10-14 22:42:54 +08:00
|
|
|
cache = 2; /* always use write-back with snapshot */
|
|
|
|
}
|
|
|
|
if (cache == 0) /* no caching */
|
|
|
|
bdrv_flags |= BDRV_O_NOCACHE;
|
|
|
|
else if (cache == 2) /* write-back */
|
|
|
|
bdrv_flags |= BDRV_O_CACHE_WB;
|
2009-08-20 22:58:35 +08:00
|
|
|
|
|
|
|
if (aio == 1) {
|
|
|
|
bdrv_flags |= BDRV_O_NATIVE_AIO;
|
|
|
|
} else {
|
|
|
|
bdrv_flags &= ~BDRV_O_NATIVE_AIO;
|
|
|
|
}
|
|
|
|
|
2009-10-26 22:25:16 +08:00
|
|
|
if (ro == 1) {
|
2010-01-17 22:48:13 +08:00
|
|
|
if (type != IF_SCSI && type != IF_VIRTIO && type != IF_FLOPPY) {
|
|
|
|
fprintf(stderr, "qemu: readonly flag not supported for drive with this interface\n");
|
2009-10-26 22:25:16 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
2010-01-17 22:48:12 +08:00
|
|
|
}
|
|
|
|
/*
|
|
|
|
* cdrom is read-only. Set it now, after above interface checking
|
|
|
|
* since readonly attribute not explicitly required, so no error.
|
|
|
|
*/
|
|
|
|
if (media == MEDIA_CDROM) {
|
2010-01-17 22:48:13 +08:00
|
|
|
ro = 1;
|
2009-10-26 22:25:16 +08:00
|
|
|
}
|
2010-01-17 22:48:13 +08:00
|
|
|
bdrv_flags |= ro ? 0 : BDRV_O_RDWR;
|
2009-10-26 22:25:16 +08:00
|
|
|
|
2009-07-22 22:42:58 +08:00
|
|
|
if (bdrv_open2(dinfo->bdrv, file, bdrv_flags, drv) < 0) {
|
2009-10-01 22:42:56 +08:00
|
|
|
fprintf(stderr, "qemu: could not open disk image %s: %s\n",
|
|
|
|
file, strerror(errno));
|
2009-07-22 22:42:57 +08:00
|
|
|
return NULL;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
2009-08-20 22:58:35 +08:00
|
|
|
|
2009-07-22 22:42:58 +08:00
|
|
|
if (bdrv_key_required(dinfo->bdrv))
|
2009-03-06 07:01:01 +08:00
|
|
|
autostart = 0;
|
2009-07-22 22:42:57 +08:00
|
|
|
*fatal_error = 0;
|
|
|
|
return dinfo;
|
2007-12-02 12:51:10 +08:00
|
|
|
}
|
|
|
|
|
2009-07-22 22:43:04 +08:00
|
|
|
static int drive_init_func(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
QEMUMachine *machine = opaque;
|
|
|
|
int fatal_error = 0;
|
|
|
|
|
|
|
|
if (drive_init(opts, machine, &fatal_error) == NULL) {
|
|
|
|
if (fatal_error)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int drive_enable_snapshot(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
if (NULL == qemu_opt_get(opts, "snapshot")) {
|
|
|
|
qemu_opt_set(opts, "snapshot", "on");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-07-02 06:19:02 +08:00
|
|
|
void qemu_register_boot_set(QEMUBootSetHandler *func, void *opaque)
|
|
|
|
{
|
|
|
|
boot_set_handler = func;
|
|
|
|
boot_set_opaque = opaque;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qemu_boot_set(const char *boot_devices)
|
|
|
|
{
|
|
|
|
if (!boot_set_handler) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
return boot_set_handler(boot_set_opaque, boot_devices);
|
|
|
|
}
|
|
|
|
|
2009-07-02 06:19:02 +08:00
|
|
|
static int parse_bootdevices(char *devices)
|
|
|
|
{
|
|
|
|
/* We just do some generic consistency checks */
|
|
|
|
const char *p;
|
|
|
|
int bitmap = 0;
|
|
|
|
|
|
|
|
for (p = devices; *p != '\0'; p++) {
|
|
|
|
/* Allowed boot devices are:
|
|
|
|
* a-b: floppy disk drives
|
|
|
|
* c-f: IDE disk drives
|
|
|
|
* g-m: machine implementation dependant drives
|
|
|
|
* n-p: network devices
|
|
|
|
* It's up to each machine implementation to check if the given boot
|
|
|
|
* devices match the actual hardware implementation and firmware
|
|
|
|
* features.
|
|
|
|
*/
|
|
|
|
if (*p < 'a' || *p > 'p') {
|
|
|
|
fprintf(stderr, "Invalid boot device '%c'\n", *p);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (bitmap & (1 << (*p - 'a'))) {
|
|
|
|
fprintf(stderr, "Boot device '%c' was given twice\n", *p);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
bitmap |= 1 << (*p - 'a');
|
|
|
|
}
|
|
|
|
return bitmap;
|
|
|
|
}
|
|
|
|
|
2009-07-02 06:19:02 +08:00
|
|
|
static void restore_boot_devices(void *opaque)
|
|
|
|
{
|
|
|
|
char *standard_boot_devices = opaque;
|
|
|
|
|
|
|
|
qemu_boot_set(standard_boot_devices);
|
|
|
|
|
|
|
|
qemu_unregister_reset(restore_boot_devices, standard_boot_devices);
|
|
|
|
qemu_free(standard_boot_devices);
|
|
|
|
}
|
|
|
|
|
2009-04-22 06:30:27 +08:00
|
|
|
static void numa_add(const char *optarg)
|
|
|
|
{
|
|
|
|
char option[128];
|
|
|
|
char *endptr;
|
|
|
|
unsigned long long value, endvalue;
|
|
|
|
int nodenr;
|
|
|
|
|
|
|
|
optarg = get_opt_name(option, 128, optarg, ',') + 1;
|
|
|
|
if (!strcmp(option, "node")) {
|
|
|
|
if (get_param_value(option, 128, "nodeid", optarg) == 0) {
|
|
|
|
nodenr = nb_numa_nodes;
|
|
|
|
} else {
|
|
|
|
nodenr = strtoull(option, NULL, 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (get_param_value(option, 128, "mem", optarg) == 0) {
|
|
|
|
node_mem[nodenr] = 0;
|
|
|
|
} else {
|
|
|
|
value = strtoull(option, &endptr, 0);
|
|
|
|
switch (*endptr) {
|
|
|
|
case 0: case 'M': case 'm':
|
|
|
|
value <<= 20;
|
|
|
|
break;
|
|
|
|
case 'G': case 'g':
|
|
|
|
value <<= 30;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
node_mem[nodenr] = value;
|
|
|
|
}
|
|
|
|
if (get_param_value(option, 128, "cpus", optarg) == 0) {
|
|
|
|
node_cpumask[nodenr] = 0;
|
|
|
|
} else {
|
|
|
|
value = strtoull(option, &endptr, 10);
|
|
|
|
if (value >= 64) {
|
|
|
|
value = 63;
|
|
|
|
fprintf(stderr, "only 64 CPUs in NUMA mode supported.\n");
|
|
|
|
} else {
|
|
|
|
if (*endptr == '-') {
|
|
|
|
endvalue = strtoull(endptr+1, &endptr, 10);
|
|
|
|
if (endvalue >= 63) {
|
|
|
|
endvalue = 62;
|
|
|
|
fprintf(stderr,
|
|
|
|
"only 63 CPUs in NUMA mode supported.\n");
|
|
|
|
}
|
2010-02-04 21:31:50 +08:00
|
|
|
value = (2ULL << endvalue) - (1ULL << value);
|
2009-04-22 06:30:27 +08:00
|
|
|
} else {
|
2010-02-04 21:31:50 +08:00
|
|
|
value = 1ULL << value;
|
2009-04-22 06:30:27 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
node_cpumask[nodenr] = value;
|
|
|
|
}
|
|
|
|
nb_numa_nodes++;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
extend -smp parsing to include cores= and threads= options
For injecting multi-core and multi-threading CPU topology into guests
extend the -smp syntax to accommodate cores and threads specification.
Syntax: -smp smp_value[,cores=nr_cores][,threads=nr_threads]\
[,socket=nr_sockets][,maxcpus=max_cpus]
smp_value is the legacy value specifying the total number of vCPUs for
the guest. If you specify one of cores, threads or sockets this value
can be omitted. Missing values will be computed to fulfill:
smp_value = nr_cores * nr_threads * nr_sockets
where it will favour sockets over cores over threads (to mimic the
current behavior, which will only inject multiple sockets.)
So -smp 4,threads=2 will inject two sockets with 2 threads each,
-smp cores=4 is an abbreviation for -smp 4,cores=4,threads=1,sockets=1.
If max_cpus (the number of hotpluggable CPUs) is omitted, it will
be set to smp_value.
Signed-off-by: Andre Przywara <andre.przywara@amd.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-08-19 21:42:40 +08:00
|
|
|
static void smp_parse(const char *optarg)
|
|
|
|
{
|
|
|
|
int smp, sockets = 0, threads = 0, cores = 0;
|
|
|
|
char *endptr;
|
|
|
|
char option[128];
|
|
|
|
|
|
|
|
smp = strtoul(optarg, &endptr, 10);
|
|
|
|
if (endptr != optarg) {
|
|
|
|
if (*endptr == ',') {
|
|
|
|
endptr++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (get_param_value(option, 128, "sockets", endptr) != 0)
|
|
|
|
sockets = strtoull(option, NULL, 10);
|
|
|
|
if (get_param_value(option, 128, "cores", endptr) != 0)
|
|
|
|
cores = strtoull(option, NULL, 10);
|
|
|
|
if (get_param_value(option, 128, "threads", endptr) != 0)
|
|
|
|
threads = strtoull(option, NULL, 10);
|
|
|
|
if (get_param_value(option, 128, "maxcpus", endptr) != 0)
|
|
|
|
max_cpus = strtoull(option, NULL, 10);
|
|
|
|
|
|
|
|
/* compute missing values, prefer sockets over cores over threads */
|
|
|
|
if (smp == 0 || sockets == 0) {
|
|
|
|
sockets = sockets > 0 ? sockets : 1;
|
|
|
|
cores = cores > 0 ? cores : 1;
|
|
|
|
threads = threads > 0 ? threads : 1;
|
|
|
|
if (smp == 0) {
|
|
|
|
smp = cores * threads * sockets;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (cores == 0) {
|
|
|
|
threads = threads > 0 ? threads : 1;
|
|
|
|
cores = smp / (sockets * threads);
|
|
|
|
} else {
|
2010-01-13 18:54:42 +08:00
|
|
|
if (sockets) {
|
extend -smp parsing to include cores= and threads= options
For injecting multi-core and multi-threading CPU topology into guests
extend the -smp syntax to accommodate cores and threads specification.
Syntax: -smp smp_value[,cores=nr_cores][,threads=nr_threads]\
[,socket=nr_sockets][,maxcpus=max_cpus]
smp_value is the legacy value specifying the total number of vCPUs for
the guest. If you specify one of cores, threads or sockets this value
can be omitted. Missing values will be computed to fulfill:
smp_value = nr_cores * nr_threads * nr_sockets
where it will favour sockets over cores over threads (to mimic the
current behavior, which will only inject multiple sockets.)
So -smp 4,threads=2 will inject two sockets with 2 threads each,
-smp cores=4 is an abbreviation for -smp 4,cores=4,threads=1,sockets=1.
If max_cpus (the number of hotpluggable CPUs) is omitted, it will
be set to smp_value.
Signed-off-by: Andre Przywara <andre.przywara@amd.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-08-19 21:42:40 +08:00
|
|
|
threads = smp / (cores * sockets);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
smp_cpus = smp;
|
|
|
|
smp_cores = cores > 0 ? cores : 1;
|
|
|
|
smp_threads = threads > 0 ? threads : 1;
|
|
|
|
if (max_cpus == 0)
|
|
|
|
max_cpus = smp_cpus;
|
|
|
|
}
|
|
|
|
|
2005-11-07 00:13:29 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* USB devices */
|
|
|
|
|
2009-03-06 07:01:01 +08:00
|
|
|
static int usb_device_add(const char *devname, int is_hotplug)
|
2005-11-07 00:13:29 +08:00
|
|
|
{
|
|
|
|
const char *p;
|
2009-08-31 20:24:00 +08:00
|
|
|
USBDevice *dev = NULL;
|
2005-11-07 00:13:29 +08:00
|
|
|
|
2009-08-31 20:24:00 +08:00
|
|
|
if (!usb_enabled)
|
2005-11-07 00:13:29 +08:00
|
|
|
return -1;
|
|
|
|
|
2009-10-26 22:56:45 +08:00
|
|
|
/* drivers with .usbdevice_name entry in USBDeviceInfo */
|
|
|
|
dev = usbdevice_create(devname);
|
|
|
|
if (dev)
|
|
|
|
goto done;
|
|
|
|
|
2009-08-31 20:24:00 +08:00
|
|
|
/* the other ones */
|
2005-11-07 00:13:29 +08:00
|
|
|
if (strstart(devname, "host:", &p)) {
|
|
|
|
dev = usb_host_device_open(p);
|
2008-11-09 08:04:26 +08:00
|
|
|
} else if (!strcmp(devname, "bt") || strstart(devname, "bt:", &p)) {
|
|
|
|
dev = usb_bt_init(devname[2] ? hci_init(p) :
|
|
|
|
bt_new_hci(qemu_find_bt_vlan(0)));
|
2005-11-07 00:13:29 +08:00
|
|
|
} else {
|
|
|
|
return -1;
|
|
|
|
}
|
2006-05-22 00:30:15 +08:00
|
|
|
if (!dev)
|
|
|
|
return -1;
|
|
|
|
|
2009-08-31 20:24:00 +08:00
|
|
|
done:
|
2005-11-07 00:13:29 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-08-22 03:27:48 +08:00
|
|
|
static int usb_device_del(const char *devname)
|
|
|
|
{
|
|
|
|
int bus_num, addr;
|
|
|
|
const char *p;
|
|
|
|
|
2008-09-14 09:07:41 +08:00
|
|
|
if (strstart(devname, "host:", &p))
|
|
|
|
return usb_host_device_close(p);
|
|
|
|
|
2009-08-31 20:24:00 +08:00
|
|
|
if (!usb_enabled)
|
2008-08-22 03:27:48 +08:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
p = strchr(devname, '.');
|
|
|
|
if (!p)
|
|
|
|
return -1;
|
|
|
|
bus_num = strtoul(devname, NULL, 0);
|
|
|
|
addr = strtoul(p + 1, NULL, 0);
|
|
|
|
|
2009-08-31 20:24:00 +08:00
|
|
|
return usb_device_delete_addr(bus_num, addr);
|
2008-08-22 03:27:48 +08:00
|
|
|
}
|
|
|
|
|
2009-07-15 19:59:26 +08:00
|
|
|
static int usb_parse(const char *cmdline)
|
|
|
|
{
|
2009-12-23 04:30:18 +08:00
|
|
|
int r;
|
|
|
|
r = usb_device_add(cmdline, 0);
|
|
|
|
if (r < 0) {
|
|
|
|
fprintf(stderr, "qemu: could not add USB device '%s'\n", cmdline);
|
|
|
|
}
|
|
|
|
return r;
|
2009-07-15 19:59:26 +08:00
|
|
|
}
|
|
|
|
|
2009-08-29 02:27:13 +08:00
|
|
|
void do_usb_add(Monitor *mon, const QDict *qdict)
|
2005-11-07 00:13:29 +08:00
|
|
|
{
|
2009-12-23 04:30:18 +08:00
|
|
|
const char *devname = qdict_get_str(qdict, "devname");
|
|
|
|
if (usb_device_add(devname, 1) < 0) {
|
2010-02-19 00:25:24 +08:00
|
|
|
error_report("could not add USB device '%s'", devname);
|
2009-12-23 04:30:18 +08:00
|
|
|
}
|
2005-11-07 00:13:29 +08:00
|
|
|
}
|
|
|
|
|
2009-08-29 02:27:13 +08:00
|
|
|
void do_usb_del(Monitor *mon, const QDict *qdict)
|
2005-11-07 00:13:29 +08:00
|
|
|
{
|
2009-12-23 04:30:18 +08:00
|
|
|
const char *devname = qdict_get_str(qdict, "devname");
|
|
|
|
if (usb_device_del(devname) < 0) {
|
2010-02-19 00:25:24 +08:00
|
|
|
error_report("could not delete USB device '%s'", devname);
|
2009-12-23 04:30:18 +08:00
|
|
|
}
|
2005-11-07 00:13:29 +08:00
|
|
|
}
|
|
|
|
|
2007-04-30 08:51:09 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* PCMCIA/Cardbus */
|
|
|
|
|
|
|
|
static struct pcmcia_socket_entry_s {
|
2009-05-10 08:44:56 +08:00
|
|
|
PCMCIASocket *socket;
|
2007-04-30 08:51:09 +08:00
|
|
|
struct pcmcia_socket_entry_s *next;
|
|
|
|
} *pcmcia_sockets = 0;
|
|
|
|
|
2009-05-10 08:44:56 +08:00
|
|
|
void pcmcia_socket_register(PCMCIASocket *socket)
|
2007-04-30 08:51:09 +08:00
|
|
|
{
|
|
|
|
struct pcmcia_socket_entry_s *entry;
|
|
|
|
|
|
|
|
entry = qemu_malloc(sizeof(struct pcmcia_socket_entry_s));
|
|
|
|
entry->socket = socket;
|
|
|
|
entry->next = pcmcia_sockets;
|
|
|
|
pcmcia_sockets = entry;
|
|
|
|
}
|
|
|
|
|
2009-05-10 08:44:56 +08:00
|
|
|
void pcmcia_socket_unregister(PCMCIASocket *socket)
|
2007-04-30 08:51:09 +08:00
|
|
|
{
|
|
|
|
struct pcmcia_socket_entry_s *entry, **ptr;
|
|
|
|
|
|
|
|
ptr = &pcmcia_sockets;
|
|
|
|
for (entry = *ptr; entry; ptr = &entry->next, entry = *ptr)
|
|
|
|
if (entry->socket == socket) {
|
|
|
|
*ptr = entry->next;
|
|
|
|
qemu_free(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-06 07:01:23 +08:00
|
|
|
void pcmcia_info(Monitor *mon)
|
2007-04-30 08:51:09 +08:00
|
|
|
{
|
|
|
|
struct pcmcia_socket_entry_s *iter;
|
2009-03-06 07:01:23 +08:00
|
|
|
|
2007-04-30 08:51:09 +08:00
|
|
|
if (!pcmcia_sockets)
|
2009-03-06 07:01:23 +08:00
|
|
|
monitor_printf(mon, "No PCMCIA sockets\n");
|
2007-04-30 08:51:09 +08:00
|
|
|
|
|
|
|
for (iter = pcmcia_sockets; iter; iter = iter->next)
|
2009-03-06 07:01:23 +08:00
|
|
|
monitor_printf(mon, "%s: %s\n", iter->socket->slot_string,
|
|
|
|
iter->socket->attached ? iter->socket->card_string :
|
|
|
|
"Empty");
|
2007-04-30 08:51:09 +08:00
|
|
|
}
|
|
|
|
|
2007-06-22 07:34:19 +08:00
|
|
|
/***********************************************************/
|
2004-04-01 03:00:16 +08:00
|
|
|
/* I/O handling */
|
2003-06-24 21:42:40 +08:00
|
|
|
|
2004-03-15 05:44:30 +08:00
|
|
|
typedef struct IOHandlerRecord {
|
|
|
|
int fd;
|
2005-11-16 06:16:05 +08:00
|
|
|
IOCanRWHandler *fd_read_poll;
|
|
|
|
IOHandler *fd_read;
|
|
|
|
IOHandler *fd_write;
|
2007-03-01 05:59:44 +08:00
|
|
|
int deleted;
|
2004-03-15 05:44:30 +08:00
|
|
|
void *opaque;
|
|
|
|
/* temporary data */
|
|
|
|
struct pollfd *ufd;
|
2004-04-01 03:00:16 +08:00
|
|
|
struct IOHandlerRecord *next;
|
2004-03-15 05:44:30 +08:00
|
|
|
} IOHandlerRecord;
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
static IOHandlerRecord *first_io_handler;
|
2004-03-15 05:44:30 +08:00
|
|
|
|
2005-11-16 06:16:05 +08:00
|
|
|
/* XXX: fd_read_poll should be suppressed, but an API change is
|
|
|
|
necessary in the character devices to suppress fd_can_read(). */
|
2007-09-17 05:08:06 +08:00
|
|
|
int qemu_set_fd_handler2(int fd,
|
|
|
|
IOCanRWHandler *fd_read_poll,
|
|
|
|
IOHandler *fd_read,
|
|
|
|
IOHandler *fd_write,
|
2005-11-16 06:16:05 +08:00
|
|
|
void *opaque)
|
2004-03-15 05:44:30 +08:00
|
|
|
{
|
2005-11-16 06:16:05 +08:00
|
|
|
IOHandlerRecord **pioh, *ioh;
|
2004-03-15 05:44:30 +08:00
|
|
|
|
2005-11-16 06:16:05 +08:00
|
|
|
if (!fd_read && !fd_write) {
|
|
|
|
pioh = &first_io_handler;
|
|
|
|
for(;;) {
|
|
|
|
ioh = *pioh;
|
|
|
|
if (ioh == NULL)
|
|
|
|
break;
|
|
|
|
if (ioh->fd == fd) {
|
2007-03-01 05:59:44 +08:00
|
|
|
ioh->deleted = 1;
|
2005-11-16 06:16:05 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
pioh = &ioh->next;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for(ioh = first_io_handler; ioh != NULL; ioh = ioh->next) {
|
|
|
|
if (ioh->fd == fd)
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
ioh = qemu_mallocz(sizeof(IOHandlerRecord));
|
|
|
|
ioh->next = first_io_handler;
|
|
|
|
first_io_handler = ioh;
|
|
|
|
found:
|
|
|
|
ioh->fd = fd;
|
|
|
|
ioh->fd_read_poll = fd_read_poll;
|
|
|
|
ioh->fd_read = fd_read;
|
|
|
|
ioh->fd_write = fd_write;
|
|
|
|
ioh->opaque = opaque;
|
2007-03-01 05:59:44 +08:00
|
|
|
ioh->deleted = 0;
|
2005-11-16 06:16:05 +08:00
|
|
|
}
|
2004-03-15 05:44:30 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-09-17 05:08:06 +08:00
|
|
|
int qemu_set_fd_handler(int fd,
|
|
|
|
IOHandler *fd_read,
|
|
|
|
IOHandler *fd_write,
|
2005-11-16 06:16:05 +08:00
|
|
|
void *opaque)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
2005-11-16 06:16:05 +08:00
|
|
|
return qemu_set_fd_handler2(fd, NULL, fd_read, fd_write, opaque);
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
|
2008-11-01 02:07:17 +08:00
|
|
|
#ifdef _WIN32
|
2006-04-13 04:21:17 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* Polling handling */
|
|
|
|
|
|
|
|
typedef struct PollingEntry {
|
|
|
|
PollingFunc *func;
|
|
|
|
void *opaque;
|
|
|
|
struct PollingEntry *next;
|
|
|
|
} PollingEntry;
|
|
|
|
|
|
|
|
static PollingEntry *first_polling_entry;
|
|
|
|
|
|
|
|
int qemu_add_polling_cb(PollingFunc *func, void *opaque)
|
|
|
|
{
|
|
|
|
PollingEntry **ppe, *pe;
|
|
|
|
pe = qemu_mallocz(sizeof(PollingEntry));
|
|
|
|
pe->func = func;
|
|
|
|
pe->opaque = opaque;
|
|
|
|
for(ppe = &first_polling_entry; *ppe != NULL; ppe = &(*ppe)->next);
|
|
|
|
*ppe = pe;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_del_polling_cb(PollingFunc *func, void *opaque)
|
|
|
|
{
|
|
|
|
PollingEntry **ppe, *pe;
|
|
|
|
for(ppe = &first_polling_entry; *ppe != NULL; ppe = &(*ppe)->next) {
|
|
|
|
pe = *ppe;
|
|
|
|
if (pe->func == func && pe->opaque == opaque) {
|
|
|
|
*ppe = pe->next;
|
|
|
|
qemu_free(pe);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-06-26 01:18:27 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* Wait objects support */
|
|
|
|
typedef struct WaitObjects {
|
|
|
|
int num;
|
|
|
|
HANDLE events[MAXIMUM_WAIT_OBJECTS + 1];
|
|
|
|
WaitObjectFunc *func[MAXIMUM_WAIT_OBJECTS + 1];
|
|
|
|
void *opaque[MAXIMUM_WAIT_OBJECTS + 1];
|
|
|
|
} WaitObjects;
|
|
|
|
|
|
|
|
static WaitObjects wait_objects = {0};
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2006-06-26 01:18:27 +08:00
|
|
|
int qemu_add_wait_object(HANDLE handle, WaitObjectFunc *func, void *opaque)
|
|
|
|
{
|
|
|
|
WaitObjects *w = &wait_objects;
|
|
|
|
|
|
|
|
if (w->num >= MAXIMUM_WAIT_OBJECTS)
|
|
|
|
return -1;
|
|
|
|
w->events[w->num] = handle;
|
|
|
|
w->func[w->num] = func;
|
|
|
|
w->opaque[w->num] = opaque;
|
|
|
|
w->num++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_del_wait_object(HANDLE handle, WaitObjectFunc *func, void *opaque)
|
|
|
|
{
|
|
|
|
int i, found;
|
|
|
|
WaitObjects *w = &wait_objects;
|
|
|
|
|
|
|
|
found = 0;
|
|
|
|
for (i = 0; i < w->num; i++) {
|
|
|
|
if (w->events[i] == handle)
|
|
|
|
found = 1;
|
|
|
|
if (found) {
|
|
|
|
w->events[i] = w->events[i + 1];
|
|
|
|
w->func[i] = w->func[i + 1];
|
|
|
|
w->opaque[i] = w->opaque[i + 1];
|
2007-09-17 16:09:54 +08:00
|
|
|
}
|
2006-06-26 01:18:27 +08:00
|
|
|
}
|
|
|
|
if (found)
|
|
|
|
w->num--;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* ram save/restore */
|
|
|
|
|
2009-09-10 09:04:23 +08:00
|
|
|
#define RAM_SAVE_FLAG_FULL 0x01 /* Obsolete, not used anymore */
|
2008-10-07 04:21:51 +08:00
|
|
|
#define RAM_SAVE_FLAG_COMPRESS 0x02
|
|
|
|
#define RAM_SAVE_FLAG_MEM_SIZE 0x04
|
|
|
|
#define RAM_SAVE_FLAG_PAGE 0x08
|
|
|
|
#define RAM_SAVE_FLAG_EOS 0x10
|
|
|
|
|
|
|
|
static int is_dup_page(uint8_t *page, uint8_t ch)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
2008-10-07 04:21:51 +08:00
|
|
|
uint32_t val = ch << 24 | ch << 16 | ch << 8 | ch;
|
|
|
|
uint32_t *array = (uint32_t *)page;
|
|
|
|
int i;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2008-10-07 04:21:51 +08:00
|
|
|
for (i = 0; i < (TARGET_PAGE_SIZE / 4); i++) {
|
|
|
|
if (array[i] != val)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ram_save_block(QEMUFile *f)
|
|
|
|
{
|
2009-10-02 05:12:16 +08:00
|
|
|
static ram_addr_t current_addr = 0;
|
|
|
|
ram_addr_t saved_addr = current_addr;
|
|
|
|
ram_addr_t addr = 0;
|
2008-10-07 04:21:51 +08:00
|
|
|
int found = 0;
|
|
|
|
|
2009-04-12 01:15:54 +08:00
|
|
|
while (addr < last_ram_offset) {
|
2008-10-07 04:21:51 +08:00
|
|
|
if (cpu_physical_memory_get_dirty(current_addr, MIGRATION_DIRTY_FLAG)) {
|
2009-04-11 22:47:08 +08:00
|
|
|
uint8_t *p;
|
2008-10-07 04:21:51 +08:00
|
|
|
|
|
|
|
cpu_physical_memory_reset_dirty(current_addr,
|
|
|
|
current_addr + TARGET_PAGE_SIZE,
|
|
|
|
MIGRATION_DIRTY_FLAG);
|
|
|
|
|
2009-04-11 22:47:08 +08:00
|
|
|
p = qemu_get_ram_ptr(current_addr);
|
2008-10-07 04:21:51 +08:00
|
|
|
|
2009-04-11 22:47:08 +08:00
|
|
|
if (is_dup_page(p, *p)) {
|
2008-10-07 04:21:51 +08:00
|
|
|
qemu_put_be64(f, current_addr | RAM_SAVE_FLAG_COMPRESS);
|
2009-04-11 22:47:08 +08:00
|
|
|
qemu_put_byte(f, *p);
|
2008-10-07 04:21:51 +08:00
|
|
|
} else {
|
|
|
|
qemu_put_be64(f, current_addr | RAM_SAVE_FLAG_PAGE);
|
2009-04-11 22:47:08 +08:00
|
|
|
qemu_put_buffer(f, p, TARGET_PAGE_SIZE);
|
2006-08-06 21:36:11 +08:00
|
|
|
}
|
2008-10-07 04:21:51 +08:00
|
|
|
|
|
|
|
found = 1;
|
|
|
|
break;
|
2006-08-06 21:36:11 +08:00
|
|
|
}
|
2008-10-07 04:21:51 +08:00
|
|
|
addr += TARGET_PAGE_SIZE;
|
2009-04-12 01:15:54 +08:00
|
|
|
current_addr = (saved_addr + addr) % last_ram_offset;
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
2008-10-07 04:21:51 +08:00
|
|
|
|
|
|
|
return found;
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
|
2009-12-02 18:29:38 +08:00
|
|
|
static uint64_t bytes_transferred;
|
2008-10-07 04:21:51 +08:00
|
|
|
|
2009-10-02 05:12:16 +08:00
|
|
|
static ram_addr_t ram_save_remaining(void)
|
2008-10-07 04:21:51 +08:00
|
|
|
{
|
2009-10-02 05:12:16 +08:00
|
|
|
ram_addr_t addr;
|
|
|
|
ram_addr_t count = 0;
|
2008-10-07 04:21:51 +08:00
|
|
|
|
2009-04-12 01:15:54 +08:00
|
|
|
for (addr = 0; addr < last_ram_offset; addr += TARGET_PAGE_SIZE) {
|
2008-10-07 04:21:51 +08:00
|
|
|
if (cpu_physical_memory_get_dirty(addr, MIGRATION_DIRTY_FLAG))
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2009-05-22 05:38:01 +08:00
|
|
|
uint64_t ram_bytes_remaining(void)
|
|
|
|
{
|
|
|
|
return ram_save_remaining() * TARGET_PAGE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t ram_bytes_transferred(void)
|
|
|
|
{
|
|
|
|
return bytes_transferred;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t ram_bytes_total(void)
|
|
|
|
{
|
|
|
|
return last_ram_offset;
|
|
|
|
}
|
|
|
|
|
2009-12-01 01:21:21 +08:00
|
|
|
static int ram_save_live(Monitor *mon, QEMUFile *f, int stage, void *opaque)
|
2008-10-07 04:21:51 +08:00
|
|
|
{
|
2009-10-02 05:12:16 +08:00
|
|
|
ram_addr_t addr;
|
2009-05-29 03:22:57 +08:00
|
|
|
uint64_t bytes_transferred_last;
|
|
|
|
double bwidth = 0;
|
|
|
|
uint64_t expected_time = 0;
|
2008-10-07 04:21:51 +08:00
|
|
|
|
2009-12-01 01:21:21 +08:00
|
|
|
if (stage < 0) {
|
|
|
|
cpu_physical_memory_set_dirty_tracking(0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-05-23 05:51:45 +08:00
|
|
|
if (cpu_physical_sync_dirty_bitmap(0, TARGET_PHYS_ADDR_MAX) != 0) {
|
2009-05-02 06:22:51 +08:00
|
|
|
qemu_file_set_error(f);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-10-07 04:21:51 +08:00
|
|
|
if (stage == 1) {
|
2009-12-02 18:29:38 +08:00
|
|
|
bytes_transferred = 0;
|
|
|
|
|
2008-10-07 04:21:51 +08:00
|
|
|
/* Make sure all dirty bits are set */
|
2009-04-12 01:15:54 +08:00
|
|
|
for (addr = 0; addr < last_ram_offset; addr += TARGET_PAGE_SIZE) {
|
2008-10-07 04:21:51 +08:00
|
|
|
if (!cpu_physical_memory_get_dirty(addr, MIGRATION_DIRTY_FLAG))
|
|
|
|
cpu_physical_memory_set_dirty(addr);
|
|
|
|
}
|
2009-05-02 06:22:51 +08:00
|
|
|
|
2008-10-07 04:21:51 +08:00
|
|
|
/* Enable dirty memory tracking */
|
|
|
|
cpu_physical_memory_set_dirty_tracking(1);
|
|
|
|
|
2009-04-12 01:15:54 +08:00
|
|
|
qemu_put_be64(f, last_ram_offset | RAM_SAVE_FLAG_MEM_SIZE);
|
2008-10-07 04:21:51 +08:00
|
|
|
}
|
|
|
|
|
2009-05-29 03:22:57 +08:00
|
|
|
bytes_transferred_last = bytes_transferred;
|
2010-01-26 16:31:46 +08:00
|
|
|
bwidth = qemu_get_clock_ns(rt_clock);
|
2009-05-29 03:22:57 +08:00
|
|
|
|
2008-10-07 04:21:51 +08:00
|
|
|
while (!qemu_file_rate_limit(f)) {
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = ram_save_block(f);
|
2009-05-22 05:38:01 +08:00
|
|
|
bytes_transferred += ret * TARGET_PAGE_SIZE;
|
2008-10-07 04:21:51 +08:00
|
|
|
if (ret == 0) /* no more blocks */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-01-26 16:31:46 +08:00
|
|
|
bwidth = qemu_get_clock_ns(rt_clock) - bwidth;
|
2009-05-29 03:22:57 +08:00
|
|
|
bwidth = (bytes_transferred - bytes_transferred_last) / bwidth;
|
|
|
|
|
|
|
|
/* if we haven't transferred anything this round, force expected_time to a
|
|
|
|
* a very high value, but without crashing */
|
|
|
|
if (bwidth == 0)
|
|
|
|
bwidth = 0.000001;
|
|
|
|
|
2008-10-07 04:21:51 +08:00
|
|
|
/* try transferring iterative blocks of memory */
|
|
|
|
if (stage == 3) {
|
|
|
|
/* flush all remaining blocks regardless of rate limiting */
|
2009-05-22 05:38:01 +08:00
|
|
|
while (ram_save_block(f) != 0) {
|
|
|
|
bytes_transferred += TARGET_PAGE_SIZE;
|
|
|
|
}
|
2009-04-06 03:30:55 +08:00
|
|
|
cpu_physical_memory_set_dirty_tracking(0);
|
2008-10-07 04:21:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
qemu_put_be64(f, RAM_SAVE_FLAG_EOS);
|
|
|
|
|
2009-05-29 03:22:57 +08:00
|
|
|
expected_time = ram_save_remaining() * TARGET_PAGE_SIZE / bwidth;
|
|
|
|
|
|
|
|
return (stage == 2) && (expected_time <= migrate_max_downtime());
|
2008-10-07 04:21:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int ram_load(QEMUFile *f, void *opaque, int version_id)
|
|
|
|
{
|
2009-10-02 05:12:16 +08:00
|
|
|
ram_addr_t addr;
|
2008-10-07 04:21:51 +08:00
|
|
|
int flags;
|
|
|
|
|
|
|
|
if (version_id != 3)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
do {
|
|
|
|
addr = qemu_get_be64(f);
|
|
|
|
|
|
|
|
flags = addr & ~TARGET_PAGE_MASK;
|
|
|
|
addr &= TARGET_PAGE_MASK;
|
|
|
|
|
|
|
|
if (flags & RAM_SAVE_FLAG_MEM_SIZE) {
|
2009-04-12 01:15:54 +08:00
|
|
|
if (addr != last_ram_offset)
|
2008-10-07 04:21:51 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & RAM_SAVE_FLAG_COMPRESS) {
|
|
|
|
uint8_t ch = qemu_get_byte(f);
|
2009-06-23 01:39:00 +08:00
|
|
|
memset(qemu_get_ram_ptr(addr), ch, TARGET_PAGE_SIZE);
|
|
|
|
#ifndef _WIN32
|
2009-06-18 05:46:12 +08:00
|
|
|
if (ch == 0 &&
|
|
|
|
(!kvm_enabled() || kvm_has_sync_mmu())) {
|
|
|
|
madvise(qemu_get_ram_ptr(addr), TARGET_PAGE_SIZE, MADV_DONTNEED);
|
2009-06-23 01:39:00 +08:00
|
|
|
}
|
2009-06-18 05:46:12 +08:00
|
|
|
#endif
|
2009-12-01 01:21:21 +08:00
|
|
|
} else if (flags & RAM_SAVE_FLAG_PAGE) {
|
2009-04-11 22:47:08 +08:00
|
|
|
qemu_get_buffer(f, qemu_get_ram_ptr(addr), TARGET_PAGE_SIZE);
|
2009-12-01 01:21:21 +08:00
|
|
|
}
|
|
|
|
if (qemu_file_has_error(f)) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
2008-10-07 04:21:51 +08:00
|
|
|
} while (!(flags & RAM_SAVE_FLAG_EOS));
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-10-09 03:50:24 +08:00
|
|
|
void qemu_service_io(void)
|
|
|
|
{
|
2009-04-25 02:03:11 +08:00
|
|
|
qemu_notify_event();
|
2008-10-09 03:50:24 +08:00
|
|
|
}
|
|
|
|
|
2005-06-05 22:49:17 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* machine registration */
|
|
|
|
|
2008-10-04 15:24:27 +08:00
|
|
|
static QEMUMachine *first_machine = NULL;
|
2009-02-11 23:21:54 +08:00
|
|
|
QEMUMachine *current_machine = NULL;
|
2005-06-05 22:49:17 +08:00
|
|
|
|
|
|
|
int qemu_register_machine(QEMUMachine *m)
|
|
|
|
{
|
|
|
|
QEMUMachine **pm;
|
|
|
|
pm = &first_machine;
|
|
|
|
while (*pm != NULL)
|
|
|
|
pm = &(*pm)->next;
|
|
|
|
m->next = NULL;
|
|
|
|
*pm = m;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-11-18 09:44:38 +08:00
|
|
|
static QEMUMachine *find_machine(const char *name)
|
2005-06-05 22:49:17 +08:00
|
|
|
{
|
|
|
|
QEMUMachine *m;
|
|
|
|
|
|
|
|
for(m = first_machine; m != NULL; m = m->next) {
|
|
|
|
if (!strcmp(m->name, name))
|
|
|
|
return m;
|
2009-07-22 17:02:50 +08:00
|
|
|
if (m->alias && !strcmp(m->alias, name))
|
|
|
|
return m;
|
2005-06-05 22:49:17 +08:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-05-22 09:41:01 +08:00
|
|
|
static QEMUMachine *find_default_machine(void)
|
|
|
|
{
|
|
|
|
QEMUMachine *m;
|
|
|
|
|
|
|
|
for(m = first_machine; m != NULL; m = m->next) {
|
|
|
|
if (m->is_default) {
|
|
|
|
return m;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* main execution loop */
|
|
|
|
|
2007-11-18 09:44:38 +08:00
|
|
|
static void gui_update(void *opaque)
|
2004-04-01 03:00:16 +08:00
|
|
|
{
|
2009-01-16 06:14:11 +08:00
|
|
|
uint64_t interval = GUI_REFRESH_INTERVAL;
|
2007-06-08 09:57:56 +08:00
|
|
|
DisplayState *ds = opaque;
|
2009-01-16 06:14:11 +08:00
|
|
|
DisplayChangeListener *dcl = ds->listeners;
|
|
|
|
|
2010-01-26 19:21:16 +08:00
|
|
|
qemu_flush_coalesced_mmio_buffer();
|
2009-01-16 06:14:11 +08:00
|
|
|
dpy_refresh(ds);
|
|
|
|
|
|
|
|
while (dcl != NULL) {
|
|
|
|
if (dcl->gui_timer_interval &&
|
|
|
|
dcl->gui_timer_interval < interval)
|
|
|
|
interval = dcl->gui_timer_interval;
|
|
|
|
dcl = dcl->next;
|
|
|
|
}
|
|
|
|
qemu_mod_timer(ds->gui_timer, interval + qemu_get_clock(rt_clock));
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
|
2009-01-22 03:28:13 +08:00
|
|
|
static void nographic_update(void *opaque)
|
|
|
|
{
|
|
|
|
uint64_t interval = GUI_REFRESH_INTERVAL;
|
|
|
|
|
2010-01-26 19:21:16 +08:00
|
|
|
qemu_flush_coalesced_mmio_buffer();
|
2009-01-22 03:28:13 +08:00
|
|
|
qemu_mod_timer(nographic_timer, interval + qemu_get_clock(rt_clock));
|
|
|
|
}
|
|
|
|
|
2010-03-02 02:10:30 +08:00
|
|
|
void cpu_synchronize_all_states(void)
|
|
|
|
{
|
|
|
|
CPUState *cpu;
|
|
|
|
|
|
|
|
for (cpu = first_cpu; cpu; cpu = cpu->next_cpu) {
|
|
|
|
cpu_synchronize_state(cpu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cpu_synchronize_all_post_reset(void)
|
|
|
|
{
|
|
|
|
CPUState *cpu;
|
|
|
|
|
|
|
|
for (cpu = first_cpu; cpu; cpu = cpu->next_cpu) {
|
|
|
|
cpu_synchronize_post_reset(cpu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cpu_synchronize_all_post_init(void)
|
|
|
|
{
|
|
|
|
CPUState *cpu;
|
|
|
|
|
|
|
|
for (cpu = first_cpu; cpu; cpu = cpu->next_cpu) {
|
|
|
|
cpu_synchronize_post_init(cpu);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-11-11 08:00:47 +08:00
|
|
|
struct vm_change_state_entry {
|
|
|
|
VMChangeStateHandler *cb;
|
|
|
|
void *opaque;
|
2009-09-12 15:36:22 +08:00
|
|
|
QLIST_ENTRY (vm_change_state_entry) entries;
|
2005-11-11 08:00:47 +08:00
|
|
|
};
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
static QLIST_HEAD(vm_change_state_head, vm_change_state_entry) vm_change_state_head;
|
2005-11-11 08:00:47 +08:00
|
|
|
|
|
|
|
VMChangeStateEntry *qemu_add_vm_change_state_handler(VMChangeStateHandler *cb,
|
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
VMChangeStateEntry *e;
|
|
|
|
|
|
|
|
e = qemu_mallocz(sizeof (*e));
|
|
|
|
|
|
|
|
e->cb = cb;
|
|
|
|
e->opaque = opaque;
|
2009-09-12 15:36:22 +08:00
|
|
|
QLIST_INSERT_HEAD(&vm_change_state_head, e, entries);
|
2005-11-11 08:00:47 +08:00
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_del_vm_change_state_handler(VMChangeStateEntry *e)
|
|
|
|
{
|
2009-09-12 15:36:22 +08:00
|
|
|
QLIST_REMOVE (e, entries);
|
2005-11-11 08:00:47 +08:00
|
|
|
qemu_free (e);
|
|
|
|
}
|
|
|
|
|
2009-01-23 01:15:29 +08:00
|
|
|
static void vm_state_notify(int running, int reason)
|
2005-11-11 08:00:47 +08:00
|
|
|
{
|
|
|
|
VMChangeStateEntry *e;
|
|
|
|
|
|
|
|
for (e = vm_change_state_head.lh_first; e; e = e->entries.le_next) {
|
2009-01-23 01:15:29 +08:00
|
|
|
e->cb(e->opaque, running, reason);
|
2005-11-11 08:00:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
static void resume_all_vcpus(void);
|
|
|
|
static void pause_all_vcpus(void);
|
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
void vm_start(void)
|
|
|
|
{
|
|
|
|
if (!vm_running) {
|
|
|
|
cpu_enable_ticks();
|
|
|
|
vm_running = 1;
|
2009-01-23 01:15:29 +08:00
|
|
|
vm_state_notify(1, 0);
|
2009-04-25 02:04:07 +08:00
|
|
|
resume_all_vcpus();
|
2004-04-01 03:00:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-06-20 20:37:32 +08:00
|
|
|
/* reset/shutdown handler */
|
|
|
|
|
|
|
|
typedef struct QEMUResetEntry {
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_ENTRY(QEMUResetEntry) entry;
|
2004-06-20 20:37:32 +08:00
|
|
|
QEMUResetHandler *func;
|
|
|
|
void *opaque;
|
|
|
|
} QEMUResetEntry;
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
static QTAILQ_HEAD(reset_handlers, QEMUResetEntry) reset_handlers =
|
|
|
|
QTAILQ_HEAD_INITIALIZER(reset_handlers);
|
2004-06-20 20:37:32 +08:00
|
|
|
static int reset_requested;
|
|
|
|
static int shutdown_requested;
|
2005-07-02 22:31:34 +08:00
|
|
|
static int powerdown_requested;
|
2009-04-25 02:03:54 +08:00
|
|
|
static int debug_requested;
|
2009-04-25 02:04:02 +08:00
|
|
|
static int vmstop_requested;
|
2004-06-20 20:37:32 +08:00
|
|
|
|
2008-03-18 14:53:05 +08:00
|
|
|
int qemu_shutdown_requested(void)
|
|
|
|
{
|
|
|
|
int r = shutdown_requested;
|
|
|
|
shutdown_requested = 0;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qemu_reset_requested(void)
|
|
|
|
{
|
|
|
|
int r = reset_requested;
|
|
|
|
reset_requested = 0;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int qemu_powerdown_requested(void)
|
|
|
|
{
|
|
|
|
int r = powerdown_requested;
|
|
|
|
powerdown_requested = 0;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:54 +08:00
|
|
|
static int qemu_debug_requested(void)
|
|
|
|
{
|
|
|
|
int r = debug_requested;
|
|
|
|
debug_requested = 0;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:02 +08:00
|
|
|
static int qemu_vmstop_requested(void)
|
|
|
|
{
|
|
|
|
int r = vmstop_requested;
|
|
|
|
vmstop_requested = 0;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void do_vm_stop(int reason)
|
|
|
|
{
|
|
|
|
if (vm_running) {
|
|
|
|
cpu_disable_ticks();
|
|
|
|
vm_running = 0;
|
2009-04-25 02:04:07 +08:00
|
|
|
pause_all_vcpus();
|
2009-04-25 02:04:02 +08:00
|
|
|
vm_state_notify(0, reason);
|
2010-02-25 23:06:59 +08:00
|
|
|
monitor_protocol_event(QEVENT_STOP, NULL);
|
2009-04-25 02:04:02 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-27 15:25:07 +08:00
|
|
|
void qemu_register_reset(QEMUResetHandler *func, void *opaque)
|
2004-06-20 20:37:32 +08:00
|
|
|
{
|
2009-07-02 06:19:02 +08:00
|
|
|
QEMUResetEntry *re = qemu_mallocz(sizeof(QEMUResetEntry));
|
2004-06-20 20:37:32 +08:00
|
|
|
|
|
|
|
re->func = func;
|
|
|
|
re->opaque = opaque;
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_INSERT_TAIL(&reset_handlers, re, entry);
|
2004-06-20 20:37:32 +08:00
|
|
|
}
|
|
|
|
|
2009-07-02 06:19:02 +08:00
|
|
|
void qemu_unregister_reset(QEMUResetHandler *func, void *opaque)
|
2004-06-20 20:37:32 +08:00
|
|
|
{
|
|
|
|
QEMUResetEntry *re;
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(re, &reset_handlers, entry) {
|
2009-07-02 06:19:02 +08:00
|
|
|
if (re->func == func && re->opaque == opaque) {
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_REMOVE(&reset_handlers, re, entry);
|
2009-07-02 06:19:02 +08:00
|
|
|
qemu_free(re);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_system_reset(void)
|
|
|
|
{
|
|
|
|
QEMUResetEntry *re, *nre;
|
|
|
|
|
|
|
|
/* reset all devices */
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH_SAFE(re, &reset_handlers, entry, nre) {
|
2004-06-20 20:37:32 +08:00
|
|
|
re->func(re->opaque);
|
|
|
|
}
|
2010-03-10 23:06:34 +08:00
|
|
|
monitor_protocol_event(QEVENT_RESET, NULL);
|
2010-03-02 02:10:30 +08:00
|
|
|
cpu_synchronize_all_post_reset();
|
2004-06-20 20:37:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_system_reset_request(void)
|
|
|
|
{
|
2006-10-03 03:44:22 +08:00
|
|
|
if (no_reboot) {
|
|
|
|
shutdown_requested = 1;
|
|
|
|
} else {
|
|
|
|
reset_requested = 1;
|
|
|
|
}
|
2009-04-25 02:03:11 +08:00
|
|
|
qemu_notify_event();
|
2004-06-20 20:37:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_system_shutdown_request(void)
|
|
|
|
{
|
|
|
|
shutdown_requested = 1;
|
2009-04-25 02:03:11 +08:00
|
|
|
qemu_notify_event();
|
2004-06-20 20:37:32 +08:00
|
|
|
}
|
|
|
|
|
2005-07-02 22:31:34 +08:00
|
|
|
void qemu_system_powerdown_request(void)
|
|
|
|
{
|
|
|
|
powerdown_requested = 1;
|
2009-04-25 02:03:11 +08:00
|
|
|
qemu_notify_event();
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
#ifdef CONFIG_IOTHREAD
|
|
|
|
static void qemu_system_vmstop_request(int reason)
|
2009-04-25 02:03:11 +08:00
|
|
|
{
|
2009-04-25 02:04:07 +08:00
|
|
|
vmstop_requested = reason;
|
|
|
|
qemu_notify_event();
|
2004-06-20 20:37:32 +08:00
|
|
|
}
|
2009-04-25 02:04:07 +08:00
|
|
|
#endif
|
2004-06-20 20:37:32 +08:00
|
|
|
|
2009-04-25 02:03:29 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
static int io_thread_fd = -1;
|
|
|
|
|
|
|
|
static void qemu_event_increment(void)
|
2009-04-25 02:03:25 +08:00
|
|
|
{
|
2010-02-11 07:23:46 +08:00
|
|
|
/* Write 8 bytes to be compatible with eventfd. */
|
|
|
|
static uint64_t val = 1;
|
2010-01-20 07:56:18 +08:00
|
|
|
ssize_t ret;
|
2009-04-25 02:03:29 +08:00
|
|
|
|
|
|
|
if (io_thread_fd == -1)
|
|
|
|
return;
|
|
|
|
|
2010-02-03 03:33:10 +08:00
|
|
|
do {
|
2010-02-11 07:23:46 +08:00
|
|
|
ret = write(io_thread_fd, &val, sizeof(val));
|
2010-02-03 03:33:10 +08:00
|
|
|
} while (ret < 0 && errno == EINTR);
|
|
|
|
|
|
|
|
/* EAGAIN is fine, a read must be pending. */
|
|
|
|
if (ret < 0 && errno != EAGAIN) {
|
2010-01-20 07:56:18 +08:00
|
|
|
fprintf(stderr, "qemu_event_increment: write() filed: %s\n",
|
|
|
|
strerror(errno));
|
|
|
|
exit (1);
|
|
|
|
}
|
2009-04-25 02:03:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_event_read(void *opaque)
|
|
|
|
{
|
|
|
|
int fd = (unsigned long)opaque;
|
|
|
|
ssize_t len;
|
2010-02-03 03:33:09 +08:00
|
|
|
char buffer[512];
|
2009-04-25 02:03:29 +08:00
|
|
|
|
2010-02-11 07:23:46 +08:00
|
|
|
/* Drain the notify pipe. For eventfd, only 8 bytes will be read. */
|
2009-04-25 02:03:29 +08:00
|
|
|
do {
|
|
|
|
len = read(fd, buffer, sizeof(buffer));
|
2010-02-03 03:33:09 +08:00
|
|
|
} while ((len == -1 && errno == EINTR) || len == sizeof(buffer));
|
2009-04-25 02:03:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int qemu_event_init(void)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
int fds[2];
|
|
|
|
|
2010-02-11 07:23:46 +08:00
|
|
|
err = qemu_eventfd(fds);
|
2009-04-25 02:03:29 +08:00
|
|
|
if (err == -1)
|
|
|
|
return -errno;
|
|
|
|
|
|
|
|
err = fcntl_setfl(fds[0], O_NONBLOCK);
|
|
|
|
if (err < 0)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
err = fcntl_setfl(fds[1], O_NONBLOCK);
|
|
|
|
if (err < 0)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
qemu_set_fd_handler2(fds[0], NULL, qemu_event_read, NULL,
|
|
|
|
(void *)(unsigned long)fds[0]);
|
|
|
|
|
|
|
|
io_thread_fd = fds[1];
|
2009-04-30 02:38:28 +08:00
|
|
|
return 0;
|
|
|
|
|
2009-04-25 02:03:29 +08:00
|
|
|
fail:
|
|
|
|
close(fds[0]);
|
|
|
|
close(fds[1]);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
HANDLE qemu_event_handle;
|
|
|
|
|
|
|
|
static void dummy_event_handler(void *opaque)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qemu_event_init(void)
|
|
|
|
{
|
|
|
|
qemu_event_handle = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
|
|
if (!qemu_event_handle) {
|
2009-09-28 04:03:56 +08:00
|
|
|
fprintf(stderr, "Failed CreateEvent: %ld\n", GetLastError());
|
2009-04-25 02:03:29 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
qemu_add_wait_object(qemu_event_handle, dummy_event_handler, NULL);
|
2009-04-25 02:03:25 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:29 +08:00
|
|
|
static void qemu_event_increment(void)
|
|
|
|
{
|
2009-09-27 18:38:18 +08:00
|
|
|
if (!SetEvent(qemu_event_handle)) {
|
2009-09-28 04:03:56 +08:00
|
|
|
fprintf(stderr, "qemu_event_increment: SetEvent failed: %ld\n",
|
2009-09-27 18:38:18 +08:00
|
|
|
GetLastError());
|
|
|
|
exit (1);
|
|
|
|
}
|
2009-04-25 02:03:29 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
static int cpu_can_run(CPUState *env)
|
|
|
|
{
|
|
|
|
if (env->stop)
|
|
|
|
return 0;
|
|
|
|
if (env->stopped)
|
|
|
|
return 0;
|
2010-02-04 07:44:17 +08:00
|
|
|
if (!vm_running)
|
|
|
|
return 0;
|
2009-04-25 02:04:07 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef CONFIG_IOTHREAD
|
2009-04-25 02:03:29 +08:00
|
|
|
static int qemu_init_main_loop(void)
|
|
|
|
{
|
|
|
|
return qemu_event_init();
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:41 +08:00
|
|
|
void qemu_init_vcpu(void *_env)
|
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
|
|
|
|
extend -smp parsing to include cores= and threads= options
For injecting multi-core and multi-threading CPU topology into guests
extend the -smp syntax to accommodate cores and threads specification.
Syntax: -smp smp_value[,cores=nr_cores][,threads=nr_threads]\
[,socket=nr_sockets][,maxcpus=max_cpus]
smp_value is the legacy value specifying the total number of vCPUs for
the guest. If you specify one of cores, threads or sockets this value
can be omitted. Missing values will be computed to fulfill:
smp_value = nr_cores * nr_threads * nr_sockets
where it will favour sockets over cores over threads (to mimic the
current behavior, which will only inject multiple sockets.)
So -smp 4,threads=2 will inject two sockets with 2 threads each,
-smp cores=4 is an abbreviation for -smp 4,cores=4,threads=1,sockets=1.
If max_cpus (the number of hotpluggable CPUs) is omitted, it will
be set to smp_value.
Signed-off-by: Andre Przywara <andre.przywara@amd.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-08-19 21:42:40 +08:00
|
|
|
env->nr_cores = smp_cores;
|
|
|
|
env->nr_threads = smp_threads;
|
2010-01-06 00:26:34 +08:00
|
|
|
if (kvm_enabled())
|
|
|
|
kvm_init_vcpu(env);
|
2009-04-25 02:03:41 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:45 +08:00
|
|
|
int qemu_cpu_self(void *env)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
static void resume_all_vcpus(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pause_all_vcpus(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:45 +08:00
|
|
|
void qemu_cpu_kick(void *env)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
void qemu_notify_event(void)
|
|
|
|
{
|
|
|
|
CPUState *env = cpu_single_env;
|
|
|
|
|
2010-03-10 18:38:42 +08:00
|
|
|
qemu_event_increment ();
|
2009-04-25 02:04:07 +08:00
|
|
|
if (env) {
|
|
|
|
cpu_exit(env);
|
2009-08-11 06:07:24 +08:00
|
|
|
}
|
2010-03-10 18:38:43 +08:00
|
|
|
if (next_cpu && env != next_cpu) {
|
|
|
|
cpu_exit(next_cpu);
|
|
|
|
}
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
|
2009-10-08 03:38:03 +08:00
|
|
|
void qemu_mutex_lock_iothread(void) {}
|
|
|
|
void qemu_mutex_unlock_iothread(void) {}
|
2009-04-25 02:03:49 +08:00
|
|
|
|
2009-04-25 02:04:02 +08:00
|
|
|
void vm_stop(int reason)
|
|
|
|
{
|
|
|
|
do_vm_stop(reason);
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
#else /* CONFIG_IOTHREAD */
|
|
|
|
|
|
|
|
#include "qemu-thread.h"
|
|
|
|
|
|
|
|
QemuMutex qemu_global_mutex;
|
|
|
|
static QemuMutex qemu_fair_mutex;
|
|
|
|
|
|
|
|
static QemuThread io_thread;
|
|
|
|
|
|
|
|
static QemuThread *tcg_cpu_thread;
|
|
|
|
static QemuCond *tcg_halt_cond;
|
|
|
|
|
|
|
|
static int qemu_system_ready;
|
|
|
|
/* cpu creation */
|
|
|
|
static QemuCond qemu_cpu_cond;
|
|
|
|
/* system init */
|
|
|
|
static QemuCond qemu_system_cond;
|
|
|
|
static QemuCond qemu_pause_cond;
|
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
static void tcg_block_io_signals(void);
|
|
|
|
static void kvm_block_io_signals(CPUState *env);
|
2009-04-25 02:04:07 +08:00
|
|
|
static void unblock_io_signals(void);
|
|
|
|
static int tcg_has_work(void);
|
2010-02-18 06:14:41 +08:00
|
|
|
static int cpu_has_work(CPUState *env);
|
2009-04-25 02:04:07 +08:00
|
|
|
|
|
|
|
static int qemu_init_main_loop(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = qemu_event_init();
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
qemu_cond_init(&qemu_pause_cond);
|
|
|
|
qemu_mutex_init(&qemu_fair_mutex);
|
|
|
|
qemu_mutex_init(&qemu_global_mutex);
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
|
|
|
|
unblock_io_signals();
|
|
|
|
qemu_thread_self(&io_thread);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-02-18 06:14:41 +08:00
|
|
|
static void qemu_wait_io_event_common(CPUState *env)
|
|
|
|
{
|
|
|
|
if (env->stop) {
|
|
|
|
env->stop = 0;
|
|
|
|
env->stopped = 1;
|
|
|
|
qemu_cond_signal(&qemu_pause_cond);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
static void qemu_wait_io_event(CPUState *env)
|
|
|
|
{
|
|
|
|
while (!tcg_has_work())
|
|
|
|
qemu_cond_timedwait(env->halt_cond, &qemu_global_mutex, 1000);
|
|
|
|
|
|
|
|
qemu_mutex_unlock(&qemu_global_mutex);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Users of qemu_global_mutex can be starved, having no chance
|
|
|
|
* to acquire it since this path will get to it first.
|
|
|
|
* So use another lock to provide fairness.
|
|
|
|
*/
|
|
|
|
qemu_mutex_lock(&qemu_fair_mutex);
|
|
|
|
qemu_mutex_unlock(&qemu_fair_mutex);
|
|
|
|
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
2010-02-18 06:14:41 +08:00
|
|
|
qemu_wait_io_event_common(env);
|
|
|
|
}
|
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
static void qemu_kvm_eat_signal(CPUState *env, int timeout)
|
|
|
|
{
|
|
|
|
struct timespec ts;
|
|
|
|
int r, e;
|
|
|
|
siginfo_t siginfo;
|
|
|
|
sigset_t waitset;
|
|
|
|
|
|
|
|
ts.tv_sec = timeout / 1000;
|
|
|
|
ts.tv_nsec = (timeout % 1000) * 1000000;
|
|
|
|
|
|
|
|
sigemptyset(&waitset);
|
|
|
|
sigaddset(&waitset, SIG_IPI);
|
|
|
|
|
|
|
|
qemu_mutex_unlock(&qemu_global_mutex);
|
|
|
|
r = sigtimedwait(&waitset, &siginfo, &ts);
|
|
|
|
e = errno;
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
|
|
|
|
if (r == -1 && !(e == EAGAIN || e == EINTR)) {
|
|
|
|
fprintf(stderr, "sigtimedwait: %s\n", strerror(e));
|
|
|
|
exit(1);
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-18 06:14:41 +08:00
|
|
|
static void qemu_kvm_wait_io_event(CPUState *env)
|
|
|
|
{
|
|
|
|
while (!cpu_has_work(env))
|
|
|
|
qemu_cond_timedwait(env->halt_cond, &qemu_global_mutex, 1000);
|
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
qemu_kvm_eat_signal(env, 0);
|
2010-02-18 06:14:41 +08:00
|
|
|
qemu_wait_io_event_common(env);
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int qemu_cpu_exec(CPUState *env);
|
|
|
|
|
|
|
|
static void *kvm_cpu_thread_fn(void *arg)
|
|
|
|
{
|
|
|
|
CPUState *env = arg;
|
|
|
|
|
|
|
|
qemu_thread_self(env->thread);
|
2009-09-03 05:59:04 +08:00
|
|
|
if (kvm_enabled())
|
|
|
|
kvm_init_vcpu(env);
|
2009-04-25 02:04:07 +08:00
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
kvm_block_io_signals(env);
|
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
/* signal CPU creation */
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
env->created = 1;
|
|
|
|
qemu_cond_signal(&qemu_cpu_cond);
|
|
|
|
|
|
|
|
/* and wait for machine initialization */
|
|
|
|
while (!qemu_system_ready)
|
|
|
|
qemu_cond_timedwait(&qemu_system_cond, &qemu_global_mutex, 100);
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
if (cpu_can_run(env))
|
|
|
|
qemu_cpu_exec(env);
|
2010-02-18 06:14:41 +08:00
|
|
|
qemu_kvm_wait_io_event(env);
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tcg_cpu_exec(void);
|
|
|
|
|
|
|
|
static void *tcg_cpu_thread_fn(void *arg)
|
|
|
|
{
|
|
|
|
CPUState *env = arg;
|
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
tcg_block_io_signals();
|
2009-04-25 02:04:07 +08:00
|
|
|
qemu_thread_self(env->thread);
|
|
|
|
|
|
|
|
/* signal CPU creation */
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
for (env = first_cpu; env != NULL; env = env->next_cpu)
|
|
|
|
env->created = 1;
|
|
|
|
qemu_cond_signal(&qemu_cpu_cond);
|
|
|
|
|
|
|
|
/* and wait for machine initialization */
|
|
|
|
while (!qemu_system_ready)
|
|
|
|
qemu_cond_timedwait(&qemu_system_cond, &qemu_global_mutex, 100);
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
tcg_cpu_exec();
|
|
|
|
qemu_wait_io_event(cur_cpu);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_cpu_kick(void *_env)
|
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
|
|
|
qemu_cond_broadcast(env->halt_cond);
|
|
|
|
if (kvm_enabled())
|
2010-02-18 06:14:42 +08:00
|
|
|
qemu_thread_signal(env->thread, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
|
2009-09-29 02:27:44 +08:00
|
|
|
int qemu_cpu_self(void *_env)
|
2009-04-25 02:04:07 +08:00
|
|
|
{
|
2009-09-29 02:27:44 +08:00
|
|
|
CPUState *env = _env;
|
|
|
|
QemuThread this;
|
|
|
|
|
|
|
|
qemu_thread_self(&this);
|
|
|
|
|
|
|
|
return qemu_thread_equal(&this, env->thread);
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void cpu_signal(int sig)
|
|
|
|
{
|
|
|
|
if (cpu_single_env)
|
|
|
|
cpu_exit(cpu_single_env);
|
|
|
|
}
|
|
|
|
|
2010-02-18 06:14:42 +08:00
|
|
|
static void tcg_block_io_signals(void)
|
2009-04-25 02:04:07 +08:00
|
|
|
{
|
|
|
|
sigset_t set;
|
|
|
|
struct sigaction sigact;
|
|
|
|
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGUSR2);
|
|
|
|
sigaddset(&set, SIGIO);
|
|
|
|
sigaddset(&set, SIGALRM);
|
2010-02-18 06:14:40 +08:00
|
|
|
sigaddset(&set, SIGCHLD);
|
2009-04-25 02:04:07 +08:00
|
|
|
pthread_sigmask(SIG_BLOCK, &set, NULL);
|
|
|
|
|
|
|
|
sigemptyset(&set);
|
2010-02-18 06:14:42 +08:00
|
|
|
sigaddset(&set, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
pthread_sigmask(SIG_UNBLOCK, &set, NULL);
|
|
|
|
|
|
|
|
memset(&sigact, 0, sizeof(sigact));
|
|
|
|
sigact.sa_handler = cpu_signal;
|
2010-02-18 06:14:42 +08:00
|
|
|
sigaction(SIG_IPI, &sigact, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dummy_signal(int sig)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void kvm_block_io_signals(CPUState *env)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
sigset_t set;
|
|
|
|
struct sigaction sigact;
|
|
|
|
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGUSR2);
|
|
|
|
sigaddset(&set, SIGIO);
|
|
|
|
sigaddset(&set, SIGALRM);
|
|
|
|
sigaddset(&set, SIGCHLD);
|
|
|
|
sigaddset(&set, SIG_IPI);
|
|
|
|
pthread_sigmask(SIG_BLOCK, &set, NULL);
|
|
|
|
|
|
|
|
pthread_sigmask(SIG_BLOCK, NULL, &set);
|
|
|
|
sigdelset(&set, SIG_IPI);
|
|
|
|
|
|
|
|
memset(&sigact, 0, sizeof(sigact));
|
|
|
|
sigact.sa_handler = dummy_signal;
|
|
|
|
sigaction(SIG_IPI, &sigact, NULL);
|
|
|
|
|
|
|
|
r = kvm_set_signal_mask(env, &set);
|
|
|
|
if (r) {
|
|
|
|
fprintf(stderr, "kvm_set_signal_mask: %s\n", strerror(r));
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-04-25 02:04:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void unblock_io_signals(void)
|
|
|
|
{
|
|
|
|
sigset_t set;
|
|
|
|
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGUSR2);
|
|
|
|
sigaddset(&set, SIGIO);
|
|
|
|
sigaddset(&set, SIGALRM);
|
|
|
|
pthread_sigmask(SIG_UNBLOCK, &set, NULL);
|
|
|
|
|
|
|
|
sigemptyset(&set);
|
2010-02-18 06:14:42 +08:00
|
|
|
sigaddset(&set, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
pthread_sigmask(SIG_BLOCK, &set, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_signal_lock(unsigned int msecs)
|
|
|
|
{
|
|
|
|
qemu_mutex_lock(&qemu_fair_mutex);
|
|
|
|
|
|
|
|
while (qemu_mutex_trylock(&qemu_global_mutex)) {
|
2010-02-18 06:14:42 +08:00
|
|
|
qemu_thread_signal(tcg_cpu_thread, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
if (!qemu_mutex_timedlock(&qemu_global_mutex, msecs))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
qemu_mutex_unlock(&qemu_fair_mutex);
|
|
|
|
}
|
|
|
|
|
2009-10-08 03:38:03 +08:00
|
|
|
void qemu_mutex_lock_iothread(void)
|
2009-04-25 02:04:07 +08:00
|
|
|
{
|
|
|
|
if (kvm_enabled()) {
|
|
|
|
qemu_mutex_lock(&qemu_fair_mutex);
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
qemu_mutex_unlock(&qemu_fair_mutex);
|
|
|
|
} else
|
|
|
|
qemu_signal_lock(100);
|
|
|
|
}
|
|
|
|
|
2009-10-08 03:38:03 +08:00
|
|
|
void qemu_mutex_unlock_iothread(void)
|
2009-04-25 02:04:07 +08:00
|
|
|
{
|
|
|
|
qemu_mutex_unlock(&qemu_global_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int all_vcpus_paused(void)
|
|
|
|
{
|
|
|
|
CPUState *penv = first_cpu;
|
|
|
|
|
|
|
|
while (penv) {
|
|
|
|
if (!penv->stopped)
|
|
|
|
return 0;
|
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pause_all_vcpus(void)
|
|
|
|
{
|
|
|
|
CPUState *penv = first_cpu;
|
|
|
|
|
|
|
|
while (penv) {
|
|
|
|
penv->stop = 1;
|
2010-02-18 06:14:42 +08:00
|
|
|
qemu_thread_signal(penv->thread, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
qemu_cpu_kick(penv);
|
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!all_vcpus_paused()) {
|
|
|
|
qemu_cond_timedwait(&qemu_pause_cond, &qemu_global_mutex, 100);
|
|
|
|
penv = first_cpu;
|
|
|
|
while (penv) {
|
2010-02-18 06:14:42 +08:00
|
|
|
qemu_thread_signal(penv->thread, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void resume_all_vcpus(void)
|
|
|
|
{
|
|
|
|
CPUState *penv = first_cpu;
|
|
|
|
|
|
|
|
while (penv) {
|
|
|
|
penv->stop = 0;
|
|
|
|
penv->stopped = 0;
|
2010-02-18 06:14:42 +08:00
|
|
|
qemu_thread_signal(penv->thread, SIG_IPI);
|
2009-04-25 02:04:07 +08:00
|
|
|
qemu_cpu_kick(penv);
|
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tcg_init_vcpu(void *_env)
|
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
|
|
|
/* share a single thread for all cpus with TCG */
|
|
|
|
if (!tcg_cpu_thread) {
|
|
|
|
env->thread = qemu_mallocz(sizeof(QemuThread));
|
|
|
|
env->halt_cond = qemu_mallocz(sizeof(QemuCond));
|
|
|
|
qemu_cond_init(env->halt_cond);
|
|
|
|
qemu_thread_create(env->thread, tcg_cpu_thread_fn, env);
|
|
|
|
while (env->created == 0)
|
|
|
|
qemu_cond_timedwait(&qemu_cpu_cond, &qemu_global_mutex, 100);
|
|
|
|
tcg_cpu_thread = env->thread;
|
|
|
|
tcg_halt_cond = env->halt_cond;
|
|
|
|
} else {
|
|
|
|
env->thread = tcg_cpu_thread;
|
|
|
|
env->halt_cond = tcg_halt_cond;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void kvm_start_vcpu(CPUState *env)
|
|
|
|
{
|
|
|
|
env->thread = qemu_mallocz(sizeof(QemuThread));
|
|
|
|
env->halt_cond = qemu_mallocz(sizeof(QemuCond));
|
|
|
|
qemu_cond_init(env->halt_cond);
|
|
|
|
qemu_thread_create(env->thread, kvm_cpu_thread_fn, env);
|
|
|
|
while (env->created == 0)
|
|
|
|
qemu_cond_timedwait(&qemu_cpu_cond, &qemu_global_mutex, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_init_vcpu(void *_env)
|
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
|
|
|
|
2010-01-06 00:26:34 +08:00
|
|
|
env->nr_cores = smp_cores;
|
|
|
|
env->nr_threads = smp_threads;
|
2009-04-25 02:04:07 +08:00
|
|
|
if (kvm_enabled())
|
|
|
|
kvm_start_vcpu(env);
|
|
|
|
else
|
|
|
|
tcg_init_vcpu(env);
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_notify_event(void)
|
|
|
|
{
|
|
|
|
qemu_event_increment();
|
|
|
|
}
|
|
|
|
|
|
|
|
void vm_stop(int reason)
|
|
|
|
{
|
|
|
|
QemuThread me;
|
|
|
|
qemu_thread_self(&me);
|
|
|
|
|
|
|
|
if (!qemu_thread_equal(&me, &io_thread)) {
|
|
|
|
qemu_system_vmstop_request(reason);
|
|
|
|
/*
|
|
|
|
* FIXME: should not return to device code in case
|
|
|
|
* vm_stop() has been requested.
|
|
|
|
*/
|
|
|
|
if (cpu_single_env) {
|
|
|
|
cpu_exit(cpu_single_env);
|
|
|
|
cpu_single_env->stop = 1;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
do_vm_stop(reason);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2007-04-19 02:11:47 +08:00
|
|
|
#ifdef _WIN32
|
2008-12-08 03:30:18 +08:00
|
|
|
static void host_main_loop_wait(int *timeout)
|
2008-11-01 02:07:17 +08:00
|
|
|
{
|
|
|
|
int ret, ret2, i;
|
2006-04-13 04:21:17 +08:00
|
|
|
PollingEntry *pe;
|
|
|
|
|
2004-03-15 05:44:30 +08:00
|
|
|
|
2006-04-13 04:21:17 +08:00
|
|
|
/* XXX: need to suppress polling by better using win32 events */
|
|
|
|
ret = 0;
|
|
|
|
for(pe = first_polling_entry; pe != NULL; pe = pe->next) {
|
|
|
|
ret |= pe->func(pe->opaque);
|
|
|
|
}
|
2007-04-19 01:56:02 +08:00
|
|
|
if (ret == 0) {
|
2006-06-26 01:18:27 +08:00
|
|
|
int err;
|
|
|
|
WaitObjects *w = &wait_objects;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2008-11-01 02:07:17 +08:00
|
|
|
ret = WaitForMultipleObjects(w->num, w->events, FALSE, *timeout);
|
2006-06-26 01:18:27 +08:00
|
|
|
if (WAIT_OBJECT_0 + 0 <= ret && ret <= WAIT_OBJECT_0 + w->num - 1) {
|
|
|
|
if (w->func[ret - WAIT_OBJECT_0])
|
|
|
|
w->func[ret - WAIT_OBJECT_0](w->opaque[ret - WAIT_OBJECT_0]);
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2007-09-17 05:08:06 +08:00
|
|
|
/* Check for additional signaled events */
|
2007-04-19 01:56:02 +08:00
|
|
|
for(i = (ret - WAIT_OBJECT_0 + 1); i < w->num; i++) {
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2007-04-19 01:56:02 +08:00
|
|
|
/* Check if event is signaled */
|
|
|
|
ret2 = WaitForSingleObject(w->events[i], 0);
|
|
|
|
if(ret2 == WAIT_OBJECT_0) {
|
|
|
|
if (w->func[i])
|
|
|
|
w->func[i](w->opaque[i]);
|
|
|
|
} else if (ret2 == WAIT_TIMEOUT) {
|
|
|
|
} else {
|
|
|
|
err = GetLastError();
|
|
|
|
fprintf(stderr, "WaitForSingleObject error %d %d\n", i, err);
|
2007-09-17 16:09:54 +08:00
|
|
|
}
|
|
|
|
}
|
2006-06-26 01:18:27 +08:00
|
|
|
} else if (ret == WAIT_TIMEOUT) {
|
|
|
|
} else {
|
|
|
|
err = GetLastError();
|
2007-04-19 01:56:02 +08:00
|
|
|
fprintf(stderr, "WaitForMultipleObjects error %d %d\n", ret, err);
|
2006-06-26 01:18:27 +08:00
|
|
|
}
|
2006-04-13 04:21:17 +08:00
|
|
|
}
|
2008-11-01 02:07:17 +08:00
|
|
|
|
|
|
|
*timeout = 0;
|
|
|
|
}
|
|
|
|
#else
|
2008-12-08 03:30:18 +08:00
|
|
|
static void host_main_loop_wait(int *timeout)
|
2008-11-01 02:07:17 +08:00
|
|
|
{
|
|
|
|
}
|
2006-02-02 05:29:26 +08:00
|
|
|
#endif
|
2008-11-01 02:07:17 +08:00
|
|
|
|
|
|
|
void main_loop_wait(int timeout)
|
|
|
|
{
|
|
|
|
IOHandlerRecord *ioh;
|
|
|
|
fd_set rfds, wfds, xfds;
|
|
|
|
int ret, nfds;
|
|
|
|
struct timeval tv;
|
|
|
|
|
|
|
|
qemu_bh_update_timeout(&timeout);
|
|
|
|
|
|
|
|
host_main_loop_wait(&timeout);
|
|
|
|
|
2006-02-02 05:29:26 +08:00
|
|
|
/* poll any events */
|
|
|
|
/* XXX: separate device handlers from system ones */
|
2008-11-06 04:49:37 +08:00
|
|
|
nfds = -1;
|
2006-02-02 05:29:26 +08:00
|
|
|
FD_ZERO(&rfds);
|
|
|
|
FD_ZERO(&wfds);
|
2006-05-01 21:33:02 +08:00
|
|
|
FD_ZERO(&xfds);
|
2006-02-02 05:29:26 +08:00
|
|
|
for(ioh = first_io_handler; ioh != NULL; ioh = ioh->next) {
|
2007-03-01 05:59:44 +08:00
|
|
|
if (ioh->deleted)
|
|
|
|
continue;
|
2006-02-02 05:29:26 +08:00
|
|
|
if (ioh->fd_read &&
|
|
|
|
(!ioh->fd_read_poll ||
|
|
|
|
ioh->fd_read_poll(ioh->opaque) != 0)) {
|
|
|
|
FD_SET(ioh->fd, &rfds);
|
|
|
|
if (ioh->fd > nfds)
|
|
|
|
nfds = ioh->fd;
|
|
|
|
}
|
|
|
|
if (ioh->fd_write) {
|
|
|
|
FD_SET(ioh->fd, &wfds);
|
|
|
|
if (ioh->fd > nfds)
|
|
|
|
nfds = ioh->fd;
|
|
|
|
}
|
|
|
|
}
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2008-11-01 02:07:17 +08:00
|
|
|
tv.tv_sec = timeout / 1000;
|
|
|
|
tv.tv_usec = (timeout % 1000) * 1000;
|
|
|
|
|
2009-06-24 20:42:30 +08:00
|
|
|
slirp_select_fill(&nfds, &rfds, &wfds, &xfds);
|
|
|
|
|
2009-04-25 02:03:49 +08:00
|
|
|
qemu_mutex_unlock_iothread();
|
2006-05-01 21:33:02 +08:00
|
|
|
ret = select(nfds + 1, &rfds, &wfds, &xfds, &tv);
|
2009-04-25 02:03:49 +08:00
|
|
|
qemu_mutex_lock_iothread();
|
2006-02-02 05:29:26 +08:00
|
|
|
if (ret > 0) {
|
2007-03-01 05:59:44 +08:00
|
|
|
IOHandlerRecord **pioh;
|
|
|
|
|
|
|
|
for(ioh = first_io_handler; ioh != NULL; ioh = ioh->next) {
|
2007-08-25 09:34:19 +08:00
|
|
|
if (!ioh->deleted && ioh->fd_read && FD_ISSET(ioh->fd, &rfds)) {
|
2006-02-02 05:29:26 +08:00
|
|
|
ioh->fd_read(ioh->opaque);
|
2005-11-16 06:16:05 +08:00
|
|
|
}
|
2007-08-25 09:34:19 +08:00
|
|
|
if (!ioh->deleted && ioh->fd_write && FD_ISSET(ioh->fd, &wfds)) {
|
2006-02-02 05:29:26 +08:00
|
|
|
ioh->fd_write(ioh->opaque);
|
2004-03-15 05:44:30 +08:00
|
|
|
}
|
2003-06-28 01:34:32 +08:00
|
|
|
}
|
2007-03-01 05:59:44 +08:00
|
|
|
|
|
|
|
/* remove deleted IO handlers */
|
|
|
|
pioh = &first_io_handler;
|
|
|
|
while (*pioh) {
|
|
|
|
ioh = *pioh;
|
|
|
|
if (ioh->deleted) {
|
|
|
|
*pioh = ioh->next;
|
|
|
|
qemu_free(ioh);
|
2007-09-17 05:08:06 +08:00
|
|
|
} else
|
2007-03-01 05:59:44 +08:00
|
|
|
pioh = &ioh->next;
|
|
|
|
}
|
2006-02-02 05:29:26 +08:00
|
|
|
}
|
2009-06-24 20:42:30 +08:00
|
|
|
|
|
|
|
slirp_select_poll(&rfds, &wfds, &xfds, (ret < 0));
|
2003-06-28 01:34:32 +08:00
|
|
|
|
2010-03-10 18:38:45 +08:00
|
|
|
qemu_run_all_timers();
|
2009-09-15 19:36:04 +08:00
|
|
|
|
2007-05-23 08:06:54 +08:00
|
|
|
/* Check bottom-halves last in case any of the earlier events triggered
|
|
|
|
them. */
|
|
|
|
qemu_bh_poll();
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2004-08-02 05:53:26 +08:00
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:33 +08:00
|
|
|
static int qemu_cpu_exec(CPUState *env)
|
2004-08-02 05:53:26 +08:00
|
|
|
{
|
2009-04-25 02:03:33 +08:00
|
|
|
int ret;
|
2006-02-09 06:46:31 +08:00
|
|
|
#ifdef CONFIG_PROFILER
|
|
|
|
int64_t ti;
|
|
|
|
#endif
|
2004-08-02 05:53:26 +08:00
|
|
|
|
2006-02-09 06:46:31 +08:00
|
|
|
#ifdef CONFIG_PROFILER
|
2009-04-25 02:03:33 +08:00
|
|
|
ti = profile_getclock();
|
2006-02-09 06:46:31 +08:00
|
|
|
#endif
|
2009-04-25 02:03:33 +08:00
|
|
|
if (use_icount) {
|
|
|
|
int64_t count;
|
|
|
|
int decr;
|
|
|
|
qemu_icount -= (env->icount_decr.u16.low + env->icount_extra);
|
|
|
|
env->icount_decr.u16.low = 0;
|
|
|
|
env->icount_extra = 0;
|
2010-03-10 18:38:49 +08:00
|
|
|
count = qemu_icount_round (qemu_next_deadline());
|
2009-04-25 02:03:33 +08:00
|
|
|
qemu_icount += count;
|
|
|
|
decr = (count > 0xffff) ? 0xffff : count;
|
|
|
|
count -= decr;
|
|
|
|
env->icount_decr.u16.low = decr;
|
|
|
|
env->icount_extra = count;
|
|
|
|
}
|
|
|
|
ret = cpu_exec(env);
|
2006-02-09 06:46:31 +08:00
|
|
|
#ifdef CONFIG_PROFILER
|
2009-04-25 02:03:33 +08:00
|
|
|
qemu_time += profile_getclock() - ti;
|
2006-02-09 06:46:31 +08:00
|
|
|
#endif
|
2009-04-25 02:03:33 +08:00
|
|
|
if (use_icount) {
|
|
|
|
/* Fold pending instructions back into the
|
|
|
|
instruction counter, and clear the interrupt flag. */
|
|
|
|
qemu_icount -= (env->icount_decr.u16.low
|
|
|
|
+ env->icount_extra);
|
|
|
|
env->icount_decr.u32 = 0;
|
|
|
|
env->icount_extra = 0;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:57 +08:00
|
|
|
static void tcg_cpu_exec(void)
|
|
|
|
{
|
2009-04-25 02:04:07 +08:00
|
|
|
int ret = 0;
|
2009-04-25 02:03:57 +08:00
|
|
|
|
|
|
|
if (next_cpu == NULL)
|
|
|
|
next_cpu = first_cpu;
|
|
|
|
for (; next_cpu != NULL; next_cpu = next_cpu->next_cpu) {
|
|
|
|
CPUState *env = cur_cpu = next_cpu;
|
|
|
|
|
2010-03-10 18:38:47 +08:00
|
|
|
qemu_clock_enable(vm_clock,
|
|
|
|
(cur_cpu->singlestep_enabled & SSTEP_NOTIMER) == 0);
|
|
|
|
|
2010-03-10 18:38:50 +08:00
|
|
|
if (qemu_alarm_pending())
|
2009-04-25 02:03:57 +08:00
|
|
|
break;
|
2009-04-25 02:04:07 +08:00
|
|
|
if (cpu_can_run(env))
|
|
|
|
ret = qemu_cpu_exec(env);
|
2010-02-09 22:49:04 +08:00
|
|
|
else if (env->stop)
|
|
|
|
break;
|
|
|
|
|
2009-04-25 02:03:57 +08:00
|
|
|
if (ret == EXCP_DEBUG) {
|
|
|
|
gdb_set_stop_cpu(env);
|
|
|
|
debug_requested = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:33 +08:00
|
|
|
static int cpu_has_work(CPUState *env)
|
|
|
|
{
|
2009-04-25 02:04:07 +08:00
|
|
|
if (env->stop)
|
|
|
|
return 1;
|
|
|
|
if (env->stopped)
|
|
|
|
return 0;
|
2009-04-25 02:03:33 +08:00
|
|
|
if (!env->halted)
|
|
|
|
return 1;
|
|
|
|
if (qemu_cpu_has_work(env))
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int tcg_has_work(void)
|
|
|
|
{
|
|
|
|
CPUState *env;
|
|
|
|
|
|
|
|
for (env = first_cpu; env != NULL; env = env->next_cpu)
|
|
|
|
if (cpu_has_work(env))
|
|
|
|
return 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qemu_calculate_timeout(void)
|
|
|
|
{
|
2009-06-10 05:24:57 +08:00
|
|
|
#ifndef CONFIG_IOTHREAD
|
2009-04-25 02:03:33 +08:00
|
|
|
int timeout;
|
|
|
|
|
|
|
|
if (!vm_running)
|
|
|
|
timeout = 5000;
|
|
|
|
else if (tcg_has_work())
|
|
|
|
timeout = 0;
|
|
|
|
else {
|
|
|
|
/* XXX: use timeout computed from timers */
|
|
|
|
int64_t add;
|
|
|
|
int64_t delta;
|
|
|
|
/* Advance virtual time to the next event. */
|
2010-03-10 18:38:51 +08:00
|
|
|
delta = qemu_icount_delta();
|
2009-04-25 02:03:33 +08:00
|
|
|
if (delta > 0) {
|
|
|
|
/* If virtual time is ahead of real time then just
|
|
|
|
wait for IO. */
|
2010-03-10 18:38:51 +08:00
|
|
|
timeout = (delta + 999999) / 1000000;
|
2009-04-25 02:03:33 +08:00
|
|
|
} else {
|
|
|
|
/* Wait for either IO to occur or the next
|
|
|
|
timer event. */
|
|
|
|
add = qemu_next_deadline();
|
|
|
|
/* We advance the timer before checking for IO.
|
|
|
|
Limit the amount we advance so that early IO
|
|
|
|
activity won't get the guest too far ahead. */
|
|
|
|
if (add > 10000000)
|
|
|
|
add = 10000000;
|
|
|
|
delta += add;
|
2010-03-10 18:38:49 +08:00
|
|
|
qemu_icount += qemu_icount_round (add);
|
2009-04-25 02:03:33 +08:00
|
|
|
timeout = delta / 1000000;
|
|
|
|
if (timeout < 0)
|
|
|
|
timeout = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return timeout;
|
2009-06-10 05:24:57 +08:00
|
|
|
#else /* CONFIG_IOTHREAD */
|
|
|
|
return 1000;
|
|
|
|
#endif
|
2009-04-25 02:03:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vm_can_run(void)
|
|
|
|
{
|
|
|
|
if (powerdown_requested)
|
|
|
|
return 0;
|
|
|
|
if (reset_requested)
|
|
|
|
return 0;
|
|
|
|
if (shutdown_requested)
|
|
|
|
return 0;
|
2009-04-25 02:03:54 +08:00
|
|
|
if (debug_requested)
|
|
|
|
return 0;
|
2009-04-25 02:03:33 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2009-08-09 16:42:19 +08:00
|
|
|
qemu_irq qemu_system_powerdown;
|
|
|
|
|
2009-04-25 02:03:33 +08:00
|
|
|
static void main_loop(void)
|
|
|
|
{
|
2009-04-25 02:04:02 +08:00
|
|
|
int r;
|
2009-04-25 02:03:57 +08:00
|
|
|
|
2009-04-25 02:04:07 +08:00
|
|
|
#ifdef CONFIG_IOTHREAD
|
|
|
|
qemu_system_ready = 1;
|
|
|
|
qemu_cond_broadcast(&qemu_system_cond);
|
|
|
|
#endif
|
|
|
|
|
2009-04-25 02:04:02 +08:00
|
|
|
for (;;) {
|
2009-04-25 02:03:33 +08:00
|
|
|
do {
|
2009-04-25 02:03:57 +08:00
|
|
|
#ifdef CONFIG_PROFILER
|
|
|
|
int64_t ti;
|
|
|
|
#endif
|
2009-04-25 02:04:07 +08:00
|
|
|
#ifndef CONFIG_IOTHREAD
|
2009-04-25 02:03:57 +08:00
|
|
|
tcg_cpu_exec();
|
2009-04-25 02:04:07 +08:00
|
|
|
#endif
|
2006-02-09 06:46:31 +08:00
|
|
|
#ifdef CONFIG_PROFILER
|
2009-04-25 02:03:33 +08:00
|
|
|
ti = profile_getclock();
|
2006-02-09 06:46:31 +08:00
|
|
|
#endif
|
2009-04-25 02:03:33 +08:00
|
|
|
main_loop_wait(qemu_calculate_timeout());
|
2006-02-09 06:46:31 +08:00
|
|
|
#ifdef CONFIG_PROFILER
|
2009-04-25 02:03:33 +08:00
|
|
|
dev_time += profile_getclock() - ti;
|
2006-02-09 06:46:31 +08:00
|
|
|
#endif
|
2009-04-25 02:03:54 +08:00
|
|
|
} while (vm_can_run());
|
2009-04-25 02:03:33 +08:00
|
|
|
|
2009-11-27 08:59:04 +08:00
|
|
|
if (qemu_debug_requested()) {
|
2009-04-25 02:03:54 +08:00
|
|
|
vm_stop(EXCP_DEBUG);
|
2009-11-27 08:59:04 +08:00
|
|
|
}
|
2009-04-25 02:03:33 +08:00
|
|
|
if (qemu_shutdown_requested()) {
|
2009-12-05 02:05:45 +08:00
|
|
|
monitor_protocol_event(QEVENT_SHUTDOWN, NULL);
|
2009-04-25 02:03:33 +08:00
|
|
|
if (no_shutdown) {
|
|
|
|
vm_stop(0);
|
|
|
|
no_shutdown = 0;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
}
|
2009-04-25 02:04:07 +08:00
|
|
|
if (qemu_reset_requested()) {
|
|
|
|
pause_all_vcpus();
|
2009-04-25 02:03:33 +08:00
|
|
|
qemu_system_reset();
|
2009-04-25 02:04:07 +08:00
|
|
|
resume_all_vcpus();
|
|
|
|
}
|
2009-08-09 16:42:19 +08:00
|
|
|
if (qemu_powerdown_requested()) {
|
2009-12-05 02:05:45 +08:00
|
|
|
monitor_protocol_event(QEVENT_POWERDOWN, NULL);
|
2009-08-09 16:42:19 +08:00
|
|
|
qemu_irq_raise(qemu_system_powerdown);
|
|
|
|
}
|
2009-11-27 08:59:04 +08:00
|
|
|
if ((r = qemu_vmstop_requested())) {
|
2009-04-25 02:04:02 +08:00
|
|
|
vm_stop(r);
|
2009-11-27 08:59:04 +08:00
|
|
|
}
|
2003-06-28 01:34:32 +08:00
|
|
|
}
|
2009-04-25 02:04:07 +08:00
|
|
|
pause_all_vcpus();
|
2003-06-28 01:34:32 +08:00
|
|
|
}
|
|
|
|
|
2009-04-08 06:58:45 +08:00
|
|
|
static void version(void)
|
|
|
|
{
|
2009-04-08 07:17:49 +08:00
|
|
|
printf("QEMU PC emulator version " QEMU_VERSION QEMU_PKGVERSION ", Copyright (c) 2003-2008 Fabrice Bellard\n");
|
2009-04-08 06:58:45 +08:00
|
|
|
}
|
|
|
|
|
2007-06-30 07:26:08 +08:00
|
|
|
static void help(int exitcode)
|
2003-06-24 21:42:40 +08:00
|
|
|
{
|
2010-02-04 23:49:59 +08:00
|
|
|
const char *options_help =
|
2009-03-28 14:44:27 +08:00
|
|
|
#define DEF(option, opt_arg, opt_enum, opt_help) \
|
|
|
|
opt_help
|
|
|
|
#define DEFHEADING(text) stringify(text) "\n"
|
|
|
|
#include "qemu-options.h"
|
|
|
|
#undef DEF
|
|
|
|
#undef DEFHEADING
|
|
|
|
#undef GEN_DOCS
|
2010-02-04 23:49:59 +08:00
|
|
|
;
|
|
|
|
version();
|
|
|
|
printf("usage: %s [options] [disk_image]\n"
|
|
|
|
"\n"
|
|
|
|
"'disk_image' is a raw hard image image for IDE hard disk 0\n"
|
2010-02-08 17:04:56 +08:00
|
|
|
"\n"
|
2010-02-04 23:49:59 +08:00
|
|
|
"%s\n"
|
2010-02-08 17:04:56 +08:00
|
|
|
"During emulation, the following keys are useful:\n"
|
|
|
|
"ctrl-alt-f toggle full screen\n"
|
|
|
|
"ctrl-alt-n switch to virtual console 'n'\n"
|
|
|
|
"ctrl-alt toggle mouse and keyboard grab\n"
|
|
|
|
"\n"
|
2010-02-04 23:49:59 +08:00
|
|
|
"When using -nographic, press 'ctrl-a h' to get some help.\n",
|
|
|
|
"qemu",
|
|
|
|
options_help);
|
2007-06-30 07:26:08 +08:00
|
|
|
exit(exitcode);
|
2003-06-24 21:42:40 +08:00
|
|
|
}
|
|
|
|
|
2004-05-14 06:02:20 +08:00
|
|
|
#define HAS_ARG 0x0001
|
|
|
|
|
|
|
|
enum {
|
2009-03-28 14:44:27 +08:00
|
|
|
#define DEF(option, opt_arg, opt_enum, opt_help) \
|
|
|
|
opt_enum,
|
|
|
|
#define DEFHEADING(text)
|
|
|
|
#include "qemu-options.h"
|
|
|
|
#undef DEF
|
|
|
|
#undef DEFHEADING
|
|
|
|
#undef GEN_DOCS
|
2004-05-14 06:02:20 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct QEMUOption {
|
|
|
|
const char *name;
|
|
|
|
int flags;
|
|
|
|
int index;
|
|
|
|
} QEMUOption;
|
|
|
|
|
2008-10-02 03:38:09 +08:00
|
|
|
static const QEMUOption qemu_options[] = {
|
2004-05-14 06:02:20 +08:00
|
|
|
{ "h", 0, QEMU_OPTION_h },
|
2009-03-28 14:44:27 +08:00
|
|
|
#define DEF(option, opt_arg, opt_enum, opt_help) \
|
|
|
|
{ option, opt_arg, opt_enum },
|
|
|
|
#define DEFHEADING(text)
|
|
|
|
#include "qemu-options.h"
|
|
|
|
#undef DEF
|
|
|
|
#undef DEFHEADING
|
|
|
|
#undef GEN_DOCS
|
2004-05-14 06:02:20 +08:00
|
|
|
{ NULL },
|
2003-06-30 18:03:06 +08:00
|
|
|
};
|
|
|
|
|
2005-10-31 02:58:22 +08:00
|
|
|
#ifdef HAS_AUDIO
|
2005-12-19 04:34:32 +08:00
|
|
|
struct soundhw soundhw[] = {
|
2007-04-30 10:22:06 +08:00
|
|
|
#ifdef HAS_AUDIO_CHOICE
|
2008-04-08 03:47:14 +08:00
|
|
|
#if defined(TARGET_I386) || defined(TARGET_MIPS)
|
2006-04-25 05:58:30 +08:00
|
|
|
{
|
|
|
|
"pcspk",
|
|
|
|
"PC speaker",
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
{ .init_isa = pcspk_audio_init }
|
|
|
|
},
|
|
|
|
#endif
|
2009-01-09 18:46:34 +08:00
|
|
|
|
|
|
|
#ifdef CONFIG_SB16
|
2005-12-19 04:34:32 +08:00
|
|
|
{
|
|
|
|
"sb16",
|
|
|
|
"Creative Sound Blaster 16",
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
{ .init_isa = SB16_init }
|
|
|
|
},
|
2009-01-09 18:46:34 +08:00
|
|
|
#endif
|
2005-12-19 04:34:32 +08:00
|
|
|
|
2008-06-13 18:48:22 +08:00
|
|
|
#ifdef CONFIG_CS4231A
|
|
|
|
{
|
|
|
|
"cs4231a",
|
|
|
|
"CS4231A",
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
{ .init_isa = cs4231a_init }
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
2005-10-31 02:58:22 +08:00
|
|
|
#ifdef CONFIG_ADLIB
|
2005-12-19 04:34:32 +08:00
|
|
|
{
|
|
|
|
"adlib",
|
2005-10-31 02:58:22 +08:00
|
|
|
#ifdef HAS_YMF262
|
2005-12-19 04:34:32 +08:00
|
|
|
"Yamaha YMF262 (OPL3)",
|
2005-10-31 02:58:22 +08:00
|
|
|
#else
|
2005-12-19 04:34:32 +08:00
|
|
|
"Yamaha YM3812 (OPL2)",
|
2005-10-31 02:58:22 +08:00
|
|
|
#endif
|
2005-12-19 04:34:32 +08:00
|
|
|
0,
|
|
|
|
1,
|
|
|
|
{ .init_isa = Adlib_init }
|
|
|
|
},
|
2005-10-31 02:58:22 +08:00
|
|
|
#endif
|
2005-12-19 04:34:32 +08:00
|
|
|
|
2005-10-31 02:58:22 +08:00
|
|
|
#ifdef CONFIG_GUS
|
2005-12-19 04:34:32 +08:00
|
|
|
{
|
|
|
|
"gus",
|
|
|
|
"Gravis Ultrasound GF1",
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
{ .init_isa = GUS_init }
|
|
|
|
},
|
2005-10-31 02:58:22 +08:00
|
|
|
#endif
|
2005-12-19 04:34:32 +08:00
|
|
|
|
2009-01-09 18:46:34 +08:00
|
|
|
#ifdef CONFIG_AC97
|
2008-01-14 12:27:55 +08:00
|
|
|
{
|
|
|
|
"ac97",
|
|
|
|
"Intel 82801AA AC97 Audio",
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
{ .init_pci = ac97_init }
|
|
|
|
},
|
2009-01-09 18:46:34 +08:00
|
|
|
#endif
|
2008-01-14 12:27:55 +08:00
|
|
|
|
2009-01-09 18:46:34 +08:00
|
|
|
#ifdef CONFIG_ES1370
|
2005-12-19 04:34:32 +08:00
|
|
|
{
|
|
|
|
"es1370",
|
|
|
|
"ENSONIQ AudioPCI ES1370",
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
{ .init_pci = es1370_init }
|
|
|
|
},
|
2007-04-30 10:22:06 +08:00
|
|
|
#endif
|
2005-12-19 04:34:32 +08:00
|
|
|
|
2009-01-09 18:46:34 +08:00
|
|
|
#endif /* HAS_AUDIO_CHOICE */
|
|
|
|
|
2005-12-19 04:34:32 +08:00
|
|
|
{ NULL, NULL, 0, 0, { NULL } }
|
|
|
|
};
|
|
|
|
|
|
|
|
static void select_soundhw (const char *optarg)
|
|
|
|
{
|
|
|
|
struct soundhw *c;
|
|
|
|
|
|
|
|
if (*optarg == '?') {
|
|
|
|
show_valid_cards:
|
|
|
|
|
|
|
|
printf ("Valid sound card names (comma separated):\n");
|
|
|
|
for (c = soundhw; c->name; ++c) {
|
|
|
|
printf ("%-11s %s\n", c->name, c->descr);
|
|
|
|
}
|
|
|
|
printf ("\n-soundhw all will enable all of the above\n");
|
2005-10-31 02:58:22 +08:00
|
|
|
exit (*optarg != '?');
|
|
|
|
}
|
|
|
|
else {
|
2005-12-19 04:34:32 +08:00
|
|
|
size_t l;
|
2005-10-31 02:58:22 +08:00
|
|
|
const char *p;
|
|
|
|
char *e;
|
|
|
|
int bad_card = 0;
|
|
|
|
|
2005-12-19 04:34:32 +08:00
|
|
|
if (!strcmp (optarg, "all")) {
|
|
|
|
for (c = soundhw; c->name; ++c) {
|
|
|
|
c->enabled = 1;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2005-10-31 02:58:22 +08:00
|
|
|
|
2005-12-19 04:34:32 +08:00
|
|
|
p = optarg;
|
2005-10-31 02:58:22 +08:00
|
|
|
while (*p) {
|
|
|
|
e = strchr (p, ',');
|
|
|
|
l = !e ? strlen (p) : (size_t) (e - p);
|
2005-12-19 04:34:32 +08:00
|
|
|
|
|
|
|
for (c = soundhw; c->name; ++c) {
|
2009-09-06 10:49:03 +08:00
|
|
|
if (!strncmp (c->name, p, l) && !c->name[l]) {
|
2005-12-19 04:34:32 +08:00
|
|
|
c->enabled = 1;
|
2005-10-31 02:58:22 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-12-19 04:34:32 +08:00
|
|
|
|
|
|
|
if (!c->name) {
|
2005-10-31 02:58:22 +08:00
|
|
|
if (l > 80) {
|
|
|
|
fprintf (stderr,
|
|
|
|
"Unknown sound card name (too big to show)\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
fprintf (stderr, "Unknown sound card name `%.*s'\n",
|
|
|
|
(int) l, p);
|
|
|
|
}
|
|
|
|
bad_card = 1;
|
|
|
|
}
|
|
|
|
p += l + (e != NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bad_card)
|
|
|
|
goto show_valid_cards;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-09-28 08:42:05 +08:00
|
|
|
static void select_vgahw (const char *p)
|
|
|
|
{
|
|
|
|
const char *opts;
|
|
|
|
|
2009-12-08 20:11:45 +08:00
|
|
|
default_vga = 0;
|
2009-07-30 18:15:02 +08:00
|
|
|
vga_interface_type = VGA_NONE;
|
2008-09-28 08:42:05 +08:00
|
|
|
if (strstart(p, "std", &opts)) {
|
2009-07-30 18:15:02 +08:00
|
|
|
vga_interface_type = VGA_STD;
|
2008-09-28 08:42:05 +08:00
|
|
|
} else if (strstart(p, "cirrus", &opts)) {
|
2009-07-30 18:15:02 +08:00
|
|
|
vga_interface_type = VGA_CIRRUS;
|
2008-09-28 08:42:05 +08:00
|
|
|
} else if (strstart(p, "vmware", &opts)) {
|
2009-07-30 18:15:02 +08:00
|
|
|
vga_interface_type = VGA_VMWARE;
|
2009-04-22 23:19:53 +08:00
|
|
|
} else if (strstart(p, "xenfb", &opts)) {
|
2009-07-30 18:15:02 +08:00
|
|
|
vga_interface_type = VGA_XENFB;
|
2009-04-22 23:19:48 +08:00
|
|
|
} else if (!strstart(p, "none", &opts)) {
|
2008-09-28 08:42:05 +08:00
|
|
|
invalid_vga:
|
|
|
|
fprintf(stderr, "Unknown vga type: %s\n", p);
|
|
|
|
exit(1);
|
|
|
|
}
|
2008-09-28 08:42:12 +08:00
|
|
|
while (*opts) {
|
|
|
|
const char *nextopt;
|
|
|
|
|
|
|
|
if (strstart(opts, ",retrace=", &nextopt)) {
|
|
|
|
opts = nextopt;
|
|
|
|
if (strstart(opts, "dumb", &nextopt))
|
|
|
|
vga_retrace_method = VGA_RETRACE_DUMB;
|
|
|
|
else if (strstart(opts, "precise", &nextopt))
|
|
|
|
vga_retrace_method = VGA_RETRACE_PRECISE;
|
|
|
|
else goto invalid_vga;
|
|
|
|
} else goto invalid_vga;
|
|
|
|
opts = nextopt;
|
|
|
|
}
|
2008-09-28 08:42:05 +08:00
|
|
|
}
|
|
|
|
|
2009-06-27 01:15:14 +08:00
|
|
|
#ifdef TARGET_I386
|
|
|
|
static int balloon_parse(const char *arg)
|
|
|
|
{
|
2009-08-14 16:34:22 +08:00
|
|
|
QemuOpts *opts;
|
2009-06-27 01:15:14 +08:00
|
|
|
|
2009-08-14 16:34:22 +08:00
|
|
|
if (strcmp(arg, "none") == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(arg, "virtio", 6)) {
|
|
|
|
if (arg[6] == ',') {
|
|
|
|
/* have params -> parse them */
|
2010-02-11 02:52:18 +08:00
|
|
|
opts = qemu_opts_parse(&qemu_device_opts, arg+7, 0);
|
2009-08-14 16:34:22 +08:00
|
|
|
if (!opts)
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
/* create empty opts */
|
|
|
|
opts = qemu_opts_create(&qemu_device_opts, NULL, 0);
|
2009-06-27 01:15:14 +08:00
|
|
|
}
|
2009-08-14 16:34:22 +08:00
|
|
|
qemu_opt_set(opts, "driver", "virtio-balloon-pci");
|
|
|
|
return 0;
|
2009-06-27 01:15:14 +08:00
|
|
|
}
|
2009-08-14 16:34:22 +08:00
|
|
|
|
|
|
|
return -1;
|
2009-06-27 01:15:14 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-06-27 04:03:44 +08:00
|
|
|
#ifdef _WIN32
|
|
|
|
static BOOL WINAPI qemu_ctrl_handler(DWORD type)
|
|
|
|
{
|
|
|
|
exit(STATUS_CONTROL_C_EXIT);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-04-18 02:58:14 +08:00
|
|
|
int qemu_uuid_parse(const char *str, uint8_t *uuid)
|
2008-09-19 02:29:08 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if(strlen(str) != 36)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = sscanf(str, UUID_FMT, &uuid[0], &uuid[1], &uuid[2], &uuid[3],
|
|
|
|
&uuid[4], &uuid[5], &uuid[6], &uuid[7], &uuid[8], &uuid[9],
|
|
|
|
&uuid[10], &uuid[11], &uuid[12], &uuid[13], &uuid[14], &uuid[15]);
|
|
|
|
|
|
|
|
if(ret != 16)
|
|
|
|
return -1;
|
|
|
|
|
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-18 02:59:56 +08:00
|
|
|
#ifdef TARGET_I386
|
|
|
|
smbios_add_field(1, offsetof(struct smbios_type_1, uuid), 16, uuid);
|
|
|
|
#endif
|
|
|
|
|
2008-09-19 02:29:08 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-08-22 04:08:03 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
|
|
|
|
static void termsig_handler(int signal)
|
|
|
|
{
|
|
|
|
qemu_system_shutdown_request();
|
|
|
|
}
|
|
|
|
|
2009-05-08 18:34:17 +08:00
|
|
|
static void sigchld_handler(int signal)
|
|
|
|
{
|
|
|
|
waitpid(-1, NULL, WNOHANG);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sighandler_setup(void)
|
2008-08-22 04:08:03 +08:00
|
|
|
{
|
|
|
|
struct sigaction act;
|
|
|
|
|
|
|
|
memset(&act, 0, sizeof(act));
|
|
|
|
act.sa_handler = termsig_handler;
|
|
|
|
sigaction(SIGINT, &act, NULL);
|
|
|
|
sigaction(SIGHUP, &act, NULL);
|
|
|
|
sigaction(SIGTERM, &act, NULL);
|
2009-05-08 18:34:17 +08:00
|
|
|
|
|
|
|
act.sa_handler = sigchld_handler;
|
|
|
|
act.sa_flags = SA_NOCLDSTOP;
|
|
|
|
sigaction(SIGCHLD, &act, NULL);
|
2008-08-22 04:08:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2009-05-30 07:52:44 +08:00
|
|
|
#ifdef _WIN32
|
|
|
|
/* Look for support files in the same directory as the executable. */
|
|
|
|
static char *find_datadir(const char *argv0)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
char buf[MAX_PATH];
|
|
|
|
DWORD len;
|
|
|
|
|
|
|
|
len = GetModuleFileName(NULL, buf, sizeof(buf) - 1);
|
|
|
|
if (len == 0) {
|
2009-06-10 01:51:21 +08:00
|
|
|
return NULL;
|
2009-05-30 07:52:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
buf[len] = 0;
|
|
|
|
p = buf + len - 1;
|
|
|
|
while (p != buf && *p != '\\')
|
|
|
|
p--;
|
|
|
|
*p = 0;
|
|
|
|
if (access(buf, R_OK) == 0) {
|
|
|
|
return qemu_strdup(buf);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#else /* !_WIN32 */
|
|
|
|
|
|
|
|
/* Find a likely location for support files using the location of the binary.
|
|
|
|
For installed binaries this will be "$bindir/../share/qemu". When
|
|
|
|
running from the build tree this will be "$bindir/../pc-bios". */
|
|
|
|
#define SHARE_SUFFIX "/share/qemu"
|
|
|
|
#define BUILD_SUFFIX "/pc-bios"
|
|
|
|
static char *find_datadir(const char *argv0)
|
|
|
|
{
|
|
|
|
char *dir;
|
|
|
|
char *p = NULL;
|
|
|
|
char *res;
|
|
|
|
char buf[PATH_MAX];
|
2009-06-10 03:12:21 +08:00
|
|
|
size_t max_len;
|
2009-05-30 07:52:44 +08:00
|
|
|
|
|
|
|
#if defined(__linux__)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
len = readlink("/proc/self/exe", buf, sizeof(buf) - 1);
|
|
|
|
if (len > 0) {
|
|
|
|
buf[len] = 0;
|
|
|
|
p = buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
len = readlink("/proc/curproc/file", buf, sizeof(buf) - 1);
|
|
|
|
if (len > 0) {
|
|
|
|
buf[len] = 0;
|
|
|
|
p = buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* If we don't have any way of figuring out the actual executable
|
|
|
|
location then try argv[0]. */
|
|
|
|
if (!p) {
|
2009-09-03 05:59:02 +08:00
|
|
|
p = realpath(argv0, buf);
|
2009-05-30 07:52:44 +08:00
|
|
|
if (!p) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dir = dirname(p);
|
|
|
|
dir = dirname(dir);
|
|
|
|
|
2009-06-10 03:12:21 +08:00
|
|
|
max_len = strlen(dir) +
|
|
|
|
MAX(strlen(SHARE_SUFFIX), strlen(BUILD_SUFFIX)) + 1;
|
|
|
|
res = qemu_mallocz(max_len);
|
|
|
|
snprintf(res, max_len, "%s%s", dir, SHARE_SUFFIX);
|
2009-05-30 07:52:44 +08:00
|
|
|
if (access(res, R_OK)) {
|
2009-06-10 03:12:21 +08:00
|
|
|
snprintf(res, max_len, "%s%s", dir, BUILD_SUFFIX);
|
2009-05-30 07:52:44 +08:00
|
|
|
if (access(res, R_OK)) {
|
|
|
|
qemu_free(res);
|
|
|
|
res = NULL;
|
|
|
|
}
|
|
|
|
}
|
2009-09-03 05:59:02 +08:00
|
|
|
|
2009-05-30 07:52:44 +08:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
#undef SHARE_SUFFIX
|
|
|
|
#undef BUILD_SUFFIX
|
|
|
|
#endif
|
|
|
|
|
|
|
|
char *qemu_find_file(int type, const char *name)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
const char *subdir;
|
|
|
|
char *buf;
|
|
|
|
|
|
|
|
/* If name contains path separators then try it as a straight path. */
|
|
|
|
if ((strchr(name, '/') || strchr(name, '\\'))
|
|
|
|
&& access(name, R_OK) == 0) {
|
2009-09-03 05:59:06 +08:00
|
|
|
return qemu_strdup(name);
|
2009-05-30 07:52:44 +08:00
|
|
|
}
|
|
|
|
switch (type) {
|
|
|
|
case QEMU_FILE_TYPE_BIOS:
|
|
|
|
subdir = "";
|
|
|
|
break;
|
|
|
|
case QEMU_FILE_TYPE_KEYMAP:
|
|
|
|
subdir = "keymaps/";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
len = strlen(data_dir) + strlen(name) + strlen(subdir) + 2;
|
|
|
|
buf = qemu_mallocz(len);
|
2009-06-10 03:12:21 +08:00
|
|
|
snprintf(buf, len, "%s/%s%s", data_dir, subdir, name);
|
2009-05-30 07:52:44 +08:00
|
|
|
if (access(buf, R_OK)) {
|
|
|
|
qemu_free(buf);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2010-01-30 02:48:57 +08:00
|
|
|
static int device_help_func(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
return qdev_device_help(opts);
|
|
|
|
}
|
|
|
|
|
2009-07-31 18:25:37 +08:00
|
|
|
static int device_init_func(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
DeviceState *dev;
|
|
|
|
|
|
|
|
dev = qdev_device_add(opts);
|
|
|
|
if (!dev)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:36 +08:00
|
|
|
static int chardev_init_func(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
CharDriverState *chr;
|
|
|
|
|
|
|
|
chr = qemu_chr_open_opts(opts, NULL);
|
|
|
|
if (!chr)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:50 +08:00
|
|
|
static int mon_init_func(QemuOpts *opts, void *opaque)
|
|
|
|
{
|
|
|
|
CharDriverState *chr;
|
|
|
|
const char *chardev;
|
|
|
|
const char *mode;
|
|
|
|
int flags;
|
|
|
|
|
|
|
|
mode = qemu_opt_get(opts, "mode");
|
|
|
|
if (mode == NULL) {
|
|
|
|
mode = "readline";
|
|
|
|
}
|
|
|
|
if (strcmp(mode, "readline") == 0) {
|
|
|
|
flags = MONITOR_USE_READLINE;
|
|
|
|
} else if (strcmp(mode, "control") == 0) {
|
|
|
|
flags = MONITOR_USE_CONTROL;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "unknown monitor mode \"%s\"\n", mode);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemu_opt_get_bool(opts, "default", 0))
|
|
|
|
flags |= MONITOR_IS_DEFAULT;
|
|
|
|
|
|
|
|
chardev = qemu_opt_get(opts, "chardev");
|
|
|
|
chr = qemu_chr_find(chardev);
|
|
|
|
if (chr == NULL) {
|
|
|
|
fprintf(stderr, "chardev \"%s\" not found\n", chardev);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
monitor_init(chr, flags);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:52 +08:00
|
|
|
static void monitor_parse(const char *optarg, const char *mode)
|
2009-12-08 20:11:50 +08:00
|
|
|
{
|
|
|
|
static int monitor_device_index = 0;
|
|
|
|
QemuOpts *opts;
|
|
|
|
const char *p;
|
|
|
|
char label[32];
|
|
|
|
int def = 0;
|
|
|
|
|
|
|
|
if (strstart(optarg, "chardev:", &p)) {
|
|
|
|
snprintf(label, sizeof(label), "%s", p);
|
|
|
|
} else {
|
|
|
|
if (monitor_device_index) {
|
|
|
|
snprintf(label, sizeof(label), "monitor%d",
|
|
|
|
monitor_device_index);
|
|
|
|
} else {
|
|
|
|
snprintf(label, sizeof(label), "monitor");
|
|
|
|
def = 1;
|
|
|
|
}
|
|
|
|
opts = qemu_chr_parse_compat(label, optarg);
|
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "parse error: %s\n", optarg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
opts = qemu_opts_create(&qemu_mon_opts, label, 1);
|
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "duplicate chardev: %s\n", label);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-12-08 20:11:52 +08:00
|
|
|
qemu_opt_set(opts, "mode", mode);
|
2009-12-08 20:11:50 +08:00
|
|
|
qemu_opt_set(opts, "chardev", label);
|
|
|
|
if (def)
|
|
|
|
qemu_opt_set(opts, "default", "on");
|
|
|
|
monitor_device_index++;
|
|
|
|
}
|
|
|
|
|
2009-07-15 19:59:26 +08:00
|
|
|
struct device_config {
|
|
|
|
enum {
|
2009-12-08 20:11:53 +08:00
|
|
|
DEV_USB, /* -usbdevice */
|
|
|
|
DEV_BT, /* -bt */
|
|
|
|
DEV_SERIAL, /* -serial */
|
|
|
|
DEV_PARALLEL, /* -parallel */
|
|
|
|
DEV_VIRTCON, /* -virtioconsole */
|
debugcon: support for debugging consoles (e.g. Bochs port 0xe9)
Add generic support for debugging consoles (simple I/O ports which
when written to cause debugging output to be written to a target.)
The current implementation matches Bochs' port 0xe9, allowing the same
debugging code to be used for both Bochs and Qemu.
There is no vm state associated with the debugging port, simply
because it has none -- the entire interface is a single, stateless,
write-only port.
Most of the code was cribbed from the serial port driver.
v2: removed non-ISA variants (they can be introduced when/if someone
wants them, using code from the serial port); added configurable
readback (Bochs returns 0xe9 on a read from this register, mimic that
by default) This retains the apparently somewhat controversial user
friendly option, however.
v3: reimplemented the user friendly option as a synthetic option
("-debugcon foo" basically ends up being a parser-level shorthand for
"-chardev stdio,id=debugcon -device isa-debugcon,chardev=debugcon") --
this dramatically reduced the complexity while keeping the same level
of user friendliness.
v4: spaces, not tabs.
v5: update to match current top of tree. Calling qemu_chr_open()
already during parsing no longer works; defer until we are parsing the
other console-like devices.
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-12-30 05:51:36 +08:00
|
|
|
DEV_DEBUGCON, /* -debugcon */
|
2009-07-15 19:59:26 +08:00
|
|
|
} type;
|
|
|
|
const char *cmdline;
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_ENTRY(device_config) next;
|
2009-07-15 19:59:26 +08:00
|
|
|
};
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_HEAD(, device_config) device_configs = QTAILQ_HEAD_INITIALIZER(device_configs);
|
2009-07-15 19:59:26 +08:00
|
|
|
|
|
|
|
static void add_device_config(int type, const char *cmdline)
|
|
|
|
{
|
|
|
|
struct device_config *conf;
|
|
|
|
|
|
|
|
conf = qemu_mallocz(sizeof(*conf));
|
|
|
|
conf->type = type;
|
|
|
|
conf->cmdline = cmdline;
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_INSERT_TAIL(&device_configs, conf, next);
|
2009-07-15 19:59:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int foreach_device_config(int type, int (*func)(const char *cmdline))
|
|
|
|
{
|
|
|
|
struct device_config *conf;
|
|
|
|
int rc;
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QTAILQ_FOREACH(conf, &device_configs, next) {
|
2009-07-15 19:59:26 +08:00
|
|
|
if (conf->type != type)
|
|
|
|
continue;
|
|
|
|
rc = func(conf->cmdline);
|
|
|
|
if (0 != rc)
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:41 +08:00
|
|
|
static int serial_parse(const char *devname)
|
|
|
|
{
|
|
|
|
static int index = 0;
|
|
|
|
char label[32];
|
|
|
|
|
|
|
|
if (strcmp(devname, "none") == 0)
|
|
|
|
return 0;
|
|
|
|
if (index == MAX_SERIAL_PORTS) {
|
|
|
|
fprintf(stderr, "qemu: too many serial ports\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
snprintf(label, sizeof(label), "serial%d", index);
|
|
|
|
serial_hds[index] = qemu_chr_open(label, devname, NULL);
|
|
|
|
if (!serial_hds[index]) {
|
|
|
|
fprintf(stderr, "qemu: could not open serial device '%s': %s\n",
|
|
|
|
devname, strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:42 +08:00
|
|
|
static int parallel_parse(const char *devname)
|
|
|
|
{
|
|
|
|
static int index = 0;
|
|
|
|
char label[32];
|
|
|
|
|
|
|
|
if (strcmp(devname, "none") == 0)
|
|
|
|
return 0;
|
|
|
|
if (index == MAX_PARALLEL_PORTS) {
|
|
|
|
fprintf(stderr, "qemu: too many parallel ports\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
snprintf(label, sizeof(label), "parallel%d", index);
|
|
|
|
parallel_hds[index] = qemu_chr_open(label, devname, NULL);
|
|
|
|
if (!parallel_hds[index]) {
|
|
|
|
fprintf(stderr, "qemu: could not open parallel device '%s': %s\n",
|
|
|
|
devname, strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:53 +08:00
|
|
|
static int virtcon_parse(const char *devname)
|
|
|
|
{
|
|
|
|
static int index = 0;
|
|
|
|
char label[32];
|
2010-01-21 18:49:23 +08:00
|
|
|
QemuOpts *bus_opts, *dev_opts;
|
2009-12-08 20:11:53 +08:00
|
|
|
|
|
|
|
if (strcmp(devname, "none") == 0)
|
|
|
|
return 0;
|
|
|
|
if (index == MAX_VIRTIO_CONSOLES) {
|
|
|
|
fprintf(stderr, "qemu: too many virtio consoles\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-01-21 18:49:23 +08:00
|
|
|
|
|
|
|
bus_opts = qemu_opts_create(&qemu_device_opts, NULL, 0);
|
|
|
|
qemu_opt_set(bus_opts, "driver", "virtio-serial");
|
|
|
|
|
|
|
|
dev_opts = qemu_opts_create(&qemu_device_opts, NULL, 0);
|
|
|
|
qemu_opt_set(dev_opts, "driver", "virtconsole");
|
|
|
|
|
2009-12-08 20:11:53 +08:00
|
|
|
snprintf(label, sizeof(label), "virtcon%d", index);
|
|
|
|
virtcon_hds[index] = qemu_chr_open(label, devname, NULL);
|
|
|
|
if (!virtcon_hds[index]) {
|
|
|
|
fprintf(stderr, "qemu: could not open virtio console '%s': %s\n",
|
|
|
|
devname, strerror(errno));
|
|
|
|
return -1;
|
|
|
|
}
|
2010-01-21 18:49:23 +08:00
|
|
|
qemu_opt_set(dev_opts, "chardev", label);
|
|
|
|
|
2009-12-08 20:11:53 +08:00
|
|
|
index++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
debugcon: support for debugging consoles (e.g. Bochs port 0xe9)
Add generic support for debugging consoles (simple I/O ports which
when written to cause debugging output to be written to a target.)
The current implementation matches Bochs' port 0xe9, allowing the same
debugging code to be used for both Bochs and Qemu.
There is no vm state associated with the debugging port, simply
because it has none -- the entire interface is a single, stateless,
write-only port.
Most of the code was cribbed from the serial port driver.
v2: removed non-ISA variants (they can be introduced when/if someone
wants them, using code from the serial port); added configurable
readback (Bochs returns 0xe9 on a read from this register, mimic that
by default) This retains the apparently somewhat controversial user
friendly option, however.
v3: reimplemented the user friendly option as a synthetic option
("-debugcon foo" basically ends up being a parser-level shorthand for
"-chardev stdio,id=debugcon -device isa-debugcon,chardev=debugcon") --
this dramatically reduced the complexity while keeping the same level
of user friendliness.
v4: spaces, not tabs.
v5: update to match current top of tree. Calling qemu_chr_open()
already during parsing no longer works; defer until we are parsing the
other console-like devices.
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-12-30 05:51:36 +08:00
|
|
|
static int debugcon_parse(const char *devname)
|
|
|
|
{
|
|
|
|
QemuOpts *opts;
|
|
|
|
|
|
|
|
if (!qemu_chr_open("debugcon", devname, NULL)) {
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
opts = qemu_opts_create(&qemu_device_opts, "debugcon", 1);
|
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "qemu: already have a debugcon device\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
qemu_opt_set(opts, "driver", "isa-debugcon");
|
|
|
|
qemu_opt_set(opts, "chardev", "debugcon");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-01-22 23:18:06 +08:00
|
|
|
static const QEMUOption *lookup_opt(int argc, char **argv,
|
|
|
|
const char **poptarg, int *poptind)
|
|
|
|
{
|
|
|
|
const QEMUOption *popt;
|
|
|
|
int optind = *poptind;
|
|
|
|
char *r = argv[optind];
|
|
|
|
const char *optarg;
|
|
|
|
|
2010-02-19 03:13:51 +08:00
|
|
|
loc_set_cmdline(argv, optind, 1);
|
2010-01-22 23:18:06 +08:00
|
|
|
optind++;
|
|
|
|
/* Treat --foo the same as -foo. */
|
|
|
|
if (r[1] == '-')
|
|
|
|
r++;
|
|
|
|
popt = qemu_options;
|
|
|
|
for(;;) {
|
|
|
|
if (!popt->name) {
|
2010-02-19 03:13:51 +08:00
|
|
|
error_report("invalid option");
|
2010-01-22 23:18:06 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (!strcmp(popt->name, r + 1))
|
|
|
|
break;
|
|
|
|
popt++;
|
|
|
|
}
|
|
|
|
if (popt->flags & HAS_ARG) {
|
|
|
|
if (optind >= argc) {
|
2010-02-19 03:13:51 +08:00
|
|
|
error_report("requires an argument");
|
2010-01-22 23:18:06 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
optarg = argv[optind++];
|
2010-02-19 03:13:51 +08:00
|
|
|
loc_set_cmdline(argv, optind - 2, 2);
|
2010-01-22 23:18:06 +08:00
|
|
|
} else {
|
|
|
|
optarg = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*poptarg = optarg;
|
|
|
|
*poptind = optind;
|
|
|
|
|
|
|
|
return popt;
|
|
|
|
}
|
|
|
|
|
2008-12-11 03:18:40 +08:00
|
|
|
int main(int argc, char **argv, char **envp)
|
2003-06-24 21:42:40 +08:00
|
|
|
{
|
2009-04-06 02:43:41 +08:00
|
|
|
const char *gdbstub_dev = NULL;
|
2007-11-11 09:50:45 +08:00
|
|
|
uint32_t boot_devices_bitmap = 0;
|
2007-12-02 12:51:10 +08:00
|
|
|
int i;
|
2007-11-11 09:50:45 +08:00
|
|
|
int snapshot, linux_boot, net_boot;
|
2010-03-10 18:38:48 +08:00
|
|
|
const char *icount_option = NULL;
|
2003-10-30 09:11:23 +08:00
|
|
|
const char *initrd_filename;
|
2003-10-01 05:07:02 +08:00
|
|
|
const char *kernel_filename, *kernel_cmdline;
|
2009-10-31 01:42:29 +08:00
|
|
|
char boot_devices[33] = "cad"; /* default to HD->floppy->CD-ROM */
|
2009-01-17 03:04:14 +08:00
|
|
|
DisplayState *ds;
|
2009-01-16 06:14:11 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2004-11-16 09:45:27 +08:00
|
|
|
int cyls, heads, secs, translation;
|
2009-07-31 18:25:37 +08:00
|
|
|
QemuOpts *hda_opts = NULL, *opts;
|
2004-05-14 06:02:20 +08:00
|
|
|
int optind;
|
2010-01-22 23:18:06 +08:00
|
|
|
const char *optarg;
|
2004-10-03 21:29:03 +08:00
|
|
|
const char *loadvm = NULL;
|
2005-06-05 22:49:17 +08:00
|
|
|
QEMUMachine *machine;
|
2007-03-06 03:44:02 +08:00
|
|
|
const char *cpu_model;
|
2009-04-06 02:03:31 +08:00
|
|
|
#ifndef _WIN32
|
2006-12-22 10:11:31 +08:00
|
|
|
int fds[2];
|
2009-04-06 02:03:31 +08:00
|
|
|
#endif
|
2008-05-28 20:30:31 +08:00
|
|
|
int tb_size;
|
2007-03-19 23:58:31 +08:00
|
|
|
const char *pid_file = NULL;
|
2008-10-13 11:12:02 +08:00
|
|
|
const char *incoming = NULL;
|
2009-04-06 02:03:31 +08:00
|
|
|
#ifndef _WIN32
|
2009-02-28 06:16:47 +08:00
|
|
|
int fd = 0;
|
|
|
|
struct passwd *pwd = NULL;
|
2009-02-28 06:09:45 +08:00
|
|
|
const char *chroot_dir = NULL;
|
|
|
|
const char *run_as = NULL;
|
2009-04-06 02:03:31 +08:00
|
|
|
#endif
|
2009-04-22 06:30:27 +08:00
|
|
|
CPUState *env;
|
2009-05-22 05:54:00 +08:00
|
|
|
int show_vnc_port = 0;
|
2010-01-22 00:57:58 +08:00
|
|
|
int defconfig = 1;
|
2005-11-11 08:00:47 +08:00
|
|
|
|
2010-02-24 21:37:14 +08:00
|
|
|
error_set_progname(argv[0]);
|
|
|
|
|
2009-09-15 19:36:04 +08:00
|
|
|
init_clocks();
|
|
|
|
|
2008-12-11 03:18:40 +08:00
|
|
|
qemu_cache_utils_init(envp);
|
|
|
|
|
2009-09-12 15:36:22 +08:00
|
|
|
QLIST_INIT (&vm_change_state_head);
|
2006-06-26 00:25:21 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
{
|
|
|
|
struct sigaction act;
|
|
|
|
sigfillset(&act.sa_mask);
|
|
|
|
act.sa_flags = 0;
|
|
|
|
act.sa_handler = SIG_IGN;
|
|
|
|
sigaction(SIGPIPE, &act, NULL);
|
|
|
|
}
|
2006-06-27 04:03:44 +08:00
|
|
|
#else
|
|
|
|
SetConsoleCtrlHandler(qemu_ctrl_handler, TRUE);
|
2006-07-14 17:36:13 +08:00
|
|
|
/* Note: cpu_interrupt() is currently not SMP safe, so we force
|
|
|
|
QEMU to run on a single CPU */
|
|
|
|
{
|
|
|
|
HANDLE h;
|
|
|
|
DWORD mask, smask;
|
|
|
|
int i;
|
|
|
|
h = GetCurrentProcess();
|
|
|
|
if (GetProcessAffinityMask(h, &mask, &smask)) {
|
|
|
|
for(i = 0; i < 32; i++) {
|
|
|
|
if (mask & (1 << i))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (i != 32) {
|
|
|
|
mask = 1 << i;
|
|
|
|
SetProcessAffinityMask(h, mask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-04-01 07:37:16 +08:00
|
|
|
#endif
|
2006-06-26 00:25:21 +08:00
|
|
|
|
2009-05-21 07:38:09 +08:00
|
|
|
module_call_init(MODULE_INIT_MACHINE);
|
2009-05-22 09:41:01 +08:00
|
|
|
machine = find_default_machine();
|
2007-03-06 03:44:02 +08:00
|
|
|
cpu_model = NULL;
|
2003-06-30 18:03:06 +08:00
|
|
|
initrd_filename = NULL;
|
2008-04-28 05:39:40 +08:00
|
|
|
ram_size = 0;
|
2003-07-07 01:15:21 +08:00
|
|
|
snapshot = 0;
|
2003-10-01 05:07:02 +08:00
|
|
|
kernel_filename = NULL;
|
|
|
|
kernel_cmdline = "";
|
2004-03-15 05:44:30 +08:00
|
|
|
cyls = heads = secs = 0;
|
2004-11-16 09:45:27 +08:00
|
|
|
translation = BIOS_ATA_TRANSLATION_AUTO;
|
2004-03-15 05:44:30 +08:00
|
|
|
|
2009-04-22 06:30:27 +08:00
|
|
|
for (i = 0; i < MAX_NODES; i++) {
|
|
|
|
node_mem[i] = 0;
|
|
|
|
node_cpumask[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
nb_numa_nodes = 0;
|
2005-11-16 06:16:05 +08:00
|
|
|
nb_nics = 0;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2008-05-28 20:30:31 +08:00
|
|
|
tb_size = 0;
|
2008-10-05 17:56:21 +08:00
|
|
|
autostart= 1;
|
|
|
|
|
2010-01-22 00:57:58 +08:00
|
|
|
/* first pass of option parsing */
|
|
|
|
optind = 1;
|
|
|
|
while (optind < argc) {
|
|
|
|
if (argv[optind][0] != '-') {
|
|
|
|
/* disk image */
|
2010-01-28 00:46:00 +08:00
|
|
|
optind++;
|
2010-01-22 00:57:58 +08:00
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
const QEMUOption *popt;
|
|
|
|
|
|
|
|
popt = lookup_opt(argc, argv, &optarg, &optind);
|
|
|
|
switch (popt->index) {
|
|
|
|
case QEMU_OPTION_nodefconfig:
|
|
|
|
defconfig=0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defconfig) {
|
2010-02-19 02:48:33 +08:00
|
|
|
const char *fname;
|
2010-01-22 00:57:58 +08:00
|
|
|
FILE *fp;
|
2010-02-19 02:48:33 +08:00
|
|
|
|
|
|
|
fname = CONFIG_QEMU_CONFDIR "/qemu.conf";
|
|
|
|
fp = fopen(fname, "r");
|
2010-01-22 00:57:58 +08:00
|
|
|
if (fp) {
|
2010-02-19 02:48:33 +08:00
|
|
|
if (qemu_config_parse(fp, fname) != 0) {
|
2010-01-22 00:57:58 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
|
2010-02-19 02:48:33 +08:00
|
|
|
fname = CONFIG_QEMU_CONFDIR "/target-" TARGET_ARCH ".conf";
|
|
|
|
fp = fopen(fname, "r");
|
2010-01-22 00:57:58 +08:00
|
|
|
if (fp) {
|
2010-02-19 02:48:33 +08:00
|
|
|
if (qemu_config_parse(fp, fname) != 0) {
|
2010-01-22 00:57:58 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
}
|
2010-02-21 01:14:59 +08:00
|
|
|
#if defined(cpudef_setup)
|
|
|
|
cpudef_setup(); /* parse cpu definitions in target config file */
|
|
|
|
#endif
|
2010-01-22 00:57:58 +08:00
|
|
|
|
|
|
|
/* second pass of option parsing */
|
2004-05-14 06:02:20 +08:00
|
|
|
optind = 1;
|
2003-06-24 21:42:40 +08:00
|
|
|
for(;;) {
|
2004-05-14 06:02:20 +08:00
|
|
|
if (optind >= argc)
|
2003-06-24 21:42:40 +08:00
|
|
|
break;
|
2010-01-22 23:18:06 +08:00
|
|
|
if (argv[optind][0] != '-') {
|
2009-07-22 22:43:04 +08:00
|
|
|
hda_opts = drive_add(argv[optind++], HD_ALIAS, 0);
|
2004-05-14 06:02:20 +08:00
|
|
|
} else {
|
|
|
|
const QEMUOption *popt;
|
|
|
|
|
2010-01-22 23:18:06 +08:00
|
|
|
popt = lookup_opt(argc, argv, &optarg, &optind);
|
2004-05-14 06:02:20 +08:00
|
|
|
switch(popt->index) {
|
2005-06-05 22:49:17 +08:00
|
|
|
case QEMU_OPTION_M:
|
|
|
|
machine = find_machine(optarg);
|
|
|
|
if (!machine) {
|
|
|
|
QEMUMachine *m;
|
|
|
|
printf("Supported machines are:\n");
|
|
|
|
for(m = first_machine; m != NULL; m = m->next) {
|
2009-07-22 17:02:50 +08:00
|
|
|
if (m->alias)
|
|
|
|
printf("%-10s %s (alias of %s)\n",
|
|
|
|
m->alias, m->desc, m->name);
|
2005-06-05 22:49:17 +08:00
|
|
|
printf("%-10s %s%s\n",
|
2007-09-17 05:08:06 +08:00
|
|
|
m->name, m->desc,
|
2009-05-22 09:41:01 +08:00
|
|
|
m->is_default ? " (default)" : "");
|
2005-06-05 22:49:17 +08:00
|
|
|
}
|
2007-06-30 07:26:08 +08:00
|
|
|
exit(*optarg != '?');
|
2005-06-05 22:49:17 +08:00
|
|
|
}
|
|
|
|
break;
|
2007-03-06 03:44:02 +08:00
|
|
|
case QEMU_OPTION_cpu:
|
|
|
|
/* hw initialization will check this */
|
2007-06-30 07:26:08 +08:00
|
|
|
if (*optarg == '?') {
|
2007-10-12 14:47:46 +08:00
|
|
|
/* XXX: implement xxx_cpu_list for targets that still miss it */
|
2010-02-21 01:14:59 +08:00
|
|
|
#if defined(cpu_list_id)
|
|
|
|
cpu_list_id(stdout, &fprintf, optarg);
|
|
|
|
#elif defined(cpu_list)
|
|
|
|
cpu_list(stdout, &fprintf); /* deprecated */
|
2007-03-06 03:44:02 +08:00
|
|
|
#endif
|
2007-06-30 07:26:08 +08:00
|
|
|
exit(0);
|
2007-03-06 03:44:02 +08:00
|
|
|
} else {
|
|
|
|
cpu_model = optarg;
|
|
|
|
}
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_initrd:
|
2003-06-30 18:03:06 +08:00
|
|
|
initrd_filename = optarg;
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_hda:
|
2007-12-02 12:51:10 +08:00
|
|
|
if (cyls == 0)
|
2009-07-22 22:43:04 +08:00
|
|
|
hda_opts = drive_add(optarg, HD_ALIAS, 0);
|
2007-12-02 12:51:10 +08:00
|
|
|
else
|
2009-07-22 22:43:04 +08:00
|
|
|
hda_opts = drive_add(optarg, HD_ALIAS
|
2007-12-02 12:51:10 +08:00
|
|
|
",cyls=%d,heads=%d,secs=%d%s",
|
2008-01-14 10:56:53 +08:00
|
|
|
0, cyls, heads, secs,
|
2007-12-02 12:51:10 +08:00
|
|
|
translation == BIOS_ATA_TRANSLATION_LBA ?
|
|
|
|
",trans=lba" :
|
|
|
|
translation == BIOS_ATA_TRANSLATION_NONE ?
|
|
|
|
",trans=none" : "");
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_hdb:
|
2005-06-05 22:49:17 +08:00
|
|
|
case QEMU_OPTION_hdc:
|
|
|
|
case QEMU_OPTION_hdd:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(optarg, HD_ALIAS, popt->index - QEMU_OPTION_hda);
|
2003-06-30 18:03:06 +08:00
|
|
|
break;
|
2007-12-02 12:51:10 +08:00
|
|
|
case QEMU_OPTION_drive:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(NULL, "%s", optarg);
|
2007-12-02 12:51:10 +08:00
|
|
|
break;
|
2009-07-31 18:25:36 +08:00
|
|
|
case QEMU_OPTION_set:
|
|
|
|
if (qemu_set_option(optarg) != 0)
|
|
|
|
exit(1);
|
|
|
|
break;
|
2009-12-08 20:11:34 +08:00
|
|
|
case QEMU_OPTION_global:
|
|
|
|
if (qemu_global_option(optarg) != 0)
|
|
|
|
exit(1);
|
|
|
|
break;
|
2007-04-30 10:09:25 +08:00
|
|
|
case QEMU_OPTION_mtdblock:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(optarg, MTD_ALIAS);
|
2007-04-30 10:09:25 +08:00
|
|
|
break;
|
2007-04-07 00:49:48 +08:00
|
|
|
case QEMU_OPTION_sd:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(optarg, SD_ALIAS);
|
2007-04-07 00:49:48 +08:00
|
|
|
break;
|
2007-04-24 14:52:59 +08:00
|
|
|
case QEMU_OPTION_pflash:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(optarg, PFLASH_ALIAS);
|
2007-04-24 14:52:59 +08:00
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_snapshot:
|
2003-07-07 01:15:21 +08:00
|
|
|
snapshot = 1;
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_hdachs:
|
2003-07-27 02:11:40 +08:00
|
|
|
{
|
|
|
|
const char *p;
|
|
|
|
p = optarg;
|
|
|
|
cyls = strtol(p, (char **)&p, 0);
|
2004-11-16 09:45:27 +08:00
|
|
|
if (cyls < 1 || cyls > 16383)
|
|
|
|
goto chs_fail;
|
2003-07-27 02:11:40 +08:00
|
|
|
if (*p != ',')
|
|
|
|
goto chs_fail;
|
|
|
|
p++;
|
|
|
|
heads = strtol(p, (char **)&p, 0);
|
2004-11-16 09:45:27 +08:00
|
|
|
if (heads < 1 || heads > 16)
|
|
|
|
goto chs_fail;
|
2003-07-27 02:11:40 +08:00
|
|
|
if (*p != ',')
|
|
|
|
goto chs_fail;
|
|
|
|
p++;
|
|
|
|
secs = strtol(p, (char **)&p, 0);
|
2004-11-16 09:45:27 +08:00
|
|
|
if (secs < 1 || secs > 63)
|
|
|
|
goto chs_fail;
|
|
|
|
if (*p == ',') {
|
|
|
|
p++;
|
|
|
|
if (!strcmp(p, "none"))
|
|
|
|
translation = BIOS_ATA_TRANSLATION_NONE;
|
|
|
|
else if (!strcmp(p, "lba"))
|
|
|
|
translation = BIOS_ATA_TRANSLATION_LBA;
|
|
|
|
else if (!strcmp(p, "auto"))
|
|
|
|
translation = BIOS_ATA_TRANSLATION_AUTO;
|
|
|
|
else
|
|
|
|
goto chs_fail;
|
|
|
|
} else if (*p != '\0') {
|
2004-03-15 05:44:30 +08:00
|
|
|
chs_fail:
|
2004-11-16 09:45:27 +08:00
|
|
|
fprintf(stderr, "qemu: invalid physical CHS format\n");
|
|
|
|
exit(1);
|
2004-03-15 05:44:30 +08:00
|
|
|
}
|
2009-07-22 22:43:04 +08:00
|
|
|
if (hda_opts != NULL) {
|
|
|
|
char num[16];
|
|
|
|
snprintf(num, sizeof(num), "%d", cyls);
|
|
|
|
qemu_opt_set(hda_opts, "cyls", num);
|
|
|
|
snprintf(num, sizeof(num), "%d", heads);
|
|
|
|
qemu_opt_set(hda_opts, "heads", num);
|
|
|
|
snprintf(num, sizeof(num), "%d", secs);
|
|
|
|
qemu_opt_set(hda_opts, "secs", num);
|
|
|
|
if (translation == BIOS_ATA_TRANSLATION_LBA)
|
|
|
|
qemu_opt_set(hda_opts, "trans", "lba");
|
|
|
|
if (translation == BIOS_ATA_TRANSLATION_NONE)
|
|
|
|
qemu_opt_set(hda_opts, "trans", "none");
|
|
|
|
}
|
2003-07-27 02:11:40 +08:00
|
|
|
}
|
|
|
|
break;
|
2009-04-22 06:30:27 +08:00
|
|
|
case QEMU_OPTION_numa:
|
|
|
|
if (nb_numa_nodes >= MAX_NODES) {
|
|
|
|
fprintf(stderr, "qemu: too many NUMA nodes\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
numa_add(optarg);
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_nographic:
|
2009-05-22 05:54:00 +08:00
|
|
|
display_type = DT_NOGRAPHIC;
|
2003-10-01 05:07:02 +08:00
|
|
|
break;
|
2008-02-11 00:33:14 +08:00
|
|
|
#ifdef CONFIG_CURSES
|
|
|
|
case QEMU_OPTION_curses:
|
2009-05-22 05:54:00 +08:00
|
|
|
display_type = DT_CURSES;
|
2008-02-11 00:33:14 +08:00
|
|
|
break;
|
|
|
|
#endif
|
2007-04-30 09:48:07 +08:00
|
|
|
case QEMU_OPTION_portrait:
|
|
|
|
graphic_rotate = 1;
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_kernel:
|
2003-10-01 05:07:02 +08:00
|
|
|
kernel_filename = optarg;
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_append:
|
2003-10-01 05:07:02 +08:00
|
|
|
kernel_cmdline = optarg;
|
2003-08-11 05:52:11 +08:00
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_cdrom:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(optarg, CDROM_ALIAS);
|
2003-11-11 21:36:08 +08:00
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_boot:
|
2007-11-11 09:50:45 +08:00
|
|
|
{
|
2009-07-02 06:19:02 +08:00
|
|
|
static const char * const params[] = {
|
2009-07-02 06:19:02 +08:00
|
|
|
"order", "once", "menu", NULL
|
2009-07-02 06:19:02 +08:00
|
|
|
};
|
|
|
|
char buf[sizeof(boot_devices)];
|
2009-07-02 06:19:02 +08:00
|
|
|
char *standard_boot_devices;
|
2009-07-02 06:19:02 +08:00
|
|
|
int legacy = 0;
|
|
|
|
|
|
|
|
if (!strchr(optarg, '=')) {
|
|
|
|
legacy = 1;
|
|
|
|
pstrcpy(buf, sizeof(buf), optarg);
|
|
|
|
} else if (check_params(buf, sizeof(buf), params, optarg) < 0) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"qemu: unknown boot parameter '%s' in '%s'\n",
|
|
|
|
buf, optarg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (legacy ||
|
|
|
|
get_param_value(buf, sizeof(buf), "order", optarg)) {
|
|
|
|
boot_devices_bitmap = parse_bootdevices(buf);
|
|
|
|
pstrcpy(boot_devices, sizeof(boot_devices), buf);
|
2007-11-11 09:50:45 +08:00
|
|
|
}
|
2009-07-02 06:19:02 +08:00
|
|
|
if (!legacy) {
|
|
|
|
if (get_param_value(buf, sizeof(buf),
|
|
|
|
"once", optarg)) {
|
|
|
|
boot_devices_bitmap |= parse_bootdevices(buf);
|
|
|
|
standard_boot_devices = qemu_strdup(boot_devices);
|
|
|
|
pstrcpy(boot_devices, sizeof(boot_devices), buf);
|
|
|
|
qemu_register_reset(restore_boot_devices,
|
|
|
|
standard_boot_devices);
|
|
|
|
}
|
2009-07-02 06:19:02 +08:00
|
|
|
if (get_param_value(buf, sizeof(buf),
|
|
|
|
"menu", optarg)) {
|
|
|
|
if (!strcmp(buf, "on")) {
|
|
|
|
boot_menu = 1;
|
|
|
|
} else if (!strcmp(buf, "off")) {
|
|
|
|
boot_menu = 0;
|
|
|
|
} else {
|
|
|
|
fprintf(stderr,
|
|
|
|
"qemu: invalid option value '%s'\n",
|
|
|
|
buf);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
2009-07-02 06:19:02 +08:00
|
|
|
}
|
2003-11-11 21:36:08 +08:00
|
|
|
}
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_fda:
|
|
|
|
case QEMU_OPTION_fdb:
|
2008-01-14 10:56:53 +08:00
|
|
|
drive_add(optarg, FD_ALIAS, popt->index - QEMU_OPTION_fda);
|
2004-01-05 08:02:06 +08:00
|
|
|
break;
|
2006-06-15 00:03:05 +08:00
|
|
|
#ifdef TARGET_I386
|
|
|
|
case QEMU_OPTION_no_fd_bootchk:
|
|
|
|
fd_bootchk = 0;
|
|
|
|
break;
|
|
|
|
#endif
|
2009-10-09 02:58:26 +08:00
|
|
|
case QEMU_OPTION_netdev:
|
|
|
|
if (net_client_parse(&qemu_netdev_opts, optarg) == -1) {
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
break;
|
2005-11-16 06:16:05 +08:00
|
|
|
case QEMU_OPTION_net:
|
2009-10-09 02:58:25 +08:00
|
|
|
if (net_client_parse(&qemu_net_opts, optarg) == -1) {
|
2004-03-15 05:44:30 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
2004-04-03 05:21:32 +08:00
|
|
|
break;
|
2004-08-25 05:57:12 +08:00
|
|
|
#ifdef CONFIG_SLIRP
|
|
|
|
case QEMU_OPTION_tftp:
|
2009-06-24 20:42:28 +08:00
|
|
|
legacy_tftp_prefix = optarg;
|
2004-08-26 06:12:49 +08:00
|
|
|
break;
|
2007-02-20 08:05:08 +08:00
|
|
|
case QEMU_OPTION_bootp:
|
2009-06-24 20:42:28 +08:00
|
|
|
legacy_bootp_filename = optarg;
|
2007-02-20 08:05:08 +08:00
|
|
|
break;
|
2004-09-14 05:37:34 +08:00
|
|
|
#ifndef _WIN32
|
2004-09-06 07:09:03 +08:00
|
|
|
case QEMU_OPTION_smb:
|
Don't exit() in config_error()
Propagating errors up the call chain is tedious. In startup code, we
can take a shortcut: terminate the program. This is wrong elsewhere,
the monitor in particular.
config_error() tries to cater for both customers: it terminates the
program unless its mon parameter tells it it's working for the
monitor.
Its users need to return status anyway (unless passing a null mon
argument, which none do), which their users need to check. So this
automatic exit buys us exactly nothing useful. Only the dangerous
delusion that we can get away without returning status. Some of its
users fell for that. Their callers continue executing after failure
when working for the monitor.
This bites monitor command host_net_add in two places:
* net_slirp_init() continues after slirp_hostfwd(), slirp_guestfwd(),
or slirp_smb() failed, and may end up reporting success. This
happens for "host_net_add user guestfwd=foo": it complains about the
invalid guest forwarding rule, then happily creates the user network
without guest forwarding.
* net_client_init() can't detect slirp_guestfwd() failure, and gets
fooled by net_slirp_init() lying about success. Suppresses its
"Could not initialize device" message.
Add the missing error reporting, make sure errors are checked, and
drop the exit() from config_error().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Mark McLoughlin <markmc@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-10-06 19:16:57 +08:00
|
|
|
if (net_slirp_smb(optarg) < 0)
|
|
|
|
exit(1);
|
2004-09-06 07:09:03 +08:00
|
|
|
break;
|
2004-09-14 05:37:34 +08:00
|
|
|
#endif
|
2004-08-26 06:12:49 +08:00
|
|
|
case QEMU_OPTION_redir:
|
Don't exit() in config_error()
Propagating errors up the call chain is tedious. In startup code, we
can take a shortcut: terminate the program. This is wrong elsewhere,
the monitor in particular.
config_error() tries to cater for both customers: it terminates the
program unless its mon parameter tells it it's working for the
monitor.
Its users need to return status anyway (unless passing a null mon
argument, which none do), which their users need to check. So this
automatic exit buys us exactly nothing useful. Only the dangerous
delusion that we can get away without returning status. Some of its
users fell for that. Their callers continue executing after failure
when working for the monitor.
This bites monitor command host_net_add in two places:
* net_slirp_init() continues after slirp_hostfwd(), slirp_guestfwd(),
or slirp_smb() failed, and may end up reporting success. This
happens for "host_net_add user guestfwd=foo": it complains about the
invalid guest forwarding rule, then happily creates the user network
without guest forwarding.
* net_client_init() can't detect slirp_guestfwd() failure, and gets
fooled by net_slirp_init() lying about success. Suppresses its
"Could not initialize device" message.
Add the missing error reporting, make sure errors are checked, and
drop the exit() from config_error().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Mark McLoughlin <markmc@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-10-06 19:16:57 +08:00
|
|
|
if (net_slirp_redir(optarg) < 0)
|
|
|
|
exit(1);
|
2004-08-26 06:12:49 +08:00
|
|
|
break;
|
2004-08-25 05:57:12 +08:00
|
|
|
#endif
|
2008-11-09 08:04:26 +08:00
|
|
|
case QEMU_OPTION_bt:
|
2009-07-15 19:59:26 +08:00
|
|
|
add_device_config(DEV_BT, optarg);
|
2008-11-09 08:04:26 +08:00
|
|
|
break;
|
2005-10-31 02:58:22 +08:00
|
|
|
#ifdef HAS_AUDIO
|
|
|
|
case QEMU_OPTION_audio_help:
|
|
|
|
AUD_help ();
|
|
|
|
exit (0);
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_soundhw:
|
|
|
|
select_soundhw (optarg);
|
|
|
|
break;
|
|
|
|
#endif
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_h:
|
2007-06-30 07:26:08 +08:00
|
|
|
help(0);
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
2009-04-08 06:58:45 +08:00
|
|
|
case QEMU_OPTION_version:
|
|
|
|
version();
|
|
|
|
exit(0);
|
|
|
|
break;
|
2008-04-28 05:12:55 +08:00
|
|
|
case QEMU_OPTION_m: {
|
|
|
|
uint64_t value;
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
value = strtoul(optarg, &ptr, 10);
|
|
|
|
switch (*ptr) {
|
|
|
|
case 0: case 'M': case 'm':
|
|
|
|
value <<= 20;
|
|
|
|
break;
|
|
|
|
case 'G': case 'g':
|
|
|
|
value <<= 30;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "qemu: invalid ram size: %s\n", optarg);
|
2004-05-14 06:02:20 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
2008-04-28 05:12:55 +08:00
|
|
|
|
|
|
|
/* On 32-bit hosts, QEMU is limited by virtual address space */
|
2009-08-11 06:07:24 +08:00
|
|
|
if (value > (2047 << 20) && HOST_LONG_BITS == 32) {
|
2008-04-28 05:12:55 +08:00
|
|
|
fprintf(stderr, "qemu: at most 2047 MB RAM can be simulated\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-02 05:12:16 +08:00
|
|
|
if (value != (uint64_t)(ram_addr_t)value) {
|
2008-04-28 05:12:55 +08:00
|
|
|
fprintf(stderr, "qemu: ram size too large\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
ram_size = value;
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
2008-04-28 05:12:55 +08:00
|
|
|
}
|
2010-03-02 07:25:08 +08:00
|
|
|
case QEMU_OPTION_mempath:
|
|
|
|
mem_path = optarg;
|
|
|
|
break;
|
|
|
|
#ifdef MAP_POPULATE
|
|
|
|
case QEMU_OPTION_mem_prealloc:
|
|
|
|
mem_prealloc = 1;
|
|
|
|
break;
|
|
|
|
#endif
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_d:
|
|
|
|
{
|
|
|
|
int mask;
|
2008-10-03 02:27:46 +08:00
|
|
|
const CPULogItem *item;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2004-05-14 06:02:20 +08:00
|
|
|
mask = cpu_str_to_log_mask(optarg);
|
|
|
|
if (!mask) {
|
|
|
|
printf("Log items (comma separated):\n");
|
2004-03-22 01:06:25 +08:00
|
|
|
for(item = cpu_log_items; item->mask != 0; item++) {
|
|
|
|
printf("%-10s %s\n", item->name, item->help);
|
|
|
|
}
|
|
|
|
exit(1);
|
2004-05-14 06:02:20 +08:00
|
|
|
}
|
|
|
|
cpu_set_log(mask);
|
2004-03-22 01:06:25 +08:00
|
|
|
}
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
|
|
|
case QEMU_OPTION_s:
|
2009-04-06 02:43:41 +08:00
|
|
|
gdbstub_dev = "tcp::" DEFAULT_GDBSTUB_PORT;
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
2009-04-06 02:43:41 +08:00
|
|
|
case QEMU_OPTION_gdb:
|
|
|
|
gdbstub_dev = optarg;
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
|
|
|
case QEMU_OPTION_L:
|
2009-05-30 07:52:44 +08:00
|
|
|
data_dir = optarg;
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
2007-10-05 21:08:35 +08:00
|
|
|
case QEMU_OPTION_bios:
|
|
|
|
bios_name = optarg;
|
|
|
|
break;
|
2009-04-06 04:08:59 +08:00
|
|
|
case QEMU_OPTION_singlestep:
|
|
|
|
singlestep = 1;
|
|
|
|
break;
|
2004-05-14 06:02:20 +08:00
|
|
|
case QEMU_OPTION_S:
|
2007-01-22 00:47:01 +08:00
|
|
|
autostart = 0;
|
2004-05-14 06:02:20 +08:00
|
|
|
break;
|
2004-12-13 00:56:30 +08:00
|
|
|
case QEMU_OPTION_k:
|
|
|
|
keyboard_layout = optarg;
|
|
|
|
break;
|
2004-06-03 20:49:50 +08:00
|
|
|
case QEMU_OPTION_localtime:
|
|
|
|
rtc_utc = 0;
|
|
|
|
break;
|
2008-09-28 08:42:05 +08:00
|
|
|
case QEMU_OPTION_vga:
|
|
|
|
select_vgahw (optarg);
|
2004-07-09 05:17:50 +08:00
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#if defined(TARGET_PPC) || defined(TARGET_SPARC)
|
2004-06-22 00:46:10 +08:00
|
|
|
case QEMU_OPTION_g:
|
|
|
|
{
|
|
|
|
const char *p;
|
|
|
|
int w, h, depth;
|
|
|
|
p = optarg;
|
|
|
|
w = strtol(p, (char **)&p, 10);
|
|
|
|
if (w <= 0) {
|
|
|
|
graphic_error:
|
|
|
|
fprintf(stderr, "qemu: invalid resolution or depth\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (*p != 'x')
|
|
|
|
goto graphic_error;
|
|
|
|
p++;
|
|
|
|
h = strtol(p, (char **)&p, 10);
|
|
|
|
if (h <= 0)
|
|
|
|
goto graphic_error;
|
|
|
|
if (*p == 'x') {
|
|
|
|
p++;
|
|
|
|
depth = strtol(p, (char **)&p, 10);
|
2007-09-17 05:08:06 +08:00
|
|
|
if (depth != 8 && depth != 15 && depth != 16 &&
|
2004-06-22 00:46:10 +08:00
|
|
|
depth != 24 && depth != 32)
|
|
|
|
goto graphic_error;
|
|
|
|
} else if (*p == '\0') {
|
|
|
|
depth = graphic_depth;
|
|
|
|
} else {
|
|
|
|
goto graphic_error;
|
|
|
|
}
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2004-06-22 00:46:10 +08:00
|
|
|
graphic_width = w;
|
|
|
|
graphic_height = h;
|
|
|
|
graphic_depth = depth;
|
|
|
|
}
|
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#endif
|
2007-02-19 01:04:49 +08:00
|
|
|
case QEMU_OPTION_echr:
|
|
|
|
{
|
|
|
|
char *r;
|
|
|
|
term_escape_char = strtol(optarg, &r, 0);
|
|
|
|
if (r == optarg)
|
|
|
|
printf("Bad argument to echr\n");
|
|
|
|
break;
|
|
|
|
}
|
2004-07-15 01:28:13 +08:00
|
|
|
case QEMU_OPTION_monitor:
|
2009-12-08 20:11:52 +08:00
|
|
|
monitor_parse(optarg, "readline");
|
|
|
|
default_monitor = 0;
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_qmp:
|
|
|
|
monitor_parse(optarg, "control");
|
2009-12-08 20:11:43 +08:00
|
|
|
default_monitor = 0;
|
2004-07-15 01:28:13 +08:00
|
|
|
break;
|
2009-12-08 20:11:51 +08:00
|
|
|
case QEMU_OPTION_mon:
|
2010-02-11 02:52:18 +08:00
|
|
|
opts = qemu_opts_parse(&qemu_mon_opts, optarg, 1);
|
2009-12-08 20:11:51 +08:00
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "parse error: %s\n", optarg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
default_monitor = 0;
|
|
|
|
break;
|
2009-09-10 16:58:35 +08:00
|
|
|
case QEMU_OPTION_chardev:
|
2010-02-11 02:52:18 +08:00
|
|
|
opts = qemu_opts_parse(&qemu_chardev_opts, optarg, 1);
|
2009-09-10 16:58:35 +08:00
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "parse error: %s\n", optarg);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
break;
|
2004-07-15 01:28:13 +08:00
|
|
|
case QEMU_OPTION_serial:
|
2009-12-08 20:11:41 +08:00
|
|
|
add_device_config(DEV_SERIAL, optarg);
|
|
|
|
default_serial = 0;
|
2010-03-07 18:28:40 +08:00
|
|
|
if (strncmp(optarg, "mon:", 4) == 0) {
|
|
|
|
default_monitor = 0;
|
|
|
|
}
|
2004-07-15 01:28:13 +08:00
|
|
|
break;
|
2009-04-25 20:56:19 +08:00
|
|
|
case QEMU_OPTION_watchdog:
|
2009-08-21 16:31:34 +08:00
|
|
|
if (watchdog) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"qemu: only one watchdog option may be given\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
watchdog = optarg;
|
2009-04-25 20:56:19 +08:00
|
|
|
break;
|
|
|
|
case QEMU_OPTION_watchdog_action:
|
|
|
|
if (select_watchdog_action(optarg) == -1) {
|
|
|
|
fprintf(stderr, "Unknown -watchdog-action parameter\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
break;
|
2009-01-16 04:06:40 +08:00
|
|
|
case QEMU_OPTION_virtiocon:
|
2009-12-08 20:11:53 +08:00
|
|
|
add_device_config(DEV_VIRTCON, optarg);
|
|
|
|
default_virtcon = 0;
|
2010-03-07 18:28:40 +08:00
|
|
|
if (strncmp(optarg, "mon:", 4) == 0) {
|
|
|
|
default_monitor = 0;
|
|
|
|
}
|
2009-01-16 04:06:40 +08:00
|
|
|
break;
|
2005-01-15 20:02:56 +08:00
|
|
|
case QEMU_OPTION_parallel:
|
2009-12-08 20:11:42 +08:00
|
|
|
add_device_config(DEV_PARALLEL, optarg);
|
|
|
|
default_parallel = 0;
|
2010-03-07 18:28:40 +08:00
|
|
|
if (strncmp(optarg, "mon:", 4) == 0) {
|
|
|
|
default_monitor = 0;
|
|
|
|
}
|
2005-01-15 20:02:56 +08:00
|
|
|
break;
|
debugcon: support for debugging consoles (e.g. Bochs port 0xe9)
Add generic support for debugging consoles (simple I/O ports which
when written to cause debugging output to be written to a target.)
The current implementation matches Bochs' port 0xe9, allowing the same
debugging code to be used for both Bochs and Qemu.
There is no vm state associated with the debugging port, simply
because it has none -- the entire interface is a single, stateless,
write-only port.
Most of the code was cribbed from the serial port driver.
v2: removed non-ISA variants (they can be introduced when/if someone
wants them, using code from the serial port); added configurable
readback (Bochs returns 0xe9 on a read from this register, mimic that
by default) This retains the apparently somewhat controversial user
friendly option, however.
v3: reimplemented the user friendly option as a synthetic option
("-debugcon foo" basically ends up being a parser-level shorthand for
"-chardev stdio,id=debugcon -device isa-debugcon,chardev=debugcon") --
this dramatically reduced the complexity while keeping the same level
of user friendliness.
v4: spaces, not tabs.
v5: update to match current top of tree. Calling qemu_chr_open()
already during parsing no longer works; defer until we are parsing the
other console-like devices.
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-12-30 05:51:36 +08:00
|
|
|
case QEMU_OPTION_debugcon:
|
|
|
|
add_device_config(DEV_DEBUGCON, optarg);
|
|
|
|
break;
|
2004-10-03 21:29:03 +08:00
|
|
|
case QEMU_OPTION_loadvm:
|
|
|
|
loadvm = optarg;
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_full_screen:
|
|
|
|
full_screen = 1;
|
|
|
|
break;
|
2006-12-11 10:08:05 +08:00
|
|
|
#ifdef CONFIG_SDL
|
2007-02-19 02:19:32 +08:00
|
|
|
case QEMU_OPTION_no_frame:
|
|
|
|
no_frame = 1;
|
|
|
|
break;
|
2007-06-22 05:08:02 +08:00
|
|
|
case QEMU_OPTION_alt_grab:
|
|
|
|
alt_grab = 1;
|
|
|
|
break;
|
2009-09-18 04:48:04 +08:00
|
|
|
case QEMU_OPTION_ctrl_grab:
|
|
|
|
ctrl_grab = 1;
|
|
|
|
break;
|
2006-12-11 10:08:05 +08:00
|
|
|
case QEMU_OPTION_no_quit:
|
|
|
|
no_quit = 1;
|
|
|
|
break;
|
2009-01-16 06:14:11 +08:00
|
|
|
case QEMU_OPTION_sdl:
|
2009-05-22 05:54:00 +08:00
|
|
|
display_type = DT_SDL;
|
2009-01-16 06:14:11 +08:00
|
|
|
break;
|
2006-12-11 10:08:05 +08:00
|
|
|
#endif
|
2004-12-09 06:21:25 +08:00
|
|
|
case QEMU_OPTION_pidfile:
|
2007-03-19 23:58:31 +08:00
|
|
|
pid_file = optarg;
|
2004-12-09 06:21:25 +08:00
|
|
|
break;
|
2005-05-01 00:10:35 +08:00
|
|
|
#ifdef TARGET_I386
|
|
|
|
case QEMU_OPTION_win2k_hack:
|
|
|
|
win2k_install_hack = 1;
|
|
|
|
break;
|
2009-01-16 04:11:34 +08:00
|
|
|
case QEMU_OPTION_rtc_td_hack:
|
|
|
|
rtc_td_hack = 1;
|
|
|
|
break;
|
2009-02-28 04:12:36 +08:00
|
|
|
case QEMU_OPTION_acpitable:
|
|
|
|
if(acpi_table_add(optarg) < 0) {
|
|
|
|
fprintf(stderr, "Wrong acpi table provided\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
break;
|
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-18 02:59:56 +08:00
|
|
|
case QEMU_OPTION_smbios:
|
|
|
|
if(smbios_entry_add(optarg) < 0) {
|
|
|
|
fprintf(stderr, "Wrong smbios provided\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
break;
|
2005-05-01 00:10:35 +08:00
|
|
|
#endif
|
2008-11-06 00:04:33 +08:00
|
|
|
#ifdef CONFIG_KVM
|
|
|
|
case QEMU_OPTION_enable_kvm:
|
|
|
|
kvm_allowed = 1;
|
|
|
|
break;
|
2005-02-11 06:00:06 +08:00
|
|
|
#endif
|
2005-11-05 22:22:28 +08:00
|
|
|
case QEMU_OPTION_usb:
|
|
|
|
usb_enabled = 1;
|
|
|
|
break;
|
2005-11-07 00:13:29 +08:00
|
|
|
case QEMU_OPTION_usbdevice:
|
|
|
|
usb_enabled = 1;
|
2009-07-15 19:59:26 +08:00
|
|
|
add_device_config(DEV_USB, optarg);
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_device:
|
2010-02-11 02:52:18 +08:00
|
|
|
if (!qemu_opts_parse(&qemu_device_opts, optarg, 1)) {
|
2009-07-31 18:25:37 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
2005-11-07 00:13:29 +08:00
|
|
|
break;
|
2005-11-22 07:25:50 +08:00
|
|
|
case QEMU_OPTION_smp:
|
extend -smp parsing to include cores= and threads= options
For injecting multi-core and multi-threading CPU topology into guests
extend the -smp syntax to accommodate cores and threads specification.
Syntax: -smp smp_value[,cores=nr_cores][,threads=nr_threads]\
[,socket=nr_sockets][,maxcpus=max_cpus]
smp_value is the legacy value specifying the total number of vCPUs for
the guest. If you specify one of cores, threads or sockets this value
can be omitted. Missing values will be computed to fulfill:
smp_value = nr_cores * nr_threads * nr_sockets
where it will favour sockets over cores over threads (to mimic the
current behavior, which will only inject multiple sockets.)
So -smp 4,threads=2 will inject two sockets with 2 threads each,
-smp cores=4 is an abbreviation for -smp 4,cores=4,threads=1,sockets=1.
If max_cpus (the number of hotpluggable CPUs) is omitted, it will
be set to smp_value.
Signed-off-by: Andre Przywara <andre.przywara@amd.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-08-19 21:42:40 +08:00
|
|
|
smp_parse(optarg);
|
2008-10-08 04:39:39 +08:00
|
|
|
if (smp_cpus < 1) {
|
2005-11-22 07:25:50 +08:00
|
|
|
fprintf(stderr, "Invalid number of CPUs\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-07-23 23:03:42 +08:00
|
|
|
if (max_cpus < smp_cpus) {
|
|
|
|
fprintf(stderr, "maxcpus must be equal to or greater than "
|
|
|
|
"smp\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (max_cpus > 255) {
|
|
|
|
fprintf(stderr, "Unsupported number of maxcpus\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2005-11-22 07:25:50 +08:00
|
|
|
break;
|
2006-05-01 05:28:36 +08:00
|
|
|
case QEMU_OPTION_vnc:
|
2009-05-22 05:54:00 +08:00
|
|
|
display_type = DT_VNC;
|
2006-12-22 10:09:07 +08:00
|
|
|
vnc_display = optarg;
|
2006-05-01 05:28:36 +08:00
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#ifdef TARGET_I386
|
2006-05-04 06:02:44 +08:00
|
|
|
case QEMU_OPTION_no_acpi:
|
|
|
|
acpi_enabled = 0;
|
|
|
|
break;
|
2008-12-18 07:28:44 +08:00
|
|
|
case QEMU_OPTION_no_hpet:
|
|
|
|
no_hpet = 1;
|
|
|
|
break;
|
2009-06-27 01:15:14 +08:00
|
|
|
case QEMU_OPTION_balloon:
|
|
|
|
if (balloon_parse(optarg) < 0) {
|
|
|
|
fprintf(stderr, "Unknown -balloon argument %s\n", optarg);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-06-11 03:34:08 +08:00
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#endif
|
2006-10-03 03:44:22 +08:00
|
|
|
case QEMU_OPTION_no_reboot:
|
|
|
|
no_reboot = 1;
|
|
|
|
break;
|
2008-04-12 05:35:52 +08:00
|
|
|
case QEMU_OPTION_no_shutdown:
|
|
|
|
no_shutdown = 1;
|
|
|
|
break;
|
2007-05-01 09:34:14 +08:00
|
|
|
case QEMU_OPTION_show_cursor:
|
|
|
|
cursor_hide = 0;
|
|
|
|
break;
|
2008-09-19 02:29:08 +08:00
|
|
|
case QEMU_OPTION_uuid:
|
|
|
|
if(qemu_uuid_parse(optarg, qemu_uuid) < 0) {
|
|
|
|
fprintf(stderr, "Fail to parse UUID string."
|
|
|
|
" Wrong format.\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#ifndef _WIN32
|
2006-12-22 10:11:31 +08:00
|
|
|
case QEMU_OPTION_daemonize:
|
|
|
|
daemonize = 1;
|
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#endif
|
2007-01-06 01:39:04 +08:00
|
|
|
case QEMU_OPTION_option_rom:
|
|
|
|
if (nb_option_roms >= MAX_OPTION_ROMS) {
|
|
|
|
fprintf(stderr, "Too many option ROMs\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
option_rom[nb_option_roms] = optarg;
|
|
|
|
nb_option_roms++;
|
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#if defined(TARGET_ARM) || defined(TARGET_M68K)
|
2007-01-21 01:12:09 +08:00
|
|
|
case QEMU_OPTION_semihosting:
|
|
|
|
semihosting_enabled = 1;
|
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#endif
|
2007-03-19 23:17:08 +08:00
|
|
|
case QEMU_OPTION_name:
|
2009-07-02 15:34:17 +08:00
|
|
|
qemu_name = qemu_strdup(optarg);
|
|
|
|
{
|
|
|
|
char *p = strchr(qemu_name, ',');
|
|
|
|
if (p != NULL) {
|
|
|
|
*p++ = 0;
|
|
|
|
if (strncmp(p, "process=", 8)) {
|
|
|
|
fprintf(stderr, "Unknown subargument %s to -name", p);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
p += 8;
|
|
|
|
set_proc_name(p);
|
|
|
|
}
|
|
|
|
}
|
2007-03-19 23:17:08 +08:00
|
|
|
break;
|
2008-12-25 04:26:14 +08:00
|
|
|
#if defined(TARGET_SPARC) || defined(TARGET_PPC)
|
2007-05-01 22:16:52 +08:00
|
|
|
case QEMU_OPTION_prom_env:
|
|
|
|
if (nb_prom_envs >= MAX_PROM_ENVS) {
|
|
|
|
fprintf(stderr, "Too many prom variables\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
prom_envs[nb_prom_envs] = optarg;
|
|
|
|
nb_prom_envs++;
|
|
|
|
break;
|
2007-07-28 06:08:46 +08:00
|
|
|
#endif
|
|
|
|
#ifdef TARGET_ARM
|
|
|
|
case QEMU_OPTION_old_param:
|
|
|
|
old_param = 1;
|
2008-01-09 03:32:16 +08:00
|
|
|
break;
|
2007-05-01 22:16:52 +08:00
|
|
|
#endif
|
2007-08-24 09:26:02 +08:00
|
|
|
case QEMU_OPTION_clock:
|
|
|
|
configure_alarms(optarg);
|
|
|
|
break;
|
2007-11-08 00:24:33 +08:00
|
|
|
case QEMU_OPTION_startdate:
|
2009-09-15 19:36:04 +08:00
|
|
|
configure_rtc_date_offset(optarg, 1);
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_rtc:
|
2010-02-11 02:52:18 +08:00
|
|
|
opts = qemu_opts_parse(&qemu_rtc_opts, optarg, 0);
|
2009-09-15 19:36:04 +08:00
|
|
|
if (!opts) {
|
|
|
|
fprintf(stderr, "parse error: %s\n", optarg);
|
|
|
|
exit(1);
|
2007-11-08 00:24:33 +08:00
|
|
|
}
|
2009-09-15 19:36:04 +08:00
|
|
|
configure_rtc(opts);
|
2007-11-08 00:24:33 +08:00
|
|
|
break;
|
2008-05-28 20:30:31 +08:00
|
|
|
case QEMU_OPTION_tb_size:
|
|
|
|
tb_size = strtol(optarg, NULL, 0);
|
|
|
|
if (tb_size < 0)
|
|
|
|
tb_size = 0;
|
|
|
|
break;
|
2008-06-29 09:03:05 +08:00
|
|
|
case QEMU_OPTION_icount:
|
2010-03-10 18:38:48 +08:00
|
|
|
icount_option = optarg;
|
2008-06-29 09:03:05 +08:00
|
|
|
break;
|
2008-10-13 11:12:02 +08:00
|
|
|
case QEMU_OPTION_incoming:
|
|
|
|
incoming = optarg;
|
|
|
|
break;
|
2009-12-08 20:11:46 +08:00
|
|
|
case QEMU_OPTION_nodefaults:
|
|
|
|
default_serial = 0;
|
|
|
|
default_parallel = 0;
|
2009-12-08 20:11:53 +08:00
|
|
|
default_virtcon = 0;
|
2009-12-08 20:11:46 +08:00
|
|
|
default_monitor = 0;
|
|
|
|
default_vga = 0;
|
2009-12-08 20:11:47 +08:00
|
|
|
default_net = 0;
|
2009-12-16 21:25:39 +08:00
|
|
|
default_floppy = 0;
|
|
|
|
default_cdrom = 0;
|
|
|
|
default_sdcard = 0;
|
2009-12-08 20:11:46 +08:00
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#ifndef _WIN32
|
2009-02-28 06:09:45 +08:00
|
|
|
case QEMU_OPTION_chroot:
|
|
|
|
chroot_dir = optarg;
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_runas:
|
|
|
|
run_as = optarg;
|
|
|
|
break;
|
2009-04-22 23:19:10 +08:00
|
|
|
#endif
|
|
|
|
#ifdef CONFIG_XEN
|
|
|
|
case QEMU_OPTION_xen_domid:
|
|
|
|
xen_domid = atoi(optarg);
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_xen_create:
|
|
|
|
xen_mode = XEN_CREATE;
|
|
|
|
break;
|
|
|
|
case QEMU_OPTION_xen_attach:
|
|
|
|
xen_mode = XEN_ATTACH;
|
|
|
|
break;
|
2009-03-28 14:44:27 +08:00
|
|
|
#endif
|
2009-10-14 16:39:28 +08:00
|
|
|
case QEMU_OPTION_readconfig:
|
|
|
|
{
|
|
|
|
FILE *fp;
|
|
|
|
fp = fopen(optarg, "r");
|
|
|
|
if (fp == NULL) {
|
|
|
|
fprintf(stderr, "open %s: %s\n", optarg, strerror(errno));
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-02-19 02:48:33 +08:00
|
|
|
if (qemu_config_parse(fp, optarg) != 0) {
|
2009-10-14 16:39:28 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case QEMU_OPTION_writeconfig:
|
|
|
|
{
|
|
|
|
FILE *fp;
|
|
|
|
if (strcmp(optarg, "-") == 0) {
|
|
|
|
fp = stdout;
|
|
|
|
} else {
|
|
|
|
fp = fopen(optarg, "w");
|
|
|
|
if (fp == NULL) {
|
|
|
|
fprintf(stderr, "open %s: %s\n", optarg, strerror(errno));
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
qemu_config_write(fp);
|
|
|
|
fclose(fp);
|
|
|
|
break;
|
|
|
|
}
|
2004-05-14 06:02:20 +08:00
|
|
|
}
|
2003-06-24 21:42:40 +08:00
|
|
|
}
|
|
|
|
}
|
2010-02-19 03:13:51 +08:00
|
|
|
loc_set_none();
|
2003-07-27 02:11:40 +08:00
|
|
|
|
2009-05-30 07:52:44 +08:00
|
|
|
/* If no data_dir is specified then try to find it relative to the
|
|
|
|
executable path. */
|
|
|
|
if (!data_dir) {
|
|
|
|
data_dir = find_datadir(argv[0]);
|
|
|
|
}
|
|
|
|
/* If all else fails use the install patch specified when building. */
|
|
|
|
if (!data_dir) {
|
|
|
|
data_dir = CONFIG_QEMU_SHAREDIR;
|
|
|
|
}
|
|
|
|
|
2009-07-23 23:03:42 +08:00
|
|
|
/*
|
|
|
|
* Default to max_cpus = smp_cpus, in case the user doesn't
|
|
|
|
* specify a max_cpus value.
|
|
|
|
*/
|
|
|
|
if (!max_cpus)
|
|
|
|
max_cpus = smp_cpus;
|
|
|
|
|
2008-10-28 18:59:59 +08:00
|
|
|
machine->max_cpus = machine->max_cpus ?: 1; /* Default to UP */
|
2008-10-08 04:39:39 +08:00
|
|
|
if (smp_cpus > machine->max_cpus) {
|
|
|
|
fprintf(stderr, "Number of SMP cpus requested (%d), exceeds max cpus "
|
|
|
|
"supported by machine `%s' (%d)\n", smp_cpus, machine->name,
|
|
|
|
machine->max_cpus);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:41 +08:00
|
|
|
qemu_opts_foreach(&qemu_device_opts, default_driver_check, NULL, 0);
|
2009-12-16 21:25:40 +08:00
|
|
|
qemu_opts_foreach(&qemu_global_opts, default_driver_check, NULL, 0);
|
2009-12-08 20:11:41 +08:00
|
|
|
|
2009-12-08 20:11:54 +08:00
|
|
|
if (machine->no_serial) {
|
|
|
|
default_serial = 0;
|
|
|
|
}
|
|
|
|
if (machine->no_parallel) {
|
|
|
|
default_parallel = 0;
|
|
|
|
}
|
|
|
|
if (!machine->use_virtcon) {
|
|
|
|
default_virtcon = 0;
|
|
|
|
}
|
|
|
|
if (machine->no_vga) {
|
|
|
|
default_vga = 0;
|
|
|
|
}
|
2009-12-16 21:25:39 +08:00
|
|
|
if (machine->no_floppy) {
|
|
|
|
default_floppy = 0;
|
|
|
|
}
|
|
|
|
if (machine->no_cdrom) {
|
|
|
|
default_cdrom = 0;
|
|
|
|
}
|
|
|
|
if (machine->no_sdcard) {
|
|
|
|
default_sdcard = 0;
|
|
|
|
}
|
2009-12-08 20:11:54 +08:00
|
|
|
|
2009-05-22 05:54:00 +08:00
|
|
|
if (display_type == DT_NOGRAPHIC) {
|
2009-12-08 20:11:42 +08:00
|
|
|
if (default_parallel)
|
|
|
|
add_device_config(DEV_PARALLEL, "null");
|
2009-12-08 20:11:44 +08:00
|
|
|
if (default_serial && default_monitor) {
|
|
|
|
add_device_config(DEV_SERIAL, "mon:stdio");
|
2009-12-08 20:11:54 +08:00
|
|
|
} else if (default_virtcon && default_monitor) {
|
|
|
|
add_device_config(DEV_VIRTCON, "mon:stdio");
|
2009-12-08 20:11:44 +08:00
|
|
|
} else {
|
|
|
|
if (default_serial)
|
|
|
|
add_device_config(DEV_SERIAL, "stdio");
|
2009-12-08 20:11:54 +08:00
|
|
|
if (default_virtcon)
|
|
|
|
add_device_config(DEV_VIRTCON, "stdio");
|
2009-12-08 20:11:44 +08:00
|
|
|
if (default_monitor)
|
2009-12-08 20:11:52 +08:00
|
|
|
monitor_parse("stdio", "readline");
|
2009-12-08 20:11:44 +08:00
|
|
|
}
|
2009-12-08 20:11:41 +08:00
|
|
|
} else {
|
|
|
|
if (default_serial)
|
|
|
|
add_device_config(DEV_SERIAL, "vc:80Cx24C");
|
2009-12-08 20:11:42 +08:00
|
|
|
if (default_parallel)
|
|
|
|
add_device_config(DEV_PARALLEL, "vc:80Cx24C");
|
2009-12-08 20:11:43 +08:00
|
|
|
if (default_monitor)
|
2009-12-08 20:11:52 +08:00
|
|
|
monitor_parse("vc:80Cx24C", "readline");
|
2009-12-17 20:06:08 +08:00
|
|
|
if (default_virtcon)
|
|
|
|
add_device_config(DEV_VIRTCON, "vc:80Cx24C");
|
2008-08-01 23:12:34 +08:00
|
|
|
}
|
2009-12-08 20:11:45 +08:00
|
|
|
if (default_vga)
|
|
|
|
vga_interface_type = VGA_CIRRUS;
|
2008-08-01 23:12:34 +08:00
|
|
|
|
2009-12-08 20:11:36 +08:00
|
|
|
if (qemu_opts_foreach(&qemu_chardev_opts, chardev_init_func, NULL, 1) != 0)
|
|
|
|
exit(1);
|
|
|
|
|
2006-12-22 10:11:31 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
if (daemonize) {
|
|
|
|
pid_t pid;
|
|
|
|
|
|
|
|
if (pipe(fds) == -1)
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
pid = fork();
|
|
|
|
if (pid > 0) {
|
|
|
|
uint8_t status;
|
|
|
|
ssize_t len;
|
|
|
|
|
|
|
|
close(fds[1]);
|
|
|
|
|
|
|
|
again:
|
2007-03-19 23:58:31 +08:00
|
|
|
len = read(fds[0], &status, 1);
|
|
|
|
if (len == -1 && (errno == EINTR))
|
|
|
|
goto again;
|
|
|
|
|
|
|
|
if (len != 1)
|
|
|
|
exit(1);
|
|
|
|
else if (status == 1) {
|
2009-10-01 22:42:56 +08:00
|
|
|
fprintf(stderr, "Could not acquire pidfile: %s\n", strerror(errno));
|
2007-03-19 23:58:31 +08:00
|
|
|
exit(1);
|
|
|
|
} else
|
|
|
|
exit(0);
|
2006-12-22 10:11:31 +08:00
|
|
|
} else if (pid < 0)
|
2007-03-19 23:58:31 +08:00
|
|
|
exit(1);
|
2006-12-22 10:11:31 +08:00
|
|
|
|
2009-12-02 19:24:42 +08:00
|
|
|
close(fds[0]);
|
|
|
|
qemu_set_cloexec(fds[1]);
|
|
|
|
|
2006-12-22 10:11:31 +08:00
|
|
|
setsid();
|
|
|
|
|
|
|
|
pid = fork();
|
|
|
|
if (pid > 0)
|
|
|
|
exit(0);
|
|
|
|
else if (pid < 0)
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
umask(027);
|
|
|
|
|
|
|
|
signal(SIGTSTP, SIG_IGN);
|
|
|
|
signal(SIGTTOU, SIG_IGN);
|
|
|
|
signal(SIGTTIN, SIG_IGN);
|
|
|
|
}
|
2009-12-03 21:56:03 +08:00
|
|
|
#endif
|
2006-12-22 10:11:31 +08:00
|
|
|
|
2007-03-26 05:33:06 +08:00
|
|
|
if (pid_file && qemu_create_pidfile(pid_file) != 0) {
|
2009-12-03 21:56:03 +08:00
|
|
|
#ifndef _WIN32
|
2007-03-19 23:58:31 +08:00
|
|
|
if (daemonize) {
|
|
|
|
uint8_t status = 1;
|
2010-01-20 07:56:18 +08:00
|
|
|
if (write(fds[1], &status, 1) != 1) {
|
|
|
|
perror("daemonize. Writing to pipe\n");
|
|
|
|
}
|
2007-03-19 23:58:31 +08:00
|
|
|
} else
|
2009-12-03 21:56:03 +08:00
|
|
|
#endif
|
2009-10-01 22:42:56 +08:00
|
|
|
fprintf(stderr, "Could not acquire pid file: %s\n", strerror(errno));
|
2007-03-19 23:58:31 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2009-09-18 13:41:23 +08:00
|
|
|
if (kvm_enabled()) {
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = kvm_init(smp_cpus);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to initialize KVM\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-25 02:03:25 +08:00
|
|
|
if (qemu_init_main_loop()) {
|
|
|
|
fprintf(stderr, "qemu_init_main_loop failed\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2003-10-01 05:07:02 +08:00
|
|
|
linux_boot = (kernel_filename != NULL);
|
2007-11-17 20:12:29 +08:00
|
|
|
|
2008-07-03 18:01:15 +08:00
|
|
|
if (!linux_boot && *kernel_cmdline != '\0') {
|
|
|
|
fprintf(stderr, "-append only allowed with -kernel option\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!linux_boot && initrd_filename != NULL) {
|
|
|
|
fprintf(stderr, "-initrd only allowed with -kernel option\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2009-07-27 23:02:04 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
/* Win32 doesn't support line-buffering and requires size >= 2 */
|
2003-07-01 07:36:21 +08:00
|
|
|
setvbuf(stdout, NULL, _IOLBF, 0);
|
2009-07-27 23:02:04 +08:00
|
|
|
#endif
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2008-11-06 04:40:18 +08:00
|
|
|
if (init_timer_alarm() < 0) {
|
|
|
|
fprintf(stderr, "could not initialize alarm timer\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-03-10 18:38:48 +08:00
|
|
|
configure_icount(icount_option);
|
2006-07-16 01:40:09 +08:00
|
|
|
|
2006-02-02 05:29:26 +08:00
|
|
|
#ifdef _WIN32
|
|
|
|
socket_init();
|
|
|
|
#endif
|
|
|
|
|
2009-10-06 19:17:16 +08:00
|
|
|
if (net_init_clients() < 0) {
|
|
|
|
exit(1);
|
2004-04-03 05:21:32 +08:00
|
|
|
}
|
2003-06-25 08:07:40 +08:00
|
|
|
|
2009-06-17 21:05:30 +08:00
|
|
|
net_boot = (boot_devices_bitmap >> ('n' - 'a')) & 0xF;
|
|
|
|
net_set_boot_mask(net_boot);
|
|
|
|
|
2008-11-09 08:04:26 +08:00
|
|
|
/* init the bluetooth world */
|
2009-07-15 19:59:26 +08:00
|
|
|
if (foreach_device_config(DEV_BT, bt_parse))
|
|
|
|
exit(1);
|
2008-11-09 08:04:26 +08:00
|
|
|
|
2003-06-24 21:42:40 +08:00
|
|
|
/* init the memory */
|
2009-04-12 01:15:54 +08:00
|
|
|
if (ram_size == 0)
|
|
|
|
ram_size = DEFAULT_RAM_SIZE * 1024 * 1024;
|
2007-01-06 01:39:04 +08:00
|
|
|
|
2008-05-28 20:30:31 +08:00
|
|
|
/* init the dynamic translator */
|
|
|
|
cpu_exec_init_all(tb_size * 1024 * 1024);
|
|
|
|
|
2009-10-28 01:41:44 +08:00
|
|
|
bdrv_init_with_whitelist();
|
2004-03-15 05:44:30 +08:00
|
|
|
|
2009-11-02 21:40:58 +08:00
|
|
|
blk_mig_init();
|
|
|
|
|
2009-12-16 21:25:39 +08:00
|
|
|
if (default_cdrom) {
|
2009-12-08 20:11:48 +08:00
|
|
|
/* we always create the cdrom drive, even if no disk is there */
|
|
|
|
drive_add(NULL, CDROM_ALIAS);
|
2009-12-16 21:25:39 +08:00
|
|
|
}
|
2004-03-15 05:44:30 +08:00
|
|
|
|
2009-12-16 21:25:39 +08:00
|
|
|
if (default_floppy) {
|
2009-12-08 20:11:48 +08:00
|
|
|
/* we always create at least one floppy */
|
|
|
|
drive_add(NULL, FD_ALIAS, 0);
|
2009-12-16 21:25:39 +08:00
|
|
|
}
|
2007-04-24 14:52:59 +08:00
|
|
|
|
2009-12-16 21:25:39 +08:00
|
|
|
if (default_sdcard) {
|
2009-12-08 20:11:48 +08:00
|
|
|
/* we always create one sd slot, even if no card is in it */
|
|
|
|
drive_add(NULL, SD_ALIAS);
|
|
|
|
}
|
2007-12-04 08:10:34 +08:00
|
|
|
|
2007-12-02 12:51:10 +08:00
|
|
|
/* open the virtual block devices */
|
2009-07-22 22:43:04 +08:00
|
|
|
if (snapshot)
|
2009-07-31 18:25:35 +08:00
|
|
|
qemu_opts_foreach(&qemu_drive_opts, drive_enable_snapshot, NULL, 0);
|
|
|
|
if (qemu_opts_foreach(&qemu_drive_opts, drive_init_func, machine, 1) != 0)
|
2009-07-22 22:43:04 +08:00
|
|
|
exit(1);
|
2007-04-30 10:09:25 +08:00
|
|
|
|
2009-11-02 21:40:58 +08:00
|
|
|
register_savevm_live("ram", 0, 3, NULL, ram_save_live, NULL,
|
|
|
|
ram_load, NULL);
|
2004-04-01 03:00:16 +08:00
|
|
|
|
2009-04-22 06:30:27 +08:00
|
|
|
if (nb_numa_nodes > 0) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (nb_numa_nodes > smp_cpus) {
|
|
|
|
nb_numa_nodes = smp_cpus;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If no memory size if given for any node, assume the default case
|
|
|
|
* and distribute the available memory equally across all nodes
|
|
|
|
*/
|
|
|
|
for (i = 0; i < nb_numa_nodes; i++) {
|
|
|
|
if (node_mem[i] != 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (i == nb_numa_nodes) {
|
|
|
|
uint64_t usedmem = 0;
|
|
|
|
|
|
|
|
/* On Linux, the each node's border has to be 8MB aligned,
|
|
|
|
* the final node gets the rest.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < nb_numa_nodes - 1; i++) {
|
|
|
|
node_mem[i] = (ram_size / nb_numa_nodes) & ~((1 << 23UL) - 1);
|
|
|
|
usedmem += node_mem[i];
|
|
|
|
}
|
|
|
|
node_mem[i] = ram_size - usedmem;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < nb_numa_nodes; i++) {
|
|
|
|
if (node_cpumask[i] != 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* assigning the VCPUs round-robin is easier to implement, guest OSes
|
|
|
|
* must cope with this anyway, because there are BIOSes out there in
|
|
|
|
* real machines which also use this scheme.
|
|
|
|
*/
|
|
|
|
if (i == nb_numa_nodes) {
|
|
|
|
for (i = 0; i < smp_cpus; i++) {
|
|
|
|
node_cpumask[i % nb_numa_nodes] |= 1 << i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-08 20:11:41 +08:00
|
|
|
if (foreach_device_config(DEV_SERIAL, serial_parse) < 0)
|
|
|
|
exit(1);
|
2009-12-08 20:11:42 +08:00
|
|
|
if (foreach_device_config(DEV_PARALLEL, parallel_parse) < 0)
|
|
|
|
exit(1);
|
2009-12-08 20:11:53 +08:00
|
|
|
if (foreach_device_config(DEV_VIRTCON, virtcon_parse) < 0)
|
|
|
|
exit(1);
|
debugcon: support for debugging consoles (e.g. Bochs port 0xe9)
Add generic support for debugging consoles (simple I/O ports which
when written to cause debugging output to be written to a target.)
The current implementation matches Bochs' port 0xe9, allowing the same
debugging code to be used for both Bochs and Qemu.
There is no vm state associated with the debugging port, simply
because it has none -- the entire interface is a single, stateless,
write-only port.
Most of the code was cribbed from the serial port driver.
v2: removed non-ISA variants (they can be introduced when/if someone
wants them, using code from the serial port); added configurable
readback (Bochs returns 0xe9 on a read from this register, mimic that
by default) This retains the apparently somewhat controversial user
friendly option, however.
v3: reimplemented the user friendly option as a synthetic option
("-debugcon foo" basically ends up being a parser-level shorthand for
"-chardev stdio,id=debugcon -device isa-debugcon,chardev=debugcon") --
this dramatically reduced the complexity while keeping the same level
of user friendliness.
v4: spaces, not tabs.
v5: update to match current top of tree. Calling qemu_chr_open()
already during parsing no longer works; defer until we are parsing the
other console-like devices.
Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-12-30 05:51:36 +08:00
|
|
|
if (foreach_device_config(DEV_DEBUGCON, debugcon_parse) < 0)
|
|
|
|
exit(1);
|
2009-01-17 04:23:27 +08:00
|
|
|
|
2009-05-15 05:35:06 +08:00
|
|
|
module_call_init(MODULE_INIT_DEVICE);
|
|
|
|
|
2010-01-30 02:48:57 +08:00
|
|
|
if (qemu_opts_foreach(&qemu_device_opts, device_help_func, NULL, 0) != 0)
|
|
|
|
exit(0);
|
|
|
|
|
2009-08-21 16:31:34 +08:00
|
|
|
if (watchdog) {
|
|
|
|
i = select_watchdog(watchdog);
|
|
|
|
if (i > 0)
|
|
|
|
exit (i == 1 ? 1 : 0);
|
|
|
|
}
|
|
|
|
|
2009-07-15 19:48:21 +08:00
|
|
|
if (machine->compat_props) {
|
2009-12-08 20:11:33 +08:00
|
|
|
qdev_prop_register_global_list(machine->compat_props);
|
2009-07-15 19:48:21 +08:00
|
|
|
}
|
2009-12-08 20:11:34 +08:00
|
|
|
qemu_add_globals();
|
|
|
|
|
2009-05-14 00:56:25 +08:00
|
|
|
machine->init(ram_size, boot_devices,
|
2009-01-17 03:04:14 +08:00
|
|
|
kernel_filename, kernel_cmdline, initrd_filename, cpu_model);
|
|
|
|
|
2010-03-02 02:10:30 +08:00
|
|
|
cpu_synchronize_all_post_init();
|
2009-04-22 06:30:27 +08:00
|
|
|
|
2009-08-29 01:25:15 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
/* must be after terminal init, SDL library changes signal handlers */
|
|
|
|
sighandler_setup();
|
|
|
|
#endif
|
|
|
|
|
2009-04-22 06:30:27 +08:00
|
|
|
for (env = first_cpu; env != NULL; env = env->next_cpu) {
|
|
|
|
for (i = 0; i < nb_numa_nodes; i++) {
|
|
|
|
if (node_cpumask[i] & (1 << env->cpu_index)) {
|
|
|
|
env->numa_node = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-02-11 23:21:54 +08:00
|
|
|
current_machine = machine;
|
|
|
|
|
2009-01-17 03:04:14 +08:00
|
|
|
/* init USB devices */
|
|
|
|
if (usb_enabled) {
|
Don't exit() in config_error()
Propagating errors up the call chain is tedious. In startup code, we
can take a shortcut: terminate the program. This is wrong elsewhere,
the monitor in particular.
config_error() tries to cater for both customers: it terminates the
program unless its mon parameter tells it it's working for the
monitor.
Its users need to return status anyway (unless passing a null mon
argument, which none do), which their users need to check. So this
automatic exit buys us exactly nothing useful. Only the dangerous
delusion that we can get away without returning status. Some of its
users fell for that. Their callers continue executing after failure
when working for the monitor.
This bites monitor command host_net_add in two places:
* net_slirp_init() continues after slirp_hostfwd(), slirp_guestfwd(),
or slirp_smb() failed, and may end up reporting success. This
happens for "host_net_add user guestfwd=foo": it complains about the
invalid guest forwarding rule, then happily creates the user network
without guest forwarding.
* net_client_init() can't detect slirp_guestfwd() failure, and gets
fooled by net_slirp_init() lying about success. Suppresses its
"Could not initialize device" message.
Add the missing error reporting, make sure errors are checked, and
drop the exit() from config_error().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Mark McLoughlin <markmc@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2009-10-06 19:16:57 +08:00
|
|
|
if (foreach_device_config(DEV_USB, usb_parse) < 0)
|
|
|
|
exit(1);
|
2009-01-17 03:04:14 +08:00
|
|
|
}
|
|
|
|
|
2009-07-15 19:59:26 +08:00
|
|
|
/* init generic devices */
|
2009-07-31 18:25:37 +08:00
|
|
|
if (qemu_opts_foreach(&qemu_device_opts, device_init_func, NULL, 1) != 0)
|
2009-07-15 19:59:26 +08:00
|
|
|
exit(1);
|
|
|
|
|
2010-02-11 21:44:58 +08:00
|
|
|
net_check_clients();
|
|
|
|
|
2009-01-17 03:04:14 +08:00
|
|
|
/* just use the first displaystate for the moment */
|
2010-02-11 07:29:55 +08:00
|
|
|
ds = get_displaystate();
|
2009-05-22 05:54:00 +08:00
|
|
|
|
|
|
|
if (display_type == DT_DEFAULT) {
|
|
|
|
#if defined(CONFIG_SDL) || defined(CONFIG_COCOA)
|
|
|
|
display_type = DT_SDL;
|
|
|
|
#else
|
|
|
|
display_type = DT_VNC;
|
|
|
|
vnc_display = "localhost:0,to=99";
|
|
|
|
show_vnc_port = 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
switch (display_type) {
|
|
|
|
case DT_NOGRAPHIC:
|
|
|
|
break;
|
2008-02-11 00:33:14 +08:00
|
|
|
#if defined(CONFIG_CURSES)
|
2009-05-22 05:54:00 +08:00
|
|
|
case DT_CURSES:
|
|
|
|
curses_display_init(ds, full_screen);
|
|
|
|
break;
|
2008-02-11 00:33:14 +08:00
|
|
|
#endif
|
2005-03-02 05:37:28 +08:00
|
|
|
#if defined(CONFIG_SDL)
|
2009-05-22 05:54:00 +08:00
|
|
|
case DT_SDL:
|
|
|
|
sdl_display_init(ds, full_screen, no_frame);
|
|
|
|
break;
|
2005-03-02 05:37:28 +08:00
|
|
|
#elif defined(CONFIG_COCOA)
|
2009-05-22 05:54:00 +08:00
|
|
|
case DT_SDL:
|
|
|
|
cocoa_display_init(ds, full_screen);
|
|
|
|
break;
|
2003-08-11 05:52:11 +08:00
|
|
|
#endif
|
2009-05-22 05:54:00 +08:00
|
|
|
case DT_VNC:
|
|
|
|
vnc_display_init(ds);
|
|
|
|
if (vnc_display_open(ds, vnc_display) < 0)
|
|
|
|
exit(1);
|
2009-05-21 02:01:02 +08:00
|
|
|
|
2009-05-22 05:54:00 +08:00
|
|
|
if (show_vnc_port) {
|
|
|
|
printf("VNC server running on `%s'\n", vnc_display_local_addr(ds));
|
2009-05-21 02:01:02 +08:00
|
|
|
}
|
2009-05-22 05:54:00 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2003-08-11 05:52:11 +08:00
|
|
|
}
|
2009-01-16 06:14:11 +08:00
|
|
|
dpy_resize(ds);
|
2008-08-22 04:08:03 +08:00
|
|
|
|
2009-01-17 03:04:14 +08:00
|
|
|
dcl = ds->listeners;
|
|
|
|
while (dcl != NULL) {
|
|
|
|
if (dcl->dpy_refresh != NULL) {
|
|
|
|
ds->gui_timer = qemu_new_timer(rt_clock, gui_update, ds);
|
|
|
|
qemu_mod_timer(ds->gui_timer, qemu_get_clock(rt_clock));
|
2007-02-19 01:04:49 +08:00
|
|
|
}
|
2009-01-17 03:04:14 +08:00
|
|
|
dcl = dcl->next;
|
2007-02-19 01:04:49 +08:00
|
|
|
}
|
2009-01-17 03:04:14 +08:00
|
|
|
|
2009-05-22 05:54:00 +08:00
|
|
|
if (display_type == DT_NOGRAPHIC || display_type == DT_VNC) {
|
2009-01-22 03:28:13 +08:00
|
|
|
nographic_timer = qemu_new_timer(rt_clock, nographic_update, NULL);
|
|
|
|
qemu_mod_timer(nographic_timer, qemu_get_clock(rt_clock));
|
|
|
|
}
|
|
|
|
|
2010-02-11 07:29:55 +08:00
|
|
|
text_consoles_set_display(ds);
|
2009-01-17 04:23:27 +08:00
|
|
|
|
2009-12-08 20:11:50 +08:00
|
|
|
if (qemu_opts_foreach(&qemu_mon_opts, mon_init_func, NULL, 1) != 0)
|
|
|
|
exit(1);
|
2004-07-15 01:28:13 +08:00
|
|
|
|
2009-04-06 02:43:41 +08:00
|
|
|
if (gdbstub_dev && gdbserver_start(gdbstub_dev) < 0) {
|
|
|
|
fprintf(stderr, "qemu: could not open gdbserver on device '%s'\n",
|
|
|
|
gdbstub_dev);
|
|
|
|
exit(1);
|
2007-07-02 21:20:17 +08:00
|
|
|
}
|
|
|
|
|
2009-09-26 03:42:41 +08:00
|
|
|
qdev_machine_creation_done();
|
|
|
|
|
2009-12-14 23:07:35 +08:00
|
|
|
if (rom_load_all() != 0) {
|
|
|
|
fprintf(stderr, "rom loading failed\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-01 22:42:33 +08:00
|
|
|
|
2009-11-12 07:39:13 +08:00
|
|
|
qemu_system_reset();
|
2009-08-21 01:42:22 +08:00
|
|
|
if (loadvm) {
|
2010-02-17 23:24:10 +08:00
|
|
|
if (load_vmstate(loadvm) < 0) {
|
2009-08-21 01:42:22 +08:00
|
|
|
autostart = 0;
|
|
|
|
}
|
|
|
|
}
|
2004-10-03 21:29:03 +08:00
|
|
|
|
2009-07-25 04:20:23 +08:00
|
|
|
if (incoming) {
|
2008-10-13 11:12:02 +08:00
|
|
|
qemu_start_incoming_migration(incoming);
|
2009-08-09 19:39:20 +08:00
|
|
|
} else if (autostart) {
|
2009-03-06 07:01:01 +08:00
|
|
|
vm_start();
|
2009-08-09 19:39:20 +08:00
|
|
|
}
|
2006-12-22 03:46:43 +08:00
|
|
|
|
2009-04-06 02:03:31 +08:00
|
|
|
#ifndef _WIN32
|
2006-12-22 10:11:31 +08:00
|
|
|
if (daemonize) {
|
|
|
|
uint8_t status = 0;
|
|
|
|
ssize_t len;
|
|
|
|
|
|
|
|
again1:
|
|
|
|
len = write(fds[1], &status, 1);
|
|
|
|
if (len == -1 && (errno == EINTR))
|
|
|
|
goto again1;
|
|
|
|
|
|
|
|
if (len != 1)
|
|
|
|
exit(1);
|
|
|
|
|
2010-01-20 07:56:18 +08:00
|
|
|
if (chdir("/")) {
|
|
|
|
perror("not able to chdir to /");
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-12-02 19:24:42 +08:00
|
|
|
TFR(fd = qemu_open("/dev/null", O_RDWR));
|
2006-12-22 10:11:31 +08:00
|
|
|
if (fd == -1)
|
|
|
|
exit(1);
|
2009-02-28 06:09:45 +08:00
|
|
|
}
|
2006-12-22 10:11:31 +08:00
|
|
|
|
2009-02-28 06:09:45 +08:00
|
|
|
if (run_as) {
|
|
|
|
pwd = getpwnam(run_as);
|
|
|
|
if (!pwd) {
|
|
|
|
fprintf(stderr, "User \"%s\" doesn't exist\n", run_as);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (chroot_dir) {
|
|
|
|
if (chroot(chroot_dir) < 0) {
|
|
|
|
fprintf(stderr, "chroot failed\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-01-20 07:56:18 +08:00
|
|
|
if (chdir("/")) {
|
|
|
|
perror("not able to chdir to /");
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-02-28 06:09:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (run_as) {
|
|
|
|
if (setgid(pwd->pw_gid) < 0) {
|
|
|
|
fprintf(stderr, "Failed to setgid(%d)\n", pwd->pw_gid);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (setuid(pwd->pw_uid) < 0) {
|
|
|
|
fprintf(stderr, "Failed to setuid(%d)\n", pwd->pw_uid);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (setuid(0) != -1) {
|
|
|
|
fprintf(stderr, "Dropping privileges failed\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (daemonize) {
|
|
|
|
dup2(fd, 0);
|
|
|
|
dup2(fd, 1);
|
|
|
|
dup2(fd, 2);
|
2006-12-22 10:11:31 +08:00
|
|
|
|
2009-02-28 06:09:45 +08:00
|
|
|
close(fd);
|
2006-12-22 10:11:31 +08:00
|
|
|
}
|
2009-04-06 02:03:31 +08:00
|
|
|
#endif
|
2006-12-22 10:11:31 +08:00
|
|
|
|
2004-04-01 03:00:16 +08:00
|
|
|
main_loop();
|
2004-04-04 20:56:28 +08:00
|
|
|
quit_timers();
|
2008-11-01 03:10:00 +08:00
|
|
|
net_cleanup();
|
2007-10-22 07:20:45 +08:00
|
|
|
|
2003-06-24 21:42:40 +08:00
|
|
|
return 0;
|
|
|
|
}
|