mirror of
https://github.com/torvalds/linux.git
synced 2025-01-01 07:42:07 +00:00
63bcff2a30
When Supervisor Mode Access Prevention (SMAP) is enabled, access to userspace from the kernel is controlled by the AC flag. To make the performance of manipulating that flag acceptable, there are two new instructions, STAC and CLAC, to set and clear it. This patch adds those instructions, via alternative(), when the SMAP feature is enabled. It also adds X86_EFLAGS_AC unconditionally to the SYSCALL entry mask; there is simply no reason to make that one conditional. Signed-off-by: H. Peter Anvin <hpa@linux.intel.com> Link: http://lkml.kernel.org/r/1348256595-29119-9-git-send-email-hpa@linux.intel.com
92 lines
2.1 KiB
C
92 lines
2.1 KiB
C
/*
|
|
* Supervisor Mode Access Prevention support
|
|
*
|
|
* Copyright (C) 2012 Intel Corporation
|
|
* Author: H. Peter Anvin <hpa@linux.intel.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; version 2
|
|
* of the License.
|
|
*/
|
|
|
|
#ifndef _ASM_X86_SMAP_H
|
|
#define _ASM_X86_SMAP_H
|
|
|
|
#include <linux/stringify.h>
|
|
#include <asm/nops.h>
|
|
#include <asm/cpufeature.h>
|
|
|
|
/* "Raw" instruction opcodes */
|
|
#define __ASM_CLAC .byte 0x0f,0x01,0xca
|
|
#define __ASM_STAC .byte 0x0f,0x01,0xcb
|
|
|
|
#ifdef __ASSEMBLY__
|
|
|
|
#include <asm/alternative-asm.h>
|
|
|
|
#ifdef CONFIG_X86_SMAP
|
|
|
|
#define ASM_CLAC \
|
|
661: ASM_NOP3 ; \
|
|
.pushsection .altinstr_replacement, "ax" ; \
|
|
662: __ASM_CLAC ; \
|
|
.popsection ; \
|
|
.pushsection .altinstructions, "a" ; \
|
|
altinstruction_entry 661b, 662b, X86_FEATURE_SMAP, 3, 3 ; \
|
|
.popsection
|
|
|
|
#define ASM_STAC \
|
|
661: ASM_NOP3 ; \
|
|
.pushsection .altinstr_replacement, "ax" ; \
|
|
662: __ASM_STAC ; \
|
|
.popsection ; \
|
|
.pushsection .altinstructions, "a" ; \
|
|
altinstruction_entry 661b, 662b, X86_FEATURE_SMAP, 3, 3 ; \
|
|
.popsection
|
|
|
|
#else /* CONFIG_X86_SMAP */
|
|
|
|
#define ASM_CLAC
|
|
#define ASM_STAC
|
|
|
|
#endif /* CONFIG_X86_SMAP */
|
|
|
|
#else /* __ASSEMBLY__ */
|
|
|
|
#include <asm/alternative.h>
|
|
|
|
#ifdef CONFIG_X86_SMAP
|
|
|
|
static __always_inline void clac(void)
|
|
{
|
|
/* Note: a barrier is implicit in alternative() */
|
|
alternative(ASM_NOP3, __stringify(__ASM_CLAC), X86_FEATURE_SMAP);
|
|
}
|
|
|
|
static __always_inline void stac(void)
|
|
{
|
|
/* Note: a barrier is implicit in alternative() */
|
|
alternative(ASM_NOP3, __stringify(__ASM_STAC), X86_FEATURE_SMAP);
|
|
}
|
|
|
|
/* These macros can be used in asm() statements */
|
|
#define ASM_CLAC \
|
|
ALTERNATIVE(ASM_NOP3, __stringify(__ASM_CLAC), X86_FEATURE_SMAP)
|
|
#define ASM_STAC \
|
|
ALTERNATIVE(ASM_NOP3, __stringify(__ASM_STAC), X86_FEATURE_SMAP)
|
|
|
|
#else /* CONFIG_X86_SMAP */
|
|
|
|
static inline void clac(void) { }
|
|
static inline void stac(void) { }
|
|
|
|
#define ASM_CLAC
|
|
#define ASM_STAC
|
|
|
|
#endif /* CONFIG_X86_SMAP */
|
|
|
|
#endif /* __ASSEMBLY__ */
|
|
|
|
#endif /* _ASM_X86_SMAP_H */
|