-
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.
parisc: Add static branch and JUMP_LABEL feature
Signed-off-by: Helge Deller <deller@gmx.de>
- Loading branch information
Helge Deller
committed
May 5, 2019
1 parent
bdca5d6
commit 62217be
Showing
5 changed files
with
104 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,43 @@ | ||
/* SPDX-License-Identifier: GPL-2.0 */ | ||
#ifndef _ASM_PARISC_JUMP_LABEL_H | ||
#define _ASM_PARISC_JUMP_LABEL_H | ||
|
||
#ifndef __ASSEMBLY__ | ||
|
||
#include <linux/types.h> | ||
#include <asm/assembly.h> | ||
|
||
#define JUMP_LABEL_NOP_SIZE 4 | ||
|
||
static __always_inline bool arch_static_branch(struct static_key *key, bool branch) | ||
{ | ||
asm_volatile_goto("1:\n\t" | ||
"nop\n\t" | ||
".pushsection __jump_table, \"aw\"\n\t" | ||
".word 1b - ., %l[l_yes] - .\n\t" | ||
__stringify(ASM_ULONG_INSN) " %c0 - .\n\t" | ||
".popsection\n\t" | ||
: : "i" (&((char *)key)[branch]) : : l_yes); | ||
|
||
return false; | ||
l_yes: | ||
return true; | ||
} | ||
|
||
static __always_inline bool arch_static_branch_jump(struct static_key *key, bool branch) | ||
{ | ||
asm_volatile_goto("1:\n\t" | ||
"b,n %l[l_yes]\n\t" | ||
".pushsection __jump_table, \"aw\"\n\t" | ||
".word 1b - ., %l[l_yes] - .\n\t" | ||
__stringify(ASM_ULONG_INSN) " %c0 - .\n\t" | ||
".popsection\n\t" | ||
: : "i" (&((char *)key)[branch]) : : l_yes); | ||
|
||
return false; | ||
l_yes: | ||
return true; | ||
} | ||
|
||
#endif /* __ASSEMBLY__ */ | ||
#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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
// SPDX-License-Identifier: GPL-2.0+ | ||
/* | ||
* Copyright (C) 2019 Helge Deller <deller@gmx.de> | ||
* | ||
* Based on arch/arm64/kernel/jump_label.c | ||
*/ | ||
#include <linux/kernel.h> | ||
#include <linux/jump_label.h> | ||
#include <linux/bug.h> | ||
#include <asm/alternative.h> | ||
#include <asm/patch.h> | ||
|
||
static inline int reassemble_17(int as17) | ||
{ | ||
return (((as17 & 0x10000) >> 16) | | ||
((as17 & 0x0f800) << 5) | | ||
((as17 & 0x00400) >> 8) | | ||
((as17 & 0x003ff) << 3)); | ||
} | ||
|
||
void arch_jump_label_transform(struct jump_entry *entry, | ||
enum jump_label_type type) | ||
{ | ||
void *addr = (void *)jump_entry_code(entry); | ||
u32 insn; | ||
|
||
if (type == JUMP_LABEL_JMP) { | ||
void *target = (void *)jump_entry_target(entry); | ||
int distance = target - addr; | ||
/* | ||
* Encode the PA1.1 "b,n" instruction with a 17-bit | ||
* displacement. In case we hit the BUG(), we could use | ||
* another branch instruction with a 22-bit displacement on | ||
* 64-bit CPUs instead. But this seems sufficient for now. | ||
*/ | ||
distance -= 8; | ||
BUG_ON(distance > 262143 || distance < -262144); | ||
insn = 0xe8000002 | reassemble_17(distance >> 2); | ||
} else { | ||
insn = INSN_NOP; | ||
} | ||
|
||
patch_text(addr, insn); | ||
} | ||
|
||
void arch_jump_label_transform_static(struct jump_entry *entry, | ||
enum jump_label_type type) | ||
{ | ||
/* | ||
* We use the architected NOP in arch_static_branch, so there's no | ||
* need to patch an identical NOP over the top of it here. The core | ||
* will call arch_jump_label_transform from a module notifier if the | ||
* NOP needs to be replaced by a branch. | ||
*/ | ||
} |
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