1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
2b21d55e9SRabin Vincent #include <linux/kernel.h>
3ab0615e2SRabin Vincent #include <linux/spinlock.h>
4b21d55e9SRabin Vincent #include <linux/kprobes.h>
5ab0615e2SRabin Vincent #include <linux/mm.h>
6b21d55e9SRabin Vincent #include <linux/stop_machine.h>
7b21d55e9SRabin Vincent
8b21d55e9SRabin Vincent #include <asm/cacheflush.h>
9ab0615e2SRabin Vincent #include <asm/fixmap.h>
10b21d55e9SRabin Vincent #include <asm/smp_plat.h>
11b21d55e9SRabin Vincent #include <asm/opcodes.h>
12fca08f32SWang Nan #include <asm/patch.h>
13b21d55e9SRabin Vincent
14b21d55e9SRabin Vincent struct patch {
15b21d55e9SRabin Vincent void *addr;
16b21d55e9SRabin Vincent unsigned int insn;
17b21d55e9SRabin Vincent };
18b21d55e9SRabin Vincent
19*7a7a8f54SPeter Zijlstra #ifdef CONFIG_MMU
20143c2a89SYang Shi static DEFINE_RAW_SPINLOCK(patch_lock);
21ab0615e2SRabin Vincent
patch_map(void * addr,int fixmap,unsigned long * flags)22ab0615e2SRabin Vincent static void __kprobes *patch_map(void *addr, int fixmap, unsigned long *flags)
23ab0615e2SRabin Vincent {
24ab0615e2SRabin Vincent unsigned int uintaddr = (uintptr_t) addr;
25ab0615e2SRabin Vincent bool module = !core_kernel_text(uintaddr);
26ab0615e2SRabin Vincent struct page *page;
27ab0615e2SRabin Vincent
280f5bf6d0SLaura Abbott if (module && IS_ENABLED(CONFIG_STRICT_MODULE_RWX))
29ab0615e2SRabin Vincent page = vmalloc_to_page(addr);
300f5bf6d0SLaura Abbott else if (!module && IS_ENABLED(CONFIG_STRICT_KERNEL_RWX))
31ab0615e2SRabin Vincent page = virt_to_page(addr);
32ab0615e2SRabin Vincent else
33ab0615e2SRabin Vincent return addr;
34ab0615e2SRabin Vincent
35ab0615e2SRabin Vincent if (flags)
36143c2a89SYang Shi raw_spin_lock_irqsave(&patch_lock, *flags);
37ab0615e2SRabin Vincent
38ab0615e2SRabin Vincent set_fixmap(fixmap, page_to_phys(page));
39ab0615e2SRabin Vincent
40ab0615e2SRabin Vincent return (void *) (__fix_to_virt(fixmap) + (uintaddr & ~PAGE_MASK));
41ab0615e2SRabin Vincent }
42ab0615e2SRabin Vincent
patch_unmap(int fixmap,unsigned long * flags)43ab0615e2SRabin Vincent static void __kprobes patch_unmap(int fixmap, unsigned long *flags)
44ab0615e2SRabin Vincent {
45ab0615e2SRabin Vincent clear_fixmap(fixmap);
46ab0615e2SRabin Vincent
47ab0615e2SRabin Vincent if (flags)
48143c2a89SYang Shi raw_spin_unlock_irqrestore(&patch_lock, *flags);
49ab0615e2SRabin Vincent }
50*7a7a8f54SPeter Zijlstra #else
patch_map(void * addr,int fixmap,unsigned long * flags)51*7a7a8f54SPeter Zijlstra static void __kprobes *patch_map(void *addr, int fixmap, unsigned long *flags)
52*7a7a8f54SPeter Zijlstra {
53*7a7a8f54SPeter Zijlstra return addr;
54*7a7a8f54SPeter Zijlstra }
patch_unmap(int fixmap,unsigned long * flags)55*7a7a8f54SPeter Zijlstra static void __kprobes patch_unmap(int fixmap, unsigned long *flags) { }
56*7a7a8f54SPeter Zijlstra #endif
57ab0615e2SRabin Vincent
__patch_text_real(void * addr,unsigned int insn,bool remap)58ab0615e2SRabin Vincent void __kprobes __patch_text_real(void *addr, unsigned int insn, bool remap)
59b21d55e9SRabin Vincent {
60b21d55e9SRabin Vincent bool thumb2 = IS_ENABLED(CONFIG_THUMB2_KERNEL);
61ab0615e2SRabin Vincent unsigned int uintaddr = (uintptr_t) addr;
62ab0615e2SRabin Vincent bool twopage = false;
63ab0615e2SRabin Vincent unsigned long flags;
64ab0615e2SRabin Vincent void *waddr = addr;
65b21d55e9SRabin Vincent int size;
66b21d55e9SRabin Vincent
67ab0615e2SRabin Vincent if (remap)
68ab0615e2SRabin Vincent waddr = patch_map(addr, FIX_TEXT_POKE0, &flags);
69ab0615e2SRabin Vincent
70b21d55e9SRabin Vincent if (thumb2 && __opcode_is_thumb16(insn)) {
71ab0615e2SRabin Vincent *(u16 *)waddr = __opcode_to_mem_thumb16(insn);
72b21d55e9SRabin Vincent size = sizeof(u16);
73ab0615e2SRabin Vincent } else if (thumb2 && (uintaddr & 2)) {
74b21d55e9SRabin Vincent u16 first = __opcode_thumb32_first(insn);
75b21d55e9SRabin Vincent u16 second = __opcode_thumb32_second(insn);
76ab0615e2SRabin Vincent u16 *addrh0 = waddr;
77ab0615e2SRabin Vincent u16 *addrh1 = waddr + 2;
78b21d55e9SRabin Vincent
79ab0615e2SRabin Vincent twopage = (uintaddr & ~PAGE_MASK) == PAGE_SIZE - 2;
80ab0615e2SRabin Vincent if (twopage && remap)
81ab0615e2SRabin Vincent addrh1 = patch_map(addr + 2, FIX_TEXT_POKE1, NULL);
82ab0615e2SRabin Vincent
83ab0615e2SRabin Vincent *addrh0 = __opcode_to_mem_thumb16(first);
84ab0615e2SRabin Vincent *addrh1 = __opcode_to_mem_thumb16(second);
85ab0615e2SRabin Vincent
86ab0615e2SRabin Vincent if (twopage && addrh1 != addr + 2) {
87ab0615e2SRabin Vincent flush_kernel_vmap_range(addrh1, 2);
88ab0615e2SRabin Vincent patch_unmap(FIX_TEXT_POKE1, NULL);
89ab0615e2SRabin Vincent }
90b21d55e9SRabin Vincent
91b21d55e9SRabin Vincent size = sizeof(u32);
92b21d55e9SRabin Vincent } else {
93b21d55e9SRabin Vincent if (thumb2)
94b21d55e9SRabin Vincent insn = __opcode_to_mem_thumb32(insn);
95b21d55e9SRabin Vincent else
96b21d55e9SRabin Vincent insn = __opcode_to_mem_arm(insn);
97b21d55e9SRabin Vincent
98ab0615e2SRabin Vincent *(u32 *)waddr = insn;
99b21d55e9SRabin Vincent size = sizeof(u32);
100b21d55e9SRabin Vincent }
101b21d55e9SRabin Vincent
102ab0615e2SRabin Vincent if (waddr != addr) {
103ab0615e2SRabin Vincent flush_kernel_vmap_range(waddr, twopage ? size / 2 : size);
104ab0615e2SRabin Vincent patch_unmap(FIX_TEXT_POKE0, &flags);
105*7a7a8f54SPeter Zijlstra }
106ab0615e2SRabin Vincent
107b21d55e9SRabin Vincent flush_icache_range((uintptr_t)(addr),
108b21d55e9SRabin Vincent (uintptr_t)(addr) + size);
109b21d55e9SRabin Vincent }
110b21d55e9SRabin Vincent
patch_text_stop_machine(void * data)111b21d55e9SRabin Vincent static int __kprobes patch_text_stop_machine(void *data)
112b21d55e9SRabin Vincent {
113b21d55e9SRabin Vincent struct patch *patch = data;
114b21d55e9SRabin Vincent
115b21d55e9SRabin Vincent __patch_text(patch->addr, patch->insn);
116b21d55e9SRabin Vincent
117b21d55e9SRabin Vincent return 0;
118b21d55e9SRabin Vincent }
119b21d55e9SRabin Vincent
patch_text(void * addr,unsigned int insn)120b21d55e9SRabin Vincent void __kprobes patch_text(void *addr, unsigned int insn)
121b21d55e9SRabin Vincent {
122b21d55e9SRabin Vincent struct patch patch = {
123b21d55e9SRabin Vincent .addr = addr,
124b21d55e9SRabin Vincent .insn = insn,
125b21d55e9SRabin Vincent };
126b21d55e9SRabin Vincent
1279489cc8fSThomas Gleixner stop_machine_cpuslocked(patch_text_stop_machine, &patch, NULL);
128b21d55e9SRabin Vincent }
129