aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichael Neuling <mikey@neuling.org>2013-11-20 00:18:54 -0500
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2013-11-29 14:11:52 -0500
commit826b7a8cc660c8412ae6c8c7bb258c875dd0dbc2 (patch)
tree0a9ebdd55886f71839f56960c1dc2fa40336f809
parent3ee8b4e1d91204180081d04c964ee5e0c7387f27 (diff)
powerpc/signals: Mark VSX not saved with small contexts
commit c13f20ac48328b05cd3b8c19e31ed6c132b44b42 upstream. The VSX MSR bit in the user context indicates if the context contains VSX state. Currently we set this when the process has touched VSX at any stage. Unfortunately, if the user has not provided enough space to save the VSX state, we can't save it but we currently still set the MSR VSX bit. This patch changes this to clear the MSR VSX bit when the user doesn't provide enough space. This indicates that there is no valid VSX state in the user context. This is needed to support get/set/make/swapcontext for applications that use VSX but only provide a small context. For example, getcontext in glibc provides a smaller context since the VSX registers don't need to be saved over the glibc function call. But since the program calling getcontext may have used VSX, the kernel currently says the VSX state is valid when it's not. If the returned context is then used in setcontext (ie. a small context without VSX but with MSR VSX set), the kernel will refuse the context. This situation has been reported by the glibc community. Based on patch from Carlos O'Donell. Tested-by: Haren Myneni <haren@linux.vnet.ibm.com> Signed-off-by: Michael Neuling <mikey@neuling.org> Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--arch/powerpc/kernel/signal_32.c10
1 files changed, 9 insertions, 1 deletions
diff --git a/arch/powerpc/kernel/signal_32.c b/arch/powerpc/kernel/signal_32.c
index 0f83122e6676..323309963cd3 100644
--- a/arch/powerpc/kernel/signal_32.c
+++ b/arch/powerpc/kernel/signal_32.c
@@ -454,7 +454,15 @@ static int save_user_regs(struct pt_regs *regs, struct mcontext __user *frame,
454 if (copy_vsx_to_user(&frame->mc_vsregs, current)) 454 if (copy_vsx_to_user(&frame->mc_vsregs, current))
455 return 1; 455 return 1;
456 msr |= MSR_VSX; 456 msr |= MSR_VSX;
457 } 457 } else if (!ctx_has_vsx_region)
458 /*
459 * With a small context structure we can't hold the VSX
460 * registers, hence clear the MSR value to indicate the state
461 * was not saved.
462 */
463 msr &= ~MSR_VSX;
464
465
458#endif /* CONFIG_VSX */ 466#endif /* CONFIG_VSX */
459#ifdef CONFIG_SPE 467#ifdef CONFIG_SPE
460 /* save spe registers */ 468 /* save spe registers */