diff options
Diffstat (limited to 'arch/sparc/kernel')
| -rw-r--r-- | arch/sparc/kernel/head_64.S | 37 | ||||
| -rw-r--r-- | arch/sparc/kernel/jump_label.c | 23 | ||||
| -rw-r--r-- | arch/sparc/kernel/mdesc.c | 46 | ||||
| -rw-r--r-- | arch/sparc/kernel/smp_64.c | 8 | 
4 files changed, 53 insertions, 61 deletions
| diff --git a/arch/sparc/kernel/head_64.S b/arch/sparc/kernel/head_64.S index beba6c11554c..6aa3da152c20 100644 --- a/arch/sparc/kernel/head_64.S +++ b/arch/sparc/kernel/head_64.S @@ -926,48 +926,11 @@ tlb_type:	.word	0	/* Must NOT end up in BSS */  EXPORT_SYMBOL(tlb_type)  	.section	".fixup",#alloc,#execinstr -	.globl	__ret_efault, __retl_efault, __ret_one, __retl_one -ENTRY(__ret_efault) -	ret -	 restore %g0, -EFAULT, %o0 -ENDPROC(__ret_efault) -EXPORT_SYMBOL(__ret_efault) -  ENTRY(__retl_efault)  	retl  	 mov	-EFAULT, %o0  ENDPROC(__retl_efault) -ENTRY(__retl_one) -	retl -	 mov	1, %o0 -ENDPROC(__retl_one) - -ENTRY(__retl_one_fp) -	VISExitHalf -	retl -	 mov	1, %o0 -ENDPROC(__retl_one_fp) - -ENTRY(__ret_one_asi) -	wr	%g0, ASI_AIUS, %asi -	ret -	 restore %g0, 1, %o0 -ENDPROC(__ret_one_asi) - -ENTRY(__retl_one_asi) -	wr	%g0, ASI_AIUS, %asi -	retl -	 mov	1, %o0 -ENDPROC(__retl_one_asi) - -ENTRY(__retl_one_asi_fp) -	wr	%g0, ASI_AIUS, %asi -	VISExitHalf -	retl -	 mov	1, %o0 -ENDPROC(__retl_one_asi_fp) -  ENTRY(__retl_o1)  	retl  	 mov	%o1, %o0 diff --git a/arch/sparc/kernel/jump_label.c b/arch/sparc/kernel/jump_label.c index 59bbeff55024..07933b9e9ce0 100644 --- a/arch/sparc/kernel/jump_label.c +++ b/arch/sparc/kernel/jump_label.c @@ -13,19 +13,30 @@  void arch_jump_label_transform(struct jump_entry *entry,  			       enum jump_label_type type)  { -	u32 val;  	u32 *insn = (u32 *) (unsigned long) entry->code; +	u32 val;  	if (type == JUMP_LABEL_JMP) {  		s32 off = (s32)entry->target - (s32)entry->code; +		bool use_v9_branch = false; + +		BUG_ON(off & 3);  #ifdef CONFIG_SPARC64 -		/* ba,pt %xcc, . + (off << 2) */ -		val = 0x10680000 | ((u32) off >> 2); -#else -		/* ba . + (off << 2) */ -		val = 0x10800000 | ((u32) off >> 2); +		if (off <= 0xfffff && off >= -0x100000) +			use_v9_branch = true;  #endif +		if (use_v9_branch) { +			/* WDISP19 - target is . + immed << 2 */ +			/* ba,pt %xcc, . + off */ +			val = 0x10680000 | (((u32) off >> 2) & 0x7ffff); +		} else { +			/* WDISP22 - target is . + immed << 2 */ +			BUG_ON(off > 0x7fffff); +			BUG_ON(off < -0x800000); +			/* ba . + off */ +			val = 0x10800000 | (((u32) off >> 2) & 0x3fffff); +		}  	} else {  		val = 0x01000000;  	} diff --git a/arch/sparc/kernel/mdesc.c b/arch/sparc/kernel/mdesc.c index 11228861d9b4..8a6982dfd733 100644 --- a/arch/sparc/kernel/mdesc.c +++ b/arch/sparc/kernel/mdesc.c @@ -645,13 +645,20 @@ static void __mark_core_id(struct mdesc_handle *hp, u64 node,  		cpu_data(*id).core_id = core_id;  } -static void __mark_sock_id(struct mdesc_handle *hp, u64 node, -			   int sock_id) +static void __mark_max_cache_id(struct mdesc_handle *hp, u64 node, +				int max_cache_id)  {  	const u64 *id = mdesc_get_property(hp, node, "id", NULL); -	if (*id < num_possible_cpus()) -		cpu_data(*id).sock_id = sock_id; +	if (*id < num_possible_cpus()) { +		cpu_data(*id).max_cache_id = max_cache_id; + +		/** +		 * On systems without explicit socket descriptions socket +		 * is max_cache_id +		 */ +		cpu_data(*id).sock_id = max_cache_id; +	}  }  static void mark_core_ids(struct mdesc_handle *hp, u64 mp, @@ -660,10 +667,11 @@ static void mark_core_ids(struct mdesc_handle *hp, u64 mp,  	find_back_node_value(hp, mp, "cpu", __mark_core_id, core_id, 10);  } -static void mark_sock_ids(struct mdesc_handle *hp, u64 mp, -			  int sock_id) +static void mark_max_cache_ids(struct mdesc_handle *hp, u64 mp, +			       int max_cache_id)  { -	find_back_node_value(hp, mp, "cpu", __mark_sock_id, sock_id, 10); +	find_back_node_value(hp, mp, "cpu", __mark_max_cache_id, +			     max_cache_id, 10);  }  static void set_core_ids(struct mdesc_handle *hp) @@ -694,14 +702,15 @@ static void set_core_ids(struct mdesc_handle *hp)  	}  } -static int set_sock_ids_by_cache(struct mdesc_handle *hp, int level) +static int set_max_cache_ids_by_cache(struct mdesc_handle *hp, int level)  {  	u64 mp;  	int idx = 1;  	int fnd = 0; -	/* Identify unique sockets by looking for cpus backpointed to by -	 * shared level n caches. +	/** +	 * Identify unique highest level of shared cache by looking for cpus +	 * backpointed to by shared level N caches.  	 */  	mdesc_for_each_node_by_name(hp, mp, "cache") {  		const u64 *cur_lvl; @@ -709,8 +718,7 @@ static int set_sock_ids_by_cache(struct mdesc_handle *hp, int level)  		cur_lvl = mdesc_get_property(hp, mp, "level", NULL);  		if (*cur_lvl != level)  			continue; - -		mark_sock_ids(hp, mp, idx); +		mark_max_cache_ids(hp, mp, idx);  		idx++;  		fnd = 1;  	} @@ -745,15 +753,17 @@ static void set_sock_ids(struct mdesc_handle *hp)  {  	u64 mp; -	/* If machine description exposes sockets data use it. -	 * Otherwise fallback to use shared L3 or L2 caches. +	/** +	 * Find the highest level of shared cache which pre-T7 is also +	 * the socket.  	 */ +	if (!set_max_cache_ids_by_cache(hp, 3)) +		set_max_cache_ids_by_cache(hp, 2); + +	/* If machine description exposes sockets data use it.*/  	mp = mdesc_node_by_name(hp, MDESC_NODE_NULL, "sockets");  	if (mp != MDESC_NODE_NULL) -		return set_sock_ids_by_socket(hp, mp); - -	if (!set_sock_ids_by_cache(hp, 3)) -		set_sock_ids_by_cache(hp, 2); +		set_sock_ids_by_socket(hp, mp);  }  static void mark_proc_ids(struct mdesc_handle *hp, u64 mp, int proc_id) diff --git a/arch/sparc/kernel/smp_64.c b/arch/sparc/kernel/smp_64.c index d3035ba6cd31..8182f7caf5b1 100644 --- a/arch/sparc/kernel/smp_64.c +++ b/arch/sparc/kernel/smp_64.c @@ -63,9 +63,13 @@ cpumask_t cpu_core_map[NR_CPUS] __read_mostly =  cpumask_t cpu_core_sib_map[NR_CPUS] __read_mostly = {  	[0 ... NR_CPUS-1] = CPU_MASK_NONE }; +cpumask_t cpu_core_sib_cache_map[NR_CPUS] __read_mostly = { +	[0 ... NR_CPUS - 1] = CPU_MASK_NONE }; +  EXPORT_PER_CPU_SYMBOL(cpu_sibling_map);  EXPORT_SYMBOL(cpu_core_map);  EXPORT_SYMBOL(cpu_core_sib_map); +EXPORT_SYMBOL(cpu_core_sib_cache_map);  static cpumask_t smp_commenced_mask; @@ -1265,6 +1269,10 @@ void smp_fill_in_sib_core_maps(void)  		unsigned int j;  		for_each_present_cpu(j)  { +			if (cpu_data(i).max_cache_id == +			    cpu_data(j).max_cache_id) +				cpumask_set_cpu(j, &cpu_core_sib_cache_map[i]); +  			if (cpu_data(i).sock_id == cpu_data(j).sock_id)  				cpumask_set_cpu(j, &cpu_core_sib_map[i]);  		} | 

