diff options
Diffstat (limited to 'arch/s390/kernel')
27 files changed, 189 insertions, 174 deletions
| diff --git a/arch/s390/kernel/Makefile b/arch/s390/kernel/Makefile index ea5ed6654050..eb06ff888314 100644 --- a/arch/s390/kernel/Makefile +++ b/arch/s390/kernel/Makefile @@ -41,7 +41,7 @@ obj-y	+= processor.o syscall.o ptrace.o signal.o cpcmd.o ebcdic.o nmi.o  obj-y	+= debug.o irq.o ipl.o dis.o vdso.o cpufeature.o  obj-y	+= sysinfo.o lgr.o os_info.o ctlreg.o  obj-y	+= runtime_instr.o cache.o fpu.o dumpstack.o guarded_storage.o sthyi.o -obj-y	+= entry.o reipl.o kdebugfs.o alternative.o +obj-y	+= entry.o reipl.o kdebugfs.o alternative.o skey.o  obj-y	+= nospec-branch.o ipl_vmparm.o machine_kexec_reloc.o unwind_bc.o  obj-y	+= smp.o text_amode31.o stacktrace.o abs_lowcore.o facility.o uv.o wti.o  obj-y	+= diag/ diff --git a/arch/s390/kernel/cpacf.c b/arch/s390/kernel/cpacf.c index 4b9b34f95d72..3bebc47beeab 100644 --- a/arch/s390/kernel/cpacf.c +++ b/arch/s390/kernel/cpacf.c @@ -101,7 +101,7 @@ static const struct bin_attribute *const cpacf_attrs[] = {  static const struct attribute_group cpacf_attr_grp = {  	.name = "cpacf", -	.bin_attrs_new = cpacf_attrs, +	.bin_attrs = cpacf_attrs,  };  static int __init cpacf_init(void) diff --git a/arch/s390/kernel/cpufeature.c b/arch/s390/kernel/cpufeature.c index 76210f001028..c9eef9ed876b 100644 --- a/arch/s390/kernel/cpufeature.c +++ b/arch/s390/kernel/cpufeature.c @@ -4,6 +4,7 @@   */  #include <linux/cpufeature.h> +#include <linux/export.h>  #include <linux/bug.h>  #include <asm/machine.h>  #include <asm/elf.h> diff --git a/arch/s390/kernel/crash_dump.c b/arch/s390/kernel/crash_dump.c index adb164223f8c..d4839de8ce9d 100644 --- a/arch/s390/kernel/crash_dump.c +++ b/arch/s390/kernel/crash_dump.c @@ -7,6 +7,7 @@   */  #include <linux/crash_dump.h> +#include <linux/export.h>  #include <asm/lowcore.h>  #include <linux/kernel.h>  #include <linux/init.h> diff --git a/arch/s390/kernel/ctlreg.c b/arch/s390/kernel/ctlreg.c index 8cc26cf2c64a..a0501f4c7e7a 100644 --- a/arch/s390/kernel/ctlreg.c +++ b/arch/s390/kernel/ctlreg.c @@ -5,6 +5,7 @@  #include <linux/irqflags.h>  #include <linux/spinlock.h> +#include <linux/export.h>  #include <linux/kernel.h>  #include <linux/init.h>  #include <linux/smp.h> diff --git a/arch/s390/kernel/dis.c b/arch/s390/kernel/dis.c index 94eb8168ea44..63a1d4226ff8 100644 --- a/arch/s390/kernel/dis.c +++ b/arch/s390/kernel/dis.c @@ -17,7 +17,6 @@  #include <linux/init.h>  #include <linux/interrupt.h>  #include <linux/delay.h> -#include <linux/export.h>  #include <linux/kallsyms.h>  #include <linux/reboot.h>  #include <linux/kprobes.h> diff --git a/arch/s390/kernel/early.c b/arch/s390/kernel/early.c index 54cf0923050f..9adfbdd377dc 100644 --- a/arch/s390/kernel/early.c +++ b/arch/s390/kernel/early.c @@ -105,6 +105,8 @@ static inline void strim_all(char *str)  	}  } +char arch_hw_string[128]; +  static noinline __init void setup_arch_string(void)  {  	struct sysinfo_1_1_1 *mach = (struct sysinfo_1_1_1 *)&sysinfo_page; @@ -131,6 +133,7 @@ static noinline __init void setup_arch_string(void)  			machine_is_vm() ? "z/VM" :  			machine_is_kvm() ? "KVM" : "unknown");  	} +	sprintf(arch_hw_string, "HW: %s (%s)", mstr, hvstr);  	dump_stack_set_arch_desc("%s (%s)", mstr, hvstr);  } @@ -154,6 +157,7 @@ void __init __do_early_pgm_check(struct pt_regs *regs)  	regs->int_code = lc->pgm_int_code;  	regs->int_parm_long = lc->trans_exc_code; +	regs->last_break = lc->pgm_last_break;  	ip = __rewind_psw(regs->psw, regs->int_code >> 16);  	/* Monitor Event? Might be a warning */ diff --git a/arch/s390/kernel/facility.c b/arch/s390/kernel/facility.c index f02127219a27..d028b0be5c1d 100644 --- a/arch/s390/kernel/facility.c +++ b/arch/s390/kernel/facility.c @@ -3,6 +3,7 @@   * Copyright IBM Corp. 2023   */ +#include <linux/export.h>  #include <asm/facility.h>  unsigned int stfle_size(void) diff --git a/arch/s390/kernel/fpu.c b/arch/s390/kernel/fpu.c index 6f2e87920288..03a8973aec3c 100644 --- a/arch/s390/kernel/fpu.c +++ b/arch/s390/kernel/fpu.c @@ -5,6 +5,8 @@   * Copyright IBM Corp. 2015   * Author(s): Hendrik Brueckner <brueckner@linux.vnet.ibm.com>   */ + +#include <linux/export.h>  #include <linux/kernel.h>  #include <linux/cpu.h>  #include <linux/sched.h> diff --git a/arch/s390/kernel/ipl.c b/arch/s390/kernel/ipl.c index ff15f91affde..961a3d60a4dd 100644 --- a/arch/s390/kernel/ipl.c +++ b/arch/s390/kernel/ipl.c @@ -596,7 +596,7 @@ static struct attribute *ipl_fcp_attrs[] = {  static const struct attribute_group ipl_fcp_attr_group = {  	.attrs = ipl_fcp_attrs, -	.bin_attrs_new = ipl_fcp_bin_attrs, +	.bin_attrs = ipl_fcp_bin_attrs,  };  static struct attribute *ipl_nvme_attrs[] = { @@ -610,7 +610,7 @@ static struct attribute *ipl_nvme_attrs[] = {  static const struct attribute_group ipl_nvme_attr_group = {  	.attrs = ipl_nvme_attrs, -	.bin_attrs_new = ipl_nvme_bin_attrs, +	.bin_attrs = ipl_nvme_bin_attrs,  };  static struct attribute *ipl_eckd_attrs[] = { @@ -623,7 +623,7 @@ static struct attribute *ipl_eckd_attrs[] = {  static const struct attribute_group ipl_eckd_attr_group = {  	.attrs = ipl_eckd_attrs, -	.bin_attrs_new = ipl_eckd_bin_attrs, +	.bin_attrs = ipl_eckd_bin_attrs,  };  /* CCW ipl device attributes */ @@ -920,7 +920,7 @@ static struct attribute *reipl_fcp_attrs[] = {  static const struct attribute_group reipl_fcp_attr_group = {  	.attrs = reipl_fcp_attrs, -	.bin_attrs_new = reipl_fcp_bin_attrs, +	.bin_attrs = reipl_fcp_bin_attrs,  };  static struct kobj_attribute sys_reipl_fcp_clear_attr = @@ -958,7 +958,7 @@ static struct attribute *reipl_nvme_attrs[] = {  static const struct attribute_group reipl_nvme_attr_group = {  	.attrs = reipl_nvme_attrs, -	.bin_attrs_new = reipl_nvme_bin_attrs +	.bin_attrs = reipl_nvme_bin_attrs  };  static ssize_t reipl_nvme_clear_show(struct kobject *kobj, @@ -1051,7 +1051,7 @@ static struct attribute *reipl_eckd_attrs[] = {  static const struct attribute_group reipl_eckd_attr_group = {  	.attrs = reipl_eckd_attrs, -	.bin_attrs_new = reipl_eckd_bin_attrs +	.bin_attrs = reipl_eckd_bin_attrs  };  static ssize_t reipl_eckd_clear_show(struct kobject *kobj, @@ -1596,7 +1596,7 @@ static const struct bin_attribute *const dump_fcp_bin_attrs[] = {  static const struct attribute_group dump_fcp_attr_group = {  	.name  = IPL_FCP_STR,  	.attrs = dump_fcp_attrs, -	.bin_attrs_new = dump_fcp_bin_attrs, +	.bin_attrs = dump_fcp_bin_attrs,  };  /* NVME dump device attributes */ @@ -1630,7 +1630,7 @@ static const struct bin_attribute *const dump_nvme_bin_attrs[] = {  static const struct attribute_group dump_nvme_attr_group = {  	.name  = IPL_NVME_STR,  	.attrs = dump_nvme_attrs, -	.bin_attrs_new = dump_nvme_bin_attrs, +	.bin_attrs = dump_nvme_bin_attrs,  };  /* ECKD dump device attributes */ @@ -1664,7 +1664,7 @@ static const struct bin_attribute *const dump_eckd_bin_attrs[] = {  static const struct attribute_group dump_eckd_attr_group = {  	.name  = IPL_ECKD_STR,  	.attrs = dump_eckd_attrs, -	.bin_attrs_new = dump_eckd_bin_attrs, +	.bin_attrs = dump_eckd_bin_attrs,  };  /* CCW dump device attributes */ diff --git a/arch/s390/kernel/nmi.c b/arch/s390/kernel/nmi.c index 3da371c144eb..11f33243a23f 100644 --- a/arch/s390/kernel/nmi.c +++ b/arch/s390/kernel/nmi.c @@ -9,6 +9,7 @@   */  #include <linux/kernel_stat.h> +#include <linux/utsname.h>  #include <linux/cpufeature.h>  #include <linux/init.h>  #include <linux/errno.h> @@ -21,7 +22,6 @@  #include <linux/module.h>  #include <linux/sched/signal.h>  #include <linux/kvm_host.h> -#include <linux/export.h>  #include <asm/lowcore.h>  #include <asm/ctlreg.h>  #include <asm/fpu.h> @@ -116,18 +116,82 @@ static __always_inline char *u64_to_hex(char *dest, u64 val)  	return dest;  } +static notrace void nmi_print_info(void) +{ +	struct lowcore *lc = get_lowcore(); +	char message[100]; +	char *ptr; +	int i; + +	ptr = nmi_puts(message, "Unrecoverable machine check, code: "); +	ptr = u64_to_hex(ptr, lc->mcck_interruption_code); +	ptr = nmi_puts(ptr, "\n"); +	sclp_emergency_printk(message); + +	ptr = nmi_puts(message, init_utsname()->release); +	ptr = nmi_puts(ptr, "\n"); +	sclp_emergency_printk(message); + +	ptr = nmi_puts(message, arch_hw_string); +	ptr = nmi_puts(ptr, "\n"); +	sclp_emergency_printk(message); + +	ptr = nmi_puts(message, "PSW: "); +	ptr = u64_to_hex(ptr, lc->mcck_old_psw.mask); +	ptr = nmi_puts(ptr, " "); +	ptr = u64_to_hex(ptr, lc->mcck_old_psw.addr); +	ptr = nmi_puts(ptr, " PFX: "); +	ptr = u64_to_hex(ptr, (u64)get_lowcore()); +	ptr = nmi_puts(ptr, "\n"); +	sclp_emergency_printk(message); + +	ptr = nmi_puts(message, "LBA: "); +	ptr = u64_to_hex(ptr, lc->last_break_save_area); +	ptr = nmi_puts(ptr, " EDC: "); +	ptr = u64_to_hex(ptr, lc->external_damage_code); +	ptr = nmi_puts(ptr, " FSA: "); +	ptr = u64_to_hex(ptr, lc->failing_storage_address); +	ptr = nmi_puts(ptr, "\n"); +	sclp_emergency_printk(message); + +	ptr = nmi_puts(message, "CRS:\n"); +	sclp_emergency_printk(message); +	ptr = message; +	for (i = 0; i < 16; i++) { +		ptr = u64_to_hex(ptr, lc->cregs_save_area[i].val); +		ptr = nmi_puts(ptr, " "); +		if ((i + 1) % 4 == 0) { +			ptr = nmi_puts(ptr, "\n"); +			sclp_emergency_printk(message); +			ptr = message; +		} +	} + +	ptr = nmi_puts(message, "GPRS:\n"); +	sclp_emergency_printk(message); +	ptr = message; +	for (i = 0; i < 16; i++) { +		ptr = u64_to_hex(ptr, lc->gpregs_save_area[i]); +		ptr = nmi_puts(ptr, " "); +		if ((i + 1) % 4 == 0) { +			ptr = nmi_puts(ptr, "\n"); +			sclp_emergency_printk(message); +			ptr = message; +		} +	} + +	ptr = nmi_puts(message, "System stopped\n"); +	sclp_emergency_printk(message); +} +  static notrace void s390_handle_damage(void)  {  	struct lowcore *lc = get_lowcore();  	union ctlreg0 cr0, cr0_new; -	char message[100];  	psw_t psw_save; -	char *ptr;  	smp_emergency_stop();  	diag_amode31_ops.diag308_reset(); -	ptr = nmi_puts(message, "System stopped due to unrecoverable machine check, code: 0x"); -	u64_to_hex(ptr, lc->mcck_interruption_code);  	/*  	 * Disable low address protection and make machine check new PSW a @@ -141,7 +205,7 @@ static notrace void s390_handle_damage(void)  	psw_bits(lc->mcck_new_psw).io = 0;  	psw_bits(lc->mcck_new_psw).ext = 0;  	psw_bits(lc->mcck_new_psw).wait = 1; -	sclp_emergency_printk(message); +	nmi_print_info();  	/*  	 * Restore machine check new PSW and control register 0 to original diff --git a/arch/s390/kernel/perf_cpum_cf.c b/arch/s390/kernel/perf_cpum_cf.c index 6a262e198e35..4d09954ebf49 100644 --- a/arch/s390/kernel/perf_cpum_cf.c +++ b/arch/s390/kernel/perf_cpum_cf.c @@ -14,7 +14,6 @@  #include <linux/percpu.h>  #include <linux/notifier.h>  #include <linux/init.h> -#include <linux/export.h>  #include <linux/miscdevice.h>  #include <linux/perf_event.h> diff --git a/arch/s390/kernel/perf_cpum_sf.c b/arch/s390/kernel/perf_cpum_sf.c index 91469401f2c9..f432869f8921 100644 --- a/arch/s390/kernel/perf_cpum_sf.c +++ b/arch/s390/kernel/perf_cpum_sf.c @@ -14,7 +14,6 @@  #include <linux/percpu.h>  #include <linux/pid.h>  #include <linux/notifier.h> -#include <linux/export.h>  #include <linux/slab.h>  #include <linux/mm.h>  #include <linux/moduleparam.h> diff --git a/arch/s390/kernel/perf_event.c b/arch/s390/kernel/perf_event.c index 2b9611c4718e..91b8716c883a 100644 --- a/arch/s390/kernel/perf_event.c +++ b/arch/s390/kernel/perf_event.c @@ -12,7 +12,6 @@  #include <linux/perf_event.h>  #include <linux/kvm_host.h>  #include <linux/percpu.h> -#include <linux/export.h>  #include <linux/seq_file.h>  #include <linux/spinlock.h>  #include <linux/uaccess.h> diff --git a/arch/s390/kernel/perf_pai_crypto.c b/arch/s390/kernel/perf_pai_crypto.c index 63875270941b..f373a1009c45 100644 --- a/arch/s390/kernel/perf_pai_crypto.c +++ b/arch/s390/kernel/perf_pai_crypto.c @@ -13,7 +13,6 @@  #include <linux/percpu.h>  #include <linux/notifier.h>  #include <linux/init.h> -#include <linux/export.h>  #include <linux/io.h>  #include <linux/perf_event.h>  #include <asm/ctlreg.h> @@ -696,7 +695,7 @@ static const char * const paicrypt_ctrnames[] = {  	[111] = "PCC_COMPUTE_LAST_BLOCK_CMAC_USING_AES_256",  	[112] = "PCC_COMPUTE_LAST_BLOCK_CMAC_USING_ENCRYPTED_AES_128",  	[113] = "PCC_COMPUTE_LAST_BLOCK_CMAC_USING_ENCRYPTED_AES_192", -	[114] = "PCC_COMPUTE_LAST_BLOCK_CMAC_USING_ENCRYPTED_AES_256A", +	[114] = "PCC_COMPUTE_LAST_BLOCK_CMAC_USING_ENCRYPTED_AES_256",  	[115] = "PCC_COMPUTE_XTS_PARAMETER_USING_AES_128",  	[116] = "PCC_COMPUTE_XTS_PARAMETER_USING_AES_256",  	[117] = "PCC_COMPUTE_XTS_PARAMETER_USING_ENCRYPTED_AES_128", diff --git a/arch/s390/kernel/perf_pai_ext.c b/arch/s390/kernel/perf_pai_ext.c index fd14d5ebccbc..d827473e7f87 100644 --- a/arch/s390/kernel/perf_pai_ext.c +++ b/arch/s390/kernel/perf_pai_ext.c @@ -14,7 +14,6 @@  #include <linux/percpu.h>  #include <linux/notifier.h>  #include <linux/init.h> -#include <linux/export.h>  #include <linux/io.h>  #include <linux/perf_event.h>  #include <asm/ctlreg.h> diff --git a/arch/s390/kernel/process.c b/arch/s390/kernel/process.c index 9637aee43c40..f55f09cda6f8 100644 --- a/arch/s390/kernel/process.c +++ b/arch/s390/kernel/process.c @@ -27,7 +27,6 @@  #include <linux/compat.h>  #include <linux/kprobes.h>  #include <linux/random.h> -#include <linux/export.h>  #include <linux/init_task.h>  #include <linux/entry-common.h>  #include <linux/io.h> diff --git a/arch/s390/kernel/ptrace.c b/arch/s390/kernel/ptrace.c index e1240f6b29fa..494216c4b4f3 100644 --- a/arch/s390/kernel/ptrace.c +++ b/arch/s390/kernel/ptrace.c @@ -1209,7 +1209,7 @@ static int s390_runtime_instr_set(struct task_struct *target,  static const struct user_regset s390_regsets[] = {  	{ -		.core_note_type = NT_PRSTATUS, +		USER_REGSET_NOTE_TYPE(PRSTATUS),  		.n = sizeof(s390_regs) / sizeof(long),  		.size = sizeof(long),  		.align = sizeof(long), @@ -1217,7 +1217,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_regs_set,  	},  	{ -		.core_note_type = NT_PRFPREG, +		USER_REGSET_NOTE_TYPE(PRFPREG),  		.n = sizeof(s390_fp_regs) / sizeof(long),  		.size = sizeof(long),  		.align = sizeof(long), @@ -1225,7 +1225,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_fpregs_set,  	},  	{ -		.core_note_type = NT_S390_SYSTEM_CALL, +		USER_REGSET_NOTE_TYPE(S390_SYSTEM_CALL),  		.n = 1,  		.size = sizeof(unsigned int),  		.align = sizeof(unsigned int), @@ -1233,7 +1233,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_system_call_set,  	},  	{ -		.core_note_type = NT_S390_LAST_BREAK, +		USER_REGSET_NOTE_TYPE(S390_LAST_BREAK),  		.n = 1,  		.size = sizeof(long),  		.align = sizeof(long), @@ -1241,7 +1241,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_last_break_set,  	},  	{ -		.core_note_type = NT_S390_TDB, +		USER_REGSET_NOTE_TYPE(S390_TDB),  		.n = 1,  		.size = 256,  		.align = 1, @@ -1249,7 +1249,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_tdb_set,  	},  	{ -		.core_note_type = NT_S390_VXRS_LOW, +		USER_REGSET_NOTE_TYPE(S390_VXRS_LOW),  		.n = __NUM_VXRS_LOW,  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1257,7 +1257,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_vxrs_low_set,  	},  	{ -		.core_note_type = NT_S390_VXRS_HIGH, +		USER_REGSET_NOTE_TYPE(S390_VXRS_HIGH),  		.n = __NUM_VXRS_HIGH,  		.size = sizeof(__vector128),  		.align = sizeof(__vector128), @@ -1265,7 +1265,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_vxrs_high_set,  	},  	{ -		.core_note_type = NT_S390_GS_CB, +		USER_REGSET_NOTE_TYPE(S390_GS_CB),  		.n = sizeof(struct gs_cb) / sizeof(__u64),  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1273,7 +1273,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_gs_cb_set,  	},  	{ -		.core_note_type = NT_S390_GS_BC, +		USER_REGSET_NOTE_TYPE(S390_GS_BC),  		.n = sizeof(struct gs_cb) / sizeof(__u64),  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1281,7 +1281,7 @@ static const struct user_regset s390_regsets[] = {  		.set = s390_gs_bc_set,  	},  	{ -		.core_note_type = NT_S390_RI_CB, +		USER_REGSET_NOTE_TYPE(S390_RI_CB),  		.n = sizeof(struct runtime_instr_cb) / sizeof(__u64),  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1413,7 +1413,7 @@ static int s390_compat_last_break_set(struct task_struct *target,  static const struct user_regset s390_compat_regsets[] = {  	{ -		.core_note_type = NT_PRSTATUS, +		USER_REGSET_NOTE_TYPE(PRSTATUS),  		.n = sizeof(s390_compat_regs) / sizeof(compat_long_t),  		.size = sizeof(compat_long_t),  		.align = sizeof(compat_long_t), @@ -1421,7 +1421,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_compat_regs_set,  	},  	{ -		.core_note_type = NT_PRFPREG, +		USER_REGSET_NOTE_TYPE(PRFPREG),  		.n = sizeof(s390_fp_regs) / sizeof(compat_long_t),  		.size = sizeof(compat_long_t),  		.align = sizeof(compat_long_t), @@ -1429,7 +1429,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_fpregs_set,  	},  	{ -		.core_note_type = NT_S390_SYSTEM_CALL, +		USER_REGSET_NOTE_TYPE(S390_SYSTEM_CALL),  		.n = 1,  		.size = sizeof(compat_uint_t),  		.align = sizeof(compat_uint_t), @@ -1437,7 +1437,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_system_call_set,  	},  	{ -		.core_note_type = NT_S390_LAST_BREAK, +		USER_REGSET_NOTE_TYPE(S390_LAST_BREAK),  		.n = 1,  		.size = sizeof(long),  		.align = sizeof(long), @@ -1445,7 +1445,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_compat_last_break_set,  	},  	{ -		.core_note_type = NT_S390_TDB, +		USER_REGSET_NOTE_TYPE(S390_TDB),  		.n = 1,  		.size = 256,  		.align = 1, @@ -1453,7 +1453,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_tdb_set,  	},  	{ -		.core_note_type = NT_S390_VXRS_LOW, +		USER_REGSET_NOTE_TYPE(S390_VXRS_LOW),  		.n = __NUM_VXRS_LOW,  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1461,7 +1461,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_vxrs_low_set,  	},  	{ -		.core_note_type = NT_S390_VXRS_HIGH, +		USER_REGSET_NOTE_TYPE(S390_VXRS_HIGH),  		.n = __NUM_VXRS_HIGH,  		.size = sizeof(__vector128),  		.align = sizeof(__vector128), @@ -1469,7 +1469,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_vxrs_high_set,  	},  	{ -		.core_note_type = NT_S390_HIGH_GPRS, +		USER_REGSET_NOTE_TYPE(S390_HIGH_GPRS),  		.n = sizeof(s390_compat_regs_high) / sizeof(compat_long_t),  		.size = sizeof(compat_long_t),  		.align = sizeof(compat_long_t), @@ -1477,7 +1477,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_compat_regs_high_set,  	},  	{ -		.core_note_type = NT_S390_GS_CB, +		USER_REGSET_NOTE_TYPE(S390_GS_CB),  		.n = sizeof(struct gs_cb) / sizeof(__u64),  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1485,7 +1485,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_gs_cb_set,  	},  	{ -		.core_note_type = NT_S390_GS_BC, +		USER_REGSET_NOTE_TYPE(S390_GS_BC),  		.n = sizeof(struct gs_cb) / sizeof(__u64),  		.size = sizeof(__u64),  		.align = sizeof(__u64), @@ -1493,7 +1493,7 @@ static const struct user_regset s390_compat_regsets[] = {  		.set = s390_gs_bc_set,  	},  	{ -		.core_note_type = NT_S390_RI_CB, +		USER_REGSET_NOTE_TYPE(S390_RI_CB),  		.n = sizeof(struct runtime_instr_cb) / sizeof(__u64),  		.size = sizeof(__u64),  		.align = sizeof(__u64), diff --git a/arch/s390/kernel/skey.c b/arch/s390/kernel/skey.c new file mode 100644 index 000000000000..ba049fd103c2 --- /dev/null +++ b/arch/s390/kernel/skey.c @@ -0,0 +1,48 @@ +// SPDX-License-Identifier: GPL-2.0 + +#include <asm/rwonce.h> +#include <asm/page.h> +#include <asm/skey.h> + +int skey_regions_initialized; + +static inline unsigned long load_real_address(unsigned long address) +{ +	unsigned long real; + +	asm volatile( +		"	lra	%[real],0(%[address])\n" +		: [real] "=d" (real) +		: [address] "a" (address) +		: "cc"); +	return real; +} + +/* + * Initialize storage keys of registered memory regions with the + * default key. This is useful for code which is executed with a + * non-default access key. + */ +void __skey_regions_initialize(void) +{ +	unsigned long address, real; +	struct skey_region *r, *end; + +	r = __skey_region_start; +	end = __skey_region_end; +	while (r < end) { +		address = r->start & PAGE_MASK; +		do { +			real = load_real_address(address); +			page_set_storage_key(real, PAGE_DEFAULT_KEY, 1); +			address += PAGE_SIZE; +		} while (address < r->end); +		r++; +	} +	/* +	 * Make sure storage keys are initialized before +	 * skey_regions_initialized is changed. +	 */ +	barrier(); +	WRITE_ONCE(skey_regions_initialized, 1); +} diff --git a/arch/s390/kernel/smp.c b/arch/s390/kernel/smp.c index 81f12bb77f62..e88ebe5339fc 100644 --- a/arch/s390/kernel/smp.c +++ b/arch/s390/kernel/smp.c @@ -175,13 +175,10 @@ static struct pcpu *pcpu_find_address(const struct cpumask *mask, u16 address)  static void pcpu_ec_call(struct pcpu *pcpu, int ec_bit)  { -	int order; -  	if (test_and_set_bit(ec_bit, &pcpu->ec_mask))  		return; -	order = pcpu_running(pcpu) ? SIGP_EXTERNAL_CALL : SIGP_EMERGENCY_SIGNAL;  	pcpu->ec_clk = get_tod_clock_fast(); -	pcpu_sigp_retry(pcpu, order, 0); +	pcpu_sigp_retry(pcpu, SIGP_EXTERNAL_CALL, 0);  }  static int pcpu_alloc_lowcore(struct pcpu *pcpu, int cpu) @@ -433,16 +430,16 @@ void notrace smp_emergency_stop(void)  	cpumask_copy(&cpumask, cpu_online_mask);  	cpumask_clear_cpu(smp_processor_id(), &cpumask); -	end = get_tod_clock() + (1000000UL << 12); +	end = get_tod_clock_monotonic() + (1000000UL << 12);  	for_each_cpu(cpu, &cpumask) {  		struct pcpu *pcpu = per_cpu_ptr(&pcpu_devices, cpu);  		set_bit(ec_stop_cpu, &pcpu->ec_mask);  		while (__pcpu_sigp(pcpu->address, SIGP_EMERGENCY_SIGNAL,  				   0, NULL) == SIGP_CC_BUSY && -		       get_tod_clock() < end) +		       get_tod_clock_monotonic() < end)  			cpu_relax();  	} -	while (get_tod_clock() < end) { +	while (get_tod_clock_monotonic() < end) {  		for_each_cpu(cpu, &cpumask)  			if (pcpu_stopped(per_cpu_ptr(&pcpu_devices, cpu)))  				cpumask_clear_cpu(cpu, &cpumask); diff --git a/arch/s390/kernel/sthyi.c b/arch/s390/kernel/sthyi.c index d40f0b983e74..f4ccdbed4b89 100644 --- a/arch/s390/kernel/sthyi.c +++ b/arch/s390/kernel/sthyi.c @@ -5,6 +5,8 @@   * Copyright IBM Corp. 2016   * Author(s): Janosch Frank <frankja@linux.vnet.ibm.com>   */ + +#include <linux/export.h>  #include <linux/errno.h>  #include <linux/pagemap.h>  #include <linux/vmalloc.h> diff --git a/arch/s390/kernel/syscalls/syscall.tbl b/arch/s390/kernel/syscalls/syscall.tbl index a4569b96ef06..8a6744d658db 100644 --- a/arch/s390/kernel/syscalls/syscall.tbl +++ b/arch/s390/kernel/syscalls/syscall.tbl @@ -470,3 +470,5 @@  465  common	listxattrat		sys_listxattrat			sys_listxattrat  466  common	removexattrat		sys_removexattrat		sys_removexattrat  467  common	open_tree_attr		sys_open_tree_attr		sys_open_tree_attr +468  common	file_getattr		sys_file_getattr		sys_file_getattr +469  common	file_setattr		sys_file_setattr		sys_file_setattr diff --git a/arch/s390/kernel/time.c b/arch/s390/kernel/time.c index fed17d407a44..63517b85f4c9 100644 --- a/arch/s390/kernel/time.c +++ b/arch/s390/kernel/time.c @@ -69,8 +69,6 @@ unsigned char ptff_function_mask[16];  static unsigned long lpar_offset;  static unsigned long initial_leap_seconds; -static unsigned long tod_steering_end; -static long tod_steering_delta;  /*   * Get time offsets with PTFF @@ -80,9 +78,7 @@ void __init time_early_init(void)  	struct ptff_qto qto;  	struct ptff_qui qui; -	/* Initialize TOD steering parameters */ -	tod_steering_end = tod_clock_base.tod; -	vdso_k_time_data->arch_data.tod_steering_end = tod_steering_end; +	vdso_k_time_data->arch_data.tod_delta = tod_clock_base.tod;  	if (!test_facility(28))  		return; @@ -226,21 +222,7 @@ void __init read_persistent_wall_and_boot_offset(struct timespec64 *wall_time,  static u64 read_tod_clock(struct clocksource *cs)  { -	unsigned long now, adj; - -	preempt_disable(); /* protect from changes to steering parameters */ -	now = get_tod_clock(); -	adj = tod_steering_end - now; -	if (unlikely((s64) adj > 0)) -		/* -		 * manually steer by 1 cycle every 2^16 cycles. This -		 * corresponds to shifting the tod delta by 15. 1s is -		 * therefore steered in ~9h. The adjust will decrease -		 * over time, until it finally reaches 0. -		 */ -		now += (tod_steering_delta < 0) ? (adj >> 15) : -(adj >> 15); -	preempt_enable(); -	return now; +	return get_tod_clock_monotonic();  }  static struct clocksource clocksource_tod = { @@ -369,26 +351,11 @@ static inline int check_sync_clock(void)   */  static void clock_sync_global(long delta)  { -	unsigned long now, adj;  	struct ptff_qto qto;  	/* Fixup the monotonic sched clock. */  	tod_clock_base.eitod += delta; -	/* Adjust TOD steering parameters. */ -	now = get_tod_clock(); -	adj = tod_steering_end - now; -	if (unlikely((s64) adj >= 0)) -		/* Calculate how much of the old adjustment is left. */ -		tod_steering_delta = (tod_steering_delta < 0) ? -			-(adj >> 15) : (adj >> 15); -	tod_steering_delta += delta; -	if ((abs(tod_steering_delta) >> 48) != 0) -		panic("TOD clock sync offset %li is too large to drift\n", -		      tod_steering_delta); -	tod_steering_end = now + (abs(tod_steering_delta) << 15); -	vdso_k_time_data->arch_data.tod_steering_end = tod_steering_end; -	vdso_k_time_data->arch_data.tod_steering_delta = tod_steering_delta; - +	vdso_k_time_data->arch_data.tod_delta = tod_clock_base.tod;  	/* Update LPAR offset. */  	if (ptff_query(PTFF_QTO) && ptff(&qto, sizeof(qto), PTFF_QTO) == 0)  		lpar_offset = qto.tod_epoch_difference; @@ -430,7 +397,7 @@ struct clock_sync_data {  /*   * Server Time Protocol (STP) code.   */ -static bool stp_online; +static bool stp_online = true;  static struct stp_sstpi stp_info;  static void *stp_page; @@ -456,7 +423,6 @@ static void __init stp_reset(void)  	if (rc == 0)  		set_bit(CLOCK_SYNC_HAS_STP, &clock_sync_flags);  	else if (stp_online) { -		pr_warn("The real or virtual hardware system does not provide an STP interface\n");  		free_page((unsigned long) stp_page);  		stp_page = NULL;  		stp_online = false; @@ -580,7 +546,7 @@ static int stp_sync_clock(void *data)  		atomic_dec(&sync->cpus);  		/* Wait for in_sync to be set. */  		while (READ_ONCE(sync->in_sync) == 0) -			__udelay(1); +			;  	}  	if (sync->in_sync != 1)  		/* Didn't work. Clear per-cpu in sync bit again. */ @@ -591,81 +557,6 @@ static int stp_sync_clock(void *data)  	return 0;  } -static int stp_clear_leap(void) -{ -	struct __kernel_timex txc; -	int ret; - -	memset(&txc, 0, sizeof(txc)); - -	ret = do_adjtimex(&txc); -	if (ret < 0) -		return ret; - -	txc.modes = ADJ_STATUS; -	txc.status &= ~(STA_INS|STA_DEL); -	return do_adjtimex(&txc); -} - -static void stp_check_leap(void) -{ -	struct stp_stzi stzi; -	struct stp_lsoib *lsoib = &stzi.lsoib; -	struct __kernel_timex txc; -	int64_t timediff; -	int leapdiff, ret; - -	if (!stp_info.lu || !check_sync_clock()) { -		/* -		 * Either a scheduled leap second was removed by the operator, -		 * or STP is out of sync. In both cases, clear the leap second -		 * kernel flags. -		 */ -		if (stp_clear_leap() < 0) -			pr_err("failed to clear leap second flags\n"); -		return; -	} - -	if (chsc_stzi(stp_page, &stzi, sizeof(stzi))) { -		pr_err("stzi failed\n"); -		return; -	} - -	timediff = tod_to_ns(lsoib->nlsout - get_tod_clock()) / NSEC_PER_SEC; -	leapdiff = lsoib->nlso - lsoib->also; - -	if (leapdiff != 1 && leapdiff != -1) { -		pr_err("Cannot schedule %d leap seconds\n", leapdiff); -		return; -	} - -	if (timediff < 0) { -		if (stp_clear_leap() < 0) -			pr_err("failed to clear leap second flags\n"); -	} else if (timediff < 7200) { -		memset(&txc, 0, sizeof(txc)); -		ret = do_adjtimex(&txc); -		if (ret < 0) -			return; - -		txc.modes = ADJ_STATUS; -		if (leapdiff > 0) -			txc.status |= STA_INS; -		else -			txc.status |= STA_DEL; -		ret = do_adjtimex(&txc); -		if (ret < 0) -			pr_err("failed to set leap second flags\n"); -		/* arm Timer to clear leap second flags */ -		mod_timer(&stp_timer, jiffies + secs_to_jiffies(14400)); -	} else { -		/* The day the leap second is scheduled for hasn't been reached. Retry -		 * in one hour. -		 */ -		mod_timer(&stp_timer, jiffies + secs_to_jiffies(3600)); -	} -} -  /*   * STP work. Check for the STP state and take over the clock   * synchronization if the STP clock source is usable. @@ -707,8 +598,6 @@ static void stp_work_fn(struct work_struct *work)  		 * Retry after a second.  		 */  		mod_timer(&stp_timer, jiffies + msecs_to_jiffies(MSEC_PER_SEC)); -	else if (stp_info.lu) -		stp_check_leap();  out_unlock:  	mutex_unlock(&stp_mutex); diff --git a/arch/s390/kernel/topology.c b/arch/s390/kernel/topology.c index 3df048e190b1..46569b8e47dd 100644 --- a/arch/s390/kernel/topology.c +++ b/arch/s390/kernel/topology.c @@ -531,11 +531,11 @@ static const struct cpumask *cpu_drawer_mask(int cpu)  }  static struct sched_domain_topology_level s390_topology[] = { -	{ cpu_thread_mask, cpu_smt_flags, SD_INIT_NAME(SMT) }, -	{ cpu_coregroup_mask, cpu_core_flags, SD_INIT_NAME(MC) }, -	{ cpu_book_mask, SD_INIT_NAME(BOOK) }, -	{ cpu_drawer_mask, SD_INIT_NAME(DRAWER) }, -	{ cpu_cpu_mask, SD_INIT_NAME(PKG) }, +	SDTL_INIT(cpu_thread_mask, cpu_smt_flags, SMT), +	SDTL_INIT(cpu_coregroup_mask, cpu_core_flags, MC), +	SDTL_INIT(cpu_book_mask, NULL, BOOK), +	SDTL_INIT(cpu_drawer_mask, NULL, DRAWER), +	SDTL_INIT(cpu_cpu_mask, NULL, PKG),  	{ NULL, },  }; diff --git a/arch/s390/kernel/unwind_bc.c b/arch/s390/kernel/unwind_bc.c index cd44be2b6ce8..0f88caca4eaf 100644 --- a/arch/s390/kernel/unwind_bc.c +++ b/arch/s390/kernel/unwind_bc.c @@ -1,4 +1,6 @@  /* SPDX-License-Identifier: GPL-2.0 */ + +#include <linux/export.h>  #include <linux/sched.h>  #include <linux/sched/task.h>  #include <linux/sched/task_stack.h> diff --git a/arch/s390/kernel/uv.c b/arch/s390/kernel/uv.c index b99478e84da4..47f574cd1728 100644 --- a/arch/s390/kernel/uv.c +++ b/arch/s390/kernel/uv.c @@ -7,6 +7,7 @@  #define KMSG_COMPONENT "prot_virt"  #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt +#include <linux/export.h>  #include <linux/kernel.h>  #include <linux/types.h>  #include <linux/sizes.h> diff --git a/arch/s390/kernel/vmlinux.lds.S b/arch/s390/kernel/vmlinux.lds.S index ff1ddba96352..1c606dfa595d 100644 --- a/arch/s390/kernel/vmlinux.lds.S +++ b/arch/s390/kernel/vmlinux.lds.S @@ -71,6 +71,13 @@ SECTIONS  	. = ALIGN(PAGE_SIZE);  	__end_ro_after_init = .; +	. = ALIGN(8); +	.skey_region_table : { +		__skey_region_start = .; +		KEEP(*(.skey_region)) +		__skey_region_end = .; +	} +  	.data.rel.ro : {  		*(.data.rel.ro .data.rel.ro.*)  	} | 
