summaryrefslogtreecommitdiff
path: root/arch/x86/kernel/static_call.c
blob: 5ff2b639a1a637a0db9fb85ee9dd7616b47b6889 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
// SPDX-License-Identifier: GPL-2.0
#include <linux/static_call.h>
#include <linux/memory.h>
#include <linux/bug.h>
#include <asm/text-patching.h>

static void __static_call_transform(void *insn, u8 opcode, void *func)
{
	const void *code = text_gen_insn(opcode, insn, func);

	if (WARN_ONCE(*(u8 *)insn != opcode,
		      "unexpected static call insn opcode 0x%x at %pS\n",
		      opcode, insn))
		return;

	if (memcmp(insn, code, CALL_INSN_SIZE) == 0)
		return;

	text_poke_bp(insn, code, CALL_INSN_SIZE, NULL);
}

void arch_static_call_transform(void *site, void *tramp, void *func)
{
	mutex_lock(&text_mutex);

	if (tramp)
		__static_call_transform(tramp, JMP32_INSN_OPCODE, func);

	if (IS_ENABLED(CONFIG_HAVE_STATIC_CALL_INLINE) && site)
		__static_call_transform(site, CALL_INSN_OPCODE, func);

	mutex_unlock(&text_mutex);
}
EXPORT_SYMBOL_GPL(arch_static_call_transform);