Commit 25eb44841e4125da67338320d8af0b4859c672de
1 parent
b333af06
better locking - added PowerPC signal handler (add it for the other archs too be…
…cause it needed for full exception support) git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@168 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
30 additions
and
88 deletions
exec-i386.c
| ... | ... | @@ -27,95 +27,16 @@ |
| 27 | 27 | |
| 28 | 28 | /* thread support */ |
| 29 | 29 | |
| 30 | -#ifdef __powerpc__ | |
| 31 | -static inline int testandset (int *p) | |
| 32 | -{ | |
| 33 | - int ret; | |
| 34 | - __asm__ __volatile__ ( | |
| 35 | - "0: lwarx %0,0,%1 ;" | |
| 36 | - " xor. %0,%3,%0;" | |
| 37 | - " bne 1f;" | |
| 38 | - " stwcx. %2,0,%1;" | |
| 39 | - " bne- 0b;" | |
| 40 | - "1: " | |
| 41 | - : "=&r" (ret) | |
| 42 | - : "r" (p), "r" (1), "r" (0) | |
| 43 | - : "cr0", "memory"); | |
| 44 | - return ret; | |
| 45 | -} | |
| 46 | -#endif | |
| 47 | - | |
| 48 | -#ifdef __i386__ | |
| 49 | -static inline int testandset (int *p) | |
| 50 | -{ | |
| 51 | - char ret; | |
| 52 | - long int readval; | |
| 53 | - | |
| 54 | - __asm__ __volatile__ ("lock; cmpxchgl %3, %1; sete %0" | |
| 55 | - : "=q" (ret), "=m" (*p), "=a" (readval) | |
| 56 | - : "r" (1), "m" (*p), "a" (0) | |
| 57 | - : "memory"); | |
| 58 | - return ret; | |
| 59 | -} | |
| 60 | -#endif | |
| 61 | - | |
| 62 | -#ifdef __s390__ | |
| 63 | -static inline int testandset (int *p) | |
| 64 | -{ | |
| 65 | - int ret; | |
| 66 | - | |
| 67 | - __asm__ __volatile__ ("0: cs %0,%1,0(%2)\n" | |
| 68 | - " jl 0b" | |
| 69 | - : "=&d" (ret) | |
| 70 | - : "r" (1), "a" (p), "0" (*p) | |
| 71 | - : "cc", "memory" ); | |
| 72 | - return ret; | |
| 73 | -} | |
| 74 | -#endif | |
| 75 | - | |
| 76 | -#ifdef __alpha__ | |
| 77 | -int testandset (int *p) | |
| 78 | -{ | |
| 79 | - int ret; | |
| 80 | - unsigned long one; | |
| 81 | - | |
| 82 | - __asm__ __volatile__ ("0: mov 1,%2\n" | |
| 83 | - " ldl_l %0,%1\n" | |
| 84 | - " stl_c %2,%1\n" | |
| 85 | - " beq %2,1f\n" | |
| 86 | - ".subsection 2\n" | |
| 87 | - "1: br 0b\n" | |
| 88 | - ".previous" | |
| 89 | - : "=r" (ret), "=m" (*p), "=r" (one) | |
| 90 | - : "m" (*p)); | |
| 91 | - return ret; | |
| 92 | -} | |
| 93 | -#endif | |
| 94 | - | |
| 95 | -#ifdef __sparc__ | |
| 96 | -static inline int testandset (int *p) | |
| 97 | -{ | |
| 98 | - int ret; | |
| 99 | - | |
| 100 | - __asm__ __volatile__("ldstub [%1], %0" | |
| 101 | - : "=r" (ret) | |
| 102 | - : "r" (p) | |
| 103 | - : "memory"); | |
| 104 | - | |
| 105 | - return (ret ? 1 : 0); | |
| 106 | -} | |
| 107 | -#endif | |
| 108 | - | |
| 109 | -int global_cpu_lock = 0; | |
| 30 | +spinlock_t global_cpu_lock = SPIN_LOCK_UNLOCKED; | |
| 110 | 31 | |
| 111 | 32 | void cpu_lock(void) |
| 112 | 33 | { |
| 113 | - while (testandset(&global_cpu_lock)); | |
| 34 | + spin_lock(&global_cpu_lock); | |
| 114 | 35 | } |
| 115 | 36 | |
| 116 | 37 | void cpu_unlock(void) |
| 117 | 38 | { |
| 118 | - global_cpu_lock = 0; | |
| 39 | + spin_unlock(&global_cpu_lock); | |
| 119 | 40 | } |
| 120 | 41 | |
| 121 | 42 | /* exception support */ |
| ... | ... | @@ -292,16 +213,16 @@ int cpu_x86_exec(CPUX86State *env1) |
| 292 | 213 | flags); |
| 293 | 214 | if (!tb) { |
| 294 | 215 | /* if no translated code available, then translate it now */ |
| 295 | - /* XXX: very inefficient: we lock all the cpus when | |
| 296 | - generating code */ | |
| 297 | - cpu_lock(); | |
| 216 | + /* very inefficient but safe: we lock all the cpus | |
| 217 | + when generating code */ | |
| 218 | + spin_lock(&tb_lock); | |
| 298 | 219 | tc_ptr = code_gen_ptr; |
| 299 | 220 | ret = cpu_x86_gen_code(code_gen_ptr, CODE_GEN_MAX_SIZE, |
| 300 | 221 | &code_gen_size, pc, cs_base, flags, |
| 301 | 222 | &code_size); |
| 302 | 223 | /* if invalid instruction, signal it */ |
| 303 | 224 | if (ret != 0) { |
| 304 | - cpu_unlock(); | |
| 225 | + spin_unlock(&tb_lock); | |
| 305 | 226 | raise_exception(EXCP06_ILLOP); |
| 306 | 227 | } |
| 307 | 228 | tb = tb_alloc((unsigned long)pc, code_size); |
| ... | ... | @@ -311,7 +232,7 @@ int cpu_x86_exec(CPUX86State *env1) |
| 311 | 232 | tb->tc_ptr = tc_ptr; |
| 312 | 233 | tb->hash_next = NULL; |
| 313 | 234 | code_gen_ptr = (void *)(((unsigned long)code_gen_ptr + code_gen_size + CODE_GEN_ALIGN - 1) & ~(CODE_GEN_ALIGN - 1)); |
| 314 | - cpu_unlock(); | |
| 235 | + spin_unlock(&tb_lock); | |
| 315 | 236 | } |
| 316 | 237 | #ifdef DEBUG_EXEC |
| 317 | 238 | if (loglevel) { |
| ... | ... | @@ -412,6 +333,7 @@ static inline int handle_cpu_signal(unsigned long pc, |
| 412 | 333 | printf("qemu: SIGSEGV pc=0x%08lx address=%08lx wr=%d oldset=0x%08lx\n", |
| 413 | 334 | pc, address, is_write, *(unsigned long *)old_set); |
| 414 | 335 | #endif |
| 336 | + /* XXX: locking issue */ | |
| 415 | 337 | if (is_write && page_unprotect(address)) { |
| 416 | 338 | sigprocmask(SIG_SETMASK, old_set, NULL); |
| 417 | 339 | return 1; |
| ... | ... | @@ -454,8 +376,28 @@ int cpu_x86_signal_handler(int host_signum, struct siginfo *info, |
| 454 | 376 | uc->uc_mcontext.gregs[REG_TRAPNO] == 0xe ? |
| 455 | 377 | (uc->uc_mcontext.gregs[REG_ERR] >> 1) & 1 : 0, |
| 456 | 378 | pold_set); |
| 379 | +#elif defined(__powerpc) | |
| 380 | + struct ucontext *uc = puc; | |
| 381 | + struct pt_regs *regs = uc->uc_mcontext.regs; | |
| 382 | + unsigned long pc; | |
| 383 | + sigset_t *pold_set; | |
| 384 | + int is_write; | |
| 385 | + | |
| 386 | + pc = regs->nip; | |
| 387 | + pold_set = &uc->uc_sigmask; | |
| 388 | + is_write = 0; | |
| 389 | +#if 0 | |
| 390 | + /* ppc 4xx case */ | |
| 391 | + if (regs->dsisr & 0x00800000) | |
| 392 | + is_write = 1; | |
| 393 | +#else | |
| 394 | + if (regs->trap != 0x400 && (regs->dsisr & 0x02000000)) | |
| 395 | + is_write = 1; | |
| 396 | +#endif | |
| 397 | + return handle_cpu_signal(pc, (unsigned long)info->si_addr, | |
| 398 | + is_write, pold_set); | |
| 457 | 399 | #else |
| 458 | -#warning No CPU specific signal handler: cannot handle target SIGSEGV events | |
| 400 | +#error CPU specific signal handler needed | |
| 459 | 401 | return 0; |
| 460 | 402 | #endif |
| 461 | 403 | } | ... | ... |