mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-06 13:16:22 +00:00
7e6b9db27d
Instead of defaulting to patching NOP opcodes at init time, and leaving it to the architectures to override this if this is not needed, switch to a model where doing nothing is the default. This is the common case by far, as only MIPS requires NOP patching at init time. On all other architectures, the correct encodings are emitted by the compiler and so no initial patching is needed. Signed-off-by: Ard Biesheuvel <ardb@kernel.org> Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org> Acked-by: Mark Rutland <mark.rutland@arm.com> Acked-by: Peter Zijlstra (Intel) <peterz@infradead.org> Link: https://lore.kernel.org/r/20220615154142.1574619-4-ardb@kernel.org
45 lines
1.1 KiB
C
45 lines
1.1 KiB
C
// 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);
|
|
}
|