diff options
Diffstat (limited to 'arch/s390/kernel')
| -rw-r--r-- | arch/s390/kernel/asm-offsets.c | 4 | ||||
| -rw-r--r-- | arch/s390/kernel/cache.c | 15 | ||||
| -rw-r--r-- | arch/s390/kernel/crash_dump.c | 51 | ||||
| -rw-r--r-- | arch/s390/kernel/entry.S | 12 | ||||
| -rw-r--r-- | arch/s390/kernel/entry.h | 2 | ||||
| -rw-r--r-- | arch/s390/kernel/entry64.S | 97 | ||||
| -rw-r--r-- | arch/s390/kernel/irq.c | 8 | ||||
| -rw-r--r-- | arch/s390/kernel/perf_cpum_cf.c | 4 | ||||
| -rw-r--r-- | arch/s390/kernel/perf_event.c | 53 | ||||
| -rw-r--r-- | arch/s390/kernel/processor.c | 2 | ||||
| -rw-r--r-- | arch/s390/kernel/ptrace.c | 50 | ||||
| -rw-r--r-- | arch/s390/kernel/s390_ksyms.c | 1 | ||||
| -rw-r--r-- | arch/s390/kernel/setup.c | 5 | ||||
| -rw-r--r-- | arch/s390/kernel/smp.c | 22 | ||||
| -rw-r--r-- | arch/s390/kernel/sysinfo.c | 2 | ||||
| -rw-r--r-- | arch/s390/kernel/vtime.c | 6 | 
16 files changed, 235 insertions, 99 deletions
| diff --git a/arch/s390/kernel/asm-offsets.c b/arch/s390/kernel/asm-offsets.c index 7a82f9f7010..2416138ebd3 100644 --- a/arch/s390/kernel/asm-offsets.c +++ b/arch/s390/kernel/asm-offsets.c @@ -7,6 +7,7 @@  #define ASM_OFFSETS_C  #include <linux/kbuild.h> +#include <linux/kvm_host.h>  #include <linux/sched.h>  #include <asm/cputime.h>  #include <asm/vdso.h> @@ -47,6 +48,7 @@ int main(void)  	DEFINE(__PT_GPRS, offsetof(struct pt_regs, gprs));  	DEFINE(__PT_ORIG_GPR2, offsetof(struct pt_regs, orig_gpr2));  	DEFINE(__PT_INT_CODE, offsetof(struct pt_regs, int_code)); +	DEFINE(__PT_INT_PARM, offsetof(struct pt_regs, int_parm));  	DEFINE(__PT_INT_PARM_LONG, offsetof(struct pt_regs, int_parm_long));  	DEFINE(__PT_SIZE, sizeof(struct pt_regs));  	BLANK(); @@ -161,6 +163,8 @@ int main(void)  	DEFINE(__LC_PGM_TDB, offsetof(struct _lowcore, pgm_tdb));  	DEFINE(__THREAD_trap_tdb, offsetof(struct task_struct, thread.trap_tdb));  	DEFINE(__GMAP_ASCE, offsetof(struct gmap, asce)); +	DEFINE(__SIE_PROG0C, offsetof(struct kvm_s390_sie_block, prog0c)); +	DEFINE(__SIE_PROG20, offsetof(struct kvm_s390_sie_block, prog20));  #endif /* CONFIG_32BIT */  	return 0;  } diff --git a/arch/s390/kernel/cache.c b/arch/s390/kernel/cache.c index 64b24650e4f..dd62071624b 100644 --- a/arch/s390/kernel/cache.c +++ b/arch/s390/kernel/cache.c @@ -173,7 +173,7 @@ error:  	}  } -static struct cache_dir *__cpuinit cache_create_cache_dir(int cpu) +static struct cache_dir *cache_create_cache_dir(int cpu)  {  	struct cache_dir *cache_dir;  	struct kobject *kobj = NULL; @@ -289,9 +289,8 @@ static struct kobj_type cache_index_type = {  	.default_attrs = cache_index_default_attrs,  }; -static int __cpuinit cache_create_index_dir(struct cache_dir *cache_dir, -					    struct cache *cache, int index, -					    int cpu) +static int cache_create_index_dir(struct cache_dir *cache_dir, +				  struct cache *cache, int index, int cpu)  {  	struct cache_index_dir *index_dir;  	int rc; @@ -313,7 +312,7 @@ out:  	return rc;  } -static int __cpuinit cache_add_cpu(int cpu) +static int cache_add_cpu(int cpu)  {  	struct cache_dir *cache_dir;  	struct cache *cache; @@ -335,7 +334,7 @@ static int __cpuinit cache_add_cpu(int cpu)  	return 0;  } -static void __cpuinit cache_remove_cpu(int cpu) +static void cache_remove_cpu(int cpu)  {  	struct cache_index_dir *index, *next;  	struct cache_dir *cache_dir; @@ -354,8 +353,8 @@ static void __cpuinit cache_remove_cpu(int cpu)  	cache_dir_cpu[cpu] = NULL;  } -static int __cpuinit cache_hotplug(struct notifier_block *nfb, -				   unsigned long action, void *hcpu) +static int cache_hotplug(struct notifier_block *nfb, unsigned long action, +			 void *hcpu)  {  	int cpu = (long)hcpu;  	int rc = 0; diff --git a/arch/s390/kernel/crash_dump.c b/arch/s390/kernel/crash_dump.c index f703d91bf72..d8f35565717 100644 --- a/arch/s390/kernel/crash_dump.c +++ b/arch/s390/kernel/crash_dump.c @@ -21,6 +21,48 @@  #define PTR_SUB(x, y) (((char *) (x)) - ((unsigned long) (y)))  #define PTR_DIFF(x, y) ((unsigned long)(((char *) (x)) - ((unsigned long) (y)))) + +/* + * Return physical address for virtual address + */ +static inline void *load_real_addr(void *addr) +{ +	unsigned long real_addr; + +	asm volatile( +		   "	lra     %0,0(%1)\n" +		   "	jz	0f\n" +		   "	la	%0,0\n" +		   "0:" +		   : "=a" (real_addr) : "a" (addr) : "cc"); +	return (void *)real_addr; +} + +/* + * Copy up to one page to vmalloc or real memory + */ +static ssize_t copy_page_real(void *buf, void *src, size_t csize) +{ +	size_t size; + +	if (is_vmalloc_addr(buf)) { +		BUG_ON(csize >= PAGE_SIZE); +		/* If buf is not page aligned, copy first part */ +		size = min(roundup(__pa(buf), PAGE_SIZE) - __pa(buf), csize); +		if (size) { +			if (memcpy_real(load_real_addr(buf), src, size)) +				return -EFAULT; +			buf += size; +			src += size; +		} +		/* Copy second part */ +		size = csize - size; +		return (size) ? memcpy_real(load_real_addr(buf), src, size) : 0; +	} else { +		return memcpy_real(buf, src, csize); +	} +} +  /*   * Copy one page from "oldmem"   * @@ -32,6 +74,7 @@ ssize_t copy_oldmem_page(unsigned long pfn, char *buf,  			 size_t csize, unsigned long offset, int userbuf)  {  	unsigned long src; +	int rc;  	if (!csize)  		return 0; @@ -43,11 +86,11 @@ ssize_t copy_oldmem_page(unsigned long pfn, char *buf,  		 src < OLDMEM_BASE + OLDMEM_SIZE)  		src -= OLDMEM_BASE;  	if (userbuf) -		copy_to_user_real((void __force __user *) buf, (void *) src, -				  csize); +		rc = copy_to_user_real((void __force __user *) buf, +				       (void *) src, csize);  	else -		memcpy_real(buf, (void *) src, csize); -	return csize; +		rc = copy_page_real(buf, (void *) src, csize); +	return (rc == 0) ? csize : rc;  }  /* diff --git a/arch/s390/kernel/entry.S b/arch/s390/kernel/entry.S index 4d5e6f8a797..be7a408be7a 100644 --- a/arch/s390/kernel/entry.S +++ b/arch/s390/kernel/entry.S @@ -429,11 +429,19 @@ io_skip:  	stm	%r0,%r7,__PT_R0(%r11)  	mvc	__PT_R8(32,%r11),__LC_SAVE_AREA_ASYNC  	stm	%r8,%r9,__PT_PSW(%r11) +	mvc	__PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID  	TRACE_IRQS_OFF  	xc	__SF_BACKCHAIN(4,%r15),__SF_BACKCHAIN(%r15) +io_loop:  	l	%r1,BASED(.Ldo_IRQ)  	lr	%r2,%r11		# pass pointer to pt_regs  	basr	%r14,%r1		# call do_IRQ +	tm	__LC_MACHINE_FLAGS+2,0x10	# MACHINE_FLAG_LPAR +	jz	io_return +	tpi	0 +	jz	io_return +	mvc	__PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID +	j	io_loop  io_return:  	LOCKDEP_SYS_EXIT  	TRACE_IRQS_ON @@ -573,10 +581,10 @@ ext_skip:  	stm	%r0,%r7,__PT_R0(%r11)  	mvc	__PT_R8(32,%r11),__LC_SAVE_AREA_ASYNC  	stm	%r8,%r9,__PT_PSW(%r11) +	mvc	__PT_INT_CODE(4,%r11),__LC_EXT_CPU_ADDR +	mvc	__PT_INT_PARM(4,%r11),__LC_EXT_PARAMS  	TRACE_IRQS_OFF  	lr	%r2,%r11		# pass pointer to pt_regs -	l	%r3,__LC_EXT_CPU_ADDR	# get cpu address + interruption code -	l	%r4,__LC_EXT_PARAMS	# get external parameters  	l	%r1,BASED(.Ldo_extint)  	basr	%r14,%r1		# call do_extint  	j	io_return diff --git a/arch/s390/kernel/entry.h b/arch/s390/kernel/entry.h index aa0ab02e959..3ddbc26d246 100644 --- a/arch/s390/kernel/entry.h +++ b/arch/s390/kernel/entry.h @@ -54,7 +54,7 @@ void handle_signal32(unsigned long sig, struct k_sigaction *ka,  void do_notify_resume(struct pt_regs *regs);  struct ext_code; -void do_extint(struct pt_regs *regs, struct ext_code, unsigned int, unsigned long); +void do_extint(struct pt_regs *regs);  void do_restart(void);  void __init startup_init(void);  void die(struct pt_regs *regs, const char *str); diff --git a/arch/s390/kernel/entry64.S b/arch/s390/kernel/entry64.S index 4c17eece707..1c039d0c24c 100644 --- a/arch/s390/kernel/entry64.S +++ b/arch/s390/kernel/entry64.S @@ -47,7 +47,6 @@ _TIF_WORK_INT = (_TIF_SIGPENDING | _TIF_NOTIFY_RESUME | _TIF_NEED_RESCHED | \  		 _TIF_MCCK_PENDING)  _TIF_TRACE    = (_TIF_SYSCALL_TRACE | _TIF_SYSCALL_AUDIT | _TIF_SECCOMP | \  		 _TIF_SYSCALL_TRACEPOINT) -_TIF_EXIT_SIE = (_TIF_SIGPENDING | _TIF_NEED_RESCHED | _TIF_MCCK_PENDING)  #define BASED(name) name-system_call(%r13) @@ -81,23 +80,27 @@ _TIF_EXIT_SIE = (_TIF_SIGPENDING | _TIF_NEED_RESCHED | _TIF_MCCK_PENDING)  #endif  	.endm -	.macro	HANDLE_SIE_INTERCEPT scratch,pgmcheck +	.macro	HANDLE_SIE_INTERCEPT scratch,reason  #if defined(CONFIG_KVM) || defined(CONFIG_KVM_MODULE)  	tmhh	%r8,0x0001		# interrupting from user ? -	jnz	.+42 +	jnz	.+62  	lgr	\scratch,%r9 -	slg	\scratch,BASED(.Lsie_loop) -	clg	\scratch,BASED(.Lsie_length) -	.if	\pgmcheck +	slg	\scratch,BASED(.Lsie_critical) +	clg	\scratch,BASED(.Lsie_critical_length) +	.if	\reason==1  	# Some program interrupts are suppressing (e.g. protection).  	# We must also check the instruction after SIE in that case.  	# do_protection_exception will rewind to rewind_pad -	jh	.+22 +	jh	.+42  	.else -	jhe	.+22 +	jhe	.+42  	.endif -	lg	%r9,BASED(.Lsie_loop) -	LPP	BASED(.Lhost_id)	# set host id +	lg	%r14,__SF_EMPTY(%r15)		# get control block pointer +	LPP	__SF_EMPTY+16(%r15)		# set host id +	ni	__SIE_PROG0C+3(%r14),0xfe	# no longer in SIE +	lctlg	%c1,%c1,__LC_USER_ASCE		# load primary asce +	larl	%r9,sie_exit			# skip forward to sie_exit +	mvi	__SF_EMPTY+31(%r15),\reason	# set exit reason  #endif  	.endm @@ -450,7 +453,7 @@ ENTRY(io_int_handler)  	lg	%r12,__LC_THREAD_INFO  	larl	%r13,system_call  	lmg	%r8,%r9,__LC_IO_OLD_PSW -	HANDLE_SIE_INTERCEPT %r14,0 +	HANDLE_SIE_INTERCEPT %r14,2  	SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT  	tmhh	%r8,0x0001		# interrupting from user?  	jz	io_skip @@ -460,10 +463,18 @@ io_skip:  	stmg	%r0,%r7,__PT_R0(%r11)  	mvc	__PT_R8(64,%r11),__LC_SAVE_AREA_ASYNC  	stmg	%r8,%r9,__PT_PSW(%r11) +	mvc	__PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID  	TRACE_IRQS_OFF  	xc	__SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15) +io_loop:  	lgr	%r2,%r11		# pass pointer to pt_regs  	brasl	%r14,do_IRQ +	tm	__LC_MACHINE_FLAGS+6,0x10	# MACHINE_FLAG_LPAR +	jz	io_return +	tpi	0 +	jz	io_return +	mvc	__PT_INT_CODE(12,%r11),__LC_SUBCHANNEL_ID +	j	io_loop  io_return:  	LOCKDEP_SYS_EXIT  	TRACE_IRQS_ON @@ -595,7 +606,7 @@ ENTRY(ext_int_handler)  	lg	%r12,__LC_THREAD_INFO  	larl	%r13,system_call  	lmg	%r8,%r9,__LC_EXT_OLD_PSW -	HANDLE_SIE_INTERCEPT %r14,0 +	HANDLE_SIE_INTERCEPT %r14,3  	SWITCH_ASYNC __LC_SAVE_AREA_ASYNC,__LC_ASYNC_STACK,STACK_SHIFT  	tmhh	%r8,0x0001		# interrupting from user ?  	jz	ext_skip @@ -605,13 +616,13 @@ ext_skip:  	stmg	%r0,%r7,__PT_R0(%r11)  	mvc	__PT_R8(64,%r11),__LC_SAVE_AREA_ASYNC  	stmg	%r8,%r9,__PT_PSW(%r11) +	lghi	%r1,__LC_EXT_PARAMS2 +	mvc	__PT_INT_CODE(4,%r11),__LC_EXT_CPU_ADDR +	mvc	__PT_INT_PARM(4,%r11),__LC_EXT_PARAMS +	mvc	__PT_INT_PARM_LONG(8,%r11),0(%r1)  	TRACE_IRQS_OFF  	xc	__SF_BACKCHAIN(8,%r15),__SF_BACKCHAIN(%r15) -	lghi	%r1,4096  	lgr	%r2,%r11		# pass pointer to pt_regs -	llgf	%r3,__LC_EXT_CPU_ADDR	# get cpu address + interruption code -	llgf	%r4,__LC_EXT_PARAMS	# get external parameter -	lg	%r5,__LC_EXT_PARAMS2-4096(%r1)	# get 64 bit external parameter  	brasl	%r14,do_extint  	j	io_return @@ -643,7 +654,7 @@ ENTRY(mcck_int_handler)  	lg	%r12,__LC_THREAD_INFO  	larl	%r13,system_call  	lmg	%r8,%r9,__LC_MCK_OLD_PSW -	HANDLE_SIE_INTERCEPT %r14,0 +	HANDLE_SIE_INTERCEPT %r14,4  	tm	__LC_MCCK_CODE,0x80	# system damage?  	jo	mcck_panic		# yes -> rest of mcck code invalid  	lghi	%r14,__LC_CPU_TIMER_SAVE_AREA @@ -937,56 +948,50 @@ ENTRY(sie64a)  	stmg	%r6,%r14,__SF_GPRS(%r15)	# save kernel registers  	stg	%r2,__SF_EMPTY(%r15)		# save control block pointer  	stg	%r3,__SF_EMPTY+8(%r15)		# save guest register save area -	xc	__SF_EMPTY+16(8,%r15),__SF_EMPTY+16(%r15) # host id == 0 +	xc	__SF_EMPTY+16(16,%r15),__SF_EMPTY+16(%r15) # host id & reason  	lmg	%r0,%r13,0(%r3)			# load guest gprs 0-13 -# some program checks are suppressing. C code (e.g. do_protection_exception) -# will rewind the PSW by the ILC, which is 4 bytes in case of SIE. Other -# instructions in the sie_loop should not cause program interrupts. So -# lets use a nop (47 00 00 00) as a landing pad. -# See also HANDLE_SIE_INTERCEPT -rewind_pad: -	nop	0 -sie_loop: -	lg	%r14,__LC_THREAD_INFO		# pointer thread_info struct -	tm	__TI_flags+7(%r14),_TIF_EXIT_SIE -	jnz	sie_exit  	lg	%r14,__LC_GMAP			# get gmap pointer  	ltgr	%r14,%r14  	jz	sie_gmap  	lctlg	%c1,%c1,__GMAP_ASCE(%r14)	# load primary asce  sie_gmap:  	lg	%r14,__SF_EMPTY(%r15)		# get control block pointer +	oi	__SIE_PROG0C+3(%r14),1		# we are going into SIE now +	tm	__SIE_PROG20+3(%r14),1		# last exit... +	jnz	sie_done  	LPP	__SF_EMPTY(%r15)		# set guest id  	sie	0(%r14)  sie_done:  	LPP	__SF_EMPTY+16(%r15)		# set host id -	lg	%r14,__LC_THREAD_INFO		# pointer thread_info struct -sie_exit: +	ni	__SIE_PROG0C+3(%r14),0xfe	# no longer in SIE  	lctlg	%c1,%c1,__LC_USER_ASCE		# load primary asce +# some program checks are suppressing. C code (e.g. do_protection_exception) +# will rewind the PSW by the ILC, which is 4 bytes in case of SIE. Other +# instructions beween sie64a and sie_done should not cause program +# interrupts. So lets use a nop (47 00 00 00) as a landing pad. +# See also HANDLE_SIE_INTERCEPT +rewind_pad: +	nop	0 +	.globl sie_exit +sie_exit:  	lg	%r14,__SF_EMPTY+8(%r15)		# load guest register save area  	stmg	%r0,%r13,0(%r14)		# save guest gprs 0-13  	lmg	%r6,%r14,__SF_GPRS(%r15)	# restore kernel registers -	lghi	%r2,0 +	lg	%r2,__SF_EMPTY+24(%r15)		# return exit reason code  	br	%r14  sie_fault: -	lctlg	%c1,%c1,__LC_USER_ASCE		# load primary asce -	lg	%r14,__LC_THREAD_INFO		# pointer thread_info struct -	lg	%r14,__SF_EMPTY+8(%r15)		# load guest register save area -	stmg	%r0,%r13,0(%r14)		# save guest gprs 0-13 -	lmg	%r6,%r14,__SF_GPRS(%r15)	# restore kernel registers -	lghi	%r2,-EFAULT -	br	%r14 +	lghi	%r14,-EFAULT +	stg	%r14,__SF_EMPTY+24(%r15)	# set exit reason code +	j	sie_exit  	.align	8 -.Lsie_loop: -	.quad	sie_loop -.Lsie_length: -	.quad	sie_done - sie_loop -.Lhost_id: -	.quad	0 +.Lsie_critical: +	.quad	sie_gmap +.Lsie_critical_length: +	.quad	sie_done - sie_gmap  	EX_TABLE(rewind_pad,sie_fault) -	EX_TABLE(sie_loop,sie_fault) +	EX_TABLE(sie_exit,sie_fault)  #endif  		.section .rodata, "a" diff --git a/arch/s390/kernel/irq.c b/arch/s390/kernel/irq.c index dd3c1994b8b..54b0995514e 100644 --- a/arch/s390/kernel/irq.c +++ b/arch/s390/kernel/irq.c @@ -234,9 +234,9 @@ int unregister_external_interrupt(u16 code, ext_int_handler_t handler)  }  EXPORT_SYMBOL(unregister_external_interrupt); -void __irq_entry do_extint(struct pt_regs *regs, struct ext_code ext_code, -			   unsigned int param32, unsigned long param64) +void __irq_entry do_extint(struct pt_regs *regs)  { +	struct ext_code ext_code;  	struct pt_regs *old_regs;  	struct ext_int_info *p;  	int index; @@ -248,6 +248,7 @@ void __irq_entry do_extint(struct pt_regs *regs, struct ext_code ext_code,  		clock_comparator_work();  	}  	kstat_incr_irqs_this_cpu(EXTERNAL_INTERRUPT, NULL); +	ext_code = *(struct ext_code *) ®s->int_code;  	if (ext_code.code != 0x1004)  		__get_cpu_var(s390_idle).nohz_delay = 1; @@ -255,7 +256,8 @@ void __irq_entry do_extint(struct pt_regs *regs, struct ext_code ext_code,  	rcu_read_lock();  	list_for_each_entry_rcu(p, &ext_int_hash[index], entry)  		if (likely(p->code == ext_code.code)) -			p->handler(ext_code, param32, param64); +			p->handler(ext_code, regs->int_parm, +				   regs->int_parm_long);  	rcu_read_unlock();  	irq_exit();  	set_irq_regs(old_regs); diff --git a/arch/s390/kernel/perf_cpum_cf.c b/arch/s390/kernel/perf_cpum_cf.c index 390d9ae57bb..fb99c2057b8 100644 --- a/arch/s390/kernel/perf_cpum_cf.c +++ b/arch/s390/kernel/perf_cpum_cf.c @@ -639,8 +639,8 @@ static struct pmu cpumf_pmu = {  	.cancel_txn   = cpumf_pmu_cancel_txn,  }; -static int __cpuinit cpumf_pmu_notifier(struct notifier_block *self, -					unsigned long action, void *hcpu) +static int cpumf_pmu_notifier(struct notifier_block *self, unsigned long action, +			      void *hcpu)  {  	unsigned int cpu = (long) hcpu;  	int flags; diff --git a/arch/s390/kernel/perf_event.c b/arch/s390/kernel/perf_event.c index f58f37f6682..500aa1029bc 100644 --- a/arch/s390/kernel/perf_event.c +++ b/arch/s390/kernel/perf_event.c @@ -13,6 +13,7 @@  #include <linux/kernel.h>  #include <linux/perf_event.h> +#include <linux/kvm_host.h>  #include <linux/percpu.h>  #include <linux/export.h>  #include <asm/irq.h> @@ -39,6 +40,58 @@ int perf_num_counters(void)  }  EXPORT_SYMBOL(perf_num_counters); +static struct kvm_s390_sie_block *sie_block(struct pt_regs *regs) +{ +	struct stack_frame *stack = (struct stack_frame *) regs->gprs[15]; + +	if (!stack) +		return NULL; + +	return (struct kvm_s390_sie_block *) stack->empty1[0]; +} + +static bool is_in_guest(struct pt_regs *regs) +{ +	if (user_mode(regs)) +		return false; +#if defined(CONFIG_KVM) || defined(CONFIG_KVM_MODULE) +	return instruction_pointer(regs) == (unsigned long) &sie_exit; +#else +	return false; +#endif +} + +static unsigned long guest_is_user_mode(struct pt_regs *regs) +{ +	return sie_block(regs)->gpsw.mask & PSW_MASK_PSTATE; +} + +static unsigned long instruction_pointer_guest(struct pt_regs *regs) +{ +	return sie_block(regs)->gpsw.addr & PSW_ADDR_INSN; +} + +unsigned long perf_instruction_pointer(struct pt_regs *regs) +{ +	return is_in_guest(regs) ? instruction_pointer_guest(regs) +				 : instruction_pointer(regs); +} + +static unsigned long perf_misc_guest_flags(struct pt_regs *regs) +{ +	return guest_is_user_mode(regs) ? PERF_RECORD_MISC_GUEST_USER +					: PERF_RECORD_MISC_GUEST_KERNEL; +} + +unsigned long perf_misc_flags(struct pt_regs *regs) +{ +	if (is_in_guest(regs)) +		return perf_misc_guest_flags(regs); + +	return user_mode(regs) ? PERF_RECORD_MISC_USER +			       : PERF_RECORD_MISC_KERNEL; +} +  void perf_event_print_debug(void)  {  	struct cpumf_ctr_info cf_info; diff --git a/arch/s390/kernel/processor.c b/arch/s390/kernel/processor.c index 753c41d0ffd..24612029f45 100644 --- a/arch/s390/kernel/processor.c +++ b/arch/s390/kernel/processor.c @@ -21,7 +21,7 @@ static DEFINE_PER_CPU(struct cpuid, cpu_id);  /*   * cpu_init - initializes state that is per-CPU.   */ -void __cpuinit cpu_init(void) +void cpu_init(void)  {  	struct s390_idle_data *idle = &__get_cpu_var(s390_idle);  	struct cpuid *id = &__get_cpu_var(cpu_id); diff --git a/arch/s390/kernel/ptrace.c b/arch/s390/kernel/ptrace.c index a314c57f4e9..e9fadb04e3c 100644 --- a/arch/s390/kernel/ptrace.c +++ b/arch/s390/kernel/ptrace.c @@ -47,7 +47,7 @@ enum s390_regset {  	REGSET_GENERAL_EXTENDED,  }; -void update_per_regs(struct task_struct *task) +void update_cr_regs(struct task_struct *task)  {  	struct pt_regs *regs = task_pt_regs(task);  	struct thread_struct *thread = &task->thread; @@ -56,17 +56,25 @@ void update_per_regs(struct task_struct *task)  #ifdef CONFIG_64BIT  	/* Take care of the enable/disable of transactional execution. */  	if (MACHINE_HAS_TE) { -		unsigned long cr0, cr0_new; +		unsigned long cr[3], cr_new[3]; -		__ctl_store(cr0, 0, 0); -		/* set or clear transaction execution bits 8 and 9. */ +		__ctl_store(cr, 0, 2); +		cr_new[1] = cr[1]; +		/* Set or clear transaction execution TXC/PIFO bits 8 and 9. */  		if (task->thread.per_flags & PER_FLAG_NO_TE) -			cr0_new = cr0 & ~(3UL << 54); +			cr_new[0] = cr[0] & ~(3UL << 54);  		else -			cr0_new = cr0 | (3UL << 54); -		/* Only load control register 0 if necessary. */ -		if (cr0 != cr0_new) -			__ctl_load(cr0_new, 0, 0); +			cr_new[0] = cr[0] | (3UL << 54); +		/* Set or clear transaction execution TDC bits 62 and 63. */ +		cr_new[2] = cr[2] & ~3UL; +		if (task->thread.per_flags & PER_FLAG_TE_ABORT_RAND) { +			if (task->thread.per_flags & PER_FLAG_TE_ABORT_RAND_TEND) +				cr_new[2] |= 1UL; +			else +				cr_new[2] |= 2UL; +		} +		if (memcmp(&cr_new, &cr, sizeof(cr))) +			__ctl_load(cr_new, 0, 2);  	}  #endif  	/* Copy user specified PER registers */ @@ -100,14 +108,14 @@ void user_enable_single_step(struct task_struct *task)  {  	set_tsk_thread_flag(task, TIF_SINGLE_STEP);  	if (task == current) -		update_per_regs(task); +		update_cr_regs(task);  }  void user_disable_single_step(struct task_struct *task)  {  	clear_tsk_thread_flag(task, TIF_SINGLE_STEP);  	if (task == current) -		update_per_regs(task); +		update_cr_regs(task);  }  /* @@ -447,6 +455,26 @@ long arch_ptrace(struct task_struct *child, long request,  		if (!MACHINE_HAS_TE)  			return -EIO;  		child->thread.per_flags |= PER_FLAG_NO_TE; +		child->thread.per_flags &= ~PER_FLAG_TE_ABORT_RAND; +		return 0; +	case PTRACE_TE_ABORT_RAND: +		if (!MACHINE_HAS_TE || (child->thread.per_flags & PER_FLAG_NO_TE)) +			return -EIO; +		switch (data) { +		case 0UL: +			child->thread.per_flags &= ~PER_FLAG_TE_ABORT_RAND; +			break; +		case 1UL: +			child->thread.per_flags |= PER_FLAG_TE_ABORT_RAND; +			child->thread.per_flags |= PER_FLAG_TE_ABORT_RAND_TEND; +			break; +		case 2UL: +			child->thread.per_flags |= PER_FLAG_TE_ABORT_RAND; +			child->thread.per_flags &= ~PER_FLAG_TE_ABORT_RAND_TEND; +			break; +		default: +			return -EINVAL; +		}  		return 0;  	default:  		/* Removing high order bit from addr (only for 31 bit). */ diff --git a/arch/s390/kernel/s390_ksyms.c b/arch/s390/kernel/s390_ksyms.c index 9bdbcef1da9..3bac589844a 100644 --- a/arch/s390/kernel/s390_ksyms.c +++ b/arch/s390/kernel/s390_ksyms.c @@ -7,6 +7,7 @@ EXPORT_SYMBOL(_mcount);  #endif  #if defined(CONFIG_KVM) || defined(CONFIG_KVM_MODULE)  EXPORT_SYMBOL(sie64a); +EXPORT_SYMBOL(sie_exit);  #endif  EXPORT_SYMBOL(memcpy);  EXPORT_SYMBOL(memset); diff --git a/arch/s390/kernel/setup.c b/arch/s390/kernel/setup.c index 0a49095104c..aeed8a61fa0 100644 --- a/arch/s390/kernel/setup.c +++ b/arch/s390/kernel/setup.c @@ -719,10 +719,6 @@ static void reserve_oldmem(void)  	}  	create_mem_hole(memory_chunk, OLDMEM_BASE, OLDMEM_SIZE);  	create_mem_hole(memory_chunk, OLDMEM_SIZE, real_size - OLDMEM_SIZE); -	if (OLDMEM_BASE + OLDMEM_SIZE == real_size) -		saved_max_pfn = PFN_DOWN(OLDMEM_BASE) - 1; -	else -		saved_max_pfn = PFN_DOWN(real_size) - 1;  #endif  } @@ -998,6 +994,7 @@ static void __init setup_hwcaps(void)  		strcpy(elf_platform, "z196");  		break;  	case 0x2827: +	case 0x2828:  		strcpy(elf_platform, "zEC12");  		break;  	} diff --git a/arch/s390/kernel/smp.c b/arch/s390/kernel/smp.c index 4f977d0d25c..d386c4e9d2e 100644 --- a/arch/s390/kernel/smp.c +++ b/arch/s390/kernel/smp.c @@ -49,7 +49,6 @@  enum {  	ec_schedule = 0, -	ec_call_function,  	ec_call_function_single,  	ec_stop_cpu,  }; @@ -166,7 +165,7 @@ static void pcpu_ec_call(struct pcpu *pcpu, int ec_bit)  	pcpu_sigp_retry(pcpu, order, 0);  } -static int __cpuinit pcpu_alloc_lowcore(struct pcpu *pcpu, int cpu) +static int pcpu_alloc_lowcore(struct pcpu *pcpu, int cpu)  {  	struct _lowcore *lc; @@ -438,8 +437,6 @@ static void smp_handle_ext_call(void)  		smp_stop_cpu();  	if (test_bit(ec_schedule, &bits))  		scheduler_ipi(); -	if (test_bit(ec_call_function, &bits)) -		generic_smp_call_function_interrupt();  	if (test_bit(ec_call_function_single, &bits))  		generic_smp_call_function_single_interrupt();  } @@ -456,7 +453,7 @@ void arch_send_call_function_ipi_mask(const struct cpumask *mask)  	int cpu;  	for_each_cpu(cpu, mask) -		pcpu_ec_call(pcpu_devices + cpu, ec_call_function); +		pcpu_ec_call(pcpu_devices + cpu, ec_call_function_single);  }  void arch_send_call_function_single_ipi(int cpu) @@ -619,10 +616,9 @@ static struct sclp_cpu_info *smp_get_cpu_info(void)  	return info;  } -static int __cpuinit smp_add_present_cpu(int cpu); +static int smp_add_present_cpu(int cpu); -static int __cpuinit __smp_rescan_cpus(struct sclp_cpu_info *info, -				       int sysfs_add) +static int __smp_rescan_cpus(struct sclp_cpu_info *info, int sysfs_add)  {  	struct pcpu *pcpu;  	cpumask_t avail; @@ -688,7 +684,7 @@ static void __init smp_detect_cpus(void)  /*   *	Activate a secondary processor.   */ -static void __cpuinit smp_start_secondary(void *cpuvoid) +static void smp_start_secondary(void *cpuvoid)  {  	S390_lowcore.last_update_clock = get_tod_clock();  	S390_lowcore.restart_stack = (unsigned long) restart_stack; @@ -711,7 +707,7 @@ static void __cpuinit smp_start_secondary(void *cpuvoid)  }  /* Upping and downing of CPUs */ -int __cpuinit __cpu_up(unsigned int cpu, struct task_struct *tidle) +int __cpu_up(unsigned int cpu, struct task_struct *tidle)  {  	struct pcpu *pcpu;  	int rc; @@ -967,8 +963,8 @@ static struct attribute_group cpu_online_attr_group = {  	.attrs = cpu_online_attrs,  }; -static int __cpuinit smp_cpu_notify(struct notifier_block *self, -				    unsigned long action, void *hcpu) +static int smp_cpu_notify(struct notifier_block *self, unsigned long action, +			  void *hcpu)  {  	unsigned int cpu = (unsigned int)(long)hcpu;  	struct cpu *c = &pcpu_devices[cpu].cpu; @@ -986,7 +982,7 @@ static int __cpuinit smp_cpu_notify(struct notifier_block *self,  	return notifier_from_errno(err);  } -static int __cpuinit smp_add_present_cpu(int cpu) +static int smp_add_present_cpu(int cpu)  {  	struct cpu *c = &pcpu_devices[cpu].cpu;  	struct device *s = &c->dev; diff --git a/arch/s390/kernel/sysinfo.c b/arch/s390/kernel/sysinfo.c index 62f89d98e88..811f542b8ed 100644 --- a/arch/s390/kernel/sysinfo.c +++ b/arch/s390/kernel/sysinfo.c @@ -418,7 +418,7 @@ void s390_adjust_jiffies(void)  /*   * calibrate the delay loop   */ -void __cpuinit calibrate_delay(void) +void calibrate_delay(void)  {  	s390_adjust_jiffies();  	/* Print the good old Bogomips line .. */ diff --git a/arch/s390/kernel/vtime.c b/arch/s390/kernel/vtime.c index 3fb09359eda..9b9c1b78ec6 100644 --- a/arch/s390/kernel/vtime.c +++ b/arch/s390/kernel/vtime.c @@ -371,14 +371,14 @@ EXPORT_SYMBOL(del_virt_timer);  /*   * Start the virtual CPU timer on the current CPU.   */ -void __cpuinit init_cpu_vtimer(void) +void init_cpu_vtimer(void)  {  	/* set initial cpu timer */  	set_vtimer(VTIMER_MAX_SLICE);  } -static int __cpuinit s390_nohz_notify(struct notifier_block *self, -				      unsigned long action, void *hcpu) +static int s390_nohz_notify(struct notifier_block *self, unsigned long action, +			    void *hcpu)  {  	struct s390_idle_data *idle;  	long cpu = (long) hcpu; | 
