cpus-common: move CPU list management to common code
Add a mutex for the CPU list to system emulation, as it will be used to manage safe work. Abstract manipulation of the CPU list in new functions cpu_list_add and cpu_list_remove. Reviewed-by: Richard Henderson <rth@twiddle.net> Reviewed-by: Alex Bennée <alex.bennee@linaro.org> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
178f94297a
commit
267f685b8b
@ -89,7 +89,7 @@ endif
|
|||||||
|
|
||||||
#######################################################################
|
#######################################################################
|
||||||
# Target-independent parts used in system and user emulation
|
# Target-independent parts used in system and user emulation
|
||||||
common-obj-y += tcg-runtime.o
|
common-obj-y += tcg-runtime.o cpus-common.o
|
||||||
common-obj-y += hw/
|
common-obj-y += hw/
|
||||||
common-obj-y += qom/
|
common-obj-y += qom/
|
||||||
common-obj-y += disas/
|
common-obj-y += disas/
|
||||||
|
@ -95,14 +95,6 @@ void fork_end(int child)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void cpu_list_lock(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void cpu_list_unlock(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef TARGET_I386
|
#ifdef TARGET_I386
|
||||||
/***********************************************************/
|
/***********************************************************/
|
||||||
/* CPUX86 core interface */
|
/* CPUX86 core interface */
|
||||||
@ -748,6 +740,7 @@ int main(int argc, char **argv)
|
|||||||
if (argc <= 1)
|
if (argc <= 1)
|
||||||
usage();
|
usage();
|
||||||
|
|
||||||
|
qemu_init_cpu_list();
|
||||||
module_call_init(MODULE_INIT_QOM);
|
module_call_init(MODULE_INIT_QOM);
|
||||||
|
|
||||||
if ((envlist = envlist_create()) == NULL) {
|
if ((envlist = envlist_create()) == NULL) {
|
||||||
|
83
cpus-common.c
Normal file
83
cpus-common.c
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
/*
|
||||||
|
* CPU thread main loop - common bits for user and system mode emulation
|
||||||
|
*
|
||||||
|
* Copyright (c) 2003-2005 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "qemu/osdep.h"
|
||||||
|
#include "exec/cpu-common.h"
|
||||||
|
#include "qom/cpu.h"
|
||||||
|
#include "sysemu/cpus.h"
|
||||||
|
|
||||||
|
static QemuMutex qemu_cpu_list_lock;
|
||||||
|
|
||||||
|
void qemu_init_cpu_list(void)
|
||||||
|
{
|
||||||
|
qemu_mutex_init(&qemu_cpu_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cpu_list_lock(void)
|
||||||
|
{
|
||||||
|
qemu_mutex_lock(&qemu_cpu_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cpu_list_unlock(void)
|
||||||
|
{
|
||||||
|
qemu_mutex_unlock(&qemu_cpu_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool cpu_index_auto_assigned;
|
||||||
|
|
||||||
|
static int cpu_get_free_index(void)
|
||||||
|
{
|
||||||
|
CPUState *some_cpu;
|
||||||
|
int cpu_index = 0;
|
||||||
|
|
||||||
|
cpu_index_auto_assigned = true;
|
||||||
|
CPU_FOREACH(some_cpu) {
|
||||||
|
cpu_index++;
|
||||||
|
}
|
||||||
|
return cpu_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
void cpu_list_add(CPUState *cpu)
|
||||||
|
{
|
||||||
|
qemu_mutex_lock(&qemu_cpu_list_lock);
|
||||||
|
if (cpu->cpu_index == UNASSIGNED_CPU_INDEX) {
|
||||||
|
cpu->cpu_index = cpu_get_free_index();
|
||||||
|
assert(cpu->cpu_index != UNASSIGNED_CPU_INDEX);
|
||||||
|
} else {
|
||||||
|
assert(!cpu_index_auto_assigned);
|
||||||
|
}
|
||||||
|
QTAILQ_INSERT_TAIL(&cpus, cpu, node);
|
||||||
|
qemu_mutex_unlock(&qemu_cpu_list_lock);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cpu_list_remove(CPUState *cpu)
|
||||||
|
{
|
||||||
|
qemu_mutex_lock(&qemu_cpu_list_lock);
|
||||||
|
if (!QTAILQ_IN_USE(cpu, node)) {
|
||||||
|
/* there is nothing to undo since cpu_exec_init() hasn't been called */
|
||||||
|
qemu_mutex_unlock(&qemu_cpu_list_lock);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(!(cpu_index_auto_assigned && cpu != QTAILQ_LAST(&cpus, CPUTailQ)));
|
||||||
|
|
||||||
|
QTAILQ_REMOVE(&cpus, cpu, node);
|
||||||
|
cpu->cpu_index = UNASSIGNED_CPU_INDEX;
|
||||||
|
qemu_mutex_unlock(&qemu_cpu_list_lock);
|
||||||
|
}
|
37
exec.c
37
exec.c
@ -598,36 +598,11 @@ AddressSpace *cpu_get_address_space(CPUState *cpu, int asidx)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
static bool cpu_index_auto_assigned;
|
|
||||||
|
|
||||||
static int cpu_get_free_index(void)
|
|
||||||
{
|
|
||||||
CPUState *some_cpu;
|
|
||||||
int cpu_index = 0;
|
|
||||||
|
|
||||||
cpu_index_auto_assigned = true;
|
|
||||||
CPU_FOREACH(some_cpu) {
|
|
||||||
cpu_index++;
|
|
||||||
}
|
|
||||||
return cpu_index;
|
|
||||||
}
|
|
||||||
|
|
||||||
void cpu_exec_exit(CPUState *cpu)
|
void cpu_exec_exit(CPUState *cpu)
|
||||||
{
|
{
|
||||||
CPUClass *cc = CPU_GET_CLASS(cpu);
|
CPUClass *cc = CPU_GET_CLASS(cpu);
|
||||||
|
|
||||||
cpu_list_lock();
|
cpu_list_remove(cpu);
|
||||||
if (!QTAILQ_IN_USE(cpu, node)) {
|
|
||||||
/* there is nothing to undo since cpu_exec_init() hasn't been called */
|
|
||||||
cpu_list_unlock();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(!(cpu_index_auto_assigned && cpu != QTAILQ_LAST(&cpus, CPUTailQ)));
|
|
||||||
|
|
||||||
QTAILQ_REMOVE(&cpus, cpu, node);
|
|
||||||
cpu->cpu_index = UNASSIGNED_CPU_INDEX;
|
|
||||||
cpu_list_unlock();
|
|
||||||
|
|
||||||
if (cc->vmsd != NULL) {
|
if (cc->vmsd != NULL) {
|
||||||
vmstate_unregister(NULL, cc->vmsd, cpu);
|
vmstate_unregister(NULL, cc->vmsd, cpu);
|
||||||
@ -663,15 +638,7 @@ void cpu_exec_init(CPUState *cpu, Error **errp)
|
|||||||
object_ref(OBJECT(cpu->memory));
|
object_ref(OBJECT(cpu->memory));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
cpu_list_lock();
|
cpu_list_add(cpu);
|
||||||
if (cpu->cpu_index == UNASSIGNED_CPU_INDEX) {
|
|
||||||
cpu->cpu_index = cpu_get_free_index();
|
|
||||||
assert(cpu->cpu_index != UNASSIGNED_CPU_INDEX);
|
|
||||||
} else {
|
|
||||||
assert(!cpu_index_auto_assigned);
|
|
||||||
}
|
|
||||||
QTAILQ_INSERT_TAIL(&cpus, cpu, node);
|
|
||||||
cpu_list_unlock();
|
|
||||||
|
|
||||||
#ifndef CONFIG_USER_ONLY
|
#ifndef CONFIG_USER_ONLY
|
||||||
if (qdev_get_vmsd(DEVICE(cpu)) == NULL) {
|
if (qdev_get_vmsd(DEVICE(cpu)) == NULL) {
|
||||||
|
@ -23,6 +23,11 @@ typedef struct CPUListState {
|
|||||||
FILE *file;
|
FILE *file;
|
||||||
} CPUListState;
|
} CPUListState;
|
||||||
|
|
||||||
|
/* The CPU list lock nests outside tb_lock/tb_unlock. */
|
||||||
|
void qemu_init_cpu_list(void);
|
||||||
|
void cpu_list_lock(void);
|
||||||
|
void cpu_list_unlock(void);
|
||||||
|
|
||||||
#if !defined(CONFIG_USER_ONLY)
|
#if !defined(CONFIG_USER_ONLY)
|
||||||
|
|
||||||
enum device_endian {
|
enum device_endian {
|
||||||
|
@ -56,17 +56,6 @@ TranslationBlock *tb_gen_code(CPUState *cpu,
|
|||||||
target_ulong pc, target_ulong cs_base,
|
target_ulong pc, target_ulong cs_base,
|
||||||
uint32_t flags,
|
uint32_t flags,
|
||||||
int cflags);
|
int cflags);
|
||||||
#if defined(CONFIG_USER_ONLY)
|
|
||||||
void cpu_list_lock(void);
|
|
||||||
void cpu_list_unlock(void);
|
|
||||||
#else
|
|
||||||
static inline void cpu_list_unlock(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
static inline void cpu_list_lock(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void cpu_exec_init(CPUState *cpu, Error **errp);
|
void cpu_exec_init(CPUState *cpu, Error **errp);
|
||||||
void QEMU_NORETURN cpu_loop_exit(CPUState *cpu);
|
void QEMU_NORETURN cpu_loop_exit(CPUState *cpu);
|
||||||
|
@ -544,6 +544,18 @@ static inline int cpu_asidx_from_attrs(CPUState *cpu, MemTxAttrs attrs)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* cpu_list_add:
|
||||||
|
* @cpu: The CPU to be added to the list of CPUs.
|
||||||
|
*/
|
||||||
|
void cpu_list_add(CPUState *cpu);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* cpu_list_remove:
|
||||||
|
* @cpu: The CPU to be removed from the list of CPUs.
|
||||||
|
*/
|
||||||
|
void cpu_list_remove(CPUState *cpu);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* cpu_reset:
|
* cpu_reset:
|
||||||
* @cpu: The CPU whose state is to be reset.
|
* @cpu: The CPU whose state is to be reset.
|
||||||
|
@ -111,7 +111,6 @@ int cpu_get_pic_interrupt(CPUX86State *env)
|
|||||||
We don't require a full sync, only that no cpus are executing guest code.
|
We don't require a full sync, only that no cpus are executing guest code.
|
||||||
The alternative is to map target atomic ops onto host equivalents,
|
The alternative is to map target atomic ops onto host equivalents,
|
||||||
which requires quite a lot of per host/target work. */
|
which requires quite a lot of per host/target work. */
|
||||||
static QemuMutex cpu_list_lock;
|
|
||||||
static QemuMutex exclusive_lock;
|
static QemuMutex exclusive_lock;
|
||||||
static QemuCond exclusive_cond;
|
static QemuCond exclusive_cond;
|
||||||
static QemuCond exclusive_resume;
|
static QemuCond exclusive_resume;
|
||||||
@ -119,7 +118,6 @@ static int pending_cpus;
|
|||||||
|
|
||||||
void qemu_init_cpu_loop(void)
|
void qemu_init_cpu_loop(void)
|
||||||
{
|
{
|
||||||
qemu_mutex_init(&cpu_list_lock);
|
|
||||||
qemu_mutex_init(&exclusive_lock);
|
qemu_mutex_init(&exclusive_lock);
|
||||||
qemu_cond_init(&exclusive_cond);
|
qemu_cond_init(&exclusive_cond);
|
||||||
qemu_cond_init(&exclusive_resume);
|
qemu_cond_init(&exclusive_resume);
|
||||||
@ -128,6 +126,7 @@ void qemu_init_cpu_loop(void)
|
|||||||
/* Make sure everything is in a consistent state for calling fork(). */
|
/* Make sure everything is in a consistent state for calling fork(). */
|
||||||
void fork_start(void)
|
void fork_start(void)
|
||||||
{
|
{
|
||||||
|
cpu_list_lock();
|
||||||
qemu_mutex_lock(&tcg_ctx.tb_ctx.tb_lock);
|
qemu_mutex_lock(&tcg_ctx.tb_ctx.tb_lock);
|
||||||
qemu_mutex_lock(&exclusive_lock);
|
qemu_mutex_lock(&exclusive_lock);
|
||||||
mmap_fork_start();
|
mmap_fork_start();
|
||||||
@ -147,14 +146,15 @@ void fork_end(int child)
|
|||||||
}
|
}
|
||||||
pending_cpus = 0;
|
pending_cpus = 0;
|
||||||
qemu_mutex_init(&exclusive_lock);
|
qemu_mutex_init(&exclusive_lock);
|
||||||
qemu_mutex_init(&cpu_list_lock);
|
|
||||||
qemu_cond_init(&exclusive_cond);
|
qemu_cond_init(&exclusive_cond);
|
||||||
qemu_cond_init(&exclusive_resume);
|
qemu_cond_init(&exclusive_resume);
|
||||||
qemu_mutex_init(&tcg_ctx.tb_ctx.tb_lock);
|
qemu_mutex_init(&tcg_ctx.tb_ctx.tb_lock);
|
||||||
|
qemu_init_cpu_list();
|
||||||
gdbserver_fork(thread_cpu);
|
gdbserver_fork(thread_cpu);
|
||||||
} else {
|
} else {
|
||||||
qemu_mutex_unlock(&exclusive_lock);
|
qemu_mutex_unlock(&exclusive_lock);
|
||||||
qemu_mutex_unlock(&tcg_ctx.tb_ctx.tb_lock);
|
qemu_mutex_unlock(&tcg_ctx.tb_ctx.tb_lock);
|
||||||
|
cpu_list_unlock();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -221,16 +221,6 @@ static inline void cpu_exec_end(CPUState *cpu)
|
|||||||
qemu_mutex_unlock(&exclusive_lock);
|
qemu_mutex_unlock(&exclusive_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cpu_list_lock(void)
|
|
||||||
{
|
|
||||||
qemu_mutex_lock(&cpu_list_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
void cpu_list_unlock(void)
|
|
||||||
{
|
|
||||||
qemu_mutex_unlock(&cpu_list_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#ifdef TARGET_I386
|
#ifdef TARGET_I386
|
||||||
/***********************************************************/
|
/***********************************************************/
|
||||||
@ -4229,6 +4219,7 @@ int main(int argc, char **argv, char **envp)
|
|||||||
int ret;
|
int ret;
|
||||||
int execfd;
|
int execfd;
|
||||||
|
|
||||||
|
qemu_init_cpu_list();
|
||||||
qemu_init_cpu_loop();
|
qemu_init_cpu_loop();
|
||||||
module_call_init(MODULE_INIT_QOM);
|
module_call_init(MODULE_INIT_QOM);
|
||||||
|
|
||||||
|
1
vl.c
1
vl.c
@ -3017,6 +3017,7 @@ int main(int argc, char **argv, char **envp)
|
|||||||
Error *err = NULL;
|
Error *err = NULL;
|
||||||
bool list_data_dirs = false;
|
bool list_data_dirs = false;
|
||||||
|
|
||||||
|
qemu_init_cpu_list();
|
||||||
qemu_init_cpu_loop();
|
qemu_init_cpu_loop();
|
||||||
qemu_mutex_lock_iothread();
|
qemu_mutex_lock_iothread();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user