diff options
Diffstat (limited to 'arch/s390/include/asm')
| -rw-r--r-- | arch/s390/include/asm/cpacf.h | 410 | ||||
| -rw-r--r-- | arch/s390/include/asm/fpu/types.h | 10 | ||||
| -rw-r--r-- | arch/s390/include/asm/ftrace.h | 4 | ||||
| -rw-r--r-- | arch/s390/include/asm/kvm_host.h | 11 | ||||
| -rw-r--r-- | arch/s390/include/asm/livepatch.h | 7 | ||||
| -rw-r--r-- | arch/s390/include/asm/mmu.h | 2 | ||||
| -rw-r--r-- | arch/s390/include/asm/mmu_context.h | 28 | ||||
| -rw-r--r-- | arch/s390/include/asm/pci.h | 36 | ||||
| -rw-r--r-- | arch/s390/include/asm/pgalloc.h | 4 | ||||
| -rw-r--r-- | arch/s390/include/asm/pgtable.h | 1 | ||||
| -rw-r--r-- | arch/s390/include/asm/processor.h | 11 | ||||
| -rw-r--r-- | arch/s390/include/asm/rwsem.h | 18 | ||||
| -rw-r--r-- | arch/s390/include/asm/sclp.h | 14 | ||||
| -rw-r--r-- | arch/s390/include/asm/seccomp.h | 2 | ||||
| -rw-r--r-- | arch/s390/include/asm/sigp.h | 1 | ||||
| -rw-r--r-- | arch/s390/include/asm/thread_info.h | 1 | ||||
| -rw-r--r-- | arch/s390/include/asm/tlbflush.h | 9 | 
17 files changed, 521 insertions, 48 deletions
| diff --git a/arch/s390/include/asm/cpacf.h b/arch/s390/include/asm/cpacf.h new file mode 100644 index 000000000000..1a82cf26ee11 --- /dev/null +++ b/arch/s390/include/asm/cpacf.h @@ -0,0 +1,410 @@ +/* + * CP Assist for Cryptographic Functions (CPACF) + * + * Copyright IBM Corp. 2003, 2016 + * Author(s): Thomas Spatzier + *	      Jan Glauber + *	      Harald Freudenberger ([email protected]) + *	      Martin Schwidefsky <[email protected]> + */ +#ifndef _ASM_S390_CPACF_H +#define _ASM_S390_CPACF_H + +#include <asm/facility.h> + +/* + * Instruction opcodes for the CPACF instructions + */ +#define CPACF_KMAC		0xb91e		/* MSA	*/ +#define CPACF_KM		0xb92e		/* MSA	*/ +#define CPACF_KMC		0xb92f		/* MSA	*/ +#define CPACF_KIMD		0xb93e		/* MSA	*/ +#define CPACF_KLMD		0xb93f		/* MSA	*/ +#define CPACF_PCC		0xb92c		/* MSA4 */ +#define CPACF_KMCTR		0xb92d		/* MSA4 */ +#define CPACF_PPNO		0xb93c		/* MSA5 */ + +/* + * Function codes for the KM (CIPHER MESSAGE) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_KM_QUERY		0x00 +#define CPACF_KM_DEA_ENC	0x01 +#define CPACF_KM_DEA_DEC	0x81 +#define CPACF_KM_TDEA_128_ENC	0x02 +#define CPACF_KM_TDEA_128_DEC	0x82 +#define CPACF_KM_TDEA_192_ENC	0x03 +#define CPACF_KM_TDEA_192_DEC	0x83 +#define CPACF_KM_AES_128_ENC	0x12 +#define CPACF_KM_AES_128_DEC	0x92 +#define CPACF_KM_AES_192_ENC	0x13 +#define CPACF_KM_AES_192_DEC	0x93 +#define CPACF_KM_AES_256_ENC	0x14 +#define CPACF_KM_AES_256_DEC	0x94 +#define CPACF_KM_XTS_128_ENC	0x32 +#define CPACF_KM_XTS_128_DEC	0xb2 +#define CPACF_KM_XTS_256_ENC	0x34 +#define CPACF_KM_XTS_256_DEC	0xb4 + +/* + * Function codes for the KMC (CIPHER MESSAGE WITH CHAINING) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_KMC_QUERY		0x00 +#define CPACF_KMC_DEA_ENC	0x01 +#define CPACF_KMC_DEA_DEC	0x81 +#define CPACF_KMC_TDEA_128_ENC	0x02 +#define CPACF_KMC_TDEA_128_DEC	0x82 +#define CPACF_KMC_TDEA_192_ENC	0x03 +#define CPACF_KMC_TDEA_192_DEC	0x83 +#define CPACF_KMC_AES_128_ENC	0x12 +#define CPACF_KMC_AES_128_DEC	0x92 +#define CPACF_KMC_AES_192_ENC	0x13 +#define CPACF_KMC_AES_192_DEC	0x93 +#define CPACF_KMC_AES_256_ENC	0x14 +#define CPACF_KMC_AES_256_DEC	0x94 +#define CPACF_KMC_PRNG		0x43 + +/* + * Function codes for the KMCTR (CIPHER MESSAGE WITH COUNTER) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_KMCTR_QUERY	 0x00 +#define CPACF_KMCTR_DEA_ENC	 0x01 +#define CPACF_KMCTR_DEA_DEC	 0x81 +#define CPACF_KMCTR_TDEA_128_ENC 0x02 +#define CPACF_KMCTR_TDEA_128_DEC 0x82 +#define CPACF_KMCTR_TDEA_192_ENC 0x03 +#define CPACF_KMCTR_TDEA_192_DEC 0x83 +#define CPACF_KMCTR_AES_128_ENC	 0x12 +#define CPACF_KMCTR_AES_128_DEC	 0x92 +#define CPACF_KMCTR_AES_192_ENC	 0x13 +#define CPACF_KMCTR_AES_192_DEC	 0x93 +#define CPACF_KMCTR_AES_256_ENC	 0x14 +#define CPACF_KMCTR_AES_256_DEC	 0x94 + +/* + * Function codes for the KIMD (COMPUTE INTERMEDIATE MESSAGE DIGEST) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_KIMD_QUERY	0x00 +#define CPACF_KIMD_SHA_1	0x01 +#define CPACF_KIMD_SHA_256	0x02 +#define CPACF_KIMD_SHA_512	0x03 +#define CPACF_KIMD_GHASH	0x41 + +/* + * Function codes for the KLMD (COMPUTE LAST MESSAGE DIGEST) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_KLMD_QUERY	0x00 +#define CPACF_KLMD_SHA_1	0x01 +#define CPACF_KLMD_SHA_256	0x02 +#define CPACF_KLMD_SHA_512	0x03 + +/* + * function codes for the KMAC (COMPUTE MESSAGE AUTHENTICATION CODE) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_KMAC_QUERY	0x00 +#define CPACF_KMAC_DEA		0x01 +#define CPACF_KMAC_TDEA_128	0x02 +#define CPACF_KMAC_TDEA_192	0x03 + +/* + * Function codes for the PPNO (PERFORM PSEUDORANDOM NUMBER OPERATION) + * instruction (0x80 is the decipher modifier bit) + */ +#define CPACF_PPNO_QUERY		0x00 +#define CPACF_PPNO_SHA512_DRNG_GEN	0x03 +#define CPACF_PPNO_SHA512_DRNG_SEED	0x83 + +/** + * cpacf_query() - check if a specific CPACF function is available + * @opcode: the opcode of the crypto instruction + * @func: the function code to test for + * + * Executes the query function for the given crypto instruction @opcode + * and checks if @func is available + * + * Returns 1 if @func is available for @opcode, 0 otherwise + */ +static inline void __cpacf_query(unsigned int opcode, unsigned char *status) +{ +	typedef struct { unsigned char _[16]; } status_type; +	register unsigned long r0 asm("0") = 0;	/* query function */ +	register unsigned long r1 asm("1") = (unsigned long) status; + +	asm volatile( +		/* Parameter registers are ignored, but may not be 0 */ +		"0:	.insn	rrf,%[opc] << 16,2,2,2,0\n" +		"	brc	1,0b\n"	/* handle partial completion */ +		: "=m" (*(status_type *) status) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (opcode) +		: "cc"); +} + +static inline int cpacf_query(unsigned int opcode, unsigned int func) +{ +	unsigned char status[16]; + +	switch (opcode) { +	case CPACF_KMAC: +	case CPACF_KM: +	case CPACF_KMC: +	case CPACF_KIMD: +	case CPACF_KLMD: +		if (!test_facility(17))	/* check for MSA */ +			return 0; +		break; +	case CPACF_PCC: +	case CPACF_KMCTR: +		if (!test_facility(77))	/* check for MSA4 */ +			return 0; +		break; +	case CPACF_PPNO: +		if (!test_facility(57))	/* check for MSA5 */ +			return 0; +		break; +	default: +		BUG(); +	} +	__cpacf_query(opcode, status); +	return (status[func >> 3] & (0x80 >> (func & 7))) != 0; +} + +/** + * cpacf_km() - executes the KM (CIPHER MESSAGE) instruction + * @func: the function code passed to KM; see CPACF_KM_xxx defines + * @param: address of parameter block; see POP for details on each func + * @dest: address of destination memory area + * @src: address of source memory area + * @src_len: length of src operand in bytes + * + * Returns 0 for the query func, number of processed bytes for + * encryption/decryption funcs + */ +static inline int cpacf_km(long func, void *param, +			   u8 *dest, const u8 *src, long src_len) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) src; +	register unsigned long r3 asm("3") = (unsigned long) src_len; +	register unsigned long r4 asm("4") = (unsigned long) dest; + +	asm volatile( +		"0:	.insn	rre,%[opc] << 16,%[dst],%[src]\n" +		"	brc	1,0b\n" /* handle partial completion */ +		: [src] "+a" (r2), [len] "+d" (r3), [dst] "+a" (r4) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_KM) +		: "cc", "memory"); + +	return src_len - r3; +} + +/** + * cpacf_kmc() - executes the KMC (CIPHER MESSAGE WITH CHAINING) instruction + * @func: the function code passed to KM; see CPACF_KMC_xxx defines + * @param: address of parameter block; see POP for details on each func + * @dest: address of destination memory area + * @src: address of source memory area + * @src_len: length of src operand in bytes + * + * Returns 0 for the query func, number of processed bytes for + * encryption/decryption funcs + */ +static inline int cpacf_kmc(long func, void *param, +			    u8 *dest, const u8 *src, long src_len) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) src; +	register unsigned long r3 asm("3") = (unsigned long) src_len; +	register unsigned long r4 asm("4") = (unsigned long) dest; + +	asm volatile( +		"0:	.insn	rre,%[opc] << 16,%[dst],%[src]\n" +		"	brc	1,0b\n" /* handle partial completion */ +		: [src] "+a" (r2), [len] "+d" (r3), [dst] "+a" (r4) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_KMC) +		: "cc", "memory"); + +	return src_len - r3; +} + +/** + * cpacf_kimd() - executes the KIMD (COMPUTE INTERMEDIATE MESSAGE DIGEST) + *		  instruction + * @func: the function code passed to KM; see CPACF_KIMD_xxx defines + * @param: address of parameter block; see POP for details on each func + * @src: address of source memory area + * @src_len: length of src operand in bytes + * + * Returns 0 for the query func, number of processed bytes for digest funcs + */ +static inline int cpacf_kimd(long func, void *param, +			     const u8 *src, long src_len) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) src; +	register unsigned long r3 asm("3") = (unsigned long) src_len; + +	asm volatile( +		"0:	.insn	rre,%[opc] << 16,0,%[src]\n" +		"	brc	1,0b\n" /* handle partial completion */ +		: [src] "+a" (r2), [len] "+d" (r3) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_KIMD) +		: "cc", "memory"); + +	return src_len - r3; +} + +/** + * cpacf_klmd() - executes the KLMD (COMPUTE LAST MESSAGE DIGEST) instruction + * @func: the function code passed to KM; see CPACF_KLMD_xxx defines + * @param: address of parameter block; see POP for details on each func + * @src: address of source memory area + * @src_len: length of src operand in bytes + * + * Returns 0 for the query func, number of processed bytes for digest funcs + */ +static inline int cpacf_klmd(long func, void *param, +			     const u8 *src, long src_len) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) src; +	register unsigned long r3 asm("3") = (unsigned long) src_len; + +	asm volatile( +		"0:	.insn	rre,%[opc] << 16,0,%[src]\n" +		"	brc	1,0b\n" /* handle partial completion */ +		: [src] "+a" (r2), [len] "+d" (r3) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_KLMD) +		: "cc", "memory"); + +	return src_len - r3; +} + +/** + * cpacf_kmac() - executes the KMAC (COMPUTE MESSAGE AUTHENTICATION CODE) + *		  instruction + * @func: the function code passed to KM; see CPACF_KMAC_xxx defines + * @param: address of parameter block; see POP for details on each func + * @src: address of source memory area + * @src_len: length of src operand in bytes + * + * Returns 0 for the query func, number of processed bytes for digest funcs + */ +static inline int cpacf_kmac(long func, void *param, +			     const u8 *src, long src_len) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) src; +	register unsigned long r3 asm("3") = (unsigned long) src_len; + +	asm volatile( +		"0:	.insn	rre,%[opc] << 16,0,%[src]\n" +		"	brc	1,0b\n" /* handle partial completion */ +		: [src] "+a" (r2), [len] "+d" (r3) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_KMAC) +		: "cc", "memory"); + +	return src_len - r3; +} + +/** + * cpacf_kmctr() - executes the KMCTR (CIPHER MESSAGE WITH COUNTER) instruction + * @func: the function code passed to KMCTR; see CPACF_KMCTR_xxx defines + * @param: address of parameter block; see POP for details on each func + * @dest: address of destination memory area + * @src: address of source memory area + * @src_len: length of src operand in bytes + * @counter: address of counter value + * + * Returns 0 for the query func, number of processed bytes for + * encryption/decryption funcs + */ +static inline int cpacf_kmctr(long func, void *param, u8 *dest, +			      const u8 *src, long src_len, u8 *counter) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) src; +	register unsigned long r3 asm("3") = (unsigned long) src_len; +	register unsigned long r4 asm("4") = (unsigned long) dest; +	register unsigned long r6 asm("6") = (unsigned long) counter; + +	asm volatile( +		"0:	.insn	rrf,%[opc] << 16,%[dst],%[src],%[ctr],0\n" +		"	brc	1,0b\n" /* handle partial completion */ +		: [src] "+a" (r2), [len] "+d" (r3), +		  [dst] "+a" (r4), [ctr] "+a" (r6) +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_KMCTR) +		: "cc", "memory"); + +	return src_len - r3; +} + +/** + * cpacf_ppno() - executes the PPNO (PERFORM PSEUDORANDOM NUMBER OPERATION) + *		  instruction + * @func: the function code passed to PPNO; see CPACF_PPNO_xxx defines + * @param: address of parameter block; see POP for details on each func + * @dest: address of destination memory area + * @dest_len: size of destination memory area in bytes + * @seed: address of seed data + * @seed_len: size of seed data in bytes + * + * Returns 0 for the query func, number of random bytes stored in + * dest buffer for generate function + */ +static inline int cpacf_ppno(long func, void *param, +			     u8 *dest, long dest_len, +			     const u8 *seed, long seed_len) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; +	register unsigned long r2 asm("2") = (unsigned long) dest; +	register unsigned long r3 asm("3") = (unsigned long) dest_len; +	register unsigned long r4 asm("4") = (unsigned long) seed; +	register unsigned long r5 asm("5") = (unsigned long) seed_len; + +	asm volatile ( +		"0:	.insn	rre,%[opc] << 16,%[dst],%[seed]\n" +		"	brc	1,0b\n"	  /* handle partial completion */ +		: [dst] "+a" (r2), [dlen] "+d" (r3) +		: [fc] "d" (r0), [pba] "a" (r1), +		  [seed] "a" (r4), [slen] "d" (r5), [opc] "i" (CPACF_PPNO) +		: "cc", "memory"); + +	return dest_len - r3; +} + +/** + * cpacf_pcc() - executes the PCC (PERFORM CRYPTOGRAPHIC COMPUTATION) + *		 instruction + * @func: the function code passed to PCC; see CPACF_KM_xxx defines + * @param: address of parameter block; see POP for details on each func + * + * Returns 0. + */ +static inline int cpacf_pcc(long func, void *param) +{ +	register unsigned long r0 asm("0") = (unsigned long) func; +	register unsigned long r1 asm("1") = (unsigned long) param; + +	asm volatile( +		"0:	.insn	rre,%[opc] << 16,0,0\n" /* PCC opcode */ +		"	brc	1,0b\n" /* handle partial completion */ +		: +		: [fc] "d" (r0), [pba] "a" (r1), [opc] "i" (CPACF_PCC) +		: "cc", "memory"); + +	return 0; +} + +#endif	/* _ASM_S390_CPACF_H */ diff --git a/arch/s390/include/asm/fpu/types.h b/arch/s390/include/asm/fpu/types.h index 14a8b0c14f87..fe937c9b6471 100644 --- a/arch/s390/include/asm/fpu/types.h +++ b/arch/s390/include/asm/fpu/types.h @@ -11,11 +11,13 @@  #include <asm/sigcontext.h>  struct fpu { -	__u32 fpc;			/* Floating-point control */ +	__u32 fpc;		/* Floating-point control */ +	void *regs;		/* Pointer to the current save area */  	union { -		void *regs; -		freg_t *fprs;		/* Floating-point register save area */ -		__vector128 *vxrs;	/* Vector register save area */ +		/* Floating-point register save area */ +		freg_t fprs[__NUM_FPRS]; +		/* Vector register save area */ +		__vector128 vxrs[__NUM_VXRS];  	};  }; diff --git a/arch/s390/include/asm/ftrace.h b/arch/s390/include/asm/ftrace.h index 836c56290499..64053d9ac3f2 100644 --- a/arch/s390/include/asm/ftrace.h +++ b/arch/s390/include/asm/ftrace.h @@ -12,7 +12,9 @@  #ifndef __ASSEMBLY__ -#define ftrace_return_address(n) __builtin_return_address(n) +unsigned long return_address(int depth); + +#define ftrace_return_address(n) return_address(n)  void _mcount(void);  void ftrace_caller(void); diff --git a/arch/s390/include/asm/kvm_host.h b/arch/s390/include/asm/kvm_host.h index 6da41fab70fb..37b9017c6a96 100644 --- a/arch/s390/include/asm/kvm_host.h +++ b/arch/s390/include/asm/kvm_host.h @@ -38,7 +38,7 @@   */  #define KVM_NR_IRQCHIPS 1  #define KVM_IRQCHIP_NUM_PINS 4096 -#define KVM_HALT_POLL_NS_DEFAULT 0 +#define KVM_HALT_POLL_NS_DEFAULT 80000  /* s390-specific vcpu->requests bit members */  #define KVM_REQ_ENABLE_IBS         8 @@ -247,6 +247,7 @@ struct kvm_vcpu_stat {  	u32 exit_instruction;  	u32 halt_successful_poll;  	u32 halt_attempted_poll; +	u32 halt_poll_invalid;  	u32 halt_wakeup;  	u32 instruction_lctl;  	u32 instruction_lctlg; @@ -544,10 +545,6 @@ struct kvm_vcpu_arch {  	struct kvm_s390_local_interrupt local_int;  	struct hrtimer    ckc_timer;  	struct kvm_s390_pgm_info pgm; -	union  { -		struct cpuid	cpu_id; -		u64		stidp_data; -	};  	struct gmap *gmap;  	struct kvm_guestdbg_info_arch guestdbg;  	unsigned long pfault_token; @@ -605,7 +602,7 @@ struct kvm_s390_cpu_model {  	__u64 fac_mask[S390_ARCH_FAC_LIST_SIZE_U64];  	/* facility list requested by guest (in dma page) */  	__u64 *fac_list; -	struct cpuid cpu_id; +	u64 cpuid;  	unsigned short ibc;  }; @@ -700,4 +697,6 @@ static inline void kvm_arch_flush_shadow_memslot(struct kvm *kvm,  static inline void kvm_arch_vcpu_blocking(struct kvm_vcpu *vcpu) {}  static inline void kvm_arch_vcpu_unblocking(struct kvm_vcpu *vcpu) {} +void kvm_arch_vcpu_block_finish(struct kvm_vcpu *vcpu); +  #endif diff --git a/arch/s390/include/asm/livepatch.h b/arch/s390/include/asm/livepatch.h index d5427c78b1b3..2c1213785892 100644 --- a/arch/s390/include/asm/livepatch.h +++ b/arch/s390/include/asm/livepatch.h @@ -24,13 +24,6 @@ static inline int klp_check_compiler_support(void)  	return 0;  } -static inline int klp_write_module_reloc(struct module *mod, unsigned long -		type, unsigned long loc, unsigned long value) -{ -	/* not supported yet */ -	return -ENOSYS; -} -  static inline void klp_arch_set_pc(struct pt_regs *regs, unsigned long ip)  {  	regs->psw.addr = ip; diff --git a/arch/s390/include/asm/mmu.h b/arch/s390/include/asm/mmu.h index d29ad9545b41..081b2ad99d73 100644 --- a/arch/s390/include/asm/mmu.h +++ b/arch/s390/include/asm/mmu.h @@ -11,7 +11,7 @@ typedef struct {  	spinlock_t list_lock;  	struct list_head pgtable_list;  	struct list_head gmap_list; -	unsigned long asce_bits; +	unsigned long asce;  	unsigned long asce_limit;  	unsigned long vdso_base;  	/* The mmu context allocates 4K page tables. */ diff --git a/arch/s390/include/asm/mmu_context.h b/arch/s390/include/asm/mmu_context.h index d321469eeda7..c837b79b455d 100644 --- a/arch/s390/include/asm/mmu_context.h +++ b/arch/s390/include/asm/mmu_context.h @@ -26,12 +26,28 @@ static inline int init_new_context(struct task_struct *tsk,  	mm->context.has_pgste = 0;  	mm->context.use_skey = 0;  #endif -	if (mm->context.asce_limit == 0) { +	switch (mm->context.asce_limit) { +	case 1UL << 42: +		/* +		 * forked 3-level task, fall through to set new asce with new +		 * mm->pgd +		 */ +	case 0:  		/* context created by exec, set asce limit to 4TB */ -		mm->context.asce_bits = _ASCE_TABLE_LENGTH | -			_ASCE_USER_BITS | _ASCE_TYPE_REGION3;  		mm->context.asce_limit = STACK_TOP_MAX; -	} else if (mm->context.asce_limit == (1UL << 31)) { +		mm->context.asce = __pa(mm->pgd) | _ASCE_TABLE_LENGTH | +				   _ASCE_USER_BITS | _ASCE_TYPE_REGION3; +		break; +	case 1UL << 53: +		/* forked 4-level task, set new asce with new mm->pgd */ +		mm->context.asce = __pa(mm->pgd) | _ASCE_TABLE_LENGTH | +				   _ASCE_USER_BITS | _ASCE_TYPE_REGION2; +		break; +	case 1UL << 31: +		/* forked 2-level compat task, set new asce with new mm->pgd */ +		mm->context.asce = __pa(mm->pgd) | _ASCE_TABLE_LENGTH | +				   _ASCE_USER_BITS | _ASCE_TYPE_SEGMENT; +		/* pgd_alloc() did not increase mm->nr_pmds */  		mm_inc_nr_pmds(mm);  	}  	crst_table_init((unsigned long *) mm->pgd, pgd_entry_type(mm)); @@ -42,7 +58,7 @@ static inline int init_new_context(struct task_struct *tsk,  static inline void set_user_asce(struct mm_struct *mm)  { -	S390_lowcore.user_asce = mm->context.asce_bits | __pa(mm->pgd); +	S390_lowcore.user_asce = mm->context.asce;  	if (current->thread.mm_segment.ar4)  		__ctl_load(S390_lowcore.user_asce, 7, 7);  	set_cpu_flag(CIF_ASCE); @@ -71,7 +87,7 @@ static inline void switch_mm(struct mm_struct *prev, struct mm_struct *next,  {  	int cpu = smp_processor_id(); -	S390_lowcore.user_asce = next->context.asce_bits | __pa(next->pgd); +	S390_lowcore.user_asce = next->context.asce;  	if (prev == next)  		return;  	if (MACHINE_HAS_TLB_LC) diff --git a/arch/s390/include/asm/pci.h b/arch/s390/include/asm/pci.h index b6bfa169a002..0da91c4d30fd 100644 --- a/arch/s390/include/asm/pci.h +++ b/arch/s390/include/asm/pci.h @@ -31,20 +31,42 @@ int pci_proc_domain(struct pci_bus *);  #define ZPCI_FC_BLOCKED			0x20  #define ZPCI_FC_DMA_ENABLED		0x10 +#define ZPCI_FMB_DMA_COUNTER_VALID	(1 << 23) + +struct zpci_fmb_fmt0 { +	u64 dma_rbytes; +	u64 dma_wbytes; +}; + +struct zpci_fmb_fmt1 { +	u64 rx_bytes; +	u64 rx_packets; +	u64 tx_bytes; +	u64 tx_packets; +}; + +struct zpci_fmb_fmt2 { +	u64 consumed_work_units; +	u64 max_work_units; +}; +  struct zpci_fmb { -	u32 format	:  8; -	u32 dma_valid	:  1; -	u32		: 23; +	u32 format	: 8; +	u32 fmt_ind	: 24;  	u32 samples;  	u64 last_update; -	/* hardware counters */ +	/* common counters */  	u64 ld_ops;  	u64 st_ops;  	u64 stb_ops;  	u64 rpcit_ops; -	u64 dma_rbytes; -	u64 dma_wbytes; -} __packed __aligned(64); +	/* format specific counters */ +	union { +		struct zpci_fmb_fmt0 fmt0; +		struct zpci_fmb_fmt1 fmt1; +		struct zpci_fmb_fmt2 fmt2; +	}; +} __packed __aligned(128);  enum zpci_state {  	ZPCI_FN_STATE_RESERVED, diff --git a/arch/s390/include/asm/pgalloc.h b/arch/s390/include/asm/pgalloc.h index 9b3d9b6099f2..da34cb6b1f3b 100644 --- a/arch/s390/include/asm/pgalloc.h +++ b/arch/s390/include/asm/pgalloc.h @@ -52,8 +52,8 @@ static inline unsigned long pgd_entry_type(struct mm_struct *mm)  	return _REGION2_ENTRY_EMPTY;  } -int crst_table_upgrade(struct mm_struct *, unsigned long limit); -void crst_table_downgrade(struct mm_struct *, unsigned long limit); +int crst_table_upgrade(struct mm_struct *); +void crst_table_downgrade(struct mm_struct *);  static inline pud_t *pud_alloc_one(struct mm_struct *mm, unsigned long address)  { diff --git a/arch/s390/include/asm/pgtable.h b/arch/s390/include/asm/pgtable.h index 2f66645587a2..18d2beb89340 100644 --- a/arch/s390/include/asm/pgtable.h +++ b/arch/s390/include/asm/pgtable.h @@ -1223,6 +1223,7 @@ static inline int pmd_trans_huge(pmd_t pmd)  	return pmd_val(pmd) & _SEGMENT_ENTRY_LARGE;  } +#define has_transparent_hugepage has_transparent_hugepage  static inline int has_transparent_hugepage(void)  {  	return MACHINE_HAS_HPAGE ? 1 : 0; diff --git a/arch/s390/include/asm/processor.h b/arch/s390/include/asm/processor.h index d6fd22ea270d..9d4d311d7e52 100644 --- a/arch/s390/include/asm/processor.h +++ b/arch/s390/include/asm/processor.h @@ -105,7 +105,6 @@ typedef struct {   * Thread structure   */  struct thread_struct { -	struct fpu fpu;			/* FP and VX register save area */  	unsigned int  acrs[NUM_ACRS];          unsigned long ksp;              /* kernel stack pointer             */  	mm_segment_t mm_segment; @@ -120,6 +119,11 @@ struct thread_struct {  	/* cpu runtime instrumentation */  	struct runtime_instr_cb *ri_cb;  	unsigned char trap_tdb[256];	/* Transaction abort diagnose block */ +	/* +	 * Warning: 'fpu' is dynamically-sized. It *MUST* be at +	 * the end. +	 */ +	struct fpu fpu;			/* FP and VX register save area */  };  /* Flag to disable transactions. */ @@ -155,10 +159,9 @@ struct stack_frame {  #define ARCH_MIN_TASKALIGN	8 -extern __vector128 init_task_fpu_regs[__NUM_VXRS];  #define INIT_THREAD {							\  	.ksp = sizeof(init_stack) + (unsigned long) &init_stack,	\ -	.fpu.regs = (void *)&init_task_fpu_regs,			\ +	.fpu.regs = (void *) init_task.thread.fpu.fprs,			\  }  /* @@ -175,7 +178,7 @@ extern __vector128 init_task_fpu_regs[__NUM_VXRS];  	regs->psw.mask	= PSW_USER_BITS | PSW_MASK_BA;			\  	regs->psw.addr	= new_psw;					\  	regs->gprs[15]	= new_stackp;					\ -	crst_table_downgrade(current->mm, 1UL << 31);			\ +	crst_table_downgrade(current->mm);				\  	execve_tail();							\  } while (0) diff --git a/arch/s390/include/asm/rwsem.h b/arch/s390/include/asm/rwsem.h index fead491dfc28..c75e4471e618 100644 --- a/arch/s390/include/asm/rwsem.h +++ b/arch/s390/include/asm/rwsem.h @@ -90,7 +90,7 @@ static inline int __down_read_trylock(struct rw_semaphore *sem)  /*   * lock for writing   */ -static inline void __down_write_nested(struct rw_semaphore *sem, int subclass) +static inline long ___down_write(struct rw_semaphore *sem)  {  	signed long old, new, tmp; @@ -104,13 +104,23 @@ static inline void __down_write_nested(struct rw_semaphore *sem, int subclass)  		: "=&d" (old), "=&d" (new), "=Q" (sem->count)  		: "Q" (sem->count), "m" (tmp)  		: "cc", "memory"); -	if (old != 0) -		rwsem_down_write_failed(sem); + +	return old;  }  static inline void __down_write(struct rw_semaphore *sem)  { -	__down_write_nested(sem, 0); +	if (___down_write(sem)) +		rwsem_down_write_failed(sem); +} + +static inline int __down_write_killable(struct rw_semaphore *sem) +{ +	if (___down_write(sem)) +		if (IS_ERR(rwsem_down_write_failed_killable(sem))) +			return -EINTR; + +	return 0;  }  /* diff --git a/arch/s390/include/asm/sclp.h b/arch/s390/include/asm/sclp.h index bab456be9a4f..e4f6f73afe2f 100644 --- a/arch/s390/include/asm/sclp.h +++ b/arch/s390/include/asm/sclp.h @@ -69,9 +69,22 @@ struct sclp_info {  	unsigned int max_cores;  	unsigned long hsa_size;  	unsigned long facilities; +	unsigned int hmfai;  };  extern struct sclp_info sclp; +struct zpci_report_error_header { +	u8 version;	/* Interface version byte */ +	u8 action;	/* Action qualifier byte +			 * 1: Deconfigure and repair action requested +			 *	(OpenCrypto Problem Call Home) +			 * 2: Informational Report +			 *	(OpenCrypto Successful Diagnostics Execution) +			 */ +	u16 length;	/* Length of Subsequent Data (up to 4K – SCLP header */ +	u8 data[0];	/* Subsequent Data passed verbatim to SCLP ET 24 */ +} __packed; +  int sclp_get_core_info(struct sclp_core_info *info);  int sclp_core_configure(u8 core);  int sclp_core_deconfigure(u8 core); @@ -83,6 +96,7 @@ int sclp_chp_read_info(struct sclp_chp_info *info);  void sclp_get_ipl_info(struct sclp_ipl_info *info);  int sclp_pci_configure(u32 fid);  int sclp_pci_deconfigure(u32 fid); +int sclp_pci_report(struct zpci_report_error_header *report, u32 fh, u32 fid);  int memcpy_hsa_kernel(void *dest, unsigned long src, size_t count);  int memcpy_hsa_user(void __user *dest, unsigned long src, size_t count);  void sclp_early_detect(void); diff --git a/arch/s390/include/asm/seccomp.h b/arch/s390/include/asm/seccomp.h index 781a9cf9b002..e10f8337367b 100644 --- a/arch/s390/include/asm/seccomp.h +++ b/arch/s390/include/asm/seccomp.h @@ -13,4 +13,6 @@  #define __NR_seccomp_exit_32	__NR_exit  #define __NR_seccomp_sigreturn_32 __NR_sigreturn +#include <asm-generic/seccomp.h> +  #endif	/* _ASM_S390_SECCOMP_H */ diff --git a/arch/s390/include/asm/sigp.h b/arch/s390/include/asm/sigp.h index ec60cf7fa0a2..1c8f33fca356 100644 --- a/arch/s390/include/asm/sigp.h +++ b/arch/s390/include/asm/sigp.h @@ -27,6 +27,7 @@  /* SIGP cpu status bits */ +#define SIGP_STATUS_INVALID_ORDER	0x00000002UL  #define SIGP_STATUS_CHECK_STOP		0x00000010UL  #define SIGP_STATUS_STOPPED		0x00000040UL  #define SIGP_STATUS_EXT_CALL_PENDING	0x00000080UL diff --git a/arch/s390/include/asm/thread_info.h b/arch/s390/include/asm/thread_info.h index 2fffc2c27581..f15c0398c363 100644 --- a/arch/s390/include/asm/thread_info.h +++ b/arch/s390/include/asm/thread_info.h @@ -62,6 +62,7 @@ static inline struct thread_info *current_thread_info(void)  }  void arch_release_task_struct(struct task_struct *tsk); +int arch_dup_task_struct(struct task_struct *dst, struct task_struct *src);  #define THREAD_SIZE_ORDER THREAD_ORDER diff --git a/arch/s390/include/asm/tlbflush.h b/arch/s390/include/asm/tlbflush.h index ca148f7c3eaa..a2e6ef32e054 100644 --- a/arch/s390/include/asm/tlbflush.h +++ b/arch/s390/include/asm/tlbflush.h @@ -110,8 +110,7 @@ static inline void __tlb_flush_asce(struct mm_struct *mm, unsigned long asce)  static inline void __tlb_flush_kernel(void)  {  	if (MACHINE_HAS_IDTE) -		__tlb_flush_idte((unsigned long) init_mm.pgd | -				 init_mm.context.asce_bits); +		__tlb_flush_idte(init_mm.context.asce);  	else  		__tlb_flush_global();  } @@ -133,8 +132,7 @@ static inline void __tlb_flush_asce(struct mm_struct *mm, unsigned long asce)  static inline void __tlb_flush_kernel(void)  {  	if (MACHINE_HAS_TLB_LC) -		__tlb_flush_idte_local((unsigned long) init_mm.pgd | -				       init_mm.context.asce_bits); +		__tlb_flush_idte_local(init_mm.context.asce);  	else  		__tlb_flush_local();  } @@ -148,8 +146,7 @@ static inline void __tlb_flush_mm(struct mm_struct * mm)  	 * only ran on the local cpu.  	 */  	if (MACHINE_HAS_IDTE && list_empty(&mm->context.gmap_list)) -		__tlb_flush_asce(mm, (unsigned long) mm->pgd | -				 mm->context.asce_bits); +		__tlb_flush_asce(mm, mm->context.asce);  	else  		__tlb_flush_full(mm);  } |