xstate.h 5.6 KB
Newer Older
1 2 3
#ifndef __ASM_X86_XSAVE_H
#define __ASM_X86_XSAVE_H

4
#include <linux/types.h>
5
#include <asm/processor.h>
6
#include <linux/uaccess.h>
7

8 9
/* Bit 63 of XCR0 is reserved for future expansion */
#define XSTATE_EXTEND_MASK	(~(XSTATE_FPSSE | (1ULL << 63)))
10

I
Ingo Molnar 已提交
11 12
#define XSTATE_CPUID		0x0000000d

13 14
#define FXSAVE_SIZE	512

15 16 17 18 19
#define XSAVE_HDR_SIZE	    64
#define XSAVE_HDR_OFFSET    FXSAVE_SIZE

#define XSAVE_YMM_SIZE	    256
#define XSAVE_YMM_OFFSET    (XSAVE_HDR_SIZE + XSAVE_HDR_OFFSET)
S
Sheng Yang 已提交
20

21
/* Supported features which support lazy state saving */
22 23
#define XSTATE_LAZY	(XSTATE_FP | XSTATE_SSE | XSTATE_YMM		      \
			| XSTATE_OPMASK | XSTATE_ZMM_Hi256 | XSTATE_Hi16_ZMM)
24 25 26 27 28 29

/* Supported features which require eager state saving */
#define XSTATE_EAGER	(XSTATE_BNDREGS | XSTATE_BNDCSR)

/* All currently supported features */
#define XCNTXT_MASK	(XSTATE_LAZY | XSTATE_EAGER)
30

31 32 33 34 35 36
#ifdef CONFIG_X86_64
#define REX_PREFIX	"0x48, "
#else
#define REX_PREFIX
#endif

37
extern unsigned int xstate_size;
38
extern u64 xfeatures_mask;
39
extern u64 xstate_fx_sw_bytes[USER_XSTATE_FX_SW_WORDS];
40

41
extern void update_regset_xstate_info(unsigned int size, u64 xstate_mask);
42

43
/* These macros all use (%edi)/(%rdi) as the single memory argument. */
44 45 46 47 48 49
#define XSAVE		".byte " REX_PREFIX "0x0f,0xae,0x27"
#define XSAVEOPT	".byte " REX_PREFIX "0x0f,0xae,0x37"
#define XSAVES		".byte " REX_PREFIX "0x0f,0xc7,0x2f"
#define XRSTOR		".byte " REX_PREFIX "0x0f,0xae,0x2f"
#define XRSTORS		".byte " REX_PREFIX "0x0f,0xc7,0x1f"

50 51 52 53 54 55 56
#define xstate_fault	".section .fixup,\"ax\"\n"	\
			"3:  movl $-1,%[err]\n"		\
			"    jmp  2b\n"			\
			".previous\n"			\
			_ASM_EXTABLE(1b, 3b)		\
			: [err] "=r" (err)

57 58 59 60
/*
 * This function is called only during boot time when x86 caps are not set
 * up and alternative can not be used yet.
 */
61
static inline int copy_xregs_to_kernel_booting(struct xsave_struct *fx)
62
{
63
	u64 mask = -1;
64 65 66 67 68 69 70 71 72
	u32 lmask = mask;
	u32 hmask = mask >> 32;
	int err = 0;

	WARN_ON(system_state != SYSTEM_BOOTING);

	if (boot_cpu_has(X86_FEATURE_XSAVES))
		asm volatile("1:"XSAVES"\n\t"
			"2:\n\t"
73 74
			     xstate_fault
			: "D" (fx), "m" (*fx), "a" (lmask), "d" (hmask)
75 76 77 78
			:   "memory");
	else
		asm volatile("1:"XSAVE"\n\t"
			"2:\n\t"
79 80
			     xstate_fault
			: "D" (fx), "m" (*fx), "a" (lmask), "d" (hmask)
81 82 83 84 85 86 87 88
			:   "memory");
	return err;
}

/*
 * This function is called only during boot time when x86 caps are not set
 * up and alternative can not be used yet.
 */
89
static inline int copy_kernel_to_xregs_booting(struct xsave_struct *fx, u64 mask)
90 91 92 93 94 95 96 97 98 99
{
	u32 lmask = mask;
	u32 hmask = mask >> 32;
	int err = 0;

	WARN_ON(system_state != SYSTEM_BOOTING);

	if (boot_cpu_has(X86_FEATURE_XSAVES))
		asm volatile("1:"XRSTORS"\n\t"
			"2:\n\t"
100 101
			     xstate_fault
			: "D" (fx), "m" (*fx), "a" (lmask), "d" (hmask)
102 103 104 105
			:   "memory");
	else
		asm volatile("1:"XRSTOR"\n\t"
			"2:\n\t"
106 107
			     xstate_fault
			: "D" (fx), "m" (*fx), "a" (lmask), "d" (hmask)
108 109 110 111
			:   "memory");
	return err;
}

112 113 114
/*
 * Save processor xstate to xsave area.
 */
115
static inline int copy_xregs_to_kernel(struct xsave_struct *fx)
116
{
117
	u64 mask = -1;
118 119 120 121
	u32 lmask = mask;
	u32 hmask = mask >> 32;
	int err = 0;

122
	WARN_ON(!alternatives_patched);
I
Ingo Molnar 已提交
123

124 125 126 127 128 129 130 131 132 133 134 135 136
	/*
	 * If xsaves is enabled, xsaves replaces xsaveopt because
	 * it supports compact format and supervisor states in addition to
	 * modified optimization in xsaveopt.
	 *
	 * Otherwise, if xsaveopt is enabled, xsaveopt replaces xsave
	 * because xsaveopt supports modified optimization which is not
	 * supported by xsave.
	 *
	 * If none of xsaves and xsaveopt is enabled, use xsave.
	 */
	alternative_input_2(
		"1:"XSAVE,
137
		XSAVEOPT,
138
		X86_FEATURE_XSAVEOPT,
139
		XSAVES,
140 141 142 143 144 145 146 147 148 149 150 151 152 153
		X86_FEATURE_XSAVES,
		[fx] "D" (fx), "a" (lmask), "d" (hmask) :
		"memory");
	asm volatile("2:\n\t"
		     xstate_fault
		     : "0" (0)
		     : "memory");

	return err;
}

/*
 * Restore processor xstate from xsave area.
 */
154
static inline int copy_kernel_to_xregs(struct xsave_struct *fx, u64 mask)
155 156 157 158 159 160 161 162 163 164 165
{
	int err = 0;
	u32 lmask = mask;
	u32 hmask = mask >> 32;

	/*
	 * Use xrstors to restore context if it is enabled. xrstors supports
	 * compacted format of xsave area which is not supported by xrstor.
	 */
	alternative_input(
		"1: " XRSTOR,
166
		XRSTORS,
167 168 169 170 171 172 173 174 175 176 177 178
		X86_FEATURE_XSAVES,
		"D" (fx), "m" (*fx), "a" (lmask), "d" (hmask)
		: "memory");

	asm volatile("2:\n"
		     xstate_fault
		     : "0" (0)
		     : "memory");

	return err;
}

179 180 181 182 183 184 185 186 187 188
/*
 * Save xstate to user space xsave area.
 *
 * We don't use modified optimization because xrstor/xrstors might track
 * a different application.
 *
 * We don't use compacted format xsave area for
 * backward compatibility for old applications which don't understand
 * compacted format of xsave area.
 */
189
static inline int copy_xregs_to_user(struct xsave_struct __user *buf)
190 191
{
	int err;
192 193 194 195 196

	/*
	 * Clear the xsave header first, so that reserved fields are
	 * initialized to zero.
	 */
197
	err = __clear_user(&buf->header, sizeof(buf->header));
198 199 200
	if (unlikely(err))
		return -EFAULT;

201
	__asm__ __volatile__(ASM_STAC "\n"
202
			     "1:"XSAVE"\n"
203
			     "2: " ASM_CLAC "\n"
204
			     xstate_fault
205 206 207 208 209
			     : "D" (buf), "a" (-1), "d" (-1), "0" (0)
			     : "memory");
	return err;
}

210 211 212
/*
 * Restore xstate from user space xsave area.
 */
213
static inline int copy_user_to_xregs(struct xsave_struct __user *buf, u64 mask)
214
{
215
	int err = 0;
216
	struct xsave_struct *xstate = ((__force struct xsave_struct *)buf);
217 218
	u32 lmask = mask;
	u32 hmask = mask >> 32;
219

220
	__asm__ __volatile__(ASM_STAC "\n"
221
			     "1:"XRSTOR"\n"
222
			     "2: " ASM_CLAC "\n"
223
			     xstate_fault
224 225 226 227 228
			     : "D" (xstate), "a" (lmask), "d" (hmask), "0" (0)
			     : "memory");	/* memory required? */
	return err;
}

229 230 231
void *get_xsave_addr(struct xsave_struct *xsave, int xstate);
void setup_xstate_comp(void);

232
#endif