| // SPDX-License-Identifier: GPL-2.0-or-later |
| |
| #include <linux/regset.h> |
| #include <linux/hw_breakpoint.h> |
| |
| #include <asm/debug.h> |
| |
| #include "ptrace-decl.h" |
| |
| void user_enable_single_step(struct task_struct *task) |
| { |
| struct pt_regs *regs = task->thread.regs; |
| |
| if (regs != NULL) { |
| regs->msr &= ~MSR_BE; |
| regs->msr |= MSR_SE; |
| } |
| set_tsk_thread_flag(task, TIF_SINGLESTEP); |
| } |
| |
| void user_enable_block_step(struct task_struct *task) |
| { |
| struct pt_regs *regs = task->thread.regs; |
| |
| if (regs != NULL) { |
| regs->msr &= ~MSR_SE; |
| regs->msr |= MSR_BE; |
| } |
| set_tsk_thread_flag(task, TIF_SINGLESTEP); |
| } |
| |
| void user_disable_single_step(struct task_struct *task) |
| { |
| struct pt_regs *regs = task->thread.regs; |
| |
| if (regs != NULL) |
| regs->msr &= ~(MSR_SE | MSR_BE); |
| |
| clear_tsk_thread_flag(task, TIF_SINGLESTEP); |
| } |
| |
| void ppc_gethwdinfo(struct ppc_debug_info *dbginfo) |
| { |
| dbginfo->version = 1; |
| dbginfo->num_instruction_bps = 0; |
| if (ppc_breakpoint_available()) |
| dbginfo->num_data_bps = nr_wp_slots(); |
| else |
| dbginfo->num_data_bps = 0; |
| dbginfo->num_condition_regs = 0; |
| dbginfo->data_bp_alignment = sizeof(long); |
| dbginfo->sizeof_condition = 0; |
| if (IS_ENABLED(CONFIG_HAVE_HW_BREAKPOINT)) { |
| dbginfo->features = PPC_DEBUG_FEATURE_DATA_BP_RANGE; |
| if (dawr_enabled()) |
| dbginfo->features |= PPC_DEBUG_FEATURE_DATA_BP_DAWR; |
| } else { |
| dbginfo->features = 0; |
| } |
| if (cpu_has_feature(CPU_FTR_ARCH_31)) |
| dbginfo->features |= PPC_DEBUG_FEATURE_DATA_BP_ARCH_31; |
| } |
| |
| int ptrace_get_debugreg(struct task_struct *child, unsigned long addr, |
| unsigned long __user *datalp) |
| { |
| unsigned long dabr_fake; |
| |
| /* We only support one DABR and no IABRS at the moment */ |
| if (addr > 0) |
| return -EINVAL; |
| dabr_fake = ((child->thread.hw_brk[0].address & (~HW_BRK_TYPE_DABR)) | |
| (child->thread.hw_brk[0].type & HW_BRK_TYPE_DABR)); |
| return put_user(dabr_fake, datalp); |
| } |
| |
| /* |
| * ptrace_set_debugreg() fakes DABR and DABR is only one. So even if |
| * internal hw supports more than one watchpoint, we support only one |
| * watchpoint with this interface. |
| */ |
| int ptrace_set_debugreg(struct task_struct *task, unsigned long addr, unsigned long data) |
| { |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| int ret; |
| struct thread_struct *thread = &task->thread; |
| struct perf_event *bp; |
| struct perf_event_attr attr; |
| #endif /* CONFIG_HAVE_HW_BREAKPOINT */ |
| bool set_bp = true; |
| struct arch_hw_breakpoint hw_brk; |
| |
| /* For ppc64 we support one DABR and no IABR's at the moment (ppc64). |
| * For embedded processors we support one DAC and no IAC's at the |
| * moment. |
| */ |
| if (addr > 0) |
| return -EINVAL; |
| |
| /* The bottom 3 bits in dabr are flags */ |
| if ((data & ~0x7UL) >= TASK_SIZE) |
| return -EIO; |
| |
| /* For processors using DABR (i.e. 970), the bottom 3 bits are flags. |
| * It was assumed, on previous implementations, that 3 bits were |
| * passed together with the data address, fitting the design of the |
| * DABR register, as follows: |
| * |
| * bit 0: Read flag |
| * bit 1: Write flag |
| * bit 2: Breakpoint translation |
| * |
| * Thus, we use them here as so. |
| */ |
| |
| /* Ensure breakpoint translation bit is set */ |
| if (data && !(data & HW_BRK_TYPE_TRANSLATE)) |
| return -EIO; |
| hw_brk.address = data & (~HW_BRK_TYPE_DABR); |
| hw_brk.type = (data & HW_BRK_TYPE_DABR) | HW_BRK_TYPE_PRIV_ALL; |
| hw_brk.len = DABR_MAX_LEN; |
| hw_brk.hw_len = DABR_MAX_LEN; |
| set_bp = (data) && (hw_brk.type & HW_BRK_TYPE_RDWR); |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| bp = thread->ptrace_bps[0]; |
| if (!set_bp) { |
| if (bp) { |
| unregister_hw_breakpoint(bp); |
| thread->ptrace_bps[0] = NULL; |
| } |
| return 0; |
| } |
| if (bp) { |
| attr = bp->attr; |
| attr.bp_addr = hw_brk.address; |
| attr.bp_len = DABR_MAX_LEN; |
| arch_bp_generic_fields(hw_brk.type, &attr.bp_type); |
| |
| /* Enable breakpoint */ |
| attr.disabled = false; |
| |
| ret = modify_user_hw_breakpoint(bp, &attr); |
| if (ret) |
| return ret; |
| |
| thread->ptrace_bps[0] = bp; |
| thread->hw_brk[0] = hw_brk; |
| return 0; |
| } |
| |
| /* Create a new breakpoint request if one doesn't exist already */ |
| hw_breakpoint_init(&attr); |
| attr.bp_addr = hw_brk.address; |
| attr.bp_len = DABR_MAX_LEN; |
| arch_bp_generic_fields(hw_brk.type, |
| &attr.bp_type); |
| |
| thread->ptrace_bps[0] = bp = register_user_hw_breakpoint(&attr, |
| ptrace_triggered, NULL, task); |
| if (IS_ERR(bp)) { |
| thread->ptrace_bps[0] = NULL; |
| return PTR_ERR(bp); |
| } |
| |
| #else /* !CONFIG_HAVE_HW_BREAKPOINT */ |
| if (set_bp && (!ppc_breakpoint_available())) |
| return -ENODEV; |
| #endif /* CONFIG_HAVE_HW_BREAKPOINT */ |
| task->thread.hw_brk[0] = hw_brk; |
| return 0; |
| } |
| |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| static int find_empty_ptrace_bp(struct thread_struct *thread) |
| { |
| int i; |
| |
| for (i = 0; i < nr_wp_slots(); i++) { |
| if (!thread->ptrace_bps[i]) |
| return i; |
| } |
| return -1; |
| } |
| #endif |
| |
| static int find_empty_hw_brk(struct thread_struct *thread) |
| { |
| int i; |
| |
| for (i = 0; i < nr_wp_slots(); i++) { |
| if (!thread->hw_brk[i].address) |
| return i; |
| } |
| return -1; |
| } |
| |
| long ppc_set_hwdebug(struct task_struct *child, struct ppc_hw_breakpoint *bp_info) |
| { |
| int i; |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| int len = 0; |
| struct thread_struct *thread = &child->thread; |
| struct perf_event *bp; |
| struct perf_event_attr attr; |
| #endif /* CONFIG_HAVE_HW_BREAKPOINT */ |
| struct arch_hw_breakpoint brk; |
| |
| if (bp_info->version != 1) |
| return -ENOTSUPP; |
| /* |
| * We only support one data breakpoint |
| */ |
| if ((bp_info->trigger_type & PPC_BREAKPOINT_TRIGGER_RW) == 0 || |
| (bp_info->trigger_type & ~PPC_BREAKPOINT_TRIGGER_RW) != 0 || |
| bp_info->condition_mode != PPC_BREAKPOINT_CONDITION_NONE) |
| return -EINVAL; |
| |
| if ((unsigned long)bp_info->addr >= TASK_SIZE) |
| return -EIO; |
| |
| brk.address = ALIGN_DOWN(bp_info->addr, HW_BREAKPOINT_SIZE); |
| brk.type = HW_BRK_TYPE_TRANSLATE | HW_BRK_TYPE_PRIV_ALL; |
| brk.len = DABR_MAX_LEN; |
| brk.hw_len = DABR_MAX_LEN; |
| if (bp_info->trigger_type & PPC_BREAKPOINT_TRIGGER_READ) |
| brk.type |= HW_BRK_TYPE_READ; |
| if (bp_info->trigger_type & PPC_BREAKPOINT_TRIGGER_WRITE) |
| brk.type |= HW_BRK_TYPE_WRITE; |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| if (bp_info->addr_mode == PPC_BREAKPOINT_MODE_RANGE_INCLUSIVE) |
| len = bp_info->addr2 - bp_info->addr; |
| else if (bp_info->addr_mode == PPC_BREAKPOINT_MODE_EXACT) |
| len = 1; |
| else |
| return -EINVAL; |
| |
| i = find_empty_ptrace_bp(thread); |
| if (i < 0) |
| return -ENOSPC; |
| |
| /* Create a new breakpoint request if one doesn't exist already */ |
| hw_breakpoint_init(&attr); |
| attr.bp_addr = (unsigned long)bp_info->addr; |
| attr.bp_len = len; |
| arch_bp_generic_fields(brk.type, &attr.bp_type); |
| |
| bp = register_user_hw_breakpoint(&attr, ptrace_triggered, NULL, child); |
| thread->ptrace_bps[i] = bp; |
| if (IS_ERR(bp)) { |
| thread->ptrace_bps[i] = NULL; |
| return PTR_ERR(bp); |
| } |
| |
| return i + 1; |
| #endif /* CONFIG_HAVE_HW_BREAKPOINT */ |
| |
| if (bp_info->addr_mode != PPC_BREAKPOINT_MODE_EXACT) |
| return -EINVAL; |
| |
| i = find_empty_hw_brk(&child->thread); |
| if (i < 0) |
| return -ENOSPC; |
| |
| if (!ppc_breakpoint_available()) |
| return -ENODEV; |
| |
| child->thread.hw_brk[i] = brk; |
| |
| return i + 1; |
| } |
| |
| long ppc_del_hwdebug(struct task_struct *child, long data) |
| { |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| int ret = 0; |
| struct thread_struct *thread = &child->thread; |
| struct perf_event *bp; |
| #endif /* CONFIG_HAVE_HW_BREAKPOINT */ |
| if (data < 1 || data > nr_wp_slots()) |
| return -EINVAL; |
| |
| #ifdef CONFIG_HAVE_HW_BREAKPOINT |
| bp = thread->ptrace_bps[data - 1]; |
| if (bp) { |
| unregister_hw_breakpoint(bp); |
| thread->ptrace_bps[data - 1] = NULL; |
| } else { |
| ret = -ENOENT; |
| } |
| return ret; |
| #else /* CONFIG_HAVE_HW_BREAKPOINT */ |
| if (!(child->thread.hw_brk[data - 1].flags & HW_BRK_FLAG_DISABLED) && |
| child->thread.hw_brk[data - 1].address == 0) |
| return -ENOENT; |
| |
| child->thread.hw_brk[data - 1].address = 0; |
| child->thread.hw_brk[data - 1].type = 0; |
| child->thread.hw_brk[data - 1].flags = 0; |
| #endif /* CONFIG_HAVE_HW_BREAKPOINT */ |
| |
| return 0; |
| } |