xref: /openbmc/linux/arch/arm/kernel/patch.c (revision 7a7a8f54)
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 
197a7a8f54SPeter 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 }
507a7a8f54SPeter Zijlstra #else
patch_map(void * addr,int fixmap,unsigned long * flags)517a7a8f54SPeter Zijlstra static void __kprobes *patch_map(void *addr, int fixmap, unsigned long *flags)
527a7a8f54SPeter Zijlstra {
537a7a8f54SPeter Zijlstra 	return addr;
547a7a8f54SPeter Zijlstra }
patch_unmap(int fixmap,unsigned long * flags)557a7a8f54SPeter Zijlstra static void __kprobes patch_unmap(int fixmap, unsigned long *flags) { }
567a7a8f54SPeter 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);
1057a7a8f54SPeter 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