-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add the arch-specific code to support jump labels for ARM and Thumb-2. This code will only be activated on compilers that are capable of building it. It has been tested with GCC 4.6 patched with the patch from GCC bug 48637. Cc: Jason Baron <jbaron@redhat.com> Signed-off-by: Rabin Vincent <rabin@rab.in> Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
- Loading branch information
Rabin Vincent
authored and
Russell King
committed
Mar 24, 2012
1 parent
a9468f3
commit 09f05d8
Showing
5 changed files
with
92 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
#ifndef _ASM_ARM_JUMP_LABEL_H | ||
#define _ASM_ARM_JUMP_LABEL_H | ||
|
||
#ifdef __KERNEL__ | ||
|
||
#include <linux/types.h> | ||
#include <asm/system.h> | ||
|
||
#define JUMP_LABEL_NOP_SIZE 4 | ||
|
||
#ifdef CONFIG_THUMB2_KERNEL | ||
#define JUMP_LABEL_NOP "nop.w" | ||
#else | ||
#define JUMP_LABEL_NOP "nop" | ||
#endif | ||
|
||
static __always_inline bool arch_static_branch(struct jump_label_key *key) | ||
{ | ||
asm goto("1:\n\t" | ||
JUMP_LABEL_NOP "\n\t" | ||
".pushsection __jump_table, \"aw\"\n\t" | ||
".word 1b, %l[l_yes], %c0\n\t" | ||
".popsection\n\t" | ||
: : "i" (key) : : l_yes); | ||
|
||
return false; | ||
l_yes: | ||
return true; | ||
} | ||
|
||
#endif /* __KERNEL__ */ | ||
|
||
typedef u32 jump_label_t; | ||
|
||
struct jump_entry { | ||
jump_label_t code; | ||
jump_label_t target; | ||
jump_label_t key; | ||
}; | ||
|
||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
#include <linux/kernel.h> | ||
#include <linux/jump_label.h> | ||
|
||
#include "insn.h" | ||
#include "patch.h" | ||
|
||
#ifdef HAVE_JUMP_LABEL | ||
|
||
static void __arch_jump_label_transform(struct jump_entry *entry, | ||
enum jump_label_type type, | ||
bool is_static) | ||
{ | ||
void *addr = (void *)entry->code; | ||
unsigned int insn; | ||
|
||
if (type == JUMP_LABEL_ENABLE) | ||
insn = arm_gen_branch(entry->code, entry->target); | ||
else | ||
insn = arm_gen_nop(); | ||
|
||
if (is_static) | ||
__patch_text(addr, insn); | ||
else | ||
patch_text(addr, insn); | ||
} | ||
|
||
void arch_jump_label_transform(struct jump_entry *entry, | ||
enum jump_label_type type) | ||
{ | ||
__arch_jump_label_transform(entry, type, false); | ||
} | ||
|
||
void arch_jump_label_transform_static(struct jump_entry *entry, | ||
enum jump_label_type type) | ||
{ | ||
__arch_jump_label_transform(entry, type, true); | ||
} | ||
|
||
#endif |