提交 6e68e076 编写于 作者: P pbrook

Move clone() register setup to target specific code. Handle fork-like clone.


git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@4623 c046a42c-6fe2-441c-8c8c-71466251a162
上级 a4a99d71
...@@ -2744,64 +2744,8 @@ int do_fork(CPUState *env, unsigned int flags, abi_ulong newsp) ...@@ -2744,64 +2744,8 @@ int do_fork(CPUState *env, unsigned int flags, abi_ulong newsp)
first_task_state = ts; first_task_state = ts;
/* we create a new CPU instance. */ /* we create a new CPU instance. */
new_env = cpu_copy(env); new_env = cpu_copy(env);
#if defined(TARGET_I386) /* Init regs that differ from the parent. */
if (!newsp) cpu_clone_regs(new_env, newsp);
newsp = env->regs[R_ESP];
new_env->regs[R_ESP] = newsp;
new_env->regs[R_EAX] = 0;
#elif defined(TARGET_ARM)
if (!newsp)
newsp = env->regs[13];
new_env->regs[13] = newsp;
new_env->regs[0] = 0;
#elif defined(TARGET_SPARC)
if (!newsp)
newsp = env->regwptr[22];
new_env->regwptr[22] = newsp;
new_env->regwptr[0] = 0;
/* XXXXX */
printf ("HELPME: %s:%d\n", __FILE__, __LINE__);
#elif defined(TARGET_M68K)
if (!newsp)
newsp = env->aregs[7];
new_env->aregs[7] = newsp;
new_env->dregs[0] = 0;
/* ??? is this sufficient? */
#elif defined(TARGET_MIPS)
if (!newsp)
newsp = env->gpr[env->current_tc][29];
new_env->gpr[env->current_tc][29] = newsp;
#elif defined(TARGET_PPC)
if (!newsp)
newsp = env->gpr[1];
new_env->gpr[1] = newsp;
{
int i;
for (i = 7; i < 32; i++)
new_env->gpr[i] = 0;
}
#elif defined(TARGET_SH4)
if (!newsp)
newsp = env->gregs[15];
new_env->gregs[15] = newsp;
/* XXXXX */
#elif defined(TARGET_ALPHA)
if (!newsp)
newsp = env->ir[30];
new_env->ir[30] = newsp;
/* ? */
{
int i;
for (i = 7; i < 30; i++)
new_env->ir[i] = 0;
}
#elif defined(TARGET_CRIS)
if (!newsp)
newsp = env->regs[14];
new_env->regs[14] = newsp;
#else
#error unsupported target CPU
#endif
new_env->opaque = ts; new_env->opaque = ts;
#ifdef __ia64__ #ifdef __ia64__
ret = __clone2(clone_func, new_stack + NEW_STACK_SIZE, flags, new_env); ret = __clone2(clone_func, new_stack + NEW_STACK_SIZE, flags, new_env);
...@@ -2813,6 +2757,9 @@ int do_fork(CPUState *env, unsigned int flags, abi_ulong newsp) ...@@ -2813,6 +2757,9 @@ int do_fork(CPUState *env, unsigned int flags, abi_ulong newsp)
if ((flags & ~CSIGNAL) != 0) if ((flags & ~CSIGNAL) != 0)
return -EINVAL; return -EINVAL;
ret = fork(); ret = fork();
if (ret == 0) {
cpu_clone_regs(env, newsp);
}
} }
return ret; return ret;
} }
......
...@@ -311,6 +311,15 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -311,6 +311,15 @@ static inline int cpu_mmu_index (CPUState *env)
return (env->ps >> 3) & 3; return (env->ps >> 3) & 3;
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->ir[30] = newsp;
/* FIXME: Zero syscall return value. */
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
enum { enum {
......
...@@ -408,6 +408,15 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -408,6 +408,15 @@ static inline int cpu_mmu_index (CPUState *env)
return (env->uncached_cpsr & CPSR_M) == ARM_CPU_MODE_USR ? 1 : 0; return (env->uncached_cpsr & CPSR_M) == ARM_CPU_MODE_USR ? 1 : 0;
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->regs[13] = newsp;
env->regs[0] = 0;
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
#endif #endif
...@@ -218,6 +218,15 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -218,6 +218,15 @@ static inline int cpu_mmu_index (CPUState *env)
return !!(env->pregs[PR_CCS] & U_FLAG); return !!(env->pregs[PR_CCS] & U_FLAG);
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->regs[14] = newsp;
env->regs[10] = 0;
}
#endif
/* Support function regs. */ /* Support function regs. */
#define SFR_RW_GC_CFG 0][0 #define SFR_RW_GC_CFG 0][0
#define SFR_RW_MM_CFG env->pregs[PR_SRS]][0 #define SFR_RW_MM_CFG env->pregs[PR_SRS]][0
......
...@@ -734,6 +734,15 @@ typedef struct CCTable { ...@@ -734,6 +734,15 @@ typedef struct CCTable {
extern CCTable cc_table[]; extern CCTable cc_table[];
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->regs[R_ESP] = newsp;
env->regs[R_EAX] = 0;
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
#include "svm.h" #include "svm.h"
......
...@@ -226,6 +226,15 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -226,6 +226,15 @@ static inline int cpu_mmu_index (CPUState *env)
return (env->sr & SR_S) == 0 ? 1 : 0; return (env->sr & SR_S) == 0 ? 1 : 0;
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->aregs[7] = newsp;
env->dregs[0] = 0;
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
#endif #endif
...@@ -500,6 +500,16 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -500,6 +500,16 @@ static inline int cpu_mmu_index (CPUState *env)
return env->hflags & MIPS_HFLAG_KSU; return env->hflags & MIPS_HFLAG_KSU;
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->gpr[env->current_tc][29] = newsp;
env->gpr[env->current_tc][7] = 0;
env->gpr[env->current_tc][2] = 0;
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
/* Memory access type : /* Memory access type :
......
...@@ -822,6 +822,17 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -822,6 +822,17 @@ static inline int cpu_mmu_index (CPUState *env)
return env->mmu_idx; return env->mmu_idx;
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
int i;
if (!newsp)
env->gpr[1] = newsp;
for (i = 7; i < 32; i++)
env->gpr[i] = 0;
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
/*****************************************************************************/ /*****************************************************************************/
......
...@@ -143,6 +143,15 @@ static inline int cpu_mmu_index (CPUState *env) ...@@ -143,6 +143,15 @@ static inline int cpu_mmu_index (CPUState *env)
return (env->sr & SR_MD) == 0 ? 1 : 0; return (env->sr & SR_MD) == 0 ? 1 : 0;
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->gregs[15] = newsp;
env->gregs[0] = 0;
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
/* Memory access type */ /* Memory access type */
......
...@@ -403,6 +403,18 @@ static inline int cpu_fpu_enabled(CPUState *env1) ...@@ -403,6 +403,18 @@ static inline int cpu_fpu_enabled(CPUState *env1)
#endif #endif
} }
#if defined(CONFIG_USER_ONLY)
static inline void cpu_clone_regs(CPUState *env, target_ulong newsp)
{
if (!newsp)
env->regwptr[22] = newsp;
env->regwptr[0] = 0;
/* FIXME: Do we also need to clear CF? */
/* XXXXX */
printf ("HELPME: %s:%d\n", __FILE__, __LINE__);
}
#endif
#include "cpu-all.h" #include "cpu-all.h"
#endif #endif
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册