diff options
Diffstat (limited to 'arch/x86/include/asm')
45 files changed, 690 insertions, 162 deletions
| diff --git a/arch/x86/include/asm/acpi.h b/arch/x86/include/asm/acpi.h index b31bf97775f..2dfac58f3b1 100644 --- a/arch/x86/include/asm/acpi.h +++ b/arch/x86/include/asm/acpi.h @@ -111,7 +111,7 @@ static inline void acpi_disable_pci(void)  }  /* Low-level suspend routine. */ -extern int acpi_suspend_lowlevel(void); +extern int (*acpi_suspend_lowlevel)(void);  /* Physical address to resume after wakeup */  #define acpi_wakeup_address ((unsigned long)(real_mode_header->wakeup_start)) diff --git a/arch/x86/include/asm/apic.h b/arch/x86/include/asm/apic.h index 33880342223..f8119b582c3 100644 --- a/arch/x86/include/asm/apic.h +++ b/arch/x86/include/asm/apic.h @@ -12,6 +12,7 @@  #include <asm/fixmap.h>  #include <asm/mpspec.h>  #include <asm/msr.h> +#include <asm/idle.h>  #define ARCH_APICTIMER_STOPS_ON_C3	1 @@ -687,5 +688,31 @@ extern int default_check_phys_apicid_present(int phys_apicid);  #endif  #endif /* CONFIG_X86_LOCAL_APIC */ +extern void irq_enter(void); +extern void irq_exit(void); + +static inline void entering_irq(void) +{ +	irq_enter(); +	exit_idle(); +} + +static inline void entering_ack_irq(void) +{ +	ack_APIC_irq(); +	entering_irq(); +} + +static inline void exiting_irq(void) +{ +	irq_exit(); +} + +static inline void exiting_ack_irq(void) +{ +	irq_exit(); +	/* Ack only at the end to avoid potential reentry */ +	ack_APIC_irq(); +}  #endif /* _ASM_X86_APIC_H */ diff --git a/arch/x86/include/asm/cpu.h b/arch/x86/include/asm/cpu.h index 5f9a1243190..d2b12988d2e 100644 --- a/arch/x86/include/asm/cpu.h +++ b/arch/x86/include/asm/cpu.h @@ -28,7 +28,7 @@ struct x86_cpu {  #ifdef CONFIG_HOTPLUG_CPU  extern int arch_register_cpu(int num);  extern void arch_unregister_cpu(int); -extern void __cpuinit start_cpu0(void); +extern void start_cpu0(void);  #ifdef CONFIG_DEBUG_HOTPLUG_CPU0  extern int _debug_hotplug_cpu(int cpu, int action);  #endif diff --git a/arch/x86/include/asm/cpufeature.h b/arch/x86/include/asm/cpufeature.h index e99ac27f95b..47538a61c91 100644 --- a/arch/x86/include/asm/cpufeature.h +++ b/arch/x86/include/asm/cpufeature.h @@ -92,7 +92,7 @@  #define X86_FEATURE_LFENCE_RDTSC (3*32+18) /* "" Lfence synchronizes RDTSC */  #define X86_FEATURE_11AP	(3*32+19) /* "" Bad local APIC aka 11AP */  #define X86_FEATURE_NOPL	(3*32+20) /* The NOPL (0F 1F) instructions */ -					  /* 21 available, was AMD_C1E */ +#define X86_FEATURE_ALWAYS	(3*32+21) /* "" Always-present feature */  #define X86_FEATURE_XTOPOLOGY	(3*32+22) /* cpu topology enum extensions */  #define X86_FEATURE_TSC_RELIABLE (3*32+23) /* TSC is known to be reliable */  #define X86_FEATURE_NONSTOP_TSC	(3*32+24) /* TSC does not stop in C states */ @@ -356,15 +356,36 @@ extern const char * const x86_power_flags[32];  #endif /* CONFIG_X86_64 */  #if __GNUC__ >= 4 +extern void warn_pre_alternatives(void); +extern bool __static_cpu_has_safe(u16 bit); +  /*   * Static testing of CPU features.  Used the same as boot_cpu_has().   * These are only valid after alternatives have run, but will statically   * patch the target code for additional performance. - *   */  static __always_inline __pure bool __static_cpu_has(u16 bit)  {  #if __GNUC__ > 4 || __GNUC_MINOR__ >= 5 + +#ifdef CONFIG_X86_DEBUG_STATIC_CPU_HAS +		/* +		 * Catch too early usage of this before alternatives +		 * have run. +		 */ +		asm goto("1: jmp %l[t_warn]\n" +			 "2:\n" +			 ".section .altinstructions,\"a\"\n" +			 " .long 1b - .\n" +			 " .long 0\n"		/* no replacement */ +			 " .word %P0\n"		/* 1: do replace */ +			 " .byte 2b - 1b\n"	/* source len */ +			 " .byte 0\n"		/* replacement len */ +			 ".previous\n" +			 /* skipping size check since replacement size = 0 */ +			 : : "i" (X86_FEATURE_ALWAYS) : : t_warn); +#endif +  		asm goto("1: jmp %l[t_no]\n"  			 "2:\n"  			 ".section .altinstructions,\"a\"\n" @@ -379,7 +400,13 @@ static __always_inline __pure bool __static_cpu_has(u16 bit)  		return true;  	t_no:  		return false; -#else + +#ifdef CONFIG_X86_DEBUG_STATIC_CPU_HAS +	t_warn: +		warn_pre_alternatives(); +		return false; +#endif +#else /* GCC_VERSION >= 40500 */  		u8 flag;  		/* Open-coded due to __stringify() in ALTERNATIVE() */  		asm volatile("1: movb $0,%0\n" @@ -411,11 +438,94 @@ static __always_inline __pure bool __static_cpu_has(u16 bit)  		__static_cpu_has(bit) :				\  		boot_cpu_has(bit)				\  ) + +static __always_inline __pure bool _static_cpu_has_safe(u16 bit) +{ +#if __GNUC__ > 4 || __GNUC_MINOR__ >= 5 +/* + * We need to spell the jumps to the compiler because, depending on the offset, + * the replacement jump can be bigger than the original jump, and this we cannot + * have. Thus, we force the jump to the widest, 4-byte, signed relative + * offset even though the last would often fit in less bytes. + */ +		asm goto("1: .byte 0xe9\n .long %l[t_dynamic] - 2f\n" +			 "2:\n" +			 ".section .altinstructions,\"a\"\n" +			 " .long 1b - .\n"		/* src offset */ +			 " .long 3f - .\n"		/* repl offset */ +			 " .word %P1\n"			/* always replace */ +			 " .byte 2b - 1b\n"		/* src len */ +			 " .byte 4f - 3f\n"		/* repl len */ +			 ".previous\n" +			 ".section .altinstr_replacement,\"ax\"\n" +			 "3: .byte 0xe9\n .long %l[t_no] - 2b\n" +			 "4:\n" +			 ".previous\n" +			 ".section .altinstructions,\"a\"\n" +			 " .long 1b - .\n"		/* src offset */ +			 " .long 0\n"			/* no replacement */ +			 " .word %P0\n"			/* feature bit */ +			 " .byte 2b - 1b\n"		/* src len */ +			 " .byte 0\n"			/* repl len */ +			 ".previous\n" +			 : : "i" (bit), "i" (X86_FEATURE_ALWAYS) +			 : : t_dynamic, t_no); +		return true; +	t_no: +		return false; +	t_dynamic: +		return __static_cpu_has_safe(bit); +#else /* GCC_VERSION >= 40500 */ +		u8 flag; +		/* Open-coded due to __stringify() in ALTERNATIVE() */ +		asm volatile("1: movb $2,%0\n" +			     "2:\n" +			     ".section .altinstructions,\"a\"\n" +			     " .long 1b - .\n"		/* src offset */ +			     " .long 3f - .\n"		/* repl offset */ +			     " .word %P2\n"		/* always replace */ +			     " .byte 2b - 1b\n"		/* source len */ +			     " .byte 4f - 3f\n"		/* replacement len */ +			     ".previous\n" +			     ".section .discard,\"aw\",@progbits\n" +			     " .byte 0xff + (4f-3f) - (2b-1b)\n" /* size check */ +			     ".previous\n" +			     ".section .altinstr_replacement,\"ax\"\n" +			     "3: movb $0,%0\n" +			     "4:\n" +			     ".previous\n" +			     ".section .altinstructions,\"a\"\n" +			     " .long 1b - .\n"		/* src offset */ +			     " .long 5f - .\n"		/* repl offset */ +			     " .word %P1\n"		/* feature bit */ +			     " .byte 4b - 3b\n"		/* src len */ +			     " .byte 6f - 5f\n"		/* repl len */ +			     ".previous\n" +			     ".section .discard,\"aw\",@progbits\n" +			     " .byte 0xff + (6f-5f) - (4b-3b)\n" /* size check */ +			     ".previous\n" +			     ".section .altinstr_replacement,\"ax\"\n" +			     "5: movb $1,%0\n" +			     "6:\n" +			     ".previous\n" +			     : "=qm" (flag) +			     : "i" (bit), "i" (X86_FEATURE_ALWAYS)); +		return (flag == 2 ? __static_cpu_has_safe(bit) : flag); +#endif +} + +#define static_cpu_has_safe(bit)				\ +(								\ +	__builtin_constant_p(boot_cpu_has(bit)) ?		\ +		boot_cpu_has(bit) :				\ +		_static_cpu_has_safe(bit)			\ +)  #else  /*   * gcc 3.x is too stupid to do the static test; fall back to dynamic.   */ -#define static_cpu_has(bit) boot_cpu_has(bit) +#define static_cpu_has(bit)		boot_cpu_has(bit) +#define static_cpu_has_safe(bit)	boot_cpu_has(bit)  #endif  #define cpu_has_bug(c, bit)	cpu_has(c, (bit)) diff --git a/arch/x86/include/asm/crypto/blowfish.h b/arch/x86/include/asm/crypto/blowfish.h deleted file mode 100644 index f097b2face1..00000000000 --- a/arch/x86/include/asm/crypto/blowfish.h +++ /dev/null @@ -1,43 +0,0 @@ -#ifndef ASM_X86_BLOWFISH_H -#define ASM_X86_BLOWFISH_H - -#include <linux/crypto.h> -#include <crypto/blowfish.h> - -#define BF_PARALLEL_BLOCKS 4 - -/* regular block cipher functions */ -asmlinkage void __blowfish_enc_blk(struct bf_ctx *ctx, u8 *dst, const u8 *src, -				   bool xor); -asmlinkage void blowfish_dec_blk(struct bf_ctx *ctx, u8 *dst, const u8 *src); - -/* 4-way parallel cipher functions */ -asmlinkage void __blowfish_enc_blk_4way(struct bf_ctx *ctx, u8 *dst, -					const u8 *src, bool xor); -asmlinkage void blowfish_dec_blk_4way(struct bf_ctx *ctx, u8 *dst, -				      const u8 *src); - -static inline void blowfish_enc_blk(struct bf_ctx *ctx, u8 *dst, const u8 *src) -{ -	__blowfish_enc_blk(ctx, dst, src, false); -} - -static inline void blowfish_enc_blk_xor(struct bf_ctx *ctx, u8 *dst, -					const u8 *src) -{ -	__blowfish_enc_blk(ctx, dst, src, true); -} - -static inline void blowfish_enc_blk_4way(struct bf_ctx *ctx, u8 *dst, -					 const u8 *src) -{ -	__blowfish_enc_blk_4way(ctx, dst, src, false); -} - -static inline void blowfish_enc_blk_xor_4way(struct bf_ctx *ctx, u8 *dst, -				      const u8 *src) -{ -	__blowfish_enc_blk_4way(ctx, dst, src, true); -} - -#endif diff --git a/arch/x86/include/asm/crypto/twofish.h b/arch/x86/include/asm/crypto/twofish.h index e655c6029b4..878c51ceebb 100644 --- a/arch/x86/include/asm/crypto/twofish.h +++ b/arch/x86/include/asm/crypto/twofish.h @@ -28,20 +28,6 @@ asmlinkage void __twofish_enc_blk_3way(struct twofish_ctx *ctx, u8 *dst,  asmlinkage void twofish_dec_blk_3way(struct twofish_ctx *ctx, u8 *dst,  				     const u8 *src); -/* 8-way parallel cipher functions */ -asmlinkage void twofish_ecb_enc_8way(struct twofish_ctx *ctx, u8 *dst, -				     const u8 *src); -asmlinkage void twofish_ecb_dec_8way(struct twofish_ctx *ctx, u8 *dst, -				     const u8 *src); -asmlinkage void twofish_cbc_dec_8way(struct twofish_ctx *ctx, u8 *dst, -				     const u8 *src); -asmlinkage void twofish_ctr_8way(struct twofish_ctx *ctx, u8 *dst, -				 const u8 *src, le128 *iv); -asmlinkage void twofish_xts_enc_8way(struct twofish_ctx *ctx, u8 *dst, -				     const u8 *src, le128 *iv); -asmlinkage void twofish_xts_dec_8way(struct twofish_ctx *ctx, u8 *dst, -				     const u8 *src, le128 *iv); -  /* helpers from twofish_x86_64-3way module */  extern void twofish_dec_blk_cbc_3way(void *ctx, u128 *dst, const u128 *src);  extern void twofish_enc_blk_ctr(void *ctx, u128 *dst, const u128 *src, @@ -57,8 +43,4 @@ extern void lrw_twofish_exit_tfm(struct crypto_tfm *tfm);  extern int xts_twofish_setkey(struct crypto_tfm *tfm, const u8 *key,  			      unsigned int keylen); -/* helpers from twofish-avx module */ -extern void twofish_xts_enc(void *ctx, u128 *dst, const u128 *src, le128 *iv); -extern void twofish_xts_dec(void *ctx, u128 *dst, const u128 *src, le128 *iv); -  #endif /* ASM_X86_TWOFISH_H */ diff --git a/arch/x86/include/asm/desc.h b/arch/x86/include/asm/desc.h index 8bf1c06070d..b90e5dfeee4 100644 --- a/arch/x86/include/asm/desc.h +++ b/arch/x86/include/asm/desc.h @@ -36,8 +36,8 @@ static inline void fill_ldt(struct desc_struct *desc, const struct user_desc *in  extern struct desc_ptr idt_descr;  extern gate_desc idt_table[]; -extern struct desc_ptr nmi_idt_descr; -extern gate_desc nmi_idt_table[]; +extern struct desc_ptr debug_idt_descr; +extern gate_desc debug_idt_table[];  struct gdt_page {  	struct desc_struct gdt[GDT_ENTRIES]; @@ -316,7 +316,20 @@ static inline void set_nmi_gate(int gate, void *addr)  	gate_desc s;  	pack_gate(&s, GATE_INTERRUPT, (unsigned long)addr, 0, 0, __KERNEL_CS); -	write_idt_entry(nmi_idt_table, gate, &s); +	write_idt_entry(debug_idt_table, gate, &s); +} +#endif + +#ifdef CONFIG_TRACING +extern struct desc_ptr trace_idt_descr; +extern gate_desc trace_idt_table[]; +static inline void write_trace_idt_entry(int entry, const gate_desc *gate) +{ +	write_idt_entry(trace_idt_table, entry, gate); +} +#else +static inline void write_trace_idt_entry(int entry, const gate_desc *gate) +{  }  #endif @@ -331,6 +344,7 @@ static inline void _set_gate(int gate, unsigned type, void *addr,  	 * setup time  	 */  	write_idt_entry(idt_table, gate, &s); +	write_trace_idt_entry(gate, &s);  }  /* @@ -360,12 +374,39 @@ static inline void alloc_system_vector(int vector)  	}  } -static inline void alloc_intr_gate(unsigned int n, void *addr) +#ifdef CONFIG_TRACING +static inline void trace_set_intr_gate(unsigned int gate, void *addr) +{ +	gate_desc s; + +	pack_gate(&s, GATE_INTERRUPT, (unsigned long)addr, 0, 0, __KERNEL_CS); +	write_idt_entry(trace_idt_table, gate, &s); +} + +static inline void __trace_alloc_intr_gate(unsigned int n, void *addr) +{ +	trace_set_intr_gate(n, addr); +} +#else +static inline void trace_set_intr_gate(unsigned int gate, void *addr) +{ +} + +#define __trace_alloc_intr_gate(n, addr) +#endif + +static inline void __alloc_intr_gate(unsigned int n, void *addr)  { -	alloc_system_vector(n);  	set_intr_gate(n, addr);  } +#define alloc_intr_gate(n, addr)				\ +	do {							\ +		alloc_system_vector(n);				\ +		__alloc_intr_gate(n, addr);			\ +		__trace_alloc_intr_gate(n, trace_##addr);	\ +	} while (0) +  /*   * This routine sets up an interrupt gate at directory privilege level 3.   */ @@ -405,4 +446,70 @@ static inline void set_system_intr_gate_ist(int n, void *addr, unsigned ist)  	_set_gate(n, GATE_INTERRUPT, addr, 0x3, ist, __KERNEL_CS);  } +#ifdef CONFIG_X86_64 +DECLARE_PER_CPU(u32, debug_idt_ctr); +static inline bool is_debug_idt_enabled(void) +{ +	if (this_cpu_read(debug_idt_ctr)) +		return true; + +	return false; +} + +static inline void load_debug_idt(void) +{ +	load_idt((const struct desc_ptr *)&debug_idt_descr); +} +#else +static inline bool is_debug_idt_enabled(void) +{ +	return false; +} + +static inline void load_debug_idt(void) +{ +} +#endif + +#ifdef CONFIG_TRACING +extern atomic_t trace_idt_ctr; +static inline bool is_trace_idt_enabled(void) +{ +	if (atomic_read(&trace_idt_ctr)) +		return true; + +	return false; +} + +static inline void load_trace_idt(void) +{ +	load_idt((const struct desc_ptr *)&trace_idt_descr); +} +#else +static inline bool is_trace_idt_enabled(void) +{ +	return false; +} + +static inline void load_trace_idt(void) +{ +} +#endif + +/* + * The load_current_idt() must be called with interrupts disabled + * to avoid races. That way the IDT will always be set back to the expected + * descriptor. It's also called when a CPU is being initialized, and + * that doesn't need to disable interrupts, as nothing should be + * bothering the CPU then. + */ +static inline void load_current_idt(void) +{ +	if (is_debug_idt_enabled()) +		load_debug_idt(); +	else if (is_trace_idt_enabled()) +		load_trace_idt(); +	else +		load_idt((const struct desc_ptr *)&idt_descr); +}  #endif /* _ASM_X86_DESC_H */ diff --git a/arch/x86/include/asm/efi.h b/arch/x86/include/asm/efi.h index 60c89f30c72..0062a012504 100644 --- a/arch/x86/include/asm/efi.h +++ b/arch/x86/include/asm/efi.h @@ -52,40 +52,40 @@ extern u64 efi_call6(void *fp, u64 arg1, u64 arg2, u64 arg3,  		     u64 arg4, u64 arg5, u64 arg6);  #define efi_call_phys0(f)			\ -	efi_call0((void *)(f)) +	efi_call0((f))  #define efi_call_phys1(f, a1)			\ -	efi_call1((void *)(f), (u64)(a1)) +	efi_call1((f), (u64)(a1))  #define efi_call_phys2(f, a1, a2)			\ -	efi_call2((void *)(f), (u64)(a1), (u64)(a2)) +	efi_call2((f), (u64)(a1), (u64)(a2))  #define efi_call_phys3(f, a1, a2, a3)				\ -	efi_call3((void *)(f), (u64)(a1), (u64)(a2), (u64)(a3)) +	efi_call3((f), (u64)(a1), (u64)(a2), (u64)(a3))  #define efi_call_phys4(f, a1, a2, a3, a4)				\ -	efi_call4((void *)(f), (u64)(a1), (u64)(a2), (u64)(a3),		\ +	efi_call4((f), (u64)(a1), (u64)(a2), (u64)(a3),		\  		  (u64)(a4))  #define efi_call_phys5(f, a1, a2, a3, a4, a5)				\ -	efi_call5((void *)(f), (u64)(a1), (u64)(a2), (u64)(a3),		\ +	efi_call5((f), (u64)(a1), (u64)(a2), (u64)(a3),		\  		  (u64)(a4), (u64)(a5))  #define efi_call_phys6(f, a1, a2, a3, a4, a5, a6)			\ -	efi_call6((void *)(f), (u64)(a1), (u64)(a2), (u64)(a3),		\ +	efi_call6((f), (u64)(a1), (u64)(a2), (u64)(a3),		\  		  (u64)(a4), (u64)(a5), (u64)(a6))  #define efi_call_virt0(f)				\ -	efi_call0((void *)(efi.systab->runtime->f)) +	efi_call0((efi.systab->runtime->f))  #define efi_call_virt1(f, a1)					\ -	efi_call1((void *)(efi.systab->runtime->f), (u64)(a1)) +	efi_call1((efi.systab->runtime->f), (u64)(a1))  #define efi_call_virt2(f, a1, a2)					\ -	efi_call2((void *)(efi.systab->runtime->f), (u64)(a1), (u64)(a2)) +	efi_call2((efi.systab->runtime->f), (u64)(a1), (u64)(a2))  #define efi_call_virt3(f, a1, a2, a3)					\ -	efi_call3((void *)(efi.systab->runtime->f), (u64)(a1), (u64)(a2), \ +	efi_call3((efi.systab->runtime->f), (u64)(a1), (u64)(a2), \  		  (u64)(a3))  #define efi_call_virt4(f, a1, a2, a3, a4)				\ -	efi_call4((void *)(efi.systab->runtime->f), (u64)(a1), (u64)(a2), \ +	efi_call4((efi.systab->runtime->f), (u64)(a1), (u64)(a2), \  		  (u64)(a3), (u64)(a4))  #define efi_call_virt5(f, a1, a2, a3, a4, a5)				\ -	efi_call5((void *)(efi.systab->runtime->f), (u64)(a1), (u64)(a2), \ +	efi_call5((efi.systab->runtime->f), (u64)(a1), (u64)(a2), \  		  (u64)(a3), (u64)(a4), (u64)(a5))  #define efi_call_virt6(f, a1, a2, a3, a4, a5, a6)			\ -	efi_call6((void *)(efi.systab->runtime->f), (u64)(a1), (u64)(a2), \ +	efi_call6((efi.systab->runtime->f), (u64)(a1), (u64)(a2), \  		  (u64)(a3), (u64)(a4), (u64)(a5), (u64)(a6))  extern void __iomem *efi_ioremap(unsigned long addr, unsigned long size, diff --git a/arch/x86/include/asm/emergency-restart.h b/arch/x86/include/asm/emergency-restart.h index 75ce3f47d20..77a99ac06d0 100644 --- a/arch/x86/include/asm/emergency-restart.h +++ b/arch/x86/include/asm/emergency-restart.h @@ -1,18 +1,6 @@  #ifndef _ASM_X86_EMERGENCY_RESTART_H  #define _ASM_X86_EMERGENCY_RESTART_H -enum reboot_type { -	BOOT_TRIPLE = 't', -	BOOT_KBD = 'k', -	BOOT_BIOS = 'b', -	BOOT_ACPI = 'a', -	BOOT_EFI = 'e', -	BOOT_CF9 = 'p', -	BOOT_CF9_COND = 'q', -}; - -extern enum reboot_type reboot_type; -  extern void machine_emergency_restart(void);  #endif /* _ASM_X86_EMERGENCY_RESTART_H */ diff --git a/arch/x86/include/asm/entry_arch.h b/arch/x86/include/asm/entry_arch.h index 9bd4ecac72b..dc5fa661465 100644 --- a/arch/x86/include/asm/entry_arch.h +++ b/arch/x86/include/asm/entry_arch.h @@ -13,14 +13,16 @@  BUILD_INTERRUPT(reschedule_interrupt,RESCHEDULE_VECTOR)  BUILD_INTERRUPT(call_function_interrupt,CALL_FUNCTION_VECTOR)  BUILD_INTERRUPT(call_function_single_interrupt,CALL_FUNCTION_SINGLE_VECTOR) -BUILD_INTERRUPT(irq_move_cleanup_interrupt,IRQ_MOVE_CLEANUP_VECTOR) -BUILD_INTERRUPT(reboot_interrupt,REBOOT_VECTOR) +BUILD_INTERRUPT3(irq_move_cleanup_interrupt, IRQ_MOVE_CLEANUP_VECTOR, +		 smp_irq_move_cleanup_interrupt) +BUILD_INTERRUPT3(reboot_interrupt, REBOOT_VECTOR, smp_reboot_interrupt)  #endif  BUILD_INTERRUPT(x86_platform_ipi, X86_PLATFORM_IPI_VECTOR)  #ifdef CONFIG_HAVE_KVM -BUILD_INTERRUPT(kvm_posted_intr_ipi, POSTED_INTR_VECTOR) +BUILD_INTERRUPT3(kvm_posted_intr_ipi, POSTED_INTR_VECTOR, +		 smp_kvm_posted_intr_ipi)  #endif  /* diff --git a/arch/x86/include/asm/fixmap.h b/arch/x86/include/asm/fixmap.h index 0dc7d9e21c3..e846225265e 100644 --- a/arch/x86/include/asm/fixmap.h +++ b/arch/x86/include/asm/fixmap.h @@ -81,11 +81,11 @@ enum fixed_addresses {  			    + ((VSYSCALL_END-VSYSCALL_START) >> PAGE_SHIFT) - 1,  	VVAR_PAGE,  	VSYSCALL_HPET, -#endif  #ifdef CONFIG_PARAVIRT_CLOCK  	PVCLOCK_FIXMAP_BEGIN,  	PVCLOCK_FIXMAP_END = PVCLOCK_FIXMAP_BEGIN+PVCLOCK_VSYSCALL_NR_PAGES-1,  #endif +#endif  	FIX_DBGP_BASE,  	FIX_EARLYCON_MEM_BASE,  #ifdef CONFIG_PROVIDE_OHCI1394_DMA_INIT diff --git a/arch/x86/include/asm/fpu-internal.h b/arch/x86/include/asm/fpu-internal.h index e25cc33ec54..4d0bda7b11e 100644 --- a/arch/x86/include/asm/fpu-internal.h +++ b/arch/x86/include/asm/fpu-internal.h @@ -62,10 +62,8 @@ extern user_regset_set_fn fpregs_set, xfpregs_set, fpregs_soft_set,  #define xstateregs_active	fpregs_active  #ifdef CONFIG_MATH_EMULATION -# define HAVE_HWFP		(boot_cpu_data.hard_math)  extern void finit_soft_fpu(struct i387_soft_struct *soft);  #else -# define HAVE_HWFP		1  static inline void finit_soft_fpu(struct i387_soft_struct *soft) {}  #endif @@ -345,7 +343,7 @@ static inline void __thread_fpu_end(struct task_struct *tsk)  static inline void __thread_fpu_begin(struct task_struct *tsk)  { -	if (!use_eager_fpu()) +	if (!static_cpu_has_safe(X86_FEATURE_EAGER_FPU))  		clts();  	__thread_set_has_fpu(tsk);  } diff --git a/arch/x86/include/asm/hw_irq.h b/arch/x86/include/asm/hw_irq.h index 1da97efad08..e4ac559c4a2 100644 --- a/arch/x86/include/asm/hw_irq.h +++ b/arch/x86/include/asm/hw_irq.h @@ -77,6 +77,23 @@ extern void threshold_interrupt(void);  extern void call_function_interrupt(void);  extern void call_function_single_interrupt(void); +#ifdef CONFIG_TRACING +/* Interrupt handlers registered during init_IRQ */ +extern void trace_apic_timer_interrupt(void); +extern void trace_x86_platform_ipi(void); +extern void trace_error_interrupt(void); +extern void trace_irq_work_interrupt(void); +extern void trace_spurious_interrupt(void); +extern void trace_thermal_interrupt(void); +extern void trace_reschedule_interrupt(void); +extern void trace_threshold_interrupt(void); +extern void trace_call_function_interrupt(void); +extern void trace_call_function_single_interrupt(void); +#define trace_irq_move_cleanup_interrupt  irq_move_cleanup_interrupt +#define trace_reboot_interrupt  reboot_interrupt +#define trace_kvm_posted_intr_ipi kvm_posted_intr_ipi +#endif /* CONFIG_TRACING */ +  /* IOAPIC */  #define IO_APIC_IRQ(x) (((x) >= NR_IRQS_LEGACY) || ((1<<(x)) & io_apic_irqs))  extern unsigned long io_apic_irqs; diff --git a/arch/x86/include/asm/io.h b/arch/x86/include/asm/io.h index d8e8eefbe24..34f69cb9350 100644 --- a/arch/x86/include/asm/io.h +++ b/arch/x86/include/asm/io.h @@ -345,4 +345,11 @@ extern bool xen_biovec_phys_mergeable(const struct bio_vec *vec1,  #define IO_SPACE_LIMIT 0xffff +#ifdef CONFIG_MTRR +extern int __must_check arch_phys_wc_add(unsigned long base, +					 unsigned long size); +extern void arch_phys_wc_del(int handle); +#define arch_phys_wc_add arch_phys_wc_add +#endif +  #endif /* _ASM_X86_IO_H */ diff --git a/arch/x86/include/asm/kvm_host.h b/arch/x86/include/asm/kvm_host.h index 3741c653767..f87f7fcefa0 100644 --- a/arch/x86/include/asm/kvm_host.h +++ b/arch/x86/include/asm/kvm_host.h @@ -59,7 +59,7 @@  	(~(unsigned long)(X86_CR4_VME | X86_CR4_PVI | X86_CR4_TSD | X86_CR4_DE\  			  | X86_CR4_PSE | X86_CR4_PAE | X86_CR4_MCE     \  			  | X86_CR4_PGE | X86_CR4_PCE | X86_CR4_OSFXSR | X86_CR4_PCIDE \ -			  | X86_CR4_OSXSAVE | X86_CR4_SMEP | X86_CR4_RDWRGSFS \ +			  | X86_CR4_OSXSAVE | X86_CR4_SMEP | X86_CR4_FSGSBASE \  			  | X86_CR4_OSXMMEXCPT | X86_CR4_VMXE))  #define CR8_RESERVED_BITS (~(unsigned long)X86_CR8_TPR) @@ -222,14 +222,22 @@ struct kvm_mmu_page {  	int root_count;          /* Currently serving as active root */  	unsigned int unsync_children;  	unsigned long parent_ptes;	/* Reverse mapping for parent_pte */ + +	/* The page is obsolete if mmu_valid_gen != kvm->arch.mmu_valid_gen.  */ +	unsigned long mmu_valid_gen; +  	DECLARE_BITMAP(unsync_child_bitmap, 512);  #ifdef CONFIG_X86_32 +	/* +	 * Used out of the mmu-lock to avoid reading spte values while an +	 * update is in progress; see the comments in __get_spte_lockless(). +	 */  	int clear_spte_count;  #endif +	/* Number of writes since the last time traversal visited this page.  */  	int write_flooding_count; -	bool mmio_cached;  };  struct kvm_pio_request { @@ -529,11 +537,14 @@ struct kvm_arch {  	unsigned int n_requested_mmu_pages;  	unsigned int n_max_mmu_pages;  	unsigned int indirect_shadow_pages; +	unsigned long mmu_valid_gen;  	struct hlist_head mmu_page_hash[KVM_NUM_MMU_PAGES];  	/*  	 * Hash table of struct kvm_mmu_page.  	 */  	struct list_head active_mmu_pages; +	struct list_head zapped_obsolete_pages; +  	struct list_head assigned_dev_head;  	struct iommu_domain *iommu_domain;  	int iommu_flags; @@ -769,7 +780,7 @@ void kvm_mmu_write_protect_pt_masked(struct kvm *kvm,  				     struct kvm_memory_slot *slot,  				     gfn_t gfn_offset, unsigned long mask);  void kvm_mmu_zap_all(struct kvm *kvm); -void kvm_mmu_zap_mmio_sptes(struct kvm *kvm); +void kvm_mmu_invalidate_mmio_sptes(struct kvm *kvm);  unsigned int kvm_mmu_calculate_mmu_pages(struct kvm *kvm);  void kvm_mmu_change_mmu_pages(struct kvm *kvm, unsigned int kvm_nr_mmu_pages); diff --git a/arch/x86/include/asm/mc146818rtc.h b/arch/x86/include/asm/mc146818rtc.h index d354fb781c5..a55c7efcc4e 100644 --- a/arch/x86/include/asm/mc146818rtc.h +++ b/arch/x86/include/asm/mc146818rtc.h @@ -95,8 +95,8 @@ static inline unsigned char current_lock_cmos_reg(void)  unsigned char rtc_cmos_read(unsigned char addr);  void rtc_cmos_write(unsigned char val, unsigned char addr); -extern int mach_set_rtc_mmss(unsigned long nowtime); -extern unsigned long mach_get_cmos_time(void); +extern int mach_set_rtc_mmss(const struct timespec *now); +extern void mach_get_cmos_time(struct timespec *now);  #define RTC_IRQ 8 diff --git a/arch/x86/include/asm/mce.h b/arch/x86/include/asm/mce.h index fa5f71e021d..29e3093bbd2 100644 --- a/arch/x86/include/asm/mce.h +++ b/arch/x86/include/asm/mce.h @@ -61,7 +61,7 @@  #define MCJ_CTX_IRQ		0x2  /* inject context: IRQ */  #define MCJ_NMI_BROADCAST	0x4  /* do NMI broadcasting */  #define MCJ_EXCEPTION		0x8  /* raise as exception */ -#define MCJ_IRQ_BRAODCAST	0x10 /* do IRQ broadcasting */ +#define MCJ_IRQ_BROADCAST	0x10 /* do IRQ broadcasting */  #define MCE_OVERFLOW 0		/* bit 0 in flags means overflow */ @@ -214,6 +214,13 @@ void mce_log_therm_throt_event(__u64 status);  /* Interrupt Handler for core thermal thresholds */  extern int (*platform_thermal_notify)(__u64 msr_val); +/* Interrupt Handler for package thermal thresholds */ +extern int (*platform_thermal_package_notify)(__u64 msr_val); + +/* Callback support of rate control, return true, if + * callback has rate control */ +extern bool (*platform_thermal_package_rate_control)(void); +  #ifdef CONFIG_X86_THERMAL_VECTOR  extern void mcheck_intel_therm_init(void);  #else diff --git a/arch/x86/include/asm/microcode.h b/arch/x86/include/asm/microcode.h index 6bc3985ee47..f98bd662531 100644 --- a/arch/x86/include/asm/microcode.h +++ b/arch/x86/include/asm/microcode.h @@ -60,11 +60,11 @@ static inline void __exit exit_amd_microcode(void) {}  #ifdef CONFIG_MICROCODE_EARLY  #define MAX_UCODE_COUNT 128  extern void __init load_ucode_bsp(void); -extern void __cpuinit load_ucode_ap(void); +extern void load_ucode_ap(void);  extern int __init save_microcode_in_initrd(void);  #else  static inline void __init load_ucode_bsp(void) {} -static inline void __cpuinit load_ucode_ap(void) {} +static inline void load_ucode_ap(void) {}  static inline int __init save_microcode_in_initrd(void)  {  	return 0; diff --git a/arch/x86/include/asm/microcode_amd.h b/arch/x86/include/asm/microcode_amd.h new file mode 100644 index 00000000000..50e5c58ced2 --- /dev/null +++ b/arch/x86/include/asm/microcode_amd.h @@ -0,0 +1,78 @@ +#ifndef _ASM_X86_MICROCODE_AMD_H +#define _ASM_X86_MICROCODE_AMD_H + +#include <asm/microcode.h> + +#define UCODE_MAGIC			0x00414d44 +#define UCODE_EQUIV_CPU_TABLE_TYPE	0x00000000 +#define UCODE_UCODE_TYPE		0x00000001 + +#define SECTION_HDR_SIZE		8 +#define CONTAINER_HDR_SZ		12 + +struct equiv_cpu_entry { +	u32	installed_cpu; +	u32	fixed_errata_mask; +	u32	fixed_errata_compare; +	u16	equiv_cpu; +	u16	res; +} __attribute__((packed)); + +struct microcode_header_amd { +	u32	data_code; +	u32	patch_id; +	u16	mc_patch_data_id; +	u8	mc_patch_data_len; +	u8	init_flag; +	u32	mc_patch_data_checksum; +	u32	nb_dev_id; +	u32	sb_dev_id; +	u16	processor_rev_id; +	u8	nb_rev_id; +	u8	sb_rev_id; +	u8	bios_api_rev; +	u8	reserved1[3]; +	u32	match_reg[8]; +} __attribute__((packed)); + +struct microcode_amd { +	struct microcode_header_amd	hdr; +	unsigned int			mpb[0]; +}; + +static inline u16 find_equiv_id(struct equiv_cpu_entry *equiv_cpu_table, +				unsigned int sig) +{ +	int i = 0; + +	if (!equiv_cpu_table) +		return 0; + +	while (equiv_cpu_table[i].installed_cpu != 0) { +		if (sig == equiv_cpu_table[i].installed_cpu) +			return equiv_cpu_table[i].equiv_cpu; + +		i++; +	} +	return 0; +} + +extern int __apply_microcode_amd(struct microcode_amd *mc_amd); +extern int apply_microcode_amd(int cpu); +extern enum ucode_state load_microcode_amd(int cpu, const u8 *data, size_t size); + +#ifdef CONFIG_MICROCODE_AMD_EARLY +#ifdef CONFIG_X86_32 +#define MPB_MAX_SIZE PAGE_SIZE +extern u8 amd_bsp_mpb[MPB_MAX_SIZE]; +#endif +extern void __init load_ucode_amd_bsp(void); +extern void load_ucode_amd_ap(void); +extern int __init save_microcode_in_initrd_amd(void); +#else +static inline void __init load_ucode_amd_bsp(void) {} +static inline void load_ucode_amd_ap(void) {} +static inline int __init save_microcode_in_initrd_amd(void) { return -EINVAL; } +#endif + +#endif /* _ASM_X86_MICROCODE_AMD_H */ diff --git a/arch/x86/include/asm/microcode_intel.h b/arch/x86/include/asm/microcode_intel.h index 5356f927d41..9067166409b 100644 --- a/arch/x86/include/asm/microcode_intel.h +++ b/arch/x86/include/asm/microcode_intel.h @@ -65,12 +65,14 @@ update_match_revision(struct microcode_header_intel *mc_header, int rev);  #ifdef CONFIG_MICROCODE_INTEL_EARLY  extern void __init load_ucode_intel_bsp(void); -extern void __cpuinit load_ucode_intel_ap(void); +extern void load_ucode_intel_ap(void);  extern void show_ucode_info_early(void); +extern int __init save_microcode_in_initrd_intel(void);  #else  static inline __init void load_ucode_intel_bsp(void) {} -static inline __cpuinit void load_ucode_intel_ap(void) {} +static inline void load_ucode_intel_ap(void) {}  static inline void show_ucode_info_early(void) {} +static inline int __init save_microcode_in_initrd_intel(void) { return -EINVAL; }  #endif  #if defined(CONFIG_MICROCODE_INTEL_EARLY) && defined(CONFIG_HOTPLUG_CPU) diff --git a/arch/x86/include/asm/mmconfig.h b/arch/x86/include/asm/mmconfig.h index 9b119da1d10..04a3fed22cf 100644 --- a/arch/x86/include/asm/mmconfig.h +++ b/arch/x86/include/asm/mmconfig.h @@ -2,8 +2,8 @@  #define _ASM_X86_MMCONFIG_H  #ifdef CONFIG_PCI_MMCONFIG -extern void __cpuinit fam10h_check_enable_mmcfg(void); -extern void __cpuinit check_enable_amd_mmconf_dmi(void); +extern void fam10h_check_enable_mmcfg(void); +extern void check_enable_amd_mmconf_dmi(void);  #else  static inline void fam10h_check_enable_mmcfg(void) { }  static inline void check_enable_amd_mmconf_dmi(void) { } diff --git a/arch/x86/include/asm/mpspec.h b/arch/x86/include/asm/mpspec.h index 3e2f42a4b87..626cf70082d 100644 --- a/arch/x86/include/asm/mpspec.h +++ b/arch/x86/include/asm/mpspec.h @@ -94,7 +94,7 @@ static inline void early_reserve_e820_mpc_new(void) { }  #define default_get_smp_config x86_init_uint_noop  #endif -void __cpuinit generic_processor_info(int apicid, int version); +void generic_processor_info(int apicid, int version);  #ifdef CONFIG_ACPI  extern void mp_register_ioapic(int id, u32 address, u32 gsi_base);  extern void mp_override_legacy_irq(u8 bus_irq, u8 polarity, u8 trigger, diff --git a/arch/x86/include/asm/mrst-vrtc.h b/arch/x86/include/asm/mrst-vrtc.h index 73668abdbed..1e69a75412a 100644 --- a/arch/x86/include/asm/mrst-vrtc.h +++ b/arch/x86/include/asm/mrst-vrtc.h @@ -3,7 +3,7 @@  extern unsigned char vrtc_cmos_read(unsigned char reg);  extern void vrtc_cmos_write(unsigned char val, unsigned char reg); -extern unsigned long vrtc_get_time(void); -extern int vrtc_set_mmss(unsigned long nowtime); +extern void vrtc_get_time(struct timespec *now); +extern int vrtc_set_mmss(const struct timespec *now);  #endif diff --git a/arch/x86/include/asm/mshyperv.h b/arch/x86/include/asm/mshyperv.h index c2934be2446..cd9c41938b8 100644 --- a/arch/x86/include/asm/mshyperv.h +++ b/arch/x86/include/asm/mshyperv.h @@ -12,6 +12,9 @@ struct ms_hyperv_info {  extern struct ms_hyperv_info ms_hyperv;  void hyperv_callback_vector(void); +#ifdef CONFIG_TRACING +#define trace_hyperv_callback_vector hyperv_callback_vector +#endif  void hyperv_vector_handler(struct pt_regs *regs);  void hv_register_vmbus_handler(int irq, irq_handler_t handler); diff --git a/arch/x86/include/asm/mtrr.h b/arch/x86/include/asm/mtrr.h index e235582f993..f768f629841 100644 --- a/arch/x86/include/asm/mtrr.h +++ b/arch/x86/include/asm/mtrr.h @@ -26,7 +26,10 @@  #include <uapi/asm/mtrr.h> -/*  The following functions are for use by other drivers  */ +/* + * The following functions are for use by other drivers that cannot use + * arch_phys_wc_add and arch_phys_wc_del. + */  # ifdef CONFIG_MTRR  extern u8 mtrr_type_lookup(u64 addr, u64 end);  extern void mtrr_save_fixed_ranges(void *); @@ -45,6 +48,7 @@ extern void mtrr_aps_init(void);  extern void mtrr_bp_restore(void);  extern int mtrr_trim_uncached_memory(unsigned long end_pfn);  extern int amd_special_default_mtrr(void); +extern int phys_wc_to_mtrr_index(int handle);  #  else  static inline u8 mtrr_type_lookup(u64 addr, u64 end)  { @@ -80,6 +84,10 @@ static inline int mtrr_trim_uncached_memory(unsigned long end_pfn)  static inline void mtrr_centaur_report_mcr(int mcr, u32 lo, u32 hi)  {  } +static inline int phys_wc_to_mtrr_index(int handle) +{ +	return -1; +}  #define mtrr_ap_init() do {} while (0)  #define mtrr_bp_init() do {} while (0) diff --git a/arch/x86/include/asm/mutex_32.h b/arch/x86/include/asm/mutex_32.h index 03f90c8a5a7..0208c3c2cbc 100644 --- a/arch/x86/include/asm/mutex_32.h +++ b/arch/x86/include/asm/mutex_32.h @@ -42,17 +42,14 @@ do {								\   *  __mutex_fastpath_lock_retval - try to take the lock by moving the count   *                                 from 1 to a 0 value   *  @count: pointer of type atomic_t - *  @fail_fn: function to call if the original value was not 1   * - * Change the count from 1 to a value lower than 1, and call <fail_fn> if it - * wasn't 1 originally. This function returns 0 if the fastpath succeeds, - * or anything the slow path function returns + * Change the count from 1 to a value lower than 1. This function returns 0 + * if the fastpath succeeds, or -1 otherwise.   */ -static inline int __mutex_fastpath_lock_retval(atomic_t *count, -					       int (*fail_fn)(atomic_t *)) +static inline int __mutex_fastpath_lock_retval(atomic_t *count)  {  	if (unlikely(atomic_dec_return(count) < 0)) -		return fail_fn(count); +		return -1;  	else  		return 0;  } diff --git a/arch/x86/include/asm/mutex_64.h b/arch/x86/include/asm/mutex_64.h index 68a87b0f8e2..2c543fff241 100644 --- a/arch/x86/include/asm/mutex_64.h +++ b/arch/x86/include/asm/mutex_64.h @@ -37,17 +37,14 @@ do {								\   *  __mutex_fastpath_lock_retval - try to take the lock by moving the count   *                                 from 1 to a 0 value   *  @count: pointer of type atomic_t - *  @fail_fn: function to call if the original value was not 1   * - * Change the count from 1 to a value lower than 1, and call <fail_fn> if - * it wasn't 1 originally. This function returns 0 if the fastpath succeeds, - * or anything the slow path function returns + * Change the count from 1 to a value lower than 1. This function returns 0 + * if the fastpath succeeds, or -1 otherwise.   */ -static inline int __mutex_fastpath_lock_retval(atomic_t *count, -					       int (*fail_fn)(atomic_t *)) +static inline int __mutex_fastpath_lock_retval(atomic_t *count)  {  	if (unlikely(atomic_dec_return(count) < 0)) -		return fail_fn(count); +		return -1;  	else  		return 0;  } diff --git a/arch/x86/include/asm/numa.h b/arch/x86/include/asm/numa.h index 1b99ee5c9f0..4064acae625 100644 --- a/arch/x86/include/asm/numa.h +++ b/arch/x86/include/asm/numa.h @@ -39,7 +39,7 @@ static inline void set_apicid_to_node(int apicid, s16 node)  	__apicid_to_node[apicid] = node;  } -extern int __cpuinit numa_cpu_node(int cpu); +extern int numa_cpu_node(int cpu);  #else	/* CONFIG_NUMA */  static inline void set_apicid_to_node(int apicid, s16 node) @@ -60,8 +60,8 @@ static inline int numa_cpu_node(int cpu)  extern void numa_set_node(int cpu, int node);  extern void numa_clear_node(int cpu);  extern void __init init_cpu_to_node(void); -extern void __cpuinit numa_add_cpu(int cpu); -extern void __cpuinit numa_remove_cpu(int cpu); +extern void numa_add_cpu(int cpu); +extern void numa_remove_cpu(int cpu);  #else	/* CONFIG_NUMA */  static inline void numa_set_node(int cpu, int node)	{ }  static inline void numa_clear_node(int cpu)		{ } diff --git a/arch/x86/include/asm/perf_event.h b/arch/x86/include/asm/perf_event.h index 57cb6340221..8249df45d2f 100644 --- a/arch/x86/include/asm/perf_event.h +++ b/arch/x86/include/asm/perf_event.h @@ -29,6 +29,9 @@  #define ARCH_PERFMON_EVENTSEL_INV			(1ULL << 23)  #define ARCH_PERFMON_EVENTSEL_CMASK			0xFF000000ULL +#define HSW_IN_TX					(1ULL << 32) +#define HSW_IN_TX_CHECKPOINTED				(1ULL << 33) +  #define AMD64_EVENTSEL_INT_CORE_ENABLE			(1ULL << 36)  #define AMD64_EVENTSEL_GUESTONLY			(1ULL << 40)  #define AMD64_EVENTSEL_HOSTONLY				(1ULL << 41) diff --git a/arch/x86/include/asm/pgtable-2level.h b/arch/x86/include/asm/pgtable-2level.h index f2b489cf160..3bf2dd0cf61 100644 --- a/arch/x86/include/asm/pgtable-2level.h +++ b/arch/x86/include/asm/pgtable-2level.h @@ -55,9 +55,53 @@ static inline pmd_t native_pmdp_get_and_clear(pmd_t *xp)  #define native_pmdp_get_and_clear(xp) native_local_pmdp_get_and_clear(xp)  #endif +#ifdef CONFIG_MEM_SOFT_DIRTY + +/* + * Bits _PAGE_BIT_PRESENT, _PAGE_BIT_FILE, _PAGE_BIT_SOFT_DIRTY and + * _PAGE_BIT_PROTNONE are taken, split up the 28 bits of offset + * into this range. + */ +#define PTE_FILE_MAX_BITS	28 +#define PTE_FILE_SHIFT1		(_PAGE_BIT_PRESENT + 1) +#define PTE_FILE_SHIFT2		(_PAGE_BIT_FILE + 1) +#define PTE_FILE_SHIFT3		(_PAGE_BIT_PROTNONE + 1) +#define PTE_FILE_SHIFT4		(_PAGE_BIT_SOFT_DIRTY + 1) +#define PTE_FILE_BITS1		(PTE_FILE_SHIFT2 - PTE_FILE_SHIFT1 - 1) +#define PTE_FILE_BITS2		(PTE_FILE_SHIFT3 - PTE_FILE_SHIFT2 - 1) +#define PTE_FILE_BITS3		(PTE_FILE_SHIFT4 - PTE_FILE_SHIFT3 - 1) + +#define pte_to_pgoff(pte)						\ +	((((pte).pte_low >> (PTE_FILE_SHIFT1))				\ +	  & ((1U << PTE_FILE_BITS1) - 1)))				\ +	+ ((((pte).pte_low >> (PTE_FILE_SHIFT2))			\ +	    & ((1U << PTE_FILE_BITS2) - 1))				\ +	   << (PTE_FILE_BITS1))						\ +	+ ((((pte).pte_low >> (PTE_FILE_SHIFT3))			\ +	    & ((1U << PTE_FILE_BITS3) - 1))				\ +	   << (PTE_FILE_BITS1 + PTE_FILE_BITS2))			\ +	+ ((((pte).pte_low >> (PTE_FILE_SHIFT4)))			\ +	    << (PTE_FILE_BITS1 + PTE_FILE_BITS2 + PTE_FILE_BITS3)) + +#define pgoff_to_pte(off)						\ +	((pte_t) { .pte_low =						\ +	 ((((off)) & ((1U << PTE_FILE_BITS1) - 1)) << PTE_FILE_SHIFT1)	\ +	 + ((((off) >> PTE_FILE_BITS1)					\ +	     & ((1U << PTE_FILE_BITS2) - 1))				\ +	    << PTE_FILE_SHIFT2)						\ +	 + ((((off) >> (PTE_FILE_BITS1 + PTE_FILE_BITS2))		\ +	     & ((1U << PTE_FILE_BITS3) - 1))				\ +	    << PTE_FILE_SHIFT3)						\ +	 + ((((off) >>							\ +	      (PTE_FILE_BITS1 + PTE_FILE_BITS2 + PTE_FILE_BITS3)))	\ +	    << PTE_FILE_SHIFT4)						\ +	 + _PAGE_FILE }) + +#else /* CONFIG_MEM_SOFT_DIRTY */ +  /*   * Bits _PAGE_BIT_PRESENT, _PAGE_BIT_FILE and _PAGE_BIT_PROTNONE are taken, - * split up the 29 bits of offset into this range: + * split up the 29 bits of offset into this range.   */  #define PTE_FILE_MAX_BITS	29  #define PTE_FILE_SHIFT1		(_PAGE_BIT_PRESENT + 1) @@ -88,6 +132,8 @@ static inline pmd_t native_pmdp_get_and_clear(pmd_t *xp)  	    << PTE_FILE_SHIFT3)						\  	 + _PAGE_FILE }) +#endif /* CONFIG_MEM_SOFT_DIRTY */ +  /* Encode and de-code a swap entry */  #if _PAGE_BIT_FILE < _PAGE_BIT_PROTNONE  #define SWP_TYPE_BITS (_PAGE_BIT_FILE - _PAGE_BIT_PRESENT - 1) diff --git a/arch/x86/include/asm/pgtable-3level.h b/arch/x86/include/asm/pgtable-3level.h index 4cc9f2b7cdc..81bb91b49a8 100644 --- a/arch/x86/include/asm/pgtable-3level.h +++ b/arch/x86/include/asm/pgtable-3level.h @@ -179,6 +179,9 @@ static inline pmd_t native_pmdp_get_and_clear(pmd_t *pmdp)  /*   * Bits 0, 6 and 7 are taken in the low part of the pte,   * put the 32 bits of offset into the high part. + * + * For soft-dirty tracking 11 bit is taken from + * the low part of pte as well.   */  #define pte_to_pgoff(pte) ((pte).pte_high)  #define pgoff_to_pte(off)						\ diff --git a/arch/x86/include/asm/pgtable.h b/arch/x86/include/asm/pgtable.h index 1e672234c4f..1c00631164c 100644 --- a/arch/x86/include/asm/pgtable.h +++ b/arch/x86/include/asm/pgtable.h @@ -207,7 +207,7 @@ static inline pte_t pte_mkexec(pte_t pte)  static inline pte_t pte_mkdirty(pte_t pte)  { -	return pte_set_flags(pte, _PAGE_DIRTY); +	return pte_set_flags(pte, _PAGE_DIRTY | _PAGE_SOFT_DIRTY);  }  static inline pte_t pte_mkyoung(pte_t pte) @@ -271,7 +271,7 @@ static inline pmd_t pmd_wrprotect(pmd_t pmd)  static inline pmd_t pmd_mkdirty(pmd_t pmd)  { -	return pmd_set_flags(pmd, _PAGE_DIRTY); +	return pmd_set_flags(pmd, _PAGE_DIRTY | _PAGE_SOFT_DIRTY);  }  static inline pmd_t pmd_mkhuge(pmd_t pmd) @@ -294,6 +294,56 @@ static inline pmd_t pmd_mknotpresent(pmd_t pmd)  	return pmd_clear_flags(pmd, _PAGE_PRESENT);  } +static inline int pte_soft_dirty(pte_t pte) +{ +	return pte_flags(pte) & _PAGE_SOFT_DIRTY; +} + +static inline int pmd_soft_dirty(pmd_t pmd) +{ +	return pmd_flags(pmd) & _PAGE_SOFT_DIRTY; +} + +static inline pte_t pte_mksoft_dirty(pte_t pte) +{ +	return pte_set_flags(pte, _PAGE_SOFT_DIRTY); +} + +static inline pmd_t pmd_mksoft_dirty(pmd_t pmd) +{ +	return pmd_set_flags(pmd, _PAGE_SOFT_DIRTY); +} + +static inline pte_t pte_swp_mksoft_dirty(pte_t pte) +{ +	return pte_set_flags(pte, _PAGE_SWP_SOFT_DIRTY); +} + +static inline int pte_swp_soft_dirty(pte_t pte) +{ +	return pte_flags(pte) & _PAGE_SWP_SOFT_DIRTY; +} + +static inline pte_t pte_swp_clear_soft_dirty(pte_t pte) +{ +	return pte_clear_flags(pte, _PAGE_SWP_SOFT_DIRTY); +} + +static inline pte_t pte_file_clear_soft_dirty(pte_t pte) +{ +	return pte_clear_flags(pte, _PAGE_SOFT_DIRTY); +} + +static inline pte_t pte_file_mksoft_dirty(pte_t pte) +{ +	return pte_set_flags(pte, _PAGE_SOFT_DIRTY); +} + +static inline int pte_file_soft_dirty(pte_t pte) +{ +	return pte_flags(pte) & _PAGE_SOFT_DIRTY; +} +  /*   * Mask out unsupported bits in a present pgprot.  Non-present pgprots   * can use those bits for other purposes, so leave them be. @@ -506,9 +556,6 @@ static inline unsigned long pages_to_mb(unsigned long npg)  	return npg >> (20 - PAGE_SHIFT);  } -#define io_remap_pfn_range(vma, vaddr, pfn, size, prot)	\ -	remap_pfn_range(vma, vaddr, pfn, size, prot) -  #if PAGETABLE_LEVELS > 2  static inline int pud_none(pud_t pud)  { diff --git a/arch/x86/include/asm/pgtable_types.h b/arch/x86/include/asm/pgtable_types.h index e6423002c10..f4843e03113 100644 --- a/arch/x86/include/asm/pgtable_types.h +++ b/arch/x86/include/asm/pgtable_types.h @@ -55,6 +55,33 @@  #define _PAGE_HIDDEN	(_AT(pteval_t, 0))  #endif +/* + * The same hidden bit is used by kmemcheck, but since kmemcheck + * works on kernel pages while soft-dirty engine on user space, + * they do not conflict with each other. + */ + +#define _PAGE_BIT_SOFT_DIRTY	_PAGE_BIT_HIDDEN + +#ifdef CONFIG_MEM_SOFT_DIRTY +#define _PAGE_SOFT_DIRTY	(_AT(pteval_t, 1) << _PAGE_BIT_SOFT_DIRTY) +#else +#define _PAGE_SOFT_DIRTY	(_AT(pteval_t, 0)) +#endif + +/* + * Tracking soft dirty bit when a page goes to a swap is tricky. + * We need a bit which can be stored in pte _and_ not conflict + * with swap entry format. On x86 bits 6 and 7 are *not* involved + * into swap entry computation, but bit 6 is used for nonlinear + * file mapping, so we borrow bit 7 for soft dirty tracking. + */ +#ifdef CONFIG_MEM_SOFT_DIRTY +#define _PAGE_SWP_SOFT_DIRTY	_PAGE_PSE +#else +#define _PAGE_SWP_SOFT_DIRTY	(_AT(pteval_t, 0)) +#endif +  #if defined(CONFIG_X86_64) || defined(CONFIG_X86_PAE)  #define _PAGE_NX	(_AT(pteval_t, 1) << _PAGE_BIT_NX)  #else diff --git a/arch/x86/include/asm/processor.h b/arch/x86/include/asm/processor.h index 22224b3b43b..24cf5aefb70 100644 --- a/arch/x86/include/asm/processor.h +++ b/arch/x86/include/asm/processor.h @@ -89,9 +89,9 @@ struct cpuinfo_x86 {  	char			wp_works_ok;	/* It doesn't on 386's */  	/* Problems on some 486Dx4's and old 386's: */ -	char			hard_math;  	char			rfu;  	char			pad0; +	char			pad1;  #else  	/* Number of 4K pages in DTLB/ITLB combined(in pages): */  	int			x86_tlbsize; @@ -164,6 +164,7 @@ extern const struct seq_operations cpuinfo_op;  #define cache_line_size()	(boot_cpu_data.x86_cache_alignment)  extern void cpu_detect(struct cpuinfo_x86 *c); +extern void fpu_detect(struct cpuinfo_x86 *c);  extern void early_cpu_init(void);  extern void identify_boot_cpu(void); @@ -981,5 +982,5 @@ bool xen_set_default_idle(void);  #endif  void stop_this_cpu(void *dummy); - +void df_debug(struct pt_regs *regs, long error_code);  #endif /* _ASM_X86_PROCESSOR_H */ diff --git a/arch/x86/include/asm/prom.h b/arch/x86/include/asm/prom.h index 60bef663609..bade6ac3b14 100644 --- a/arch/x86/include/asm/prom.h +++ b/arch/x86/include/asm/prom.h @@ -27,7 +27,7 @@ extern int of_ioapic;  extern u64 initial_dtb;  extern void add_dtb(u64 data);  extern void x86_add_irq_domains(void); -void __cpuinit x86_of_pci_init(void); +void x86_of_pci_init(void);  void x86_dtb_init(void);  #else  static inline void add_dtb(u64 data) { } diff --git a/arch/x86/include/asm/sighandling.h b/arch/x86/include/asm/sighandling.h index beff97f7df3..7a958164088 100644 --- a/arch/x86/include/asm/sighandling.h +++ b/arch/x86/include/asm/sighandling.h @@ -7,10 +7,10 @@  #include <asm/processor-flags.h> -#define __FIX_EFLAGS	(X86_EFLAGS_AC | X86_EFLAGS_OF | \ +#define FIX_EFLAGS	(X86_EFLAGS_AC | X86_EFLAGS_OF | \  			 X86_EFLAGS_DF | X86_EFLAGS_TF | X86_EFLAGS_SF | \  			 X86_EFLAGS_ZF | X86_EFLAGS_AF | X86_EFLAGS_PF | \ -			 X86_EFLAGS_CF) +			 X86_EFLAGS_CF | X86_EFLAGS_RF)  void signal_fault(struct pt_regs *regs, void __user *frame, char *where); diff --git a/arch/x86/include/asm/smp.h b/arch/x86/include/asm/smp.h index b073aaea747..4137890e88e 100644 --- a/arch/x86/include/asm/smp.h +++ b/arch/x86/include/asm/smp.h @@ -179,7 +179,7 @@ static inline int wbinvd_on_all_cpus(void)  }  #endif /* CONFIG_SMP */ -extern unsigned disabled_cpus __cpuinitdata; +extern unsigned disabled_cpus;  #ifdef CONFIG_X86_32_SMP  /* diff --git a/arch/x86/include/asm/special_insns.h b/arch/x86/include/asm/special_insns.h index 41fc93a2e22..2f4d924fe6c 100644 --- a/arch/x86/include/asm/special_insns.h +++ b/arch/x86/include/asm/special_insns.h @@ -16,7 +16,7 @@ static inline void native_clts(void)   * all loads stores around it, which can hurt performance. Solution is to   * use a variable and mimic reads and writes to it to enforce serialization   */ -static unsigned long __force_order; +extern unsigned long __force_order;  static inline unsigned long native_read_cr0(void)  { diff --git a/arch/x86/include/asm/spinlock.h b/arch/x86/include/asm/spinlock.h index 33692eaabab..e3ddd7db723 100644 --- a/arch/x86/include/asm/spinlock.h +++ b/arch/x86/include/asm/spinlock.h @@ -233,8 +233,4 @@ static inline void arch_write_unlock(arch_rwlock_t *rw)  #define arch_read_relax(lock)	cpu_relax()  #define arch_write_relax(lock)	cpu_relax() -/* The {read|write|spin}_lock() on x86 are full memory barriers. */ -static inline void smp_mb__after_lock(void) { } -#define ARCH_HAS_SMP_MB_AFTER_LOCK -  #endif /* _ASM_X86_SPINLOCK_H */ diff --git a/arch/x86/include/asm/thread_info.h b/arch/x86/include/asm/thread_info.h index a1df6e84691..27811190cbd 100644 --- a/arch/x86/include/asm/thread_info.h +++ b/arch/x86/include/asm/thread_info.h @@ -89,7 +89,6 @@ struct thread_info {  #define TIF_FORK		18	/* ret_from_fork */  #define TIF_NOHZ		19	/* in adaptive nohz mode */  #define TIF_MEMDIE		20	/* is terminating due to OOM killer */ -#define TIF_DEBUG		21	/* uses debug registers */  #define TIF_IO_BITMAP		22	/* uses I/O bitmap */  #define TIF_FORCED_TF		24	/* true if TF in eflags artificially */  #define TIF_BLOCKSTEP		25	/* set when we want DEBUGCTLMSR_BTF */ @@ -113,7 +112,6 @@ struct thread_info {  #define _TIF_IA32		(1 << TIF_IA32)  #define _TIF_FORK		(1 << TIF_FORK)  #define _TIF_NOHZ		(1 << TIF_NOHZ) -#define _TIF_DEBUG		(1 << TIF_DEBUG)  #define _TIF_IO_BITMAP		(1 << TIF_IO_BITMAP)  #define _TIF_FORCED_TF		(1 << TIF_FORCED_TF)  #define _TIF_BLOCKSTEP		(1 << TIF_BLOCKSTEP) @@ -154,7 +152,7 @@ struct thread_info {  	(_TIF_IO_BITMAP|_TIF_NOTSC|_TIF_BLOCKSTEP)  #define _TIF_WORK_CTXSW_PREV (_TIF_WORK_CTXSW|_TIF_USER_RETURN_NOTIFY) -#define _TIF_WORK_CTXSW_NEXT (_TIF_WORK_CTXSW|_TIF_DEBUG) +#define _TIF_WORK_CTXSW_NEXT (_TIF_WORK_CTXSW)  #define PREEMPT_ACTIVE		0x10000000 diff --git a/arch/x86/include/asm/tlbflush.h b/arch/x86/include/asm/tlbflush.h index 50a7fc0f824..cf512003e66 100644 --- a/arch/x86/include/asm/tlbflush.h +++ b/arch/x86/include/asm/tlbflush.h @@ -62,7 +62,7 @@ static inline void __flush_tlb_all(void)  static inline void __flush_tlb_one(unsigned long addr)  { -		__flush_tlb_single(addr); +	__flush_tlb_single(addr);  }  #define TLB_FLUSH_ALL	-1UL diff --git a/arch/x86/include/asm/trace/irq_vectors.h b/arch/x86/include/asm/trace/irq_vectors.h new file mode 100644 index 00000000000..2874df24e7a --- /dev/null +++ b/arch/x86/include/asm/trace/irq_vectors.h @@ -0,0 +1,104 @@ +#undef TRACE_SYSTEM +#define TRACE_SYSTEM irq_vectors + +#if !defined(_TRACE_IRQ_VECTORS_H) || defined(TRACE_HEADER_MULTI_READ) +#define _TRACE_IRQ_VECTORS_H + +#include <linux/tracepoint.h> + +extern void trace_irq_vector_regfunc(void); +extern void trace_irq_vector_unregfunc(void); + +DECLARE_EVENT_CLASS(x86_irq_vector, + +	TP_PROTO(int vector), + +	TP_ARGS(vector), + +	TP_STRUCT__entry( +		__field(		int,	vector	) +	), + +	TP_fast_assign( +		__entry->vector = vector; +	), + +	TP_printk("vector=%d", __entry->vector) ); + +#define DEFINE_IRQ_VECTOR_EVENT(name)		\ +DEFINE_EVENT_FN(x86_irq_vector, name##_entry,	\ +	TP_PROTO(int vector),			\ +	TP_ARGS(vector),			\ +	trace_irq_vector_regfunc,		\ +	trace_irq_vector_unregfunc);		\ +DEFINE_EVENT_FN(x86_irq_vector, name##_exit,	\ +	TP_PROTO(int vector),			\ +	TP_ARGS(vector),			\ +	trace_irq_vector_regfunc,		\ +	trace_irq_vector_unregfunc); + + +/* + * local_timer - called when entering/exiting a local timer interrupt + * vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(local_timer); + +/* + * reschedule - called when entering/exiting a reschedule vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(reschedule); + +/* + * spurious_apic - called when entering/exiting a spurious apic vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(spurious_apic); + +/* + * error_apic - called when entering/exiting an error apic vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(error_apic); + +/* + * x86_platform_ipi - called when entering/exiting a x86 platform ipi interrupt + * vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(x86_platform_ipi); + +/* + * irq_work - called when entering/exiting a irq work interrupt + * vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(irq_work); + +/* + * call_function - called when entering/exiting a call function interrupt + * vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(call_function); + +/* + * call_function_single - called when entering/exiting a call function + * single interrupt vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(call_function_single); + +/* + * threshold_apic - called when entering/exiting a threshold apic interrupt + * vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(threshold_apic); + +/* + * thermal_apic - called when entering/exiting a thermal apic interrupt + * vector handler + */ +DEFINE_IRQ_VECTOR_EVENT(thermal_apic); + +#undef TRACE_INCLUDE_PATH +#define TRACE_INCLUDE_PATH . +#define TRACE_INCLUDE_FILE irq_vectors +#endif /*  _TRACE_IRQ_VECTORS_H */ + +/* This part must be outside protection */ +#include <trace/define_trace.h> diff --git a/arch/x86/include/asm/uaccess_64.h b/arch/x86/include/asm/uaccess_64.h index 142810c457d..4f7923dd000 100644 --- a/arch/x86/include/asm/uaccess_64.h +++ b/arch/x86/include/asm/uaccess_64.h @@ -235,7 +235,7 @@ extern long __copy_user_nocache(void *dst, const void __user *src,  static inline int  __copy_from_user_nocache(void *dst, const void __user *src, unsigned size)  { -	might_sleep(); +	might_fault();  	return __copy_user_nocache(dst, src, size, 1);  } diff --git a/arch/x86/include/asm/uv/uv_bau.h b/arch/x86/include/asm/uv/uv_bau.h index a06983cdc12..0b46ef261c7 100644 --- a/arch/x86/include/asm/uv/uv_bau.h +++ b/arch/x86/include/asm/uv/uv_bau.h @@ -731,6 +731,9 @@ static inline void bau_cpubits_clear(struct bau_local_cpumask *dstp, int nbits)  }  extern void uv_bau_message_intr1(void); +#ifdef CONFIG_TRACING +#define trace_uv_bau_message_intr1 uv_bau_message_intr1 +#endif  extern void uv_bau_timeout_intr1(void);  struct atomic_short { diff --git a/arch/x86/include/asm/x86_init.h b/arch/x86/include/asm/x86_init.h index d8d99222b36..828a1565ba5 100644 --- a/arch/x86/include/asm/x86_init.h +++ b/arch/x86/include/asm/x86_init.h @@ -142,6 +142,8 @@ struct x86_cpuinit_ops {  	void (*fixup_cpu_id)(struct cpuinfo_x86 *c, int node);  }; +struct timespec; +  /**   * struct x86_platform_ops - platform specific runtime functions   * @calibrate_tsc:		calibrate TSC @@ -156,8 +158,8 @@ struct x86_cpuinit_ops {   */  struct x86_platform_ops {  	unsigned long (*calibrate_tsc)(void); -	unsigned long (*get_wallclock)(void); -	int (*set_wallclock)(unsigned long nowtime); +	void (*get_wallclock)(struct timespec *ts); +	int (*set_wallclock)(const struct timespec *ts);  	void (*iommu_shutdown)(void);  	bool (*is_untracked_pat_range)(u64 start, u64 end);  	void (*nmi_init)(void); | 
