Commit f9f57267 authored by Benjamin Herrenschmidt's avatar Benjamin Herrenschmidt Committed by Linus Torvalds

[PATCH] ppc32: Even more preempt fixes

Add a warning if enable_kernel_{fp,altivec} is called with preempt
enabled since this is always an error, and make sure the alignement
exception handler properly disables preempt when doing FP operations.
parent ff1581f7
...@@ -325,14 +325,18 @@ fix_alignment(struct pt_regs *regs) ...@@ -325,14 +325,18 @@ fix_alignment(struct pt_regs *regs)
* the kernel with -msoft-float so it doesn't use the * the kernel with -msoft-float so it doesn't use the
* fp regs for copying 8-byte objects. */ * fp regs for copying 8-byte objects. */
case LD+F+S: case LD+F+S:
preempt_disable();
enable_kernel_fp(); enable_kernel_fp();
cvt_fd(&data.f, &current->thread.fpr[reg], &current->thread.fpscr); cvt_fd(&data.f, &current->thread.fpr[reg], &current->thread.fpscr);
/* current->thread.fpr[reg] = data.f; */ /* current->thread.fpr[reg] = data.f; */
preempt_enable();
break; break;
case ST+F+S: case ST+F+S:
preempt_disable();
enable_kernel_fp(); enable_kernel_fp();
cvt_df(&current->thread.fpr[reg], &data.f, &current->thread.fpscr); cvt_df(&current->thread.fpr[reg], &data.f, &current->thread.fpscr);
/* data.f = current->thread.fpr[reg]; */ /* data.f = current->thread.fpr[reg]; */
preempt_enable();
break; break;
default: default:
printk("align: can't handle flags=%x\n", flags); printk("align: can't handle flags=%x\n", flags);
......
...@@ -163,7 +163,8 @@ dump_altivec(struct pt_regs *regs, elf_vrregset_t *vrregs) ...@@ -163,7 +163,8 @@ dump_altivec(struct pt_regs *regs, elf_vrregset_t *vrregs)
void void
enable_kernel_altivec(void) enable_kernel_altivec(void)
{ {
preempt_disable(); WARN_ON(current_thread_info()->preempt_count == 0 && !irqs_disabled());
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
if (current->thread.regs && (current->thread.regs->msr & MSR_VEC)) if (current->thread.regs && (current->thread.regs->msr & MSR_VEC))
giveup_altivec(current); giveup_altivec(current);
...@@ -172,14 +173,15 @@ enable_kernel_altivec(void) ...@@ -172,14 +173,15 @@ enable_kernel_altivec(void)
#else #else
giveup_altivec(last_task_used_altivec); giveup_altivec(last_task_used_altivec);
#endif /* __SMP __ */ #endif /* __SMP __ */
preempt_enable();
} }
EXPORT_SYMBOL(enable_kernel_altivec);
#endif /* CONFIG_ALTIVEC */ #endif /* CONFIG_ALTIVEC */
void void
enable_kernel_fp(void) enable_kernel_fp(void)
{ {
preempt_disable(); WARN_ON(current_thread_info()->preempt_count == 0 && !irqs_disabled());
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
if (current->thread.regs && (current->thread.regs->msr & MSR_FP)) if (current->thread.regs && (current->thread.regs->msr & MSR_FP))
giveup_fpu(current); giveup_fpu(current);
...@@ -188,8 +190,8 @@ enable_kernel_fp(void) ...@@ -188,8 +190,8 @@ enable_kernel_fp(void)
#else #else
giveup_fpu(last_task_used_math); giveup_fpu(last_task_used_math);
#endif /* CONFIG_SMP */ #endif /* CONFIG_SMP */
preempt_enable();
} }
EXPORT_SYMBOL(enable_kernel_fp);
int int
dump_task_fpu(struct task_struct *tsk, elf_fpregset_t *fpregs) dump_task_fpu(struct task_struct *tsk, elf_fpregset_t *fpregs)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment