diff options
| author | Ingo Molnar <mingo@elte.hu> | 2010-09-24 09:12:05 +0200 | 
|---|---|---|
| committer | Ingo Molnar <mingo@elte.hu> | 2010-09-24 09:12:05 +0200 | 
| commit | a5a2bad55de89a0adf7d6f783cb87ab7eb1a894f (patch) | |
| tree | 452cb8addc26f5c79fdd4e2fea92c78dc17fc8e8 /arch/sparc | |
| parent | d0303d71c2fb9bcb90a8d48e6462c78c86f70ce6 (diff) | |
| parent | 46eb3b64dddd20f44e76b08676fa642dd374bf1d (diff) | |
| download | kernel_samsung_smdk4412-a5a2bad55de89a0adf7d6f783cb87ab7eb1a894f.zip kernel_samsung_smdk4412-a5a2bad55de89a0adf7d6f783cb87ab7eb1a894f.tar.gz kernel_samsung_smdk4412-a5a2bad55de89a0adf7d6f783cb87ab7eb1a894f.tar.bz2 | |
Merge branch 'tip/perf/core' of git://git.kernel.org/pub/scm/linux/kernel/git/rostedt/linux-2.6-trace into perf/core
Diffstat (limited to 'arch/sparc')
| -rw-r--r-- | arch/sparc/Kconfig | 1 | ||||
| -rw-r--r-- | arch/sparc/include/asm/jump_label.h | 32 | ||||
| -rw-r--r-- | arch/sparc/kernel/Makefile | 2 | ||||
| -rw-r--r-- | arch/sparc/kernel/jump_label.c | 47 | ||||
| -rw-r--r-- | arch/sparc/kernel/module.c | 6 | 
5 files changed, 88 insertions, 0 deletions
| diff --git a/arch/sparc/Kconfig b/arch/sparc/Kconfig index 491e9d6..9212cd4 100644 --- a/arch/sparc/Kconfig +++ b/arch/sparc/Kconfig @@ -30,6 +30,7 @@ config SPARC  	select PERF_USE_VMALLOC  	select HAVE_DMA_ATTRS  	select HAVE_DMA_API_DEBUG +	select HAVE_ARCH_JUMP_LABEL  config SPARC32  	def_bool !64BIT diff --git a/arch/sparc/include/asm/jump_label.h b/arch/sparc/include/asm/jump_label.h new file mode 100644 index 0000000..62e66d7 --- /dev/null +++ b/arch/sparc/include/asm/jump_label.h @@ -0,0 +1,32 @@ +#ifndef _ASM_SPARC_JUMP_LABEL_H +#define _ASM_SPARC_JUMP_LABEL_H + +#ifdef __KERNEL__ + +#include <linux/types.h> +#include <asm/system.h> + +#define JUMP_LABEL_NOP_SIZE 4 + +#define JUMP_LABEL(key, label)					\ +	do {							\ +		asm goto("1:\n\t"				\ +			 "nop\n\t"				\ +			 "nop\n\t"				\ +			 ".pushsection __jump_table,  \"a\"\n\t"\ +			 ".word 1b, %l[" #label "], %c0\n\t"	\ +			 ".popsection \n\t"			\ +			 : :  "i" (key) :  : label);\ +	} while (0) + +#endif /* __KERNEL__ */ + +typedef u32 jump_label_t; + +struct jump_entry { +	jump_label_t code; +	jump_label_t target; +	jump_label_t key; +}; + +#endif diff --git a/arch/sparc/kernel/Makefile b/arch/sparc/kernel/Makefile index 0c2dc1f..599398f 100644 --- a/arch/sparc/kernel/Makefile +++ b/arch/sparc/kernel/Makefile @@ -119,3 +119,5 @@ obj-$(CONFIG_COMPAT)    += $(audit--y)  pc--$(CONFIG_PERF_EVENTS) := perf_event.o  obj-$(CONFIG_SPARC64)	+= $(pc--y) + +obj-$(CONFIG_SPARC64)	+= jump_label.o diff --git a/arch/sparc/kernel/jump_label.c b/arch/sparc/kernel/jump_label.c new file mode 100644 index 0000000..ea2dafc --- /dev/null +++ b/arch/sparc/kernel/jump_label.c @@ -0,0 +1,47 @@ +#include <linux/kernel.h> +#include <linux/types.h> +#include <linux/mutex.h> +#include <linux/cpu.h> + +#include <linux/jump_label.h> +#include <linux/memory.h> + +#ifdef HAVE_JUMP_LABEL + +void arch_jump_label_transform(struct jump_entry *entry, +			       enum jump_label_type type) +{ +	u32 val; +	u32 *insn = (u32 *) (unsigned long) entry->code; + +	if (type == JUMP_LABEL_ENABLE) { +		s32 off = (s32)entry->target - (s32)entry->code; + +#ifdef CONFIG_SPARC64 +		/* ba,pt %xcc, . + (off << 2) */ +		val = 0x10680000 | ((u32) off >> 2); +#else +		/* ba . + (off << 2) */ +		val = 0x10800000 | ((u32) off >> 2); +#endif +	} else { +		val = 0x01000000; +	} + +	get_online_cpus(); +	mutex_lock(&text_mutex); +	*insn = val; +	flushi(insn); +	mutex_unlock(&text_mutex); +	put_online_cpus(); +} + +void arch_jump_label_text_poke_early(jump_label_t addr) +{ +	u32 *insn_p = (u32 *) (unsigned long) addr; + +	*insn_p = 0x01000000; +	flushi(insn_p); +} + +#endif diff --git a/arch/sparc/kernel/module.c b/arch/sparc/kernel/module.c index f848aad..ee3c7dd 100644 --- a/arch/sparc/kernel/module.c +++ b/arch/sparc/kernel/module.c @@ -18,6 +18,9 @@  #include <asm/spitfire.h>  #ifdef CONFIG_SPARC64 + +#include <linux/jump_label.h> +  static void *module_map(unsigned long size)  {  	struct vm_struct *area; @@ -227,6 +230,9 @@ int module_finalize(const Elf_Ehdr *hdr,  		    const Elf_Shdr *sechdrs,  		    struct module *me)  { +	/* make jump label nops */ +	jump_label_apply_nops(me); +  	/* Cheetah's I-cache is fully coherent.  */  	if (tlb_type == spitfire) {  		unsigned long va; | 
