2010-03-30 03:23:50 +08:00
|
|
|
/*
|
|
|
|
* QEMU System Emulator
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003-2008 Fabrice Bellard
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Needed early for CONFIG_BSD etc. */
|
|
|
|
#include "config-host.h"
|
|
|
|
|
|
|
|
#include "monitor.h"
|
|
|
|
#include "sysemu.h"
|
|
|
|
#include "gdbstub.h"
|
|
|
|
#include "dma.h"
|
|
|
|
#include "kvm.h"
|
|
|
|
|
2011-03-13 00:43:53 +08:00
|
|
|
#include "qemu-thread.h"
|
2010-03-30 03:23:50 +08:00
|
|
|
#include "cpus.h"
|
2011-06-23 16:15:55 +08:00
|
|
|
|
|
|
|
#ifndef _WIN32
|
2010-10-12 02:31:16 +08:00
|
|
|
#include "compatfd.h"
|
2011-06-23 16:15:55 +08:00
|
|
|
#endif
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2010-04-13 01:19:06 +08:00
|
|
|
#ifdef SIGRTMIN
|
|
|
|
#define SIG_IPI (SIGRTMIN+4)
|
|
|
|
#else
|
|
|
|
#define SIG_IPI SIGUSR1
|
|
|
|
#endif
|
|
|
|
|
2011-02-02 05:15:58 +08:00
|
|
|
#ifdef CONFIG_LINUX
|
|
|
|
|
|
|
|
#include <sys/prctl.h>
|
|
|
|
|
2010-10-12 02:31:21 +08:00
|
|
|
#ifndef PR_MCE_KILL
|
|
|
|
#define PR_MCE_KILL 33
|
|
|
|
#endif
|
|
|
|
|
2011-02-02 05:15:58 +08:00
|
|
|
#ifndef PR_MCE_KILL_SET
|
|
|
|
#define PR_MCE_KILL_SET 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef PR_MCE_KILL_EARLY
|
|
|
|
#define PR_MCE_KILL_EARLY 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* CONFIG_LINUX */
|
|
|
|
|
2010-03-30 03:23:50 +08:00
|
|
|
static CPUState *next_cpu;
|
|
|
|
|
|
|
|
/***********************************************************/
|
|
|
|
void hw_error(const char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
CPUState *env;
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
fprintf(stderr, "qemu: hardware error: ");
|
|
|
|
vfprintf(stderr, fmt, ap);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
for(env = first_cpu; env != NULL; env = env->next_cpu) {
|
|
|
|
fprintf(stderr, "CPU #%d:\n", env->cpu_index);
|
|
|
|
#ifdef TARGET_I386
|
|
|
|
cpu_dump_state(env, stderr, fprintf, X86_DUMP_FPU);
|
|
|
|
#else
|
|
|
|
cpu_dump_state(env, stderr, fprintf, 0);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
va_end(ap);
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-04 20:45:24 +08:00
|
|
|
int cpu_is_stopped(CPUState *env)
|
|
|
|
{
|
|
|
|
return !vm_running || env->stopped;
|
|
|
|
}
|
|
|
|
|
2011-07-30 01:26:33 +08:00
|
|
|
static void do_vm_stop(RunState state)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
if (vm_running) {
|
|
|
|
cpu_disable_ticks();
|
|
|
|
vm_running = 0;
|
|
|
|
pause_all_vcpus();
|
2011-07-30 02:04:45 +08:00
|
|
|
runstate_set(state);
|
2011-07-30 01:26:33 +08:00
|
|
|
vm_state_notify(0, state);
|
2010-11-23 01:52:22 +08:00
|
|
|
qemu_aio_flush();
|
|
|
|
bdrv_flush_all();
|
2010-03-30 03:23:50 +08:00
|
|
|
monitor_protocol_event(QEVENT_STOP, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int cpu_can_run(CPUState *env)
|
|
|
|
{
|
2011-02-07 19:19:14 +08:00
|
|
|
if (env->stop) {
|
2010-03-30 03:23:50 +08:00
|
|
|
return 0;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
|
|
|
if (env->stopped || !vm_running) {
|
2010-03-30 03:23:50 +08:00
|
|
|
return 0;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-02-09 23:29:37 +08:00
|
|
|
static bool cpu_thread_is_idle(CPUState *env)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2011-02-09 23:29:37 +08:00
|
|
|
if (env->stop || env->queued_work_first) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (env->stopped || !vm_running) {
|
|
|
|
return true;
|
|
|
|
}
|
2011-03-15 19:26:18 +08:00
|
|
|
if (!env->halted || qemu_cpu_has_work(env) ||
|
|
|
|
(kvm_enabled() && kvm_irqchip_in_kernel())) {
|
2011-02-09 23:29:37 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
enable vm_clock to "warp" in the iothread+icount case
The previous patch however is not enough, because if the virtual CPU
goes to sleep waiting for a future timer interrupt to wake it up, qemu
deadlocks. The timer interrupt never comes because time is driven by
icount, but the vCPU doesn't run any insns.
You could say that VCPUs should never go to sleep in icount
mode if there is a pending vm_clock timer; rather time should
just warp to the next vm_clock event with no sleep ever taking place.
Even better, you can sleep for some time related to the
time left until the next event, to avoid that the warps are too visible
externally; for example, you could be sending network packets continously
instead of every 100ms.
This is what this patch implements. qemu_clock_warp is called: 1)
whenever a vm_clock timer is adjusted, to ensure the warp_timer is
synchronized; 2) at strategic points in the CPU thread, to make sure
the insn counter is synchronized before the CPU starts running.
In any case, the warp_timer is disabled while the CPU is running,
because the insn counter will then be making progress on its own.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Tested-by: Edgar E. Iglesias <edgar.iglesias@gmail.com>
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@gmail.com>
2011-04-13 16:03:44 +08:00
|
|
|
bool all_cpu_threads_idle(void)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
CPUState *env;
|
|
|
|
|
2011-02-09 23:29:37 +08:00
|
|
|
for (env = first_cpu; env != NULL; env = env->next_cpu) {
|
|
|
|
if (!cpu_thread_is_idle(env)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
2011-03-15 19:26:13 +08:00
|
|
|
static void cpu_handle_guest_debug(CPUState *env)
|
2011-02-07 19:19:17 +08:00
|
|
|
{
|
2010-06-25 22:56:56 +08:00
|
|
|
gdb_set_stop_cpu(env);
|
2011-02-07 19:19:16 +08:00
|
|
|
qemu_system_debug_request();
|
2011-02-07 19:19:17 +08:00
|
|
|
env->stopped = 1;
|
2010-06-25 22:56:56 +08:00
|
|
|
}
|
|
|
|
|
2011-03-13 00:44:06 +08:00
|
|
|
static void cpu_signal(int sig)
|
|
|
|
{
|
|
|
|
if (cpu_single_env) {
|
|
|
|
cpu_exit(cpu_single_env);
|
|
|
|
}
|
|
|
|
exit_request = 1;
|
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:58 +08:00
|
|
|
#ifdef CONFIG_LINUX
|
|
|
|
static void sigbus_reraise(void)
|
|
|
|
{
|
|
|
|
sigset_t set;
|
|
|
|
struct sigaction action;
|
|
|
|
|
|
|
|
memset(&action, 0, sizeof(action));
|
|
|
|
action.sa_handler = SIG_DFL;
|
|
|
|
if (!sigaction(SIGBUS, &action, NULL)) {
|
|
|
|
raise(SIGBUS);
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGBUS);
|
|
|
|
sigprocmask(SIG_UNBLOCK, &set, NULL);
|
|
|
|
}
|
|
|
|
perror("Failed to re-raise SIGBUS!\n");
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sigbus_handler(int n, struct qemu_signalfd_siginfo *siginfo,
|
|
|
|
void *ctx)
|
|
|
|
{
|
|
|
|
if (kvm_on_sigbus(siginfo->ssi_code,
|
|
|
|
(void *)(intptr_t)siginfo->ssi_addr)) {
|
|
|
|
sigbus_reraise();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_init_sigbus(void)
|
|
|
|
{
|
|
|
|
struct sigaction action;
|
|
|
|
|
|
|
|
memset(&action, 0, sizeof(action));
|
|
|
|
action.sa_flags = SA_SIGINFO;
|
|
|
|
action.sa_sigaction = (void (*)(int, siginfo_t*, void*))sigbus_handler;
|
|
|
|
sigaction(SIGBUS, &action, NULL);
|
|
|
|
|
|
|
|
prctl(PR_MCE_KILL, PR_MCE_KILL_SET, PR_MCE_KILL_EARLY, 0, 0);
|
|
|
|
}
|
|
|
|
|
2011-03-15 19:26:12 +08:00
|
|
|
static void qemu_kvm_eat_signals(CPUState *env)
|
|
|
|
{
|
|
|
|
struct timespec ts = { 0, 0 };
|
|
|
|
siginfo_t siginfo;
|
|
|
|
sigset_t waitset;
|
|
|
|
sigset_t chkset;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
sigemptyset(&waitset);
|
|
|
|
sigaddset(&waitset, SIG_IPI);
|
|
|
|
sigaddset(&waitset, SIGBUS);
|
|
|
|
|
|
|
|
do {
|
|
|
|
r = sigtimedwait(&waitset, &siginfo, &ts);
|
|
|
|
if (r == -1 && !(errno == EAGAIN || errno == EINTR)) {
|
|
|
|
perror("sigtimedwait");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (r) {
|
|
|
|
case SIGBUS:
|
|
|
|
if (kvm_on_sigbus_vcpu(env, siginfo.si_code, siginfo.si_addr)) {
|
|
|
|
sigbus_reraise();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
r = sigpending(&chkset);
|
|
|
|
if (r == -1) {
|
|
|
|
perror("sigpending");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
} while (sigismember(&chkset, SIG_IPI) || sigismember(&chkset, SIGBUS));
|
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:58 +08:00
|
|
|
#else /* !CONFIG_LINUX */
|
|
|
|
|
|
|
|
static void qemu_init_sigbus(void)
|
|
|
|
{
|
|
|
|
}
|
2011-03-15 19:26:12 +08:00
|
|
|
|
|
|
|
static void qemu_kvm_eat_signals(CPUState *env)
|
|
|
|
{
|
|
|
|
}
|
2011-02-02 05:15:58 +08:00
|
|
|
#endif /* !CONFIG_LINUX */
|
|
|
|
|
2010-03-30 03:23:50 +08:00
|
|
|
#ifndef _WIN32
|
|
|
|
static int io_thread_fd = -1;
|
|
|
|
|
|
|
|
static void qemu_event_increment(void)
|
|
|
|
{
|
|
|
|
/* Write 8 bytes to be compatible with eventfd. */
|
2010-05-15 03:32:21 +08:00
|
|
|
static const uint64_t val = 1;
|
2010-03-30 03:23:50 +08:00
|
|
|
ssize_t ret;
|
|
|
|
|
2011-02-07 19:19:14 +08:00
|
|
|
if (io_thread_fd == -1) {
|
2010-03-30 03:23:50 +08:00
|
|
|
return;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
do {
|
|
|
|
ret = write(io_thread_fd, &val, sizeof(val));
|
|
|
|
} while (ret < 0 && errno == EINTR);
|
|
|
|
|
|
|
|
/* EAGAIN is fine, a read must be pending. */
|
|
|
|
if (ret < 0 && errno != EAGAIN) {
|
2011-06-14 10:16:36 +08:00
|
|
|
fprintf(stderr, "qemu_event_increment: write() failed: %s\n",
|
2010-03-30 03:23:50 +08:00
|
|
|
strerror(errno));
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_event_read(void *opaque)
|
|
|
|
{
|
2011-02-24 02:09:16 +08:00
|
|
|
int fd = (intptr_t)opaque;
|
2010-03-30 03:23:50 +08:00
|
|
|
ssize_t len;
|
|
|
|
char buffer[512];
|
|
|
|
|
|
|
|
/* Drain the notify pipe. For eventfd, only 8 bytes will be read. */
|
|
|
|
do {
|
|
|
|
len = read(fd, buffer, sizeof(buffer));
|
|
|
|
} while ((len == -1 && errno == EINTR) || len == sizeof(buffer));
|
|
|
|
}
|
|
|
|
|
|
|
|
static int qemu_event_init(void)
|
|
|
|
{
|
|
|
|
int err;
|
|
|
|
int fds[2];
|
|
|
|
|
|
|
|
err = qemu_eventfd(fds);
|
2011-02-07 19:19:14 +08:00
|
|
|
if (err == -1) {
|
2010-03-30 03:23:50 +08:00
|
|
|
return -errno;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
err = fcntl_setfl(fds[0], O_NONBLOCK);
|
2011-02-07 19:19:14 +08:00
|
|
|
if (err < 0) {
|
2010-03-30 03:23:50 +08:00
|
|
|
goto fail;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
err = fcntl_setfl(fds[1], O_NONBLOCK);
|
2011-02-07 19:19:14 +08:00
|
|
|
if (err < 0) {
|
2010-03-30 03:23:50 +08:00
|
|
|
goto fail;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_set_fd_handler2(fds[0], NULL, qemu_event_read, NULL,
|
2011-02-24 02:09:16 +08:00
|
|
|
(void *)(intptr_t)fds[0]);
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
io_thread_fd = fds[1];
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
close(fds[0]);
|
|
|
|
close(fds[1]);
|
|
|
|
return err;
|
|
|
|
}
|
2011-02-02 05:15:52 +08:00
|
|
|
|
|
|
|
static void dummy_signal(int sig)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:56 +08:00
|
|
|
/* If we have signalfd, we mask out the signals we want to handle and then
|
|
|
|
* use signalfd to listen for them. We rely on whatever the current signal
|
|
|
|
* handler is to dispatch the signals when we receive them.
|
|
|
|
*/
|
|
|
|
static void sigfd_handler(void *opaque)
|
|
|
|
{
|
2011-02-24 02:09:16 +08:00
|
|
|
int fd = (intptr_t)opaque;
|
2011-02-02 05:15:56 +08:00
|
|
|
struct qemu_signalfd_siginfo info;
|
|
|
|
struct sigaction action;
|
|
|
|
ssize_t len;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
do {
|
|
|
|
len = read(fd, &info, sizeof(info));
|
|
|
|
} while (len == -1 && errno == EINTR);
|
|
|
|
|
|
|
|
if (len == -1 && errno == EAGAIN) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len != sizeof(info)) {
|
|
|
|
printf("read from sigfd returned %zd: %m\n", len);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sigaction(info.ssi_signo, NULL, &action);
|
|
|
|
if ((action.sa_flags & SA_SIGINFO) && action.sa_sigaction) {
|
|
|
|
action.sa_sigaction(info.ssi_signo,
|
|
|
|
(siginfo_t *)&info, NULL);
|
|
|
|
} else if (action.sa_handler) {
|
|
|
|
action.sa_handler(info.ssi_signo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-13 00:44:05 +08:00
|
|
|
static int qemu_signal_init(void)
|
2011-02-02 05:15:56 +08:00
|
|
|
{
|
|
|
|
int sigfd;
|
2011-03-13 00:44:05 +08:00
|
|
|
sigset_t set;
|
2011-02-02 05:15:56 +08:00
|
|
|
|
2011-03-13 00:44:05 +08:00
|
|
|
/* SIGUSR2 used by posix-aio-compat.c */
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGUSR2);
|
|
|
|
pthread_sigmask(SIG_UNBLOCK, &set, NULL);
|
|
|
|
|
2011-06-15 13:20:31 +08:00
|
|
|
/*
|
|
|
|
* SIG_IPI must be blocked in the main thread and must not be caught
|
|
|
|
* by sigwait() in the signal thread. Otherwise, the cpu thread will
|
|
|
|
* not catch it reliably.
|
|
|
|
*/
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIG_IPI);
|
|
|
|
pthread_sigmask(SIG_BLOCK, &set, NULL);
|
|
|
|
|
2011-03-13 00:44:05 +08:00
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIGIO);
|
|
|
|
sigaddset(&set, SIGALRM);
|
|
|
|
sigaddset(&set, SIGBUS);
|
2011-06-14 22:05:36 +08:00
|
|
|
pthread_sigmask(SIG_BLOCK, &set, NULL);
|
2011-03-13 00:44:05 +08:00
|
|
|
|
|
|
|
sigfd = qemu_signalfd(&set);
|
2011-02-02 05:15:56 +08:00
|
|
|
if (sigfd == -1) {
|
|
|
|
fprintf(stderr, "failed to create signalfd\n");
|
|
|
|
return -errno;
|
|
|
|
}
|
|
|
|
|
|
|
|
fcntl_setfl(sigfd, O_NONBLOCK);
|
|
|
|
|
|
|
|
qemu_set_fd_handler2(sigfd, NULL, sigfd_handler, NULL,
|
2011-02-24 02:09:16 +08:00
|
|
|
(void *)(intptr_t)sigfd);
|
2011-02-02 05:15:56 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-03-13 00:44:06 +08:00
|
|
|
static void qemu_kvm_init_cpu_signals(CPUState *env)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
sigset_t set;
|
|
|
|
struct sigaction sigact;
|
|
|
|
|
|
|
|
memset(&sigact, 0, sizeof(sigact));
|
|
|
|
sigact.sa_handler = dummy_signal;
|
|
|
|
sigaction(SIG_IPI, &sigact, NULL);
|
|
|
|
|
|
|
|
pthread_sigmask(SIG_BLOCK, NULL, &set);
|
|
|
|
sigdelset(&set, SIG_IPI);
|
|
|
|
sigdelset(&set, SIGBUS);
|
|
|
|
r = kvm_set_signal_mask(env, &set);
|
|
|
|
if (r) {
|
|
|
|
fprintf(stderr, "kvm_set_signal_mask: %s\n", strerror(-r));
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
sigdelset(&set, SIG_IPI);
|
|
|
|
sigdelset(&set, SIGBUS);
|
|
|
|
r = kvm_set_signal_mask(env, &set);
|
|
|
|
if (r) {
|
|
|
|
fprintf(stderr, "kvm_set_signal_mask: %s\n", strerror(-r));
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_tcg_init_cpu_signals(void)
|
|
|
|
{
|
|
|
|
sigset_t set;
|
|
|
|
struct sigaction sigact;
|
|
|
|
|
|
|
|
memset(&sigact, 0, sizeof(sigact));
|
|
|
|
sigact.sa_handler = cpu_signal;
|
|
|
|
sigaction(SIG_IPI, &sigact, NULL);
|
|
|
|
|
|
|
|
sigemptyset(&set);
|
|
|
|
sigaddset(&set, SIG_IPI);
|
|
|
|
pthread_sigmask(SIG_UNBLOCK, &set, NULL);
|
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:52 +08:00
|
|
|
#else /* _WIN32 */
|
|
|
|
|
2010-03-30 03:23:50 +08:00
|
|
|
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) {
|
|
|
|
fprintf(stderr, "Failed CreateEvent: %ld\n", GetLastError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
qemu_add_wait_object(qemu_event_handle, dummy_event_handler, NULL);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_event_increment(void)
|
|
|
|
{
|
|
|
|
if (!SetEvent(qemu_event_handle)) {
|
|
|
|
fprintf(stderr, "qemu_event_increment: SetEvent failed: %ld\n",
|
|
|
|
GetLastError());
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
}
|
2011-02-02 05:15:55 +08:00
|
|
|
|
2011-03-13 00:44:05 +08:00
|
|
|
static int qemu_signal_init(void)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:53 +08:00
|
|
|
static void qemu_kvm_init_cpu_signals(CPUState *env)
|
|
|
|
{
|
2011-03-13 00:44:06 +08:00
|
|
|
abort();
|
|
|
|
}
|
2011-02-02 05:15:53 +08:00
|
|
|
|
2011-03-13 00:44:06 +08:00
|
|
|
static void qemu_tcg_init_cpu_signals(void)
|
|
|
|
{
|
2011-02-02 05:15:53 +08:00
|
|
|
}
|
2011-03-13 00:44:06 +08:00
|
|
|
#endif /* _WIN32 */
|
2011-02-02 05:15:53 +08:00
|
|
|
|
2010-03-30 03:23:50 +08:00
|
|
|
QemuMutex qemu_global_mutex;
|
2011-06-09 19:10:24 +08:00
|
|
|
static QemuCond qemu_io_proceeded_cond;
|
|
|
|
static bool iothread_requesting_mutex;
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
static QemuThread io_thread;
|
|
|
|
|
|
|
|
static QemuThread *tcg_cpu_thread;
|
|
|
|
static QemuCond *tcg_halt_cond;
|
|
|
|
|
|
|
|
/* cpu creation */
|
|
|
|
static QemuCond qemu_cpu_cond;
|
|
|
|
/* system init */
|
|
|
|
static QemuCond qemu_pause_cond;
|
2010-05-04 20:45:22 +08:00
|
|
|
static QemuCond qemu_work_cond;
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
int qemu_init_main_loop(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
2011-02-02 05:15:58 +08:00
|
|
|
qemu_init_sigbus();
|
2010-06-25 22:56:56 +08:00
|
|
|
|
2011-03-13 00:44:05 +08:00
|
|
|
ret = qemu_signal_init();
|
2011-02-07 19:19:14 +08:00
|
|
|
if (ret) {
|
2010-10-12 02:31:16 +08:00
|
|
|
return ret;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-10-12 02:31:16 +08:00
|
|
|
|
|
|
|
/* Note eventfd must be drained before signalfd handlers run */
|
2010-03-30 03:23:50 +08:00
|
|
|
ret = qemu_event_init();
|
2011-02-07 19:19:14 +08:00
|
|
|
if (ret) {
|
2010-03-30 03:23:50 +08:00
|
|
|
return ret;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2011-02-09 01:18:18 +08:00
|
|
|
qemu_cond_init(&qemu_cpu_cond);
|
|
|
|
qemu_cond_init(&qemu_pause_cond);
|
|
|
|
qemu_cond_init(&qemu_work_cond);
|
2011-06-09 19:10:24 +08:00
|
|
|
qemu_cond_init(&qemu_io_proceeded_cond);
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_mutex_init(&qemu_global_mutex);
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
|
2011-03-13 00:43:51 +08:00
|
|
|
qemu_thread_get_self(&io_thread);
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-04-13 01:19:06 +08:00
|
|
|
void qemu_main_loop_start(void)
|
|
|
|
{
|
2011-08-23 00:35:25 +08:00
|
|
|
resume_all_vcpus();
|
2010-04-13 01:19:06 +08:00
|
|
|
}
|
|
|
|
|
2010-05-04 20:45:22 +08:00
|
|
|
void run_on_cpu(CPUState *env, void (*func)(void *data), void *data)
|
|
|
|
{
|
|
|
|
struct qemu_work_item wi;
|
|
|
|
|
2011-03-13 00:43:51 +08:00
|
|
|
if (qemu_cpu_is_self(env)) {
|
2010-05-04 20:45:22 +08:00
|
|
|
func(data);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wi.func = func;
|
|
|
|
wi.data = data;
|
2011-02-07 19:19:14 +08:00
|
|
|
if (!env->queued_work_first) {
|
2010-05-04 20:45:22 +08:00
|
|
|
env->queued_work_first = &wi;
|
2011-02-07 19:19:14 +08:00
|
|
|
} else {
|
2010-05-04 20:45:22 +08:00
|
|
|
env->queued_work_last->next = &wi;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-05-04 20:45:22 +08:00
|
|
|
env->queued_work_last = &wi;
|
|
|
|
wi.next = NULL;
|
|
|
|
wi.done = false;
|
|
|
|
|
|
|
|
qemu_cpu_kick(env);
|
|
|
|
while (!wi.done) {
|
|
|
|
CPUState *self_env = cpu_single_env;
|
|
|
|
|
|
|
|
qemu_cond_wait(&qemu_work_cond, &qemu_global_mutex);
|
|
|
|
cpu_single_env = self_env;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void flush_queued_work(CPUState *env)
|
|
|
|
{
|
|
|
|
struct qemu_work_item *wi;
|
|
|
|
|
2011-02-07 19:19:14 +08:00
|
|
|
if (!env->queued_work_first) {
|
2010-05-04 20:45:22 +08:00
|
|
|
return;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-05-04 20:45:22 +08:00
|
|
|
|
|
|
|
while ((wi = env->queued_work_first)) {
|
|
|
|
env->queued_work_first = wi->next;
|
|
|
|
wi->func(wi->data);
|
|
|
|
wi->done = true;
|
|
|
|
}
|
|
|
|
env->queued_work_last = NULL;
|
|
|
|
qemu_cond_broadcast(&qemu_work_cond);
|
|
|
|
}
|
|
|
|
|
2010-03-30 03:23:50 +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);
|
|
|
|
}
|
2010-05-04 20:45:22 +08:00
|
|
|
flush_queued_work(env);
|
2011-02-02 05:15:42 +08:00
|
|
|
env->thread_kicked = false;
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
2010-06-25 22:56:53 +08:00
|
|
|
static void qemu_tcg_wait_io_event(void)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2010-06-25 22:56:53 +08:00
|
|
|
CPUState *env;
|
|
|
|
|
2011-02-09 23:29:37 +08:00
|
|
|
while (all_cpu_threads_idle()) {
|
enable vm_clock to "warp" in the iothread+icount case
The previous patch however is not enough, because if the virtual CPU
goes to sleep waiting for a future timer interrupt to wake it up, qemu
deadlocks. The timer interrupt never comes because time is driven by
icount, but the vCPU doesn't run any insns.
You could say that VCPUs should never go to sleep in icount
mode if there is a pending vm_clock timer; rather time should
just warp to the next vm_clock event with no sleep ever taking place.
Even better, you can sleep for some time related to the
time left until the next event, to avoid that the warps are too visible
externally; for example, you could be sending network packets continously
instead of every 100ms.
This is what this patch implements. qemu_clock_warp is called: 1)
whenever a vm_clock timer is adjusted, to ensure the warp_timer is
synchronized; 2) at strategic points in the CPU thread, to make sure
the insn counter is synchronized before the CPU starts running.
In any case, the warp_timer is disabled while the CPU is running,
because the insn counter will then be making progress on its own.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Tested-by: Edgar E. Iglesias <edgar.iglesias@gmail.com>
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@gmail.com>
2011-04-13 16:03:44 +08:00
|
|
|
/* Start accounting real time to the virtual clock if the CPUs
|
|
|
|
are idle. */
|
|
|
|
qemu_clock_warp(vm_clock);
|
2011-03-13 00:44:00 +08:00
|
|
|
qemu_cond_wait(tcg_halt_cond, &qemu_global_mutex);
|
2011-02-09 23:29:37 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2011-06-09 19:10:24 +08:00
|
|
|
while (iothread_requesting_mutex) {
|
|
|
|
qemu_cond_wait(&qemu_io_proceeded_cond, &qemu_global_mutex);
|
|
|
|
}
|
2010-06-25 22:56:53 +08:00
|
|
|
|
|
|
|
for (env = first_cpu; env != NULL; env = env->next_cpu) {
|
|
|
|
qemu_wait_io_event_common(env);
|
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void qemu_kvm_wait_io_event(CPUState *env)
|
|
|
|
{
|
2011-02-09 23:29:37 +08:00
|
|
|
while (cpu_thread_is_idle(env)) {
|
2011-03-13 00:44:00 +08:00
|
|
|
qemu_cond_wait(env->halt_cond, &qemu_global_mutex);
|
2011-02-09 23:29:37 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2011-02-02 05:15:54 +08:00
|
|
|
qemu_kvm_eat_signals(env);
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_wait_io_event_common(env);
|
|
|
|
}
|
|
|
|
|
2011-02-07 19:19:12 +08:00
|
|
|
static void *qemu_kvm_cpu_thread_fn(void *arg)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
CPUState *env = arg;
|
2011-02-02 05:15:50 +08:00
|
|
|
int r;
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2010-03-24 00:37:13 +08:00
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
2011-03-13 00:43:51 +08:00
|
|
|
qemu_thread_get_self(env->thread);
|
2011-03-15 19:26:31 +08:00
|
|
|
env->thread_id = qemu_get_thread_id();
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2011-02-02 05:15:50 +08:00
|
|
|
r = kvm_init_vcpu(env);
|
|
|
|
if (r < 0) {
|
|
|
|
fprintf(stderr, "kvm_init_vcpu failed: %s\n", strerror(-r));
|
|
|
|
exit(1);
|
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2011-02-02 05:15:52 +08:00
|
|
|
qemu_kvm_init_cpu_signals(env);
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
/* signal CPU creation */
|
|
|
|
env->created = 1;
|
|
|
|
qemu_cond_signal(&qemu_cpu_cond);
|
|
|
|
|
|
|
|
while (1) {
|
2011-02-07 19:19:14 +08:00
|
|
|
if (cpu_can_run(env)) {
|
2011-02-07 19:19:18 +08:00
|
|
|
r = kvm_cpu_exec(env);
|
2011-02-07 19:19:17 +08:00
|
|
|
if (r == EXCP_DEBUG) {
|
2011-03-15 19:26:13 +08:00
|
|
|
cpu_handle_guest_debug(env);
|
2011-02-07 19:19:17 +08:00
|
|
|
}
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_kvm_wait_io_event(env);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-02-07 19:19:12 +08:00
|
|
|
static void *qemu_tcg_cpu_thread_fn(void *arg)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
CPUState *env = arg;
|
|
|
|
|
2011-02-02 05:15:52 +08:00
|
|
|
qemu_tcg_init_cpu_signals();
|
2011-03-13 00:43:51 +08:00
|
|
|
qemu_thread_get_self(env->thread);
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
/* signal CPU creation */
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
2011-02-07 19:19:14 +08:00
|
|
|
for (env = first_cpu; env != NULL; env = env->next_cpu) {
|
2011-03-15 19:26:31 +08:00
|
|
|
env->thread_id = qemu_get_thread_id();
|
2010-03-30 03:23:50 +08:00
|
|
|
env->created = 1;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_cond_signal(&qemu_cpu_cond);
|
|
|
|
|
2011-08-23 00:35:25 +08:00
|
|
|
/* wait for initial kick-off after machine start */
|
|
|
|
while (first_cpu->stopped) {
|
|
|
|
qemu_cond_wait(tcg_halt_cond, &qemu_global_mutex);
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
while (1) {
|
2010-06-25 22:56:55 +08:00
|
|
|
cpu_exec_all();
|
2011-04-13 16:03:46 +08:00
|
|
|
if (use_icount && qemu_next_icount_deadline() <= 0) {
|
2011-04-13 16:03:43 +08:00
|
|
|
qemu_notify_event();
|
|
|
|
}
|
2010-06-25 22:56:53 +08:00
|
|
|
qemu_tcg_wait_io_event();
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-03-13 00:44:08 +08:00
|
|
|
static void qemu_cpu_kick_thread(CPUState *env)
|
|
|
|
{
|
|
|
|
#ifndef _WIN32
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = pthread_kill(env->thread->thread, SIG_IPI);
|
|
|
|
if (err) {
|
|
|
|
fprintf(stderr, "qemu:%s: %s", __func__, strerror(err));
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
#else /* _WIN32 */
|
|
|
|
if (!qemu_cpu_is_self(env)) {
|
|
|
|
SuspendThread(env->thread->thread);
|
|
|
|
cpu_signal(0);
|
|
|
|
ResumeThread(env->thread->thread);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-03-30 03:23:50 +08:00
|
|
|
void qemu_cpu_kick(void *_env)
|
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
|
|
|
|
|
|
|
qemu_cond_broadcast(env->halt_cond);
|
2011-08-22 23:46:03 +08:00
|
|
|
if (kvm_enabled() && !env->thread_kicked) {
|
2011-03-13 00:44:08 +08:00
|
|
|
qemu_cpu_kick_thread(env);
|
2011-02-02 05:15:42 +08:00
|
|
|
env->thread_kicked = true;
|
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:59 +08:00
|
|
|
void qemu_cpu_kick_self(void)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2011-03-13 00:44:07 +08:00
|
|
|
#ifndef _WIN32
|
2011-02-02 05:15:59 +08:00
|
|
|
assert(cpu_single_env);
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2011-02-02 05:15:59 +08:00
|
|
|
if (!cpu_single_env->thread_kicked) {
|
2011-03-13 00:44:08 +08:00
|
|
|
qemu_cpu_kick_thread(cpu_single_env);
|
2011-02-02 05:15:59 +08:00
|
|
|
cpu_single_env->thread_kicked = true;
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
2011-03-13 00:44:07 +08:00
|
|
|
#else
|
|
|
|
abort();
|
|
|
|
#endif
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
2011-03-13 00:43:51 +08:00
|
|
|
int qemu_cpu_is_self(void *_env)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
2010-10-12 02:31:16 +08:00
|
|
|
|
2011-03-13 00:43:51 +08:00
|
|
|
return qemu_thread_is_self(env->thread);
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_mutex_lock_iothread(void)
|
|
|
|
{
|
|
|
|
if (kvm_enabled()) {
|
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
2010-05-04 20:45:20 +08:00
|
|
|
} else {
|
2011-06-09 19:10:24 +08:00
|
|
|
iothread_requesting_mutex = true;
|
2010-05-04 20:45:20 +08:00
|
|
|
if (qemu_mutex_trylock(&qemu_global_mutex)) {
|
2011-03-13 00:44:08 +08:00
|
|
|
qemu_cpu_kick_thread(first_cpu);
|
2010-05-04 20:45:20 +08:00
|
|
|
qemu_mutex_lock(&qemu_global_mutex);
|
|
|
|
}
|
2011-06-09 19:10:24 +08:00
|
|
|
iothread_requesting_mutex = false;
|
|
|
|
qemu_cond_broadcast(&qemu_io_proceeded_cond);
|
2010-05-04 20:45:20 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_mutex_unlock_iothread(void)
|
|
|
|
{
|
|
|
|
qemu_mutex_unlock(&qemu_global_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int all_vcpus_paused(void)
|
|
|
|
{
|
|
|
|
CPUState *penv = first_cpu;
|
|
|
|
|
|
|
|
while (penv) {
|
2011-02-07 19:19:14 +08:00
|
|
|
if (!penv->stopped) {
|
2010-03-30 03:23:50 +08:00
|
|
|
return 0;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pause_all_vcpus(void)
|
|
|
|
{
|
|
|
|
CPUState *penv = first_cpu;
|
|
|
|
|
|
|
|
while (penv) {
|
|
|
|
penv->stop = 1;
|
|
|
|
qemu_cpu_kick(penv);
|
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!all_vcpus_paused()) {
|
2011-03-13 00:44:02 +08:00
|
|
|
qemu_cond_wait(&qemu_pause_cond, &qemu_global_mutex);
|
2010-03-30 03:23:50 +08:00
|
|
|
penv = first_cpu;
|
|
|
|
while (penv) {
|
2010-05-04 20:45:21 +08:00
|
|
|
qemu_cpu_kick(penv);
|
2010-03-30 03:23:50 +08:00
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void resume_all_vcpus(void)
|
|
|
|
{
|
|
|
|
CPUState *penv = first_cpu;
|
|
|
|
|
|
|
|
while (penv) {
|
|
|
|
penv->stop = 0;
|
|
|
|
penv->stopped = 0;
|
|
|
|
qemu_cpu_kick(penv);
|
|
|
|
penv = (CPUState *)penv->next_cpu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-07 19:19:12 +08:00
|
|
|
static void qemu_tcg_init_vcpu(void *_env)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
2011-02-07 19:19:14 +08:00
|
|
|
|
2010-03-30 03:23:50 +08:00
|
|
|
/* share a single thread for all cpus with TCG */
|
|
|
|
if (!tcg_cpu_thread) {
|
2011-08-21 11:09:37 +08:00
|
|
|
env->thread = g_malloc0(sizeof(QemuThread));
|
|
|
|
env->halt_cond = g_malloc0(sizeof(QemuCond));
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_cond_init(env->halt_cond);
|
2011-08-23 00:35:25 +08:00
|
|
|
tcg_halt_cond = env->halt_cond;
|
2011-02-07 19:19:12 +08:00
|
|
|
qemu_thread_create(env->thread, qemu_tcg_cpu_thread_fn, env);
|
2011-02-07 19:19:14 +08:00
|
|
|
while (env->created == 0) {
|
2011-03-13 00:44:03 +08:00
|
|
|
qemu_cond_wait(&qemu_cpu_cond, &qemu_global_mutex);
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
tcg_cpu_thread = env->thread;
|
|
|
|
} else {
|
|
|
|
env->thread = tcg_cpu_thread;
|
|
|
|
env->halt_cond = tcg_halt_cond;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-07 19:19:12 +08:00
|
|
|
static void qemu_kvm_start_vcpu(CPUState *env)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2011-08-21 11:09:37 +08:00
|
|
|
env->thread = g_malloc0(sizeof(QemuThread));
|
|
|
|
env->halt_cond = g_malloc0(sizeof(QemuCond));
|
2010-03-30 03:23:50 +08:00
|
|
|
qemu_cond_init(env->halt_cond);
|
2011-02-07 19:19:12 +08:00
|
|
|
qemu_thread_create(env->thread, qemu_kvm_cpu_thread_fn, env);
|
2011-02-07 19:19:14 +08:00
|
|
|
while (env->created == 0) {
|
2011-03-13 00:44:03 +08:00
|
|
|
qemu_cond_wait(&qemu_cpu_cond, &qemu_global_mutex);
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_init_vcpu(void *_env)
|
|
|
|
{
|
|
|
|
CPUState *env = _env;
|
|
|
|
|
|
|
|
env->nr_cores = smp_cores;
|
|
|
|
env->nr_threads = smp_threads;
|
2011-08-23 00:35:25 +08:00
|
|
|
env->stopped = 1;
|
2011-02-07 19:19:14 +08:00
|
|
|
if (kvm_enabled()) {
|
2011-02-07 19:19:12 +08:00
|
|
|
qemu_kvm_start_vcpu(env);
|
2011-02-07 19:19:14 +08:00
|
|
|
} else {
|
2011-02-07 19:19:12 +08:00
|
|
|
qemu_tcg_init_vcpu(env);
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_notify_event(void)
|
|
|
|
{
|
|
|
|
qemu_event_increment();
|
|
|
|
}
|
|
|
|
|
2011-02-02 05:15:43 +08:00
|
|
|
void cpu_stop_current(void)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2011-02-02 05:15:43 +08:00
|
|
|
if (cpu_single_env) {
|
2011-03-13 00:43:59 +08:00
|
|
|
cpu_single_env->stop = 0;
|
2011-02-02 05:15:43 +08:00
|
|
|
cpu_single_env->stopped = 1;
|
|
|
|
cpu_exit(cpu_single_env);
|
2011-03-13 00:43:59 +08:00
|
|
|
qemu_cond_signal(&qemu_pause_cond);
|
2011-02-02 05:15:43 +08:00
|
|
|
}
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
2011-07-30 01:26:33 +08:00
|
|
|
void vm_stop(RunState state)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2011-03-13 00:43:51 +08:00
|
|
|
if (!qemu_thread_is_self(&io_thread)) {
|
2011-07-30 01:26:33 +08:00
|
|
|
qemu_system_vmstop_request(state);
|
2010-03-30 03:23:50 +08:00
|
|
|
/*
|
|
|
|
* FIXME: should not return to device code in case
|
|
|
|
* vm_stop() has been requested.
|
|
|
|
*/
|
2011-02-02 05:15:43 +08:00
|
|
|
cpu_stop_current();
|
2010-03-30 03:23:50 +08:00
|
|
|
return;
|
|
|
|
}
|
2011-07-30 01:26:33 +08:00
|
|
|
do_vm_stop(state);
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
2011-02-07 19:19:18 +08:00
|
|
|
static int tcg_cpu_exec(CPUState *env)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
#ifdef CONFIG_PROFILER
|
|
|
|
int64_t ti;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_PROFILER
|
|
|
|
ti = profile_getclock();
|
|
|
|
#endif
|
|
|
|
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;
|
2011-04-13 16:03:46 +08:00
|
|
|
count = qemu_icount_round(qemu_next_icount_deadline());
|
2010-03-30 03:23:50 +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);
|
|
|
|
#ifdef CONFIG_PROFILER
|
|
|
|
qemu_time += profile_getclock() - ti;
|
|
|
|
#endif
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2010-06-25 22:56:55 +08:00
|
|
|
bool cpu_exec_all(void)
|
2010-03-30 03:23:50 +08:00
|
|
|
{
|
2011-02-02 05:15:55 +08:00
|
|
|
int r;
|
|
|
|
|
enable vm_clock to "warp" in the iothread+icount case
The previous patch however is not enough, because if the virtual CPU
goes to sleep waiting for a future timer interrupt to wake it up, qemu
deadlocks. The timer interrupt never comes because time is driven by
icount, but the vCPU doesn't run any insns.
You could say that VCPUs should never go to sleep in icount
mode if there is a pending vm_clock timer; rather time should
just warp to the next vm_clock event with no sleep ever taking place.
Even better, you can sleep for some time related to the
time left until the next event, to avoid that the warps are too visible
externally; for example, you could be sending network packets continously
instead of every 100ms.
This is what this patch implements. qemu_clock_warp is called: 1)
whenever a vm_clock timer is adjusted, to ensure the warp_timer is
synchronized; 2) at strategic points in the CPU thread, to make sure
the insn counter is synchronized before the CPU starts running.
In any case, the warp_timer is disabled while the CPU is running,
because the insn counter will then be making progress on its own.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Tested-by: Edgar E. Iglesias <edgar.iglesias@gmail.com>
Signed-off-by: Edgar E. Iglesias <edgar.iglesias@gmail.com>
2011-04-13 16:03:44 +08:00
|
|
|
/* Account partial waits to the vm_clock. */
|
|
|
|
qemu_clock_warp(vm_clock);
|
|
|
|
|
2011-02-07 19:19:14 +08:00
|
|
|
if (next_cpu == NULL) {
|
2010-03-30 03:23:50 +08:00
|
|
|
next_cpu = first_cpu;
|
2011-02-07 19:19:14 +08:00
|
|
|
}
|
2010-06-25 22:56:52 +08:00
|
|
|
for (; next_cpu != NULL && !exit_request; next_cpu = next_cpu->next_cpu) {
|
2010-06-25 22:56:54 +08:00
|
|
|
CPUState *env = next_cpu;
|
2010-03-30 03:23:50 +08:00
|
|
|
|
|
|
|
qemu_clock_enable(vm_clock,
|
2010-06-25 22:56:54 +08:00
|
|
|
(env->singlestep_enabled & SSTEP_NOTIMER) == 0);
|
2010-03-30 03:23:50 +08:00
|
|
|
|
2010-06-25 22:56:56 +08:00
|
|
|
if (cpu_can_run(env)) {
|
2011-02-02 05:15:55 +08:00
|
|
|
if (kvm_enabled()) {
|
2011-02-07 19:19:18 +08:00
|
|
|
r = kvm_cpu_exec(env);
|
2011-02-02 05:15:55 +08:00
|
|
|
qemu_kvm_eat_signals(env);
|
2011-02-07 19:19:18 +08:00
|
|
|
} else {
|
|
|
|
r = tcg_cpu_exec(env);
|
2011-02-02 05:15:55 +08:00
|
|
|
}
|
|
|
|
if (r == EXCP_DEBUG) {
|
2011-03-15 19:26:13 +08:00
|
|
|
cpu_handle_guest_debug(env);
|
2010-06-25 22:56:56 +08:00
|
|
|
break;
|
|
|
|
}
|
2011-03-13 00:43:58 +08:00
|
|
|
} else if (env->stop || env->stopped) {
|
2010-03-30 03:23:50 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2010-06-25 22:56:52 +08:00
|
|
|
exit_request = 0;
|
2011-02-09 23:29:37 +08:00
|
|
|
return !all_cpu_threads_idle();
|
2010-03-30 03:23:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void set_numa_modes(void)
|
|
|
|
{
|
|
|
|
CPUState *env;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_cpu_log(const char *optarg)
|
|
|
|
{
|
|
|
|
int mask;
|
|
|
|
const CPULogItem *item;
|
|
|
|
|
|
|
|
mask = cpu_str_to_log_mask(optarg);
|
|
|
|
if (!mask) {
|
|
|
|
printf("Log items (comma separated):\n");
|
|
|
|
for (item = cpu_log_items; item->mask != 0; item++) {
|
|
|
|
printf("%-10s %s\n", item->name, item->help);
|
|
|
|
}
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
cpu_set_log(mask);
|
|
|
|
}
|
2010-03-30 03:24:00 +08:00
|
|
|
|
2011-06-08 00:32:40 +08:00
|
|
|
void set_cpu_log_filename(const char *optarg)
|
|
|
|
{
|
|
|
|
cpu_set_log_filename(optarg);
|
|
|
|
}
|
|
|
|
|
2010-03-30 03:24:00 +08:00
|
|
|
/* Return the virtual CPU time, based on the instruction counter. */
|
|
|
|
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);
|
|
|
|
}
|
2010-05-05 03:55:35 +08:00
|
|
|
|
2010-10-23 05:03:33 +08:00
|
|
|
void list_cpus(FILE *f, fprintf_function cpu_fprintf, const char *optarg)
|
2010-05-05 03:55:35 +08:00
|
|
|
{
|
|
|
|
/* XXX: implement xxx_cpu_list for targets that still miss it */
|
|
|
|
#if defined(cpu_list_id)
|
|
|
|
cpu_list_id(f, cpu_fprintf, optarg);
|
|
|
|
#elif defined(cpu_list)
|
|
|
|
cpu_list(f, cpu_fprintf); /* deprecated */
|
|
|
|
#endif
|
|
|
|
}
|