xref: /openbmc/linux/arch/arm/probes/kprobes/core.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
11802d0beSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2fca08f32SWang Nan /*
3fca08f32SWang Nan  * arch/arm/kernel/kprobes.c
4fca08f32SWang Nan  *
5fca08f32SWang Nan  * Kprobes on ARM
6fca08f32SWang Nan  *
7fca08f32SWang Nan  * Abhishek Sagar <sagar.abhishek@gmail.com>
8fca08f32SWang Nan  * Copyright (C) 2006, 2007 Motorola Inc.
9fca08f32SWang Nan  *
10fca08f32SWang Nan  * Nicolas Pitre <nico@marvell.com>
11fca08f32SWang Nan  * Copyright (C) 2007 Marvell Ltd.
12fca08f32SWang Nan  */
13fca08f32SWang Nan 
149c89bb8eSMasami Hiramatsu #define pr_fmt(fmt) "kprobes: " fmt
159c89bb8eSMasami Hiramatsu 
16fca08f32SWang Nan #include <linux/kernel.h>
17fca08f32SWang Nan #include <linux/kprobes.h>
18fca08f32SWang Nan #include <linux/module.h>
19fca08f32SWang Nan #include <linux/slab.h>
20fca08f32SWang Nan #include <linux/stop_machine.h>
21b17b0153SIngo Molnar #include <linux/sched/debug.h>
22fca08f32SWang Nan #include <linux/stringify.h>
23fca08f32SWang Nan #include <asm/traps.h>
24fca08f32SWang Nan #include <asm/opcodes.h>
25fca08f32SWang Nan #include <asm/cacheflush.h>
26fca08f32SWang Nan #include <linux/percpu.h>
27fca08f32SWang Nan #include <linux/bug.h>
28fca08f32SWang Nan #include <asm/patch.h>
29c6089061SRussell King #include <asm/sections.h>
30fca08f32SWang Nan 
31fca08f32SWang Nan #include "../decode-arm.h"
32fca08f32SWang Nan #include "../decode-thumb.h"
33fca08f32SWang Nan #include "core.h"
34fca08f32SWang Nan 
35fca08f32SWang Nan #define MIN_STACK_SIZE(addr) 				\
36fca08f32SWang Nan 	min((unsigned long)MAX_STACK_SIZE,		\
37fca08f32SWang Nan 	    (unsigned long)current_thread_info() + THREAD_START_SP - (addr))
38fca08f32SWang Nan 
39fca08f32SWang Nan #define flush_insns(addr, size)				\
40fca08f32SWang Nan 	flush_icache_range((unsigned long)(addr),	\
41fca08f32SWang Nan 			   (unsigned long)(addr) +	\
42fca08f32SWang Nan 			   (size))
43fca08f32SWang Nan 
44fca08f32SWang Nan DEFINE_PER_CPU(struct kprobe *, current_kprobe) = NULL;
45fca08f32SWang Nan DEFINE_PER_CPU(struct kprobe_ctlblk, kprobe_ctlblk);
46fca08f32SWang Nan 
47fca08f32SWang Nan 
arch_prepare_kprobe(struct kprobe * p)48fca08f32SWang Nan int __kprobes arch_prepare_kprobe(struct kprobe *p)
49fca08f32SWang Nan {
50fca08f32SWang Nan 	kprobe_opcode_t insn;
51fca08f32SWang Nan 	kprobe_opcode_t tmp_insn[MAX_INSN_SIZE];
52fca08f32SWang Nan 	unsigned long addr = (unsigned long)p->addr;
53fca08f32SWang Nan 	bool thumb;
54fca08f32SWang Nan 	kprobe_decode_insn_t *decode_insn;
55fca08f32SWang Nan 	const union decode_action *actions;
56fca08f32SWang Nan 	int is;
5783803d97SWang Nan 	const struct decode_checker **checkers;
58fca08f32SWang Nan 
59fca08f32SWang Nan #ifdef CONFIG_THUMB2_KERNEL
60fca08f32SWang Nan 	thumb = true;
61fca08f32SWang Nan 	addr &= ~1; /* Bit 0 would normally be set to indicate Thumb code */
62fca08f32SWang Nan 	insn = __mem_to_opcode_thumb16(((u16 *)addr)[0]);
63fca08f32SWang Nan 	if (is_wide_instruction(insn)) {
64fca08f32SWang Nan 		u16 inst2 = __mem_to_opcode_thumb16(((u16 *)addr)[1]);
65fca08f32SWang Nan 		insn = __opcode_thumb32_compose(insn, inst2);
66fca08f32SWang Nan 		decode_insn = thumb32_probes_decode_insn;
67fca08f32SWang Nan 		actions = kprobes_t32_actions;
6883803d97SWang Nan 		checkers = kprobes_t32_checkers;
69fca08f32SWang Nan 	} else {
70fca08f32SWang Nan 		decode_insn = thumb16_probes_decode_insn;
71fca08f32SWang Nan 		actions = kprobes_t16_actions;
7283803d97SWang Nan 		checkers = kprobes_t16_checkers;
73fca08f32SWang Nan 	}
74fca08f32SWang Nan #else /* !CONFIG_THUMB2_KERNEL */
75fca08f32SWang Nan 	thumb = false;
76fca08f32SWang Nan 	if (addr & 0x3)
77fca08f32SWang Nan 		return -EINVAL;
78fca08f32SWang Nan 	insn = __mem_to_opcode_arm(*p->addr);
79fca08f32SWang Nan 	decode_insn = arm_probes_decode_insn;
80fca08f32SWang Nan 	actions = kprobes_arm_actions;
8183803d97SWang Nan 	checkers = kprobes_arm_checkers;
82fca08f32SWang Nan #endif
83fca08f32SWang Nan 
84fca08f32SWang Nan 	p->opcode = insn;
85fca08f32SWang Nan 	p->ainsn.insn = tmp_insn;
86fca08f32SWang Nan 
8783803d97SWang Nan 	switch ((*decode_insn)(insn, &p->ainsn, true, actions, checkers)) {
88fca08f32SWang Nan 	case INSN_REJECTED:	/* not supported */
89fca08f32SWang Nan 		return -EINVAL;
90fca08f32SWang Nan 
91fca08f32SWang Nan 	case INSN_GOOD:		/* instruction uses slot */
92fca08f32SWang Nan 		p->ainsn.insn = get_insn_slot();
93fca08f32SWang Nan 		if (!p->ainsn.insn)
94fca08f32SWang Nan 			return -ENOMEM;
95fca08f32SWang Nan 		for (is = 0; is < MAX_INSN_SIZE; ++is)
96fca08f32SWang Nan 			p->ainsn.insn[is] = tmp_insn[is];
97fca08f32SWang Nan 		flush_insns(p->ainsn.insn,
98fca08f32SWang Nan 				sizeof(p->ainsn.insn[0]) * MAX_INSN_SIZE);
99fca08f32SWang Nan 		p->ainsn.insn_fn = (probes_insn_fn_t *)
100fca08f32SWang Nan 					((uintptr_t)p->ainsn.insn | thumb);
101fca08f32SWang Nan 		break;
102fca08f32SWang Nan 
103fca08f32SWang Nan 	case INSN_GOOD_NO_SLOT:	/* instruction doesn't need insn slot */
104fca08f32SWang Nan 		p->ainsn.insn = NULL;
105fca08f32SWang Nan 		break;
106fca08f32SWang Nan 	}
107fca08f32SWang Nan 
108a0266c21SWang Nan 	/*
109a0266c21SWang Nan 	 * Never instrument insn like 'str r0, [sp, +/-r1]'. Also, insn likes
110a0266c21SWang Nan 	 * 'str r0, [sp, #-68]' should also be prohibited.
111a0266c21SWang Nan 	 * See __und_svc.
112a0266c21SWang Nan 	 */
113a0266c21SWang Nan 	if ((p->ainsn.stack_space < 0) ||
114a0266c21SWang Nan 			(p->ainsn.stack_space > MAX_STACK_SIZE))
115a0266c21SWang Nan 		return -EINVAL;
116a0266c21SWang Nan 
117fca08f32SWang Nan 	return 0;
118fca08f32SWang Nan }
119fca08f32SWang Nan 
arch_arm_kprobe(struct kprobe * p)120fca08f32SWang Nan void __kprobes arch_arm_kprobe(struct kprobe *p)
121fca08f32SWang Nan {
122fca08f32SWang Nan 	unsigned int brkp;
123fca08f32SWang Nan 	void *addr;
124fca08f32SWang Nan 
125fca08f32SWang Nan 	if (IS_ENABLED(CONFIG_THUMB2_KERNEL)) {
126fca08f32SWang Nan 		/* Remove any Thumb flag */
127fca08f32SWang Nan 		addr = (void *)((uintptr_t)p->addr & ~1);
128fca08f32SWang Nan 
129fca08f32SWang Nan 		if (is_wide_instruction(p->opcode))
130fca08f32SWang Nan 			brkp = KPROBE_THUMB32_BREAKPOINT_INSTRUCTION;
131fca08f32SWang Nan 		else
132fca08f32SWang Nan 			brkp = KPROBE_THUMB16_BREAKPOINT_INSTRUCTION;
133fca08f32SWang Nan 	} else {
134fca08f32SWang Nan 		kprobe_opcode_t insn = p->opcode;
135fca08f32SWang Nan 
136fca08f32SWang Nan 		addr = p->addr;
137fca08f32SWang Nan 		brkp = KPROBE_ARM_BREAKPOINT_INSTRUCTION;
138fca08f32SWang Nan 
139fca08f32SWang Nan 		if (insn >= 0xe0000000)
140fca08f32SWang Nan 			brkp |= 0xe0000000;  /* Unconditional instruction */
141fca08f32SWang Nan 		else
142fca08f32SWang Nan 			brkp |= insn & 0xf0000000;  /* Copy condition from insn */
143fca08f32SWang Nan 	}
144fca08f32SWang Nan 
145fca08f32SWang Nan 	patch_text(addr, brkp);
146fca08f32SWang Nan }
147fca08f32SWang Nan 
148fca08f32SWang Nan /*
149fca08f32SWang Nan  * The actual disarming is done here on each CPU and synchronized using
150fca08f32SWang Nan  * stop_machine. This synchronization is necessary on SMP to avoid removing
151fca08f32SWang Nan  * a probe between the moment the 'Undefined Instruction' exception is raised
152fca08f32SWang Nan  * and the moment the exception handler reads the faulting instruction from
153fca08f32SWang Nan  * memory. It is also needed to atomically set the two half-words of a 32-bit
154fca08f32SWang Nan  * Thumb breakpoint.
155fca08f32SWang Nan  */
1560dc016dbSWang Nan struct patch {
1570dc016dbSWang Nan 	void *addr;
1580dc016dbSWang Nan 	unsigned int insn;
1590dc016dbSWang Nan };
1600dc016dbSWang Nan 
__kprobes_remove_breakpoint(void * data)1610dc016dbSWang Nan static int __kprobes_remove_breakpoint(void *data)
162fca08f32SWang Nan {
1630dc016dbSWang Nan 	struct patch *p = data;
1640dc016dbSWang Nan 	__patch_text(p->addr, p->insn);
165fca08f32SWang Nan 	return 0;
166fca08f32SWang Nan }
167fca08f32SWang Nan 
kprobes_remove_breakpoint(void * addr,unsigned int insn)1680dc016dbSWang Nan void __kprobes kprobes_remove_breakpoint(void *addr, unsigned int insn)
1690dc016dbSWang Nan {
1700dc016dbSWang Nan 	struct patch p = {
1710dc016dbSWang Nan 		.addr = addr,
1720dc016dbSWang Nan 		.insn = insn,
1730dc016dbSWang Nan 	};
1749489cc8fSThomas Gleixner 	stop_machine_cpuslocked(__kprobes_remove_breakpoint, &p,
1759489cc8fSThomas Gleixner 				cpu_online_mask);
1760dc016dbSWang Nan }
1770dc016dbSWang Nan 
arch_disarm_kprobe(struct kprobe * p)178fca08f32SWang Nan void __kprobes arch_disarm_kprobe(struct kprobe *p)
179fca08f32SWang Nan {
1800dc016dbSWang Nan 	kprobes_remove_breakpoint((void *)((uintptr_t)p->addr & ~1),
1810dc016dbSWang Nan 			p->opcode);
182fca08f32SWang Nan }
183fca08f32SWang Nan 
arch_remove_kprobe(struct kprobe * p)184fca08f32SWang Nan void __kprobes arch_remove_kprobe(struct kprobe *p)
185fca08f32SWang Nan {
186fca08f32SWang Nan 	if (p->ainsn.insn) {
187fca08f32SWang Nan 		free_insn_slot(p->ainsn.insn, 0);
188fca08f32SWang Nan 		p->ainsn.insn = NULL;
189fca08f32SWang Nan 	}
190fca08f32SWang Nan }
191fca08f32SWang Nan 
save_previous_kprobe(struct kprobe_ctlblk * kcb)192fca08f32SWang Nan static void __kprobes save_previous_kprobe(struct kprobe_ctlblk *kcb)
193fca08f32SWang Nan {
194fca08f32SWang Nan 	kcb->prev_kprobe.kp = kprobe_running();
195fca08f32SWang Nan 	kcb->prev_kprobe.status = kcb->kprobe_status;
196fca08f32SWang Nan }
197fca08f32SWang Nan 
restore_previous_kprobe(struct kprobe_ctlblk * kcb)198fca08f32SWang Nan static void __kprobes restore_previous_kprobe(struct kprobe_ctlblk *kcb)
199fca08f32SWang Nan {
200fca08f32SWang Nan 	__this_cpu_write(current_kprobe, kcb->prev_kprobe.kp);
201fca08f32SWang Nan 	kcb->kprobe_status = kcb->prev_kprobe.status;
202fca08f32SWang Nan }
203fca08f32SWang Nan 
set_current_kprobe(struct kprobe * p)204fca08f32SWang Nan static void __kprobes set_current_kprobe(struct kprobe *p)
205fca08f32SWang Nan {
206fca08f32SWang Nan 	__this_cpu_write(current_kprobe, p);
207fca08f32SWang Nan }
208fca08f32SWang Nan 
209fca08f32SWang Nan static void __kprobes
singlestep_skip(struct kprobe * p,struct pt_regs * regs)210fca08f32SWang Nan singlestep_skip(struct kprobe *p, struct pt_regs *regs)
211fca08f32SWang Nan {
212fca08f32SWang Nan #ifdef CONFIG_THUMB2_KERNEL
213fca08f32SWang Nan 	regs->ARM_cpsr = it_advance(regs->ARM_cpsr);
214fca08f32SWang Nan 	if (is_wide_instruction(p->opcode))
215fca08f32SWang Nan 		regs->ARM_pc += 4;
216fca08f32SWang Nan 	else
217fca08f32SWang Nan 		regs->ARM_pc += 2;
218fca08f32SWang Nan #else
219fca08f32SWang Nan 	regs->ARM_pc += 4;
220fca08f32SWang Nan #endif
221fca08f32SWang Nan }
222fca08f32SWang Nan 
223fca08f32SWang Nan static inline void __kprobes
singlestep(struct kprobe * p,struct pt_regs * regs,struct kprobe_ctlblk * kcb)224fca08f32SWang Nan singlestep(struct kprobe *p, struct pt_regs *regs, struct kprobe_ctlblk *kcb)
225fca08f32SWang Nan {
226fca08f32SWang Nan 	p->ainsn.insn_singlestep(p->opcode, &p->ainsn, regs);
227fca08f32SWang Nan }
228fca08f32SWang Nan 
229fca08f32SWang Nan /*
230fca08f32SWang Nan  * Called with IRQs disabled. IRQs must remain disabled from that point
231fca08f32SWang Nan  * all the way until processing this kprobe is complete.  The current
232fca08f32SWang Nan  * kprobes implementation cannot process more than one nested level of
233fca08f32SWang Nan  * kprobe, and that level is reserved for user kprobe handlers, so we can't
234fca08f32SWang Nan  * risk encountering a new kprobe in an interrupt handler.
235fca08f32SWang Nan  */
kprobe_handler(struct pt_regs * regs)236*1b9c3ddcSArnd Bergmann static void __kprobes kprobe_handler(struct pt_regs *regs)
237fca08f32SWang Nan {
238fca08f32SWang Nan 	struct kprobe *p, *cur;
239fca08f32SWang Nan 	struct kprobe_ctlblk *kcb;
240fca08f32SWang Nan 
241fca08f32SWang Nan 	kcb = get_kprobe_ctlblk();
242fca08f32SWang Nan 	cur = kprobe_running();
243fca08f32SWang Nan 
244fca08f32SWang Nan #ifdef CONFIG_THUMB2_KERNEL
245fca08f32SWang Nan 	/*
246fca08f32SWang Nan 	 * First look for a probe which was registered using an address with
247fca08f32SWang Nan 	 * bit 0 set, this is the usual situation for pointers to Thumb code.
248fca08f32SWang Nan 	 * If not found, fallback to looking for one with bit 0 clear.
249fca08f32SWang Nan 	 */
250fca08f32SWang Nan 	p = get_kprobe((kprobe_opcode_t *)(regs->ARM_pc | 1));
251fca08f32SWang Nan 	if (!p)
252fca08f32SWang Nan 		p = get_kprobe((kprobe_opcode_t *)regs->ARM_pc);
253fca08f32SWang Nan 
254fca08f32SWang Nan #else /* ! CONFIG_THUMB2_KERNEL */
255fca08f32SWang Nan 	p = get_kprobe((kprobe_opcode_t *)regs->ARM_pc);
256fca08f32SWang Nan #endif
257fca08f32SWang Nan 
258fca08f32SWang Nan 	if (p) {
25991fc862cSMasami Hiramatsu 		if (!p->ainsn.insn_check_cc(regs->ARM_cpsr)) {
26091fc862cSMasami Hiramatsu 			/*
26191fc862cSMasami Hiramatsu 			 * Probe hit but conditional execution check failed,
26291fc862cSMasami Hiramatsu 			 * so just skip the instruction and continue as if
26391fc862cSMasami Hiramatsu 			 * nothing had happened.
26491fc862cSMasami Hiramatsu 			 * In this case, we can skip recursing check too.
26591fc862cSMasami Hiramatsu 			 */
26691fc862cSMasami Hiramatsu 			singlestep_skip(p, regs);
26791fc862cSMasami Hiramatsu 		} else if (cur) {
268fca08f32SWang Nan 			/* Kprobe is pending, so we're recursing. */
269fca08f32SWang Nan 			switch (kcb->kprobe_status) {
270fca08f32SWang Nan 			case KPROBE_HIT_ACTIVE:
271fca08f32SWang Nan 			case KPROBE_HIT_SSDONE:
272f3fbd7ecSMasami Hiramatsu 			case KPROBE_HIT_SS:
273fca08f32SWang Nan 				/* A pre- or post-handler probe got us here. */
274fca08f32SWang Nan 				kprobes_inc_nmissed_count(p);
275fca08f32SWang Nan 				save_previous_kprobe(kcb);
276fca08f32SWang Nan 				set_current_kprobe(p);
277fca08f32SWang Nan 				kcb->kprobe_status = KPROBE_REENTER;
278fca08f32SWang Nan 				singlestep(p, regs, kcb);
279fca08f32SWang Nan 				restore_previous_kprobe(kcb);
280fca08f32SWang Nan 				break;
281f3fbd7ecSMasami Hiramatsu 			case KPROBE_REENTER:
282f3fbd7ecSMasami Hiramatsu 				/* A nested probe was hit in FIQ, it is a BUG */
2839c89bb8eSMasami Hiramatsu 				pr_warn("Failed to recover from reentered kprobes.\n");
28475b2f5f5SMasami Hiramatsu 				dump_kprobe(p);
285df561f66SGustavo A. R. Silva 				fallthrough;
286fca08f32SWang Nan 			default:
287fca08f32SWang Nan 				/* impossible cases */
288fca08f32SWang Nan 				BUG();
289fca08f32SWang Nan 			}
29091fc862cSMasami Hiramatsu 		} else {
291fca08f32SWang Nan 			/* Probe hit and conditional execution check ok. */
292fca08f32SWang Nan 			set_current_kprobe(p);
293fca08f32SWang Nan 			kcb->kprobe_status = KPROBE_HIT_ACTIVE;
294fca08f32SWang Nan 
295fca08f32SWang Nan 			/*
296fca08f32SWang Nan 			 * If we have no pre-handler or it returned 0, we
297fca08f32SWang Nan 			 * continue with normal processing. If we have a
298cce188bdSMasami Hiramatsu 			 * pre-handler and it returned non-zero, it will
299cce188bdSMasami Hiramatsu 			 * modify the execution path and no need to single
300cce188bdSMasami Hiramatsu 			 * stepping. Let's just reset current kprobe and exit.
301fca08f32SWang Nan 			 */
302fca08f32SWang Nan 			if (!p->pre_handler || !p->pre_handler(p, regs)) {
303fca08f32SWang Nan 				kcb->kprobe_status = KPROBE_HIT_SS;
304fca08f32SWang Nan 				singlestep(p, regs, kcb);
305fca08f32SWang Nan 				if (p->post_handler) {
306fca08f32SWang Nan 					kcb->kprobe_status = KPROBE_HIT_SSDONE;
307fca08f32SWang Nan 					p->post_handler(p, regs, 0);
308fca08f32SWang Nan 				}
309fca08f32SWang Nan 			}
310cce188bdSMasami Hiramatsu 			reset_current_kprobe();
311fca08f32SWang Nan 		}
312fca08f32SWang Nan 	} else {
313fca08f32SWang Nan 		/*
314fca08f32SWang Nan 		 * The probe was removed and a race is in progress.
315fca08f32SWang Nan 		 * There is nothing we can do about it.  Let's restart
316fca08f32SWang Nan 		 * the instruction.  By the time we can restart, the
317fca08f32SWang Nan 		 * real instruction will be there.
318fca08f32SWang Nan 		 */
319fca08f32SWang Nan 	}
320fca08f32SWang Nan }
321fca08f32SWang Nan 
kprobe_trap_handler(struct pt_regs * regs,unsigned int instr)322fca08f32SWang Nan static int __kprobes kprobe_trap_handler(struct pt_regs *regs, unsigned int instr)
323fca08f32SWang Nan {
324fca08f32SWang Nan 	unsigned long flags;
325fca08f32SWang Nan 	local_irq_save(flags);
326fca08f32SWang Nan 	kprobe_handler(regs);
327fca08f32SWang Nan 	local_irq_restore(flags);
328fca08f32SWang Nan 	return 0;
329fca08f32SWang Nan }
330fca08f32SWang Nan 
kprobe_fault_handler(struct pt_regs * regs,unsigned int fsr)331fca08f32SWang Nan int __kprobes kprobe_fault_handler(struct pt_regs *regs, unsigned int fsr)
332fca08f32SWang Nan {
333fca08f32SWang Nan 	struct kprobe *cur = kprobe_running();
334fca08f32SWang Nan 	struct kprobe_ctlblk *kcb = get_kprobe_ctlblk();
335fca08f32SWang Nan 
336fca08f32SWang Nan 	switch (kcb->kprobe_status) {
337fca08f32SWang Nan 	case KPROBE_HIT_SS:
338fca08f32SWang Nan 	case KPROBE_REENTER:
339fca08f32SWang Nan 		/*
340fca08f32SWang Nan 		 * We are here because the instruction being single
341fca08f32SWang Nan 		 * stepped caused a page fault. We reset the current
342fca08f32SWang Nan 		 * kprobe and the PC to point back to the probe address
343fca08f32SWang Nan 		 * and allow the page fault handler to continue as a
344fca08f32SWang Nan 		 * normal page fault.
345fca08f32SWang Nan 		 */
346fca08f32SWang Nan 		regs->ARM_pc = (long)cur->addr;
347fca08f32SWang Nan 		if (kcb->kprobe_status == KPROBE_REENTER) {
348fca08f32SWang Nan 			restore_previous_kprobe(kcb);
349fca08f32SWang Nan 		} else {
350fca08f32SWang Nan 			reset_current_kprobe();
351fca08f32SWang Nan 		}
352fca08f32SWang Nan 		break;
353fca08f32SWang Nan 	}
354fca08f32SWang Nan 
355fca08f32SWang Nan 	return 0;
356fca08f32SWang Nan }
357fca08f32SWang Nan 
kprobe_exceptions_notify(struct notifier_block * self,unsigned long val,void * data)358fca08f32SWang Nan int __kprobes kprobe_exceptions_notify(struct notifier_block *self,
359fca08f32SWang Nan 				       unsigned long val, void *data)
360fca08f32SWang Nan {
361fca08f32SWang Nan 	/*
362fca08f32SWang Nan 	 * notify_die() is currently never called on ARM,
363fca08f32SWang Nan 	 * so this callback is currently empty.
364fca08f32SWang Nan 	 */
365fca08f32SWang Nan 	return NOTIFY_DONE;
366fca08f32SWang Nan }
367fca08f32SWang Nan 
368fca08f32SWang Nan /*
369fca08f32SWang Nan  * When a retprobed function returns, trampoline_handler() is called,
370fca08f32SWang Nan  * calling the kretprobe's handler. We construct a struct pt_regs to
3717e9bf33bSMasami Hiramatsu  * give a view of registers r0-r11, sp, lr, and pc to the user
3727e9bf33bSMasami Hiramatsu  * return-handler. This is not a complete pt_regs structure, but that
3737e9bf33bSMasami Hiramatsu  * should be enough for stacktrace from the return handler with or
3747e9bf33bSMasami Hiramatsu  * without pt_regs.
375fca08f32SWang Nan  */
__kretprobe_trampoline(void)376adf8a61aSMasami Hiramatsu void __naked __kprobes __kretprobe_trampoline(void)
377fca08f32SWang Nan {
378fca08f32SWang Nan 	__asm__ __volatile__ (
3797e9bf33bSMasami Hiramatsu #ifdef CONFIG_FRAME_POINTER
3807e9bf33bSMasami Hiramatsu 		"ldr	lr, =__kretprobe_trampoline	\n\t"
3817e9bf33bSMasami Hiramatsu 	/* __kretprobe_trampoline makes a framepointer on pt_regs. */
3827e9bf33bSMasami Hiramatsu #ifdef CONFIG_CC_IS_CLANG
3837e9bf33bSMasami Hiramatsu 		"stmdb	sp, {sp, lr, pc}	\n\t"
3847e9bf33bSMasami Hiramatsu 		"sub	sp, sp, #12		\n\t"
3857e9bf33bSMasami Hiramatsu 		/* In clang case, pt_regs->ip = lr. */
3867e9bf33bSMasami Hiramatsu 		"stmdb	sp!, {r0 - r11, lr}	\n\t"
3877e9bf33bSMasami Hiramatsu 		/* fp points regs->r11 (fp) */
3887e9bf33bSMasami Hiramatsu 		"add	fp, sp,	#44		\n\t"
3897e9bf33bSMasami Hiramatsu #else /* !CONFIG_CC_IS_CLANG */
3907e9bf33bSMasami Hiramatsu 		/* In gcc case, pt_regs->ip = fp. */
3917e9bf33bSMasami Hiramatsu 		"stmdb	sp, {fp, sp, lr, pc}	\n\t"
3927391dd19SMasami Hiramatsu 		"sub	sp, sp, #16		\n\t"
393fca08f32SWang Nan 		"stmdb	sp!, {r0 - r11}		\n\t"
3947e9bf33bSMasami Hiramatsu 		/* fp points regs->r15 (pc) */
3957e9bf33bSMasami Hiramatsu 		"add	fp, sp, #60		\n\t"
3967e9bf33bSMasami Hiramatsu #endif /* CONFIG_CC_IS_CLANG */
3977e9bf33bSMasami Hiramatsu #else /* !CONFIG_FRAME_POINTER */
3987e9bf33bSMasami Hiramatsu 		"sub	sp, sp, #16		\n\t"
3997e9bf33bSMasami Hiramatsu 		"stmdb	sp!, {r0 - r11}		\n\t"
4007e9bf33bSMasami Hiramatsu #endif /* CONFIG_FRAME_POINTER */
401fca08f32SWang Nan 		"mov	r0, sp			\n\t"
402fca08f32SWang Nan 		"bl	trampoline_handler	\n\t"
403fca08f32SWang Nan 		"mov	lr, r0			\n\t"
404fca08f32SWang Nan 		"ldmia	sp!, {r0 - r11}		\n\t"
4057391dd19SMasami Hiramatsu 		"add	sp, sp, #16		\n\t"
406fca08f32SWang Nan #ifdef CONFIG_THUMB2_KERNEL
407fca08f32SWang Nan 		"bx	lr			\n\t"
408fca08f32SWang Nan #else
409fca08f32SWang Nan 		"mov	pc, lr			\n\t"
410fca08f32SWang Nan #endif
411fca08f32SWang Nan 		: : : "memory");
412fca08f32SWang Nan }
413fca08f32SWang Nan 
414adf8a61aSMasami Hiramatsu /* Called from __kretprobe_trampoline */
trampoline_handler(struct pt_regs * regs)415fca08f32SWang Nan static __used __kprobes void *trampoline_handler(struct pt_regs *regs)
416fca08f32SWang Nan {
41796fed8acSMasami Hiramatsu 	return (void *)kretprobe_trampoline_handler(regs, (void *)regs->ARM_fp);
418fca08f32SWang Nan }
419fca08f32SWang Nan 
arch_prepare_kretprobe(struct kretprobe_instance * ri,struct pt_regs * regs)420fca08f32SWang Nan void __kprobes arch_prepare_kretprobe(struct kretprobe_instance *ri,
421fca08f32SWang Nan 				      struct pt_regs *regs)
422fca08f32SWang Nan {
423fca08f32SWang Nan 	ri->ret_addr = (kprobe_opcode_t *)regs->ARM_lr;
42494509582SMasami Hiramatsu 	ri->fp = (void *)regs->ARM_fp;
425fca08f32SWang Nan 
426fca08f32SWang Nan 	/* Replace the return addr with trampoline addr. */
427adf8a61aSMasami Hiramatsu 	regs->ARM_lr = (unsigned long)&__kretprobe_trampoline;
428fca08f32SWang Nan }
429fca08f32SWang Nan 
arch_trampoline_kprobe(struct kprobe * p)430fca08f32SWang Nan int __kprobes arch_trampoline_kprobe(struct kprobe *p)
431fca08f32SWang Nan {
432fca08f32SWang Nan 	return 0;
433fca08f32SWang Nan }
434fca08f32SWang Nan 
435fca08f32SWang Nan #ifdef CONFIG_THUMB2_KERNEL
436fca08f32SWang Nan 
437fca08f32SWang Nan static struct undef_hook kprobes_thumb16_break_hook = {
438fca08f32SWang Nan 	.instr_mask	= 0xffff,
439fca08f32SWang Nan 	.instr_val	= KPROBE_THUMB16_BREAKPOINT_INSTRUCTION,
440fca08f32SWang Nan 	.cpsr_mask	= MODE_MASK,
441fca08f32SWang Nan 	.cpsr_val	= SVC_MODE,
442fca08f32SWang Nan 	.fn		= kprobe_trap_handler,
443fca08f32SWang Nan };
444fca08f32SWang Nan 
445fca08f32SWang Nan static struct undef_hook kprobes_thumb32_break_hook = {
446fca08f32SWang Nan 	.instr_mask	= 0xffffffff,
447fca08f32SWang Nan 	.instr_val	= KPROBE_THUMB32_BREAKPOINT_INSTRUCTION,
448fca08f32SWang Nan 	.cpsr_mask	= MODE_MASK,
449fca08f32SWang Nan 	.cpsr_val	= SVC_MODE,
450fca08f32SWang Nan 	.fn		= kprobe_trap_handler,
451fca08f32SWang Nan };
452fca08f32SWang Nan 
453fca08f32SWang Nan #else  /* !CONFIG_THUMB2_KERNEL */
454fca08f32SWang Nan 
455fca08f32SWang Nan static struct undef_hook kprobes_arm_break_hook = {
456fca08f32SWang Nan 	.instr_mask	= 0x0fffffff,
457fca08f32SWang Nan 	.instr_val	= KPROBE_ARM_BREAKPOINT_INSTRUCTION,
458fca08f32SWang Nan 	.cpsr_mask	= MODE_MASK,
459fca08f32SWang Nan 	.cpsr_val	= SVC_MODE,
460fca08f32SWang Nan 	.fn		= kprobe_trap_handler,
461fca08f32SWang Nan };
462fca08f32SWang Nan 
463fca08f32SWang Nan #endif /* !CONFIG_THUMB2_KERNEL */
464fca08f32SWang Nan 
arch_init_kprobes(void)4651f323127SArnd Bergmann int __init arch_init_kprobes(void)
466fca08f32SWang Nan {
467fca08f32SWang Nan 	arm_probes_decode_init();
468fca08f32SWang Nan #ifdef CONFIG_THUMB2_KERNEL
469fca08f32SWang Nan 	register_undef_hook(&kprobes_thumb16_break_hook);
470fca08f32SWang Nan 	register_undef_hook(&kprobes_thumb32_break_hook);
471fca08f32SWang Nan #else
472fca08f32SWang Nan 	register_undef_hook(&kprobes_arm_break_hook);
473fca08f32SWang Nan #endif
474fca08f32SWang Nan 	return 0;
475fca08f32SWang Nan }
476c6089061SRussell King 
arch_within_kprobe_blacklist(unsigned long addr)477c6089061SRussell King bool arch_within_kprobe_blacklist(unsigned long addr)
478c6089061SRussell King {
479c6089061SRussell King 	void *a = (void *)addr;
480c6089061SRussell King 
481c6089061SRussell King 	return __in_irqentry_text(addr) ||
482c6089061SRussell King 	       in_entry_text(addr) ||
483c6089061SRussell King 	       in_idmap_text(addr) ||
484c6089061SRussell King 	       memory_contains(__kprobes_text_start, __kprobes_text_end, a, 1);
485c6089061SRussell King }
486