diff options
Diffstat (limited to 'arch/mips/kernel')
38 files changed, 230 insertions, 204 deletions
| diff --git a/arch/mips/kernel/asm-offsets.c b/arch/mips/kernel/asm-offsets.c index 0845091ba480..0c2e853c3db4 100644 --- a/arch/mips/kernel/asm-offsets.c +++ b/arch/mips/kernel/asm-offsets.c @@ -82,6 +82,9 @@ void output_task_defines(void)  	OFFSET(TASK_FLAGS, task_struct, flags);  	OFFSET(TASK_MM, task_struct, mm);  	OFFSET(TASK_PID, task_struct, pid); +#if defined(CONFIG_CC_STACKPROTECTOR) +	OFFSET(TASK_STACK_CANARY, task_struct, stack_canary); +#endif  	DEFINE(TASK_STRUCT_SIZE, sizeof(struct task_struct));  	BLANK();  } diff --git a/arch/mips/kernel/bmips_vec.S b/arch/mips/kernel/bmips_vec.S index 64c4fd62cf08..f739aedcb509 100644 --- a/arch/mips/kernel/bmips_vec.S +++ b/arch/mips/kernel/bmips_vec.S @@ -28,8 +28,6 @@  	.set	mips0  	.endm -	__CPUINIT -  /***********************************************************************   * Alternate CPU1 startup vector for BMIPS4350   * @@ -216,8 +214,6 @@ END(bmips_smp_int_vec)   * Certain CPUs support extending kseg0 to 1024MB.   ***********************************************************************/ -	__CPUINIT -  LEAF(bmips_enable_xks01)  #if defined(CONFIG_XKS01) diff --git a/arch/mips/kernel/branch.c b/arch/mips/kernel/branch.c index 46c2ad0703a0..4d78bf445a9c 100644 --- a/arch/mips/kernel/branch.c +++ b/arch/mips/kernel/branch.c @@ -467,5 +467,4 @@ unaligned:  	printk("%s: unaligned epc - sending SIGBUS.\n", current->comm);  	force_sig(SIGBUS, current);  	return -EFAULT; -  } diff --git a/arch/mips/kernel/cevt-bcm1480.c b/arch/mips/kernel/cevt-bcm1480.c index 15f618b40cf6..7976457184b1 100644 --- a/arch/mips/kernel/cevt-bcm1480.c +++ b/arch/mips/kernel/cevt-bcm1480.c @@ -109,7 +109,7 @@ static DEFINE_PER_CPU(struct clock_event_device, sibyte_hpt_clockevent);  static DEFINE_PER_CPU(struct irqaction, sibyte_hpt_irqaction);  static DEFINE_PER_CPU(char [18], sibyte_hpt_name); -void __cpuinit sb1480_clockevent_init(void) +void sb1480_clockevent_init(void)  {  	unsigned int cpu = smp_processor_id();  	unsigned int irq = K_BCM1480_INT_TIMER_0 + cpu; diff --git a/arch/mips/kernel/cevt-gic.c b/arch/mips/kernel/cevt-gic.c index 730eaf92c018..594cbbf16d62 100644 --- a/arch/mips/kernel/cevt-gic.c +++ b/arch/mips/kernel/cevt-gic.c @@ -59,7 +59,7 @@ void gic_event_handler(struct clock_event_device *dev)  {  } -int __cpuinit gic_clockevent_init(void) +int gic_clockevent_init(void)  {  	unsigned int cpu = smp_processor_id();  	struct clock_event_device *cd; diff --git a/arch/mips/kernel/cevt-r4k.c b/arch/mips/kernel/cevt-r4k.c index 02033eaf8825..50d3f5a8d6bb 100644 --- a/arch/mips/kernel/cevt-r4k.c +++ b/arch/mips/kernel/cevt-r4k.c @@ -171,7 +171,7 @@ int c0_compare_int_usable(void)  }  #ifndef CONFIG_MIPS_MT_SMTC -int __cpuinit r4k_clockevent_init(void) +int r4k_clockevent_init(void)  {  	unsigned int cpu = smp_processor_id();  	struct clock_event_device *cd; diff --git a/arch/mips/kernel/cevt-sb1250.c b/arch/mips/kernel/cevt-sb1250.c index 200f2778bf36..5ea6d6b1de15 100644 --- a/arch/mips/kernel/cevt-sb1250.c +++ b/arch/mips/kernel/cevt-sb1250.c @@ -107,7 +107,7 @@ static DEFINE_PER_CPU(struct clock_event_device, sibyte_hpt_clockevent);  static DEFINE_PER_CPU(struct irqaction, sibyte_hpt_irqaction);  static DEFINE_PER_CPU(char [18], sibyte_hpt_name); -void __cpuinit sb1250_clockevent_init(void) +void sb1250_clockevent_init(void)  {  	unsigned int cpu = smp_processor_id();  	unsigned int irq = K_INT_TIMER_0 + cpu; diff --git a/arch/mips/kernel/cevt-smtc.c b/arch/mips/kernel/cevt-smtc.c index 9de5ed7ef1a3..b6cf0a60d896 100644 --- a/arch/mips/kernel/cevt-smtc.c +++ b/arch/mips/kernel/cevt-smtc.c @@ -248,7 +248,7 @@ irqreturn_t c0_compare_interrupt(int irq, void *dev_id)  } -int __cpuinit smtc_clockevent_init(void) +int smtc_clockevent_init(void)  {  	uint64_t mips_freq = mips_hpt_frequency;  	unsigned int cpu = smp_processor_id(); diff --git a/arch/mips/kernel/cpu-bugs64.c b/arch/mips/kernel/cpu-bugs64.c index de3c25ffd9f9..2d80b5f1aeae 100644 --- a/arch/mips/kernel/cpu-bugs64.c +++ b/arch/mips/kernel/cpu-bugs64.c @@ -6,6 +6,7 @@   * as published by the Free Software Foundation; either version   * 2 of the License, or (at your option) any later version.   */ +#include <linux/context_tracking.h>  #include <linux/init.h>  #include <linux/kernel.h>  #include <linux/ptrace.h> @@ -167,12 +168,16 @@ static inline void check_mult_sh(void)  	panic(bug64hit, !R4000_WAR ? r4kwar : nowar);  } -static volatile int daddi_ov __cpuinitdata; +static volatile int daddi_ov;  asmlinkage void __init do_daddi_ov(struct pt_regs *regs)  { +	enum ctx_state prev_state; + +	prev_state = exception_enter();  	daddi_ov = 1;  	regs->cp0_epc += 4; +	exception_exit(prev_state);  }  static inline void check_daddi(void) diff --git a/arch/mips/kernel/cpu-probe.c b/arch/mips/kernel/cpu-probe.c index c6568bf4b1b0..4c6167a17875 100644 --- a/arch/mips/kernel/cpu-probe.c +++ b/arch/mips/kernel/cpu-probe.c @@ -27,7 +27,7 @@  #include <asm/spram.h>  #include <asm/uaccess.h> -static int __cpuinitdata mips_fpu_disabled; +static int mips_fpu_disabled;  static int __init fpu_disable(char *s)  { @@ -39,7 +39,7 @@ static int __init fpu_disable(char *s)  __setup("nofpu", fpu_disable); -int __cpuinitdata mips_dsp_disabled; +int mips_dsp_disabled;  static int __init dsp_disable(char *s)  { @@ -134,7 +134,7 @@ static inline void cpu_probe_vmbits(struct cpuinfo_mips *c)  #endif  } -static void __cpuinit set_isa(struct cpuinfo_mips *c, unsigned int isa) +static void set_isa(struct cpuinfo_mips *c, unsigned int isa)  {  	switch (isa) {  	case MIPS_CPU_ISA_M64R2: @@ -146,8 +146,7 @@ static void __cpuinit set_isa(struct cpuinfo_mips *c, unsigned int isa)  	case MIPS_CPU_ISA_IV:  		c->isa_level |= MIPS_CPU_ISA_IV;  	case MIPS_CPU_ISA_III: -		c->isa_level |= MIPS_CPU_ISA_I | MIPS_CPU_ISA_II | -				MIPS_CPU_ISA_III; +		c->isa_level |= MIPS_CPU_ISA_II | MIPS_CPU_ISA_III;  		break;  	case MIPS_CPU_ISA_M32R2: @@ -156,13 +155,11 @@ static void __cpuinit set_isa(struct cpuinfo_mips *c, unsigned int isa)  		c->isa_level |= MIPS_CPU_ISA_M32R1;  	case MIPS_CPU_ISA_II:  		c->isa_level |= MIPS_CPU_ISA_II; -	case MIPS_CPU_ISA_I: -		c->isa_level |= MIPS_CPU_ISA_I;  		break;  	}  } -static char unknown_isa[] __cpuinitdata = KERN_ERR \ +static char unknown_isa[] = KERN_ERR \  	"Unsupported ISA type, c0.config0: %d.";  static inline unsigned int decode_config0(struct cpuinfo_mips *c) @@ -272,9 +269,6 @@ static inline unsigned int decode_config3(struct cpuinfo_mips *c)  		c->options |= MIPS_CPU_ULRI;  	if (config3 & MIPS_CONF3_ISA)  		c->options |= MIPS_CPU_MICROMIPS; -#ifdef CONFIG_CPU_MICROMIPS -	write_c0_config3(read_c0_config3() | MIPS_CONF3_ISA_OE); -#endif  	if (config3 & MIPS_CONF3_VZ)  		c->ases |= MIPS_ASE_VZ; @@ -296,7 +290,7 @@ static inline unsigned int decode_config4(struct cpuinfo_mips *c)  	return config4 & MIPS_CONF_M;  } -static void __cpuinit decode_configs(struct cpuinfo_mips *c) +static void decode_configs(struct cpuinfo_mips *c)  {  	int ok; @@ -332,7 +326,6 @@ static inline void cpu_probe_legacy(struct cpuinfo_mips *c, unsigned int cpu)  	case PRID_IMP_R2000:  		c->cputype = CPU_R2000;  		__cpu_name[cpu] = "R2000"; -		set_isa(c, MIPS_CPU_ISA_I);  		c->options = MIPS_CPU_TLB | MIPS_CPU_3K_CACHE |  			     MIPS_CPU_NOFPUEX;  		if (__cpu_has_fpu()) @@ -352,7 +345,6 @@ static inline void cpu_probe_legacy(struct cpuinfo_mips *c, unsigned int cpu)  			c->cputype = CPU_R3000;  			__cpu_name[cpu] = "R3000";  		} -		set_isa(c, MIPS_CPU_ISA_I);  		c->options = MIPS_CPU_TLB | MIPS_CPU_3K_CACHE |  			     MIPS_CPU_NOFPUEX;  		if (__cpu_has_fpu()) @@ -455,7 +447,6 @@ static inline void cpu_probe_legacy(struct cpuinfo_mips *c, unsigned int cpu)  		break;  	#endif  	case PRID_IMP_TX39: -		set_isa(c, MIPS_CPU_ISA_I);  		c->options = MIPS_CPU_TLB | MIPS_CPU_TX39_CACHE;  		if ((c->processor_id & 0xf0) == (PRID_REV_TX3927 & 0xf0)) { @@ -959,6 +950,7 @@ static inline void cpu_probe_netlogic(struct cpuinfo_mips *c, int cpu)  		set_isa(c, MIPS_CPU_ISA_M64R1);  		c->tlbsize = ((read_c0_config1() >> 25) & 0x3f) + 1;  	} +	c->kscratch_mask = 0xf;  }  #ifdef CONFIG_64BIT @@ -970,7 +962,7 @@ EXPORT_SYMBOL(__ua_limit);  const char *__cpu_name[NR_CPUS];  const char *__elf_platform; -__cpuinit void cpu_probe(void) +void cpu_probe(void)  {  	struct cpuinfo_mips *c = ¤t_cpu_data;  	unsigned int cpu = smp_processor_id(); @@ -1055,7 +1047,7 @@ __cpuinit void cpu_probe(void)  #endif  } -__cpuinit void cpu_report(void) +void cpu_report(void)  {  	struct cpuinfo_mips *c = ¤t_cpu_data; diff --git a/arch/mips/kernel/crash_dump.c b/arch/mips/kernel/crash_dump.c index 3be9e7bb30ff..f291cf99b03a 100644 --- a/arch/mips/kernel/crash_dump.c +++ b/arch/mips/kernel/crash_dump.c @@ -4,16 +4,6 @@  #include <asm/uaccess.h>  #include <linux/slab.h> -static int __init parse_savemaxmem(char *p) -{ -	if (p) -		saved_max_pfn = (memparse(p, &p) >> PAGE_SHIFT) - 1; - -	return 1; -} -__setup("savemaxmem=", parse_savemaxmem); - -  static void *kdump_buf_page;  /** diff --git a/arch/mips/kernel/head.S b/arch/mips/kernel/head.S index c61cdaed2b1d..7b6a5b3e3acf 100644 --- a/arch/mips/kernel/head.S +++ b/arch/mips/kernel/head.S @@ -28,45 +28,6 @@  #include <kernel-entry-init.h>  	/* -	 * inputs are the text nasid in t1, data nasid in t2. -	 */ -	.macro MAPPED_KERNEL_SETUP_TLB -#ifdef CONFIG_MAPPED_KERNEL -	/* -	 * This needs to read the nasid - assume 0 for now. -	 * Drop in 0xffffffffc0000000 in tlbhi, 0+VG in tlblo_0, -	 * 0+DVG in tlblo_1. -	 */ -	dli	t0, 0xffffffffc0000000 -	dmtc0	t0, CP0_ENTRYHI -	li	t0, 0x1c000		# Offset of text into node memory -	dsll	t1, NASID_SHFT		# Shift text nasid into place -	dsll	t2, NASID_SHFT		# Same for data nasid -	or	t1, t1, t0		# Physical load address of kernel text -	or	t2, t2, t0		# Physical load address of kernel data -	dsrl	t1, 12			# 4K pfn -	dsrl	t2, 12			# 4K pfn -	dsll	t1, 6			# Get pfn into place -	dsll	t2, 6			# Get pfn into place -	li	t0, ((_PAGE_GLOBAL|_PAGE_VALID| _CACHE_CACHABLE_COW) >> 6) -	or	t0, t0, t1 -	mtc0	t0, CP0_ENTRYLO0	# physaddr, VG, cach exlwr -	li	t0, ((_PAGE_GLOBAL|_PAGE_VALID| _PAGE_DIRTY|_CACHE_CACHABLE_COW) >> 6) -	or	t0, t0, t2 -	mtc0	t0, CP0_ENTRYLO1	# physaddr, DVG, cach exlwr -	li	t0, 0x1ffe000		# MAPPED_KERN_TLBMASK, TLBPGMASK_16M -	mtc0	t0, CP0_PAGEMASK -	li	t0, 0			# KMAP_INX -	mtc0	t0, CP0_INDEX -	li	t0, 1 -	mtc0	t0, CP0_WIRED -	tlbwi -#else -	mtc0	zero, CP0_WIRED -#endif -	.endm - -	/*  	 * For the moment disable interrupts, mark the kernel mode and  	 * set ST0_KX so that the CPU does not spit fire when using  	 * 64-bit addresses.  A full initialization of the CPU's status @@ -197,8 +158,6 @@ NESTED(kernel_entry, 16, sp)			# kernel entry point  	j		start_kernel  	END(kernel_entry) -	__CPUINIT -  #ifdef CONFIG_SMP  /*   * SMP slave cpus entry point.	Board specific code for bootstrap calls this @@ -227,5 +186,3 @@ NESTED(smp_bootstrap, 16, sp)  	j	start_secondary  	END(smp_bootstrap)  #endif /* CONFIG_SMP */ - -	__FINIT diff --git a/arch/mips/kernel/irq-gic.c b/arch/mips/kernel/irq-gic.c index c01b307317a9..5b5ddb231f26 100644 --- a/arch/mips/kernel/irq-gic.c +++ b/arch/mips/kernel/irq-gic.c @@ -219,16 +219,15 @@ static int gic_set_affinity(struct irq_data *d, const struct cpumask *cpumask,  	/* Assumption : cpumask refers to a single CPU */  	spin_lock_irqsave(&gic_lock, flags); -	for (;;) { -		/* Re-route this IRQ */ -		GIC_SH_MAP_TO_VPE_SMASK(irq, first_cpu(tmp)); -		/* Update the pcpu_masks */ -		for (i = 0; i < NR_CPUS; i++) -			clear_bit(irq, pcpu_masks[i].pcpu_mask); -		set_bit(irq, pcpu_masks[first_cpu(tmp)].pcpu_mask); +	/* Re-route this IRQ */ +	GIC_SH_MAP_TO_VPE_SMASK(irq, first_cpu(tmp)); + +	/* Update the pcpu_masks */ +	for (i = 0; i < NR_CPUS; i++) +		clear_bit(irq, pcpu_masks[i].pcpu_mask); +	set_bit(irq, pcpu_masks[first_cpu(tmp)].pcpu_mask); -	}  	cpumask_copy(d->affinity, cpumask);  	spin_unlock_irqrestore(&gic_lock, flags); diff --git a/arch/mips/kernel/mcount.S b/arch/mips/kernel/mcount.S index 33d067148e61..a03e93c4a946 100644 --- a/arch/mips/kernel/mcount.S +++ b/arch/mips/kernel/mcount.S @@ -168,15 +168,11 @@ NESTED(ftrace_graph_caller, PT_SIZE, ra)  #endif  	/* arg3: Get frame pointer of current stack */ -#ifdef CONFIG_FRAME_POINTER -	move	a2, fp -#else /* ! CONFIG_FRAME_POINTER */  #ifdef CONFIG_64BIT  	PTR_LA	a2, PT_SIZE(sp)  #else  	PTR_LA	a2, (PT_SIZE+8)(sp)  #endif -#endif  	jal	prepare_ftrace_return  	 nop diff --git a/arch/mips/kernel/mips-mt-fpaff.c b/arch/mips/kernel/mips-mt-fpaff.c index fd814e08c945..cb098628aee8 100644 --- a/arch/mips/kernel/mips-mt-fpaff.c +++ b/arch/mips/kernel/mips-mt-fpaff.c @@ -27,12 +27,12 @@ unsigned long mt_fpemul_threshold;   * FPU affinity with the user's requested processor affinity.   * This code is 98% identical with the sys_sched_setaffinity()   * and sys_sched_getaffinity() system calls, and should be - * updated when kernel/sched.c changes. + * updated when kernel/sched/core.c changes.   */  /*   * find_process_by_pid - find a process with a matching PID value. - * used in sys_sched_set/getaffinity() in kernel/sched.c, so + * used in sys_sched_set/getaffinity() in kernel/sched/core.c, so   * cloned here.   */  static inline struct task_struct *find_process_by_pid(pid_t pid) diff --git a/arch/mips/kernel/octeon_switch.S b/arch/mips/kernel/octeon_switch.S index 0e23343eb0a9..4204d76af854 100644 --- a/arch/mips/kernel/octeon_switch.S +++ b/arch/mips/kernel/octeon_switch.S @@ -40,33 +40,6 @@  	cpu_save_nonscratch a0  	LONG_S	ra, THREAD_REG31(a0) -	/* check if we need to save COP2 registers */ -	PTR_L	t2, TASK_THREAD_INFO(a0) -	LONG_L	t0, ST_OFF(t2) -	bbit0	t0, 30, 1f - -	/* Disable COP2 in the stored process state */ -	li	t1, ST0_CU2 -	xor	t0, t1 -	LONG_S	t0, ST_OFF(t2) - -	/* Enable COP2 so we can save it */ -	mfc0	t0, CP0_STATUS -	or	t0, t1 -	mtc0	t0, CP0_STATUS - -	/* Save COP2 */ -	daddu	a0, THREAD_CP2 -	jal octeon_cop2_save -	dsubu	a0, THREAD_CP2 - -	/* Disable COP2 now that we are done */ -	mfc0	t0, CP0_STATUS -	li	t1, ST0_CU2 -	xor	t0, t1 -	mtc0	t0, CP0_STATUS - -1:  #if CONFIG_CAVIUM_OCTEON_CVMSEG_SIZE > 0  	/* Check if we need to store CVMSEG state */  	mfc0	t0, $11,7	/* CvmMemCtl */ @@ -98,6 +71,13 @@  	mtc0	t0, $11,7	/* CvmMemCtl */  #endif  3: + +#if defined(CONFIG_CC_STACKPROTECTOR) && !defined(CONFIG_SMP) +	PTR_L	t8, __stack_chk_guard +	LONG_L	t9, TASK_STACK_CANARY(a1) +	LONG_S	t9, 0(t8) +#endif +  	/*  	 * The order of restoring the registers takes care of the race  	 * updating $28, $29 and kernelsp without disabling ints. diff --git a/arch/mips/kernel/proc.c b/arch/mips/kernel/proc.c index acb34373679e..8c58d8a84bf3 100644 --- a/arch/mips/kernel/proc.c +++ b/arch/mips/kernel/proc.c @@ -66,9 +66,7 @@ static int show_cpuinfo(struct seq_file *m, void *v)  		seq_printf(m, "]\n");  	}  	if (cpu_has_mips_r) { -		seq_printf(m, "isa\t\t\t:"); -		if (cpu_has_mips_1) -			seq_printf(m, "%s", " mips1"); +		seq_printf(m, "isa\t\t\t: mips1");  		if (cpu_has_mips_2)  			seq_printf(m, "%s", " mips2");  		if (cpu_has_mips_3) diff --git a/arch/mips/kernel/process.c b/arch/mips/kernel/process.c index c6a041d9d05d..ddc76103e78c 100644 --- a/arch/mips/kernel/process.c +++ b/arch/mips/kernel/process.c @@ -201,9 +201,12 @@ int dump_task_fpu(struct task_struct *t, elf_fpregset_t *fpr)  	return 1;  } -/* - * - */ +#ifdef CONFIG_CC_STACKPROTECTOR +#include <linux/stackprotector.h> +unsigned long __stack_chk_guard __read_mostly; +EXPORT_SYMBOL(__stack_chk_guard); +#endif +  struct mips_frame_info {  	void		*func;  	unsigned long	func_size; diff --git a/arch/mips/kernel/prom.c b/arch/mips/kernel/prom.c index 5712bb532245..7e954042f252 100644 --- a/arch/mips/kernel/prom.c +++ b/arch/mips/kernel/prom.c @@ -30,7 +30,7 @@ __init void mips_set_machine_name(const char *name)  	if (name == NULL)  		return; -	strncpy(mips_machine_name, name, sizeof(mips_machine_name)); +	strlcpy(mips_machine_name, name, sizeof(mips_machine_name));  	pr_info("MIPS: machine is %s\n", mips_get_machine_name());  } diff --git a/arch/mips/kernel/ptrace.c b/arch/mips/kernel/ptrace.c index 9c6299c733a3..8ae1ebef8b71 100644 --- a/arch/mips/kernel/ptrace.c +++ b/arch/mips/kernel/ptrace.c @@ -15,6 +15,7 @@   * binaries.   */  #include <linux/compiler.h> +#include <linux/context_tracking.h>  #include <linux/kernel.h>  #include <linux/sched.h>  #include <linux/mm.h> @@ -534,6 +535,8 @@ static inline int audit_arch(void)   */  asmlinkage void syscall_trace_enter(struct pt_regs *regs)  { +	user_exit(); +  	/* do the secure computing check first */  	secure_computing_strict(regs->regs[2]); @@ -570,6 +573,13 @@ out:   */  asmlinkage void syscall_trace_leave(struct pt_regs *regs)  { +        /* +	 * We may come here right after calling schedule_user() +	 * or do_notify_resume(), in which case we can be in RCU +	 * user mode. +	 */ +	user_exit(); +  	audit_syscall_exit(regs);  	if (!(current->ptrace & PT_PTRACED)) @@ -592,4 +602,6 @@ asmlinkage void syscall_trace_leave(struct pt_regs *regs)  		send_sig(current->exit_code, current, 1);  		current->exit_code = 0;  	} + +	user_enter();  } diff --git a/arch/mips/kernel/r2300_switch.S b/arch/mips/kernel/r2300_switch.S index 5266c6ee2b35..38af83f84c4a 100644 --- a/arch/mips/kernel/r2300_switch.S +++ b/arch/mips/kernel/r2300_switch.S @@ -65,6 +65,13 @@ LEAF(resume)  	fpu_save_single a0, t0			# clobbers t0  1: + +#if defined(CONFIG_CC_STACKPROTECTOR) && !defined(CONFIG_SMP) +	PTR_L	t8, __stack_chk_guard +	LONG_L	t9, TASK_STACK_CANARY(a1) +	LONG_S	t9, 0(t8) +#endif +  	/*  	 * The order of restoring the registers takes care of the race  	 * updating $28, $29 and kernelsp without disabling ints. diff --git a/arch/mips/kernel/r4k_switch.S b/arch/mips/kernel/r4k_switch.S index 5e51219990aa..921238a6bd26 100644 --- a/arch/mips/kernel/r4k_switch.S +++ b/arch/mips/kernel/r4k_switch.S @@ -68,6 +68,12 @@  						# clobbers t1  1: +#if defined(CONFIG_CC_STACKPROTECTOR) && !defined(CONFIG_SMP) +	PTR_L	t8, __stack_chk_guard +	LONG_L	t9, TASK_STACK_CANARY(a1) +	LONG_S	t9, 0(t8) +#endif +  	/*  	 * The order of restoring the registers takes care of the race  	 * updating $28, $29 and kernelsp without disabling ints. diff --git a/arch/mips/kernel/rtlx.c b/arch/mips/kernel/rtlx.c index 6fa198db8999..d763f11e35e2 100644 --- a/arch/mips/kernel/rtlx.c +++ b/arch/mips/kernel/rtlx.c @@ -437,7 +437,6 @@ static ssize_t file_write(struct file *file, const char __user * buffer,  			  size_t count, loff_t * ppos)  {  	int minor = iminor(file_inode(file)); -	struct rtlx_channel *rt = &rtlx->channel[minor];  	/* any space left... */  	if (!rtlx_write_poll(minor)) { diff --git a/arch/mips/kernel/scall32-o32.S b/arch/mips/kernel/scall32-o32.S index 9b36424b03c5..e774bb1088b5 100644 --- a/arch/mips/kernel/scall32-o32.S +++ b/arch/mips/kernel/scall32-o32.S @@ -52,7 +52,7 @@ NESTED(handle_sys, PT_SIZE, sp)  stack_done:  	lw	t0, TI_FLAGS($28)	# syscall tracing enabled? -	li	t1, _TIF_SYSCALL_TRACE | _TIF_SYSCALL_AUDIT +	li	t1, _TIF_WORK_SYSCALL_ENTRY  	and	t0, t1  	bnez	t0, syscall_trace_entry # -> yes @@ -476,8 +476,9 @@ einval: li	v0, -ENOSYS  	/*  	 * For FPU affinity scheduling on MIPS MT processors, we need to  	 * intercept sys_sched_xxxaffinity() calls until we get a proper hook -	 * in kernel/sched.c.  Considered only temporary we only support these -	 * hooks for the 32-bit kernel - there is no MIPS64 MT processor atm. +	 * in kernel/sched/core.c.  Considered only temporary we only support +	 * these hooks for the 32-bit kernel - there is no MIPS64 MT processor +	 * atm.  	 */  	sys	mipsmt_sys_sched_setaffinity	3  	sys	mipsmt_sys_sched_getaffinity	3 diff --git a/arch/mips/kernel/scall64-64.S b/arch/mips/kernel/scall64-64.S index 97a5909a61cf..be6627ead619 100644 --- a/arch/mips/kernel/scall64-64.S +++ b/arch/mips/kernel/scall64-64.S @@ -54,7 +54,7 @@ NESTED(handle_sys64, PT_SIZE, sp)  	sd	a3, PT_R26(sp)		# save a3 for syscall restarting -	li	t1, _TIF_SYSCALL_TRACE | _TIF_SYSCALL_AUDIT +	li	t1, _TIF_WORK_SYSCALL_ENTRY  	LONG_L	t0, TI_FLAGS($28)	# syscall tracing enabled?  	and	t0, t1, t0  	bnez	t0, syscall_trace_entry diff --git a/arch/mips/kernel/scall64-n32.S b/arch/mips/kernel/scall64-n32.S index edcb6594e7b5..cab150789c8d 100644 --- a/arch/mips/kernel/scall64-n32.S +++ b/arch/mips/kernel/scall64-n32.S @@ -47,7 +47,7 @@ NESTED(handle_sysn32, PT_SIZE, sp)  	sd	a3, PT_R26(sp)		# save a3 for syscall restarting -	li	t1, _TIF_SYSCALL_TRACE | _TIF_SYSCALL_AUDIT +	li	t1, _TIF_WORK_SYSCALL_ENTRY  	LONG_L	t0, TI_FLAGS($28)	# syscall tracing enabled?  	and	t0, t1, t0  	bnez	t0, n32_syscall_trace_entry diff --git a/arch/mips/kernel/scall64-o32.S b/arch/mips/kernel/scall64-o32.S index 74f485d3c0ef..37605dc8eef7 100644 --- a/arch/mips/kernel/scall64-o32.S +++ b/arch/mips/kernel/scall64-o32.S @@ -81,7 +81,7 @@ NESTED(handle_sys, PT_SIZE, sp)  	PTR	4b, bad_stack  	.previous -	li	t1, _TIF_SYSCALL_TRACE | _TIF_SYSCALL_AUDIT +	li	t1, _TIF_WORK_SYSCALL_ENTRY  	LONG_L	t0, TI_FLAGS($28)	# syscall tracing enabled?  	and	t0, t1, t0  	bnez	t0, trace_a_syscall diff --git a/arch/mips/kernel/signal.c b/arch/mips/kernel/signal.c index fd3ef2c2afbc..2f285abc76d5 100644 --- a/arch/mips/kernel/signal.c +++ b/arch/mips/kernel/signal.c @@ -8,6 +8,7 @@   * Copyright (C) 1999, 2000 Silicon Graphics, Inc.   */  #include <linux/cache.h> +#include <linux/context_tracking.h>  #include <linux/irqflags.h>  #include <linux/sched.h>  #include <linux/mm.h> @@ -573,6 +574,8 @@ asmlinkage void do_notify_resume(struct pt_regs *regs, void *unused,  {  	local_irq_enable(); +	user_exit(); +  	/* deal with pending signal delivery */  	if (thread_info_flags & _TIF_SIGPENDING)  		do_signal(regs); @@ -581,6 +584,8 @@ asmlinkage void do_notify_resume(struct pt_regs *regs, void *unused,  		clear_thread_flag(TIF_NOTIFY_RESUME);  		tracehook_notify_resume(regs);  	} + +	user_enter();  }  #ifdef CONFIG_SMP diff --git a/arch/mips/kernel/smp-bmips.c b/arch/mips/kernel/smp-bmips.c index 8e393b8443f7..c0bb4d59076a 100644 --- a/arch/mips/kernel/smp-bmips.c +++ b/arch/mips/kernel/smp-bmips.c @@ -63,7 +63,7 @@ static irqreturn_t bmips_ipi_interrupt(int irq, void *dev_id);  static void __init bmips_smp_setup(void)  { -	int i; +	int i, cpu = 1, boot_cpu = 0;  #if defined(CONFIG_CPU_BMIPS4350) || defined(CONFIG_CPU_BMIPS4380)  	/* arbitration priority */ @@ -72,13 +72,22 @@ static void __init bmips_smp_setup(void)  	/* NBK and weak order flags */  	set_c0_brcm_config_0(0x30000); +	/* Find out if we are running on TP0 or TP1 */ +	boot_cpu = !!(read_c0_brcm_cmt_local() & (1 << 31)); +  	/*  	 * MIPS interrupts 0,1 (SW INT 0,1) cross over to the other thread  	 * MIPS interrupt 2 (HW INT 0) is the CPU0 L1 controller output  	 * MIPS interrupt 3 (HW INT 1) is the CPU1 L1 controller output +	 * +	 * If booting from TP1, leave the existing CMT interrupt routing +	 * such that TP0 responds to SW1 and TP1 responds to SW0.  	 */ -	change_c0_brcm_cmt_intr(0xf8018000, -		(0x02 << 27) | (0x03 << 15)); +	if (boot_cpu == 0) +		change_c0_brcm_cmt_intr(0xf8018000, +					(0x02 << 27) | (0x03 << 15)); +	else +		change_c0_brcm_cmt_intr(0xf8018000, (0x1d << 27));  	/* single core, 2 threads (2 pipelines) */  	max_cpus = 2; @@ -106,9 +115,15 @@ static void __init bmips_smp_setup(void)  	if (!board_ebase_setup)  		board_ebase_setup = &bmips_ebase_setup; +	__cpu_number_map[boot_cpu] = 0; +	__cpu_logical_map[0] = boot_cpu; +  	for (i = 0; i < max_cpus; i++) { -		__cpu_number_map[i] = 1; -		__cpu_logical_map[i] = 1; +		if (i != boot_cpu) { +			__cpu_number_map[i] = cpu; +			__cpu_logical_map[cpu] = i; +			cpu++; +		}  		set_cpu_possible(i, 1);  		set_cpu_present(i, 1);  	} @@ -157,7 +172,9 @@ static void bmips_boot_secondary(int cpu, struct task_struct *idle)  		bmips_send_ipi_single(cpu, 0);  	else {  #if defined(CONFIG_CPU_BMIPS4350) || defined(CONFIG_CPU_BMIPS4380) -		set_c0_brcm_cmt_ctrl(0x01); +		/* Reset slave TP1 if booting from TP0 */ +		if (cpu_logical_map(cpu) == 1) +			set_c0_brcm_cmt_ctrl(0x01);  #elif defined(CONFIG_CPU_BMIPS5000)  		if (cpu & 0x01)  			write_c0_brcm_action(ACTION_BOOT_THREAD(cpu)); @@ -381,7 +398,7 @@ struct plat_smp_ops bmips_smp_ops = {   * UP BMIPS systems as well.   ***********************************************************************/ -static void __cpuinit bmips_wr_vec(unsigned long dst, char *start, char *end) +static void bmips_wr_vec(unsigned long dst, char *start, char *end)  {  	memcpy((void *)dst, start, end - start);  	dma_cache_wback((unsigned long)start, end - start); @@ -389,7 +406,7 @@ static void __cpuinit bmips_wr_vec(unsigned long dst, char *start, char *end)  	instruction_hazard();  } -static inline void __cpuinit bmips_nmi_handler_setup(void) +static inline void bmips_nmi_handler_setup(void)  {  	bmips_wr_vec(BMIPS_NMI_RESET_VEC, &bmips_reset_nmi_vec,  		&bmips_reset_nmi_vec_end); @@ -397,7 +414,7 @@ static inline void __cpuinit bmips_nmi_handler_setup(void)  		&bmips_smp_int_vec_end);  } -void __cpuinit bmips_ebase_setup(void) +void bmips_ebase_setup(void)  {  	unsigned long new_ebase = ebase;  	void __iomem __maybe_unused *cbr; diff --git a/arch/mips/kernel/smp-mt.c b/arch/mips/kernel/smp-mt.c index 3e5164c11cac..57a3f7a2b370 100644 --- a/arch/mips/kernel/smp-mt.c +++ b/arch/mips/kernel/smp-mt.c @@ -149,7 +149,7 @@ static void vsmp_send_ipi_mask(const struct cpumask *mask, unsigned int action)  		vsmp_send_ipi_single(i, action);  } -static void __cpuinit vsmp_init_secondary(void) +static void vsmp_init_secondary(void)  {  #ifdef CONFIG_IRQ_GIC  	/* This is Malta specific: IPI,performance and timer interrupts */ @@ -162,7 +162,7 @@ static void __cpuinit vsmp_init_secondary(void)  					 STATUSF_IP6 | STATUSF_IP7);  } -static void __cpuinit vsmp_smp_finish(void) +static void vsmp_smp_finish(void)  {  	/* CDFIXME: remove this? */  	write_c0_compare(read_c0_count() + (8* mips_hpt_frequency/HZ)); @@ -188,7 +188,7 @@ static void vsmp_cpus_done(void)   * (unsigned long)idle->thread_info the gp   * assumes a 1:1 mapping of TC => VPE   */ -static void __cpuinit vsmp_boot_secondary(int cpu, struct task_struct *idle) +static void vsmp_boot_secondary(int cpu, struct task_struct *idle)  {  	struct thread_info *gp = task_thread_info(idle);  	dvpe(); diff --git a/arch/mips/kernel/smp-up.c b/arch/mips/kernel/smp-up.c index 00500fea2750..7fde3e4d978f 100644 --- a/arch/mips/kernel/smp-up.c +++ b/arch/mips/kernel/smp-up.c @@ -28,11 +28,11 @@ static inline void up_send_ipi_mask(const struct cpumask *mask,   *  After we've done initial boot, this function is called to allow the   *  board code to clean up state, if needed   */ -static void __cpuinit up_init_secondary(void) +static void up_init_secondary(void)  {  } -static void __cpuinit up_smp_finish(void) +static void up_smp_finish(void)  {  } @@ -44,7 +44,7 @@ static void up_cpus_done(void)  /*   * Firmware CPU startup hook   */ -static void __cpuinit up_boot_secondary(int cpu, struct task_struct *idle) +static void up_boot_secondary(int cpu, struct task_struct *idle)  {  } diff --git a/arch/mips/kernel/smp.c b/arch/mips/kernel/smp.c index 6e7862ab46cc..5c208ed8f856 100644 --- a/arch/mips/kernel/smp.c +++ b/arch/mips/kernel/smp.c @@ -86,7 +86,7 @@ static inline void set_cpu_sibling_map(int cpu)  struct plat_smp_ops *mp_ops;  EXPORT_SYMBOL(mp_ops); -__cpuinit void register_smp_ops(struct plat_smp_ops *ops) +void register_smp_ops(struct plat_smp_ops *ops)  {  	if (mp_ops)  		printk(KERN_WARNING "Overriding previously set SMP ops\n"); @@ -98,7 +98,7 @@ __cpuinit void register_smp_ops(struct plat_smp_ops *ops)   * First C code run on the secondary CPUs after being started up by   * the master.   */ -asmlinkage __cpuinit void start_secondary(void) +asmlinkage void start_secondary(void)  {  	unsigned int cpu; @@ -197,7 +197,7 @@ void smp_prepare_boot_cpu(void)  	cpu_set(0, cpu_callin_map);  } -int __cpuinit __cpu_up(unsigned int cpu, struct task_struct *tidle) +int __cpu_up(unsigned int cpu, struct task_struct *tidle)  {  	mp_ops->boot_secondary(cpu, tidle); diff --git a/arch/mips/kernel/smtc.c b/arch/mips/kernel/smtc.c index 75a4fd709841..dfc1b911be04 100644 --- a/arch/mips/kernel/smtc.c +++ b/arch/mips/kernel/smtc.c @@ -645,7 +645,7 @@ void smtc_prepare_cpus(int cpus)   * (unsigned long)idle->thread_info the gp   *   */ -void __cpuinit smtc_boot_secondary(int cpu, struct task_struct *idle) +void smtc_boot_secondary(int cpu, struct task_struct *idle)  {  	extern u32 kernelsp[NR_CPUS];  	unsigned long flags; diff --git a/arch/mips/kernel/spram.c b/arch/mips/kernel/spram.c index 6af08d896e20..93f86817f20a 100644 --- a/arch/mips/kernel/spram.c +++ b/arch/mips/kernel/spram.c @@ -37,7 +37,7 @@  /*   * Different semantics to the set_c0_* function built by __BUILD_SET_C0   */ -static __cpuinit unsigned int bis_c0_errctl(unsigned int set) +static unsigned int bis_c0_errctl(unsigned int set)  {  	unsigned int res;  	res = read_c0_errctl(); @@ -45,7 +45,7 @@ static __cpuinit unsigned int bis_c0_errctl(unsigned int set)  	return res;  } -static __cpuinit void ispram_store_tag(unsigned int offset, unsigned int data) +static void ispram_store_tag(unsigned int offset, unsigned int data)  {  	unsigned int errctl; @@ -64,7 +64,7 @@ static __cpuinit void ispram_store_tag(unsigned int offset, unsigned int data)  } -static __cpuinit unsigned int ispram_load_tag(unsigned int offset) +static unsigned int ispram_load_tag(unsigned int offset)  {  	unsigned int data;  	unsigned int errctl; @@ -82,7 +82,7 @@ static __cpuinit unsigned int ispram_load_tag(unsigned int offset)  	return data;  } -static __cpuinit void dspram_store_tag(unsigned int offset, unsigned int data) +static void dspram_store_tag(unsigned int offset, unsigned int data)  {  	unsigned int errctl; @@ -98,7 +98,7 @@ static __cpuinit void dspram_store_tag(unsigned int offset, unsigned int data)  } -static __cpuinit unsigned int dspram_load_tag(unsigned int offset) +static unsigned int dspram_load_tag(unsigned int offset)  {  	unsigned int data;  	unsigned int errctl; @@ -115,7 +115,7 @@ static __cpuinit unsigned int dspram_load_tag(unsigned int offset)  	return data;  } -static __cpuinit void probe_spram(char *type, +static void probe_spram(char *type,  	    unsigned int base,  	    unsigned int (*read)(unsigned int),  	    void (*write)(unsigned int, unsigned int)) @@ -196,7 +196,7 @@ static __cpuinit void probe_spram(char *type,  		offset += 2 * SPRAM_TAG_STRIDE;  	}  } -void __cpuinit spram_config(void) +void spram_config(void)  {  	struct cpuinfo_mips *c = ¤t_cpu_data;  	unsigned int config0; diff --git a/arch/mips/kernel/sync-r4k.c b/arch/mips/kernel/sync-r4k.c index 1ff43d5ac2c4..84536bf4a154 100644 --- a/arch/mips/kernel/sync-r4k.c +++ b/arch/mips/kernel/sync-r4k.c @@ -20,15 +20,15 @@  #include <asm/barrier.h>  #include <asm/mipsregs.h> -static atomic_t __cpuinitdata count_start_flag = ATOMIC_INIT(0); -static atomic_t __cpuinitdata count_count_start = ATOMIC_INIT(0); -static atomic_t __cpuinitdata count_count_stop = ATOMIC_INIT(0); -static atomic_t __cpuinitdata count_reference = ATOMIC_INIT(0); +static atomic_t count_start_flag = ATOMIC_INIT(0); +static atomic_t count_count_start = ATOMIC_INIT(0); +static atomic_t count_count_stop = ATOMIC_INIT(0); +static atomic_t count_reference = ATOMIC_INIT(0);  #define COUNTON 100  #define NR_LOOPS 5 -void __cpuinit synchronise_count_master(int cpu) +void synchronise_count_master(int cpu)  {  	int i;  	unsigned long flags; @@ -106,7 +106,7 @@ void __cpuinit synchronise_count_master(int cpu)  	printk("done.\n");  } -void __cpuinit synchronise_count_slave(int cpu) +void synchronise_count_slave(int cpu)  {  	int i;  	unsigned int initcount; diff --git a/arch/mips/kernel/traps.c b/arch/mips/kernel/traps.c index a75ae40184aa..aec3408edd4b 100644 --- a/arch/mips/kernel/traps.c +++ b/arch/mips/kernel/traps.c @@ -13,6 +13,7 @@   */  #include <linux/bug.h>  #include <linux/compiler.h> +#include <linux/context_tracking.h>  #include <linux/kexec.h>  #include <linux/init.h>  #include <linux/kernel.h> @@ -89,7 +90,7 @@ void (*board_nmi_handler_setup)(void);  void (*board_ejtag_handler_setup)(void);  void (*board_bind_eic_interrupt)(int irq, int regset);  void (*board_ebase_setup)(void); -void __cpuinitdata(*board_cache_error_setup)(void); +void(*board_cache_error_setup)(void);  static void show_raw_backtrace(unsigned long reg29)  { @@ -264,7 +265,7 @@ static void __show_regs(const struct pt_regs *regs)  	printk("Status: %08x	", (uint32_t) regs->cp0_status); -	if (current_cpu_data.isa_level == MIPS_CPU_ISA_I) { +	if (cpu_has_3kex) {  		if (regs->cp0_status & ST0_KUO)  			printk("KUo ");  		if (regs->cp0_status & ST0_IEO) @@ -277,7 +278,7 @@ static void __show_regs(const struct pt_regs *regs)  			printk("KUc ");  		if (regs->cp0_status & ST0_IEC)  			printk("IEc "); -	} else { +	} else if (cpu_has_4kex) {  		if (regs->cp0_status & ST0_KX)  			printk("KX ");  		if (regs->cp0_status & ST0_SX) @@ -423,7 +424,9 @@ asmlinkage void do_be(struct pt_regs *regs)  	const struct exception_table_entry *fixup = NULL;  	int data = regs->cp0_cause & 4;  	int action = MIPS_BE_FATAL; +	enum ctx_state prev_state; +	prev_state = exception_enter();  	/* XXX For now.	 Fixme, this searches the wrong table ...  */  	if (data && !user_mode(regs))  		fixup = search_dbe_tables(exception_epc(regs)); @@ -436,11 +439,11 @@ asmlinkage void do_be(struct pt_regs *regs)  	switch (action) {  	case MIPS_BE_DISCARD: -		return; +		goto out;  	case MIPS_BE_FIXUP:  		if (fixup) {  			regs->cp0_epc = fixup->nextinsn; -			return; +			goto out;  		}  		break;  	default: @@ -455,10 +458,13 @@ asmlinkage void do_be(struct pt_regs *regs)  	       field, regs->cp0_epc, field, regs->regs[31]);  	if (notify_die(DIE_OOPS, "bus error", regs, 0, regs_to_trapnr(regs), SIGBUS)  	    == NOTIFY_STOP) -		return; +		goto out;  	die_if_kernel("Oops", regs);  	force_sig(SIGBUS, current); + +out: +	exception_exit(prev_state);  }  /* @@ -673,8 +679,10 @@ static int simulate_sync(struct pt_regs *regs, unsigned int opcode)  asmlinkage void do_ov(struct pt_regs *regs)  { +	enum ctx_state prev_state;  	siginfo_t info; +	prev_state = exception_enter();  	die_if_kernel("Integer overflow", regs);  	info.si_code = FPE_INTOVF; @@ -682,6 +690,7 @@ asmlinkage void do_ov(struct pt_regs *regs)  	info.si_errno = 0;  	info.si_addr = (void __user *) regs->cp0_epc;  	force_sig_info(SIGFPE, &info, current); +	exception_exit(prev_state);  }  int process_fpemu_return(int sig, void __user *fault_addr) @@ -713,11 +722,13 @@ int process_fpemu_return(int sig, void __user *fault_addr)   */  asmlinkage void do_fpe(struct pt_regs *regs, unsigned long fcr31)  { +	enum ctx_state prev_state;  	siginfo_t info = {0}; +	prev_state = exception_enter();  	if (notify_die(DIE_FP, "FP exception", regs, 0, regs_to_trapnr(regs), SIGFPE)  	    == NOTIFY_STOP) -		return; +		goto out;  	die_if_kernel("FP exception in kernel code", regs);  	if (fcr31 & FPU_CSR_UNI_X) { @@ -753,7 +764,7 @@ asmlinkage void do_fpe(struct pt_regs *regs, unsigned long fcr31)  		/* If something went wrong, signal */  		process_fpemu_return(sig, fault_addr); -		return; +		goto out;  	} else if (fcr31 & FPU_CSR_INV_X)  		info.si_code = FPE_FLTINV;  	else if (fcr31 & FPU_CSR_DIV_X) @@ -770,6 +781,9 @@ asmlinkage void do_fpe(struct pt_regs *regs, unsigned long fcr31)  	info.si_errno = 0;  	info.si_addr = (void __user *) regs->cp0_epc;  	force_sig_info(SIGFPE, &info, current); + +out: +	exception_exit(prev_state);  }  static void do_trap_or_bp(struct pt_regs *regs, unsigned int code, @@ -835,9 +849,11 @@ static void do_trap_or_bp(struct pt_regs *regs, unsigned int code,  asmlinkage void do_bp(struct pt_regs *regs)  {  	unsigned int opcode, bcode; +	enum ctx_state prev_state;  	unsigned long epc;  	u16 instr[2]; +	prev_state = exception_enter();  	if (get_isa16_mode(regs->cp0_epc)) {  		/* Calculate EPC. */  		epc = exception_epc(regs); @@ -852,7 +868,7 @@ asmlinkage void do_bp(struct pt_regs *regs)  				goto out_sigsegv;  		    bcode = (instr[0] >> 6) & 0x3f;  		    do_trap_or_bp(regs, bcode, "Break"); -		    return; +		    goto out;  		}  	} else {  		if (__get_user(opcode, (unsigned int __user *) exception_epc(regs))) @@ -876,12 +892,12 @@ asmlinkage void do_bp(struct pt_regs *regs)  	switch (bcode) {  	case BRK_KPROBE_BP:  		if (notify_die(DIE_BREAK, "debug", regs, bcode, regs_to_trapnr(regs), SIGTRAP) == NOTIFY_STOP) -			return; +			goto out;  		else  			break;  	case BRK_KPROBE_SSTEPBP:  		if (notify_die(DIE_SSTEPBP, "single_step", regs, bcode, regs_to_trapnr(regs), SIGTRAP) == NOTIFY_STOP) -			return; +			goto out;  		else  			break;  	default: @@ -889,18 +905,24 @@ asmlinkage void do_bp(struct pt_regs *regs)  	}  	do_trap_or_bp(regs, bcode, "Break"); + +out: +	exception_exit(prev_state);  	return;  out_sigsegv:  	force_sig(SIGSEGV, current); +	goto out;  }  asmlinkage void do_tr(struct pt_regs *regs)  {  	u32 opcode, tcode = 0; +	enum ctx_state prev_state;  	u16 instr[2];  	unsigned long epc = msk_isa16_mode(exception_epc(regs)); +	prev_state = exception_enter();  	if (get_isa16_mode(regs->cp0_epc)) {  		if (__get_user(instr[0], (u16 __user *)(epc + 0)) ||  		    __get_user(instr[1], (u16 __user *)(epc + 2))) @@ -918,10 +940,14 @@ asmlinkage void do_tr(struct pt_regs *regs)  	}  	do_trap_or_bp(regs, tcode, "Trap"); + +out: +	exception_exit(prev_state);  	return;  out_sigsegv:  	force_sig(SIGSEGV, current); +	goto out;  }  asmlinkage void do_ri(struct pt_regs *regs) @@ -929,17 +955,19 @@ asmlinkage void do_ri(struct pt_regs *regs)  	unsigned int __user *epc = (unsigned int __user *)exception_epc(regs);  	unsigned long old_epc = regs->cp0_epc;  	unsigned long old31 = regs->regs[31]; +	enum ctx_state prev_state;  	unsigned int opcode = 0;  	int status = -1; +	prev_state = exception_enter();  	if (notify_die(DIE_RI, "RI Fault", regs, 0, regs_to_trapnr(regs), SIGILL)  	    == NOTIFY_STOP) -		return; +		goto out;  	die_if_kernel("Reserved instruction in kernel code", regs);  	if (unlikely(compute_return_epc(regs) < 0)) -		return; +		goto out;  	if (get_isa16_mode(regs->cp0_epc)) {  		unsigned short mmop[2] = { 0 }; @@ -974,6 +1002,9 @@ asmlinkage void do_ri(struct pt_regs *regs)  		regs->regs[31] = old31;  		force_sig(status, current);  	} + +out: +	exception_exit(prev_state);  }  /* @@ -1025,21 +1056,16 @@ static int default_cu2_call(struct notifier_block *nfb, unsigned long action,  {  	struct pt_regs *regs = data; -	switch (action) { -	default: -		die_if_kernel("Unhandled kernel unaligned access or invalid " +	die_if_kernel("COP2: Unhandled kernel unaligned access or invalid "  			      "instruction", regs); -		/* Fall through	 */ - -	case CU2_EXCEPTION: -		force_sig(SIGILL, current); -	} +	force_sig(SIGILL, current);  	return NOTIFY_OK;  }  asmlinkage void do_cpu(struct pt_regs *regs)  { +	enum ctx_state prev_state;  	unsigned int __user *epc;  	unsigned long old_epc, old31;  	unsigned int opcode; @@ -1047,10 +1073,12 @@ asmlinkage void do_cpu(struct pt_regs *regs)  	int status;  	unsigned long __maybe_unused flags; -	die_if_kernel("do_cpu invoked from kernel context!", regs); - +	prev_state = exception_enter();  	cpid = (regs->cp0_cause >> CAUSEB_CE) & 3; +	if (cpid != 2) +		die_if_kernel("do_cpu invoked from kernel context!", regs); +  	switch (cpid) {  	case 0:  		epc = (unsigned int __user *)exception_epc(regs); @@ -1060,7 +1088,7 @@ asmlinkage void do_cpu(struct pt_regs *regs)  		status = -1;  		if (unlikely(compute_return_epc(regs) < 0)) -			return; +			goto out;  		if (get_isa16_mode(regs->cp0_epc)) {  			unsigned short mmop[2] = { 0 }; @@ -1093,7 +1121,7 @@ asmlinkage void do_cpu(struct pt_regs *regs)  			force_sig(status, current);  		} -		return; +		goto out;  	case 3:  		/* @@ -1131,19 +1159,26 @@ asmlinkage void do_cpu(struct pt_regs *regs)  				mt_ase_fp_affinity();  		} -		return; +		goto out;  	case 2:  		raw_notifier_call_chain(&cu2_chain, CU2_EXCEPTION, regs); -		return; +		goto out;  	}  	force_sig(SIGILL, current); + +out: +	exception_exit(prev_state);  }  asmlinkage void do_mdmx(struct pt_regs *regs)  { +	enum ctx_state prev_state; + +	prev_state = exception_enter();  	force_sig(SIGILL, current); +	exception_exit(prev_state);  }  /* @@ -1151,8 +1186,10 @@ asmlinkage void do_mdmx(struct pt_regs *regs)   */  asmlinkage void do_watch(struct pt_regs *regs)  { +	enum ctx_state prev_state;  	u32 cause; +	prev_state = exception_enter();  	/*  	 * Clear WP (bit 22) bit of cause register so we don't loop  	 * forever. @@ -1174,13 +1211,16 @@ asmlinkage void do_watch(struct pt_regs *regs)  		mips_clear_watch_registers();  		local_irq_enable();  	} +	exception_exit(prev_state);  }  asmlinkage void do_mcheck(struct pt_regs *regs)  {  	const int field = 2 * sizeof(unsigned long);  	int multi_match = regs->cp0_status & ST0_TS; +	enum ctx_state prev_state; +	prev_state = exception_enter();  	show_regs(regs);  	if (multi_match) { @@ -1627,7 +1667,6 @@ void *set_vi_handler(int n, vi_handler_t addr)  }  extern void tlb_init(void); -extern void flush_tlb_handlers(void);  /*   * Timer interrupt @@ -1642,7 +1681,7 @@ int cp0_compare_irq_shift;  int cp0_perfcount_irq;  EXPORT_SYMBOL_GPL(cp0_perfcount_irq); -static int __cpuinitdata noulri; +static int noulri;  static int __init ulri_disable(char *s)  { @@ -1653,7 +1692,7 @@ static int __init ulri_disable(char *s)  }  __setup("noulri", ulri_disable); -void __cpuinit per_cpu_trap_init(bool is_boot_cpu) +void per_cpu_trap_init(bool is_boot_cpu)  {  	unsigned int cpu = smp_processor_id();  	unsigned int status_set = ST0_CU0; @@ -1770,7 +1809,7 @@ void __cpuinit per_cpu_trap_init(bool is_boot_cpu)  }  /* Install CPU exception handler */ -void __cpuinit set_handler(unsigned long offset, void *addr, unsigned long size) +void set_handler(unsigned long offset, void *addr, unsigned long size)  {  #ifdef CONFIG_CPU_MICROMIPS  	memcpy((void *)(ebase + offset), ((unsigned char *)addr - 1), size); @@ -1780,7 +1819,7 @@ void __cpuinit set_handler(unsigned long offset, void *addr, unsigned long size)  	local_flush_icache_range(ebase + offset, ebase + offset + size);  } -static char panic_null_cerr[] __cpuinitdata = +static char panic_null_cerr[] =  	"Trying to set NULL cache error exception handler";  /* @@ -1788,7 +1827,7 @@ static char panic_null_cerr[] __cpuinitdata =   * This is suitable only for the cache error exception which is the only   * exception handler that is being run uncached.   */ -void __cpuinit set_uncached_handler(unsigned long offset, void *addr, +void set_uncached_handler(unsigned long offset, void *addr,  	unsigned long size)  {  	unsigned long uncached_ebase = CKSEG1ADDR(ebase); @@ -1837,6 +1876,15 @@ void __init trap_init(void)  			ebase += (read_c0_ebase() & 0x3ffff000);  	} +	if (cpu_has_mmips) { +		unsigned int config3 = read_c0_config3(); + +		if (IS_ENABLED(CONFIG_CPU_MICROMIPS)) +			write_c0_config3(config3 | MIPS_CONF3_ISA_OE); +		else +			write_c0_config3(config3 & ~MIPS_CONF3_ISA_OE); +	} +  	if (board_ebase_setup)  		board_ebase_setup();  	per_cpu_trap_init(true); @@ -1956,7 +2004,6 @@ void __init trap_init(void)  		set_handler(0x080, &except_vec3_generic, 0x80);  	local_flush_icache_range(ebase, ebase + 0x400); -	flush_tlb_handlers();  	sort_extable(__start___dbe_table, __stop___dbe_table); diff --git a/arch/mips/kernel/unaligned.c b/arch/mips/kernel/unaligned.c index 203d8857070d..c369a5d35527 100644 --- a/arch/mips/kernel/unaligned.c +++ b/arch/mips/kernel/unaligned.c @@ -72,6 +72,7 @@   *	 A store crossing a page boundary might be executed only partially.   *	 Undo the partial store in this case.   */ +#include <linux/context_tracking.h>  #include <linux/mm.h>  #include <linux/signal.h>  #include <linux/smp.h> @@ -684,7 +685,8 @@ const int reg16to32[] = { 16, 17, 2, 3, 4, 5, 6, 7 };  /* Recode table from 16-bit STORE register notation to 32-bit GPR. */  const int reg16to32st[] = { 0, 17, 2, 3, 4, 5, 6, 7 }; -void emulate_load_store_microMIPS(struct pt_regs *regs, void __user * addr) +static void emulate_load_store_microMIPS(struct pt_regs *regs, +					 void __user *addr)  {  	unsigned long value;  	unsigned int res; @@ -1548,11 +1550,14 @@ sigill:  	    ("Unhandled kernel unaligned access or invalid instruction", regs);  	force_sig(SIGILL, current);  } +  asmlinkage void do_ade(struct pt_regs *regs)  { +	enum ctx_state prev_state;  	unsigned int __user *pc;  	mm_segment_t seg; +	prev_state = exception_enter();  	perf_sw_event(PERF_COUNT_SW_ALIGNMENT_FAULTS,  			1, regs, regs->cp0_badvaddr);  	/* @@ -1628,6 +1633,7 @@ sigbus:  	/*  	 * XXX On return from the signal handler we should advance the epc  	 */ +	exception_exit(prev_state);  }  #ifdef CONFIG_DEBUG_FS diff --git a/arch/mips/kernel/watch.c b/arch/mips/kernel/watch.c index 7726f6157d9e..2a03abb5bd2c 100644 --- a/arch/mips/kernel/watch.c +++ b/arch/mips/kernel/watch.c @@ -100,7 +100,7 @@ void mips_clear_watch_registers(void)  	}  } -__cpuinit void mips_probe_watch_registers(struct cpuinfo_mips *c) +void mips_probe_watch_registers(struct cpuinfo_mips *c)  {  	unsigned int t; @@ -111,6 +111,7 @@ __cpuinit void mips_probe_watch_registers(struct cpuinfo_mips *c)  	 * disable the register.  	 */  	write_c0_watchlo0(7); +	back_to_back_c0_hazard();  	t = read_c0_watchlo0();  	write_c0_watchlo0(0);  	c->watch_reg_masks[0] = t & 7; @@ -121,12 +122,14 @@ __cpuinit void mips_probe_watch_registers(struct cpuinfo_mips *c)  	c->watch_reg_use_cnt = 1;  	t = read_c0_watchhi0();  	write_c0_watchhi0(t | 0xff8); +	back_to_back_c0_hazard();  	t = read_c0_watchhi0();  	c->watch_reg_masks[0] |= (t & 0xff8);  	if ((t & 0x80000000) == 0)  		return;  	write_c0_watchlo1(7); +	back_to_back_c0_hazard();  	t = read_c0_watchlo1();  	write_c0_watchlo1(0);  	c->watch_reg_masks[1] = t & 7; @@ -135,12 +138,14 @@ __cpuinit void mips_probe_watch_registers(struct cpuinfo_mips *c)  	c->watch_reg_use_cnt = 2;  	t = read_c0_watchhi1();  	write_c0_watchhi1(t | 0xff8); +	back_to_back_c0_hazard();  	t = read_c0_watchhi1();  	c->watch_reg_masks[1] |= (t & 0xff8);  	if ((t & 0x80000000) == 0)  		return;  	write_c0_watchlo2(7); +	back_to_back_c0_hazard();  	t = read_c0_watchlo2();  	write_c0_watchlo2(0);  	c->watch_reg_masks[2] = t & 7; @@ -149,12 +154,14 @@ __cpuinit void mips_probe_watch_registers(struct cpuinfo_mips *c)  	c->watch_reg_use_cnt = 3;  	t = read_c0_watchhi2();  	write_c0_watchhi2(t | 0xff8); +	back_to_back_c0_hazard();  	t = read_c0_watchhi2();  	c->watch_reg_masks[2] |= (t & 0xff8);  	if ((t & 0x80000000) == 0)  		return;  	write_c0_watchlo3(7); +	back_to_back_c0_hazard();  	t = read_c0_watchlo3();  	write_c0_watchlo3(0);  	c->watch_reg_masks[3] = t & 7; @@ -163,6 +170,7 @@ __cpuinit void mips_probe_watch_registers(struct cpuinfo_mips *c)  	c->watch_reg_use_cnt = 4;  	t = read_c0_watchhi3();  	write_c0_watchhi3(t | 0xff8); +	back_to_back_c0_hazard();  	t = read_c0_watchhi3();  	c->watch_reg_masks[3] |= (t & 0xff8);  	if ((t & 0x80000000) == 0) |