diff options
Diffstat (limited to 'arch/sparc/kernel')
| -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 |
3 files changed, 55 insertions, 0 deletions
diff --git a/arch/sparc/kernel/Makefile b/arch/sparc/kernel/Makefile index 0c2dc1f24a9a..599398fbbc7c 100644 --- a/arch/sparc/kernel/Makefile +++ b/arch/sparc/kernel/Makefile | |||
| @@ -119,3 +119,5 @@ obj-$(CONFIG_COMPAT) += $(audit--y) | |||
| 119 | 119 | ||
| 120 | pc--$(CONFIG_PERF_EVENTS) := perf_event.o | 120 | pc--$(CONFIG_PERF_EVENTS) := perf_event.o |
| 121 | obj-$(CONFIG_SPARC64) += $(pc--y) | 121 | obj-$(CONFIG_SPARC64) += $(pc--y) |
| 122 | |||
| 123 | 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 000000000000..ea2dafc93d78 --- /dev/null +++ b/arch/sparc/kernel/jump_label.c | |||
| @@ -0,0 +1,47 @@ | |||
| 1 | #include <linux/kernel.h> | ||
| 2 | #include <linux/types.h> | ||
| 3 | #include <linux/mutex.h> | ||
| 4 | #include <linux/cpu.h> | ||
| 5 | |||
| 6 | #include <linux/jump_label.h> | ||
| 7 | #include <linux/memory.h> | ||
| 8 | |||
| 9 | #ifdef HAVE_JUMP_LABEL | ||
| 10 | |||
| 11 | void arch_jump_label_transform(struct jump_entry *entry, | ||
| 12 | enum jump_label_type type) | ||
| 13 | { | ||
| 14 | u32 val; | ||
| 15 | u32 *insn = (u32 *) (unsigned long) entry->code; | ||
| 16 | |||
| 17 | if (type == JUMP_LABEL_ENABLE) { | ||
| 18 | s32 off = (s32)entry->target - (s32)entry->code; | ||
| 19 | |||
| 20 | #ifdef CONFIG_SPARC64 | ||
| 21 | /* ba,pt %xcc, . + (off << 2) */ | ||
| 22 | val = 0x10680000 | ((u32) off >> 2); | ||
| 23 | #else | ||
| 24 | /* ba . + (off << 2) */ | ||
| 25 | val = 0x10800000 | ((u32) off >> 2); | ||
| 26 | #endif | ||
| 27 | } else { | ||
| 28 | val = 0x01000000; | ||
| 29 | } | ||
| 30 | |||
| 31 | get_online_cpus(); | ||
| 32 | mutex_lock(&text_mutex); | ||
| 33 | *insn = val; | ||
| 34 | flushi(insn); | ||
| 35 | mutex_unlock(&text_mutex); | ||
| 36 | put_online_cpus(); | ||
| 37 | } | ||
| 38 | |||
| 39 | void arch_jump_label_text_poke_early(jump_label_t addr) | ||
| 40 | { | ||
| 41 | u32 *insn_p = (u32 *) (unsigned long) addr; | ||
| 42 | |||
| 43 | *insn_p = 0x01000000; | ||
| 44 | flushi(insn_p); | ||
| 45 | } | ||
| 46 | |||
| 47 | #endif | ||
diff --git a/arch/sparc/kernel/module.c b/arch/sparc/kernel/module.c index f848aadf54dc..ee3c7dde8d9f 100644 --- a/arch/sparc/kernel/module.c +++ b/arch/sparc/kernel/module.c | |||
| @@ -18,6 +18,9 @@ | |||
| 18 | #include <asm/spitfire.h> | 18 | #include <asm/spitfire.h> |
| 19 | 19 | ||
| 20 | #ifdef CONFIG_SPARC64 | 20 | #ifdef CONFIG_SPARC64 |
| 21 | |||
| 22 | #include <linux/jump_label.h> | ||
| 23 | |||
| 21 | static void *module_map(unsigned long size) | 24 | static void *module_map(unsigned long size) |
| 22 | { | 25 | { |
| 23 | struct vm_struct *area; | 26 | struct vm_struct *area; |
| @@ -227,6 +230,9 @@ int module_finalize(const Elf_Ehdr *hdr, | |||
| 227 | const Elf_Shdr *sechdrs, | 230 | const Elf_Shdr *sechdrs, |
| 228 | struct module *me) | 231 | struct module *me) |
| 229 | { | 232 | { |
| 233 | /* make jump label nops */ | ||
| 234 | jump_label_apply_nops(me); | ||
| 235 | |||
| 230 | /* Cheetah's I-cache is fully coherent. */ | 236 | /* Cheetah's I-cache is fully coherent. */ |
| 231 | if (tlb_type == spitfire) { | 237 | if (tlb_type == spitfire) { |
| 232 | unsigned long va; | 238 | unsigned long va; |
