x86/mce: Fix check for processor context when machine check was taken.
Linus pointed out that there was no value is checking whether m->ip was zero - because zero is a legimate value. If we have a reliable (or faked in the VM86 case) "m->cs" we can use it to tell whether we were in user mode or kernelwhen the machine check hit. Reported-by: Linus Torvalds <torvalds@linuxfoundation.org> Cc: <stable@vger.kernel.org> Signed-off-by: Tony Luck <tony.luck@intel.com>
This commit is contained in:
@@ -165,15 +165,19 @@ static struct severity {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the EIPV bit is set, it means the saved IP is the
|
* If mcgstatus indicated that ip/cs on the stack were
|
||||||
* instruction which caused the MCE.
|
* no good, then "m->cs" will be zero and we will have
|
||||||
|
* to assume the worst case (IN_KERNEL) as we actually
|
||||||
|
* have no idea what we were executing when the machine
|
||||||
|
* check hit.
|
||||||
|
* If we do have a good "m->cs" (or a faked one in the
|
||||||
|
* case we were executing in VM86 mode) we can use it to
|
||||||
|
* distinguish an exception taken in user from from one
|
||||||
|
* taken in the kernel.
|
||||||
*/
|
*/
|
||||||
static int error_context(struct mce *m)
|
static int error_context(struct mce *m)
|
||||||
{
|
{
|
||||||
if (m->mcgstatus & MCG_STATUS_EIPV)
|
return ((m->cs & 3) == 3) ? IN_USER : IN_KERNEL;
|
||||||
return (m->ip && (m->cs & 3) == 3) ? IN_USER : IN_KERNEL;
|
|
||||||
/* Unknown, assume kernel */
|
|
||||||
return IN_KERNEL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int mce_severity(struct mce *m, int tolerant, char **msg)
|
int mce_severity(struct mce *m, int tolerant, char **msg)
|
||||||
|
Reference in New Issue
Block a user