Lines Matching +full:ctrl +full:- +full:len

1 // SPDX-License-Identifier: GPL-2.0-only
10 * HW_breakpoint: a unified kernel/user-space hardware breakpoint facility,
13 #define pr_fmt(fmt) "hw-breakpoint: " fmt
151 /* We don't support the memory-mapped interface. */ in debug_arch_supported()
216 return core_has_mismatch_brps() ? brps - 1 : brps; in get_num_brps()
257 return -ENODEV; in enable_monitor_mode()
265 return -EPERM; in enable_monitor_mode()
293 * Check if 8-bit byte-address select is available.
299 struct arch_hw_breakpoint_ctrl ctrl; in get_max_wp_len() local
305 memset(&ctrl, 0, sizeof(ctrl)); in get_max_wp_len()
306 ctrl.len = ARM_BREAKPOINT_LEN_8; in get_max_wp_len()
307 ctrl_reg = encode_ctrl_reg(ctrl); in get_max_wp_len()
331 u32 addr, ctrl; in arch_install_hw_breakpoint() local
333 addr = info->address; in arch_install_hw_breakpoint()
334 ctrl = encode_ctrl_reg(info->ctrl) | 0x1; in arch_install_hw_breakpoint()
336 if (info->ctrl.type == ARM_BREAKPOINT_EXECUTE) { in arch_install_hw_breakpoint()
361 return -EBUSY; in arch_install_hw_breakpoint()
365 if (info->step_ctrl.enabled) { in arch_install_hw_breakpoint()
366 addr = info->trigger & ~0x3; in arch_install_hw_breakpoint()
367 ctrl = encode_ctrl_reg(info->step_ctrl); in arch_install_hw_breakpoint()
368 if (info->ctrl.type != ARM_BREAKPOINT_EXECUTE) { in arch_install_hw_breakpoint()
379 write_wb_reg(ctrl_base + i, ctrl); in arch_install_hw_breakpoint()
389 if (info->ctrl.type == ARM_BREAKPOINT_EXECUTE) { in arch_uninstall_hw_breakpoint()
417 if (info->ctrl.type != ARM_BREAKPOINT_EXECUTE && in arch_uninstall_hw_breakpoint()
418 info->step_ctrl.enabled) { in arch_uninstall_hw_breakpoint()
454 unsigned int len; in arch_check_bp_in_kernelspace() local
457 va = hw->address; in arch_check_bp_in_kernelspace()
458 len = get_hbp_len(hw->ctrl.len); in arch_check_bp_in_kernelspace()
460 return (va >= TASK_SIZE) && ((va + len - 1) >= TASK_SIZE); in arch_check_bp_in_kernelspace()
468 int arch_bp_generic_fields(struct arch_hw_breakpoint_ctrl ctrl, in arch_bp_generic_fields() argument
472 switch (ctrl.type) { in arch_bp_generic_fields()
486 return -EINVAL; in arch_bp_generic_fields()
489 /* Len */ in arch_bp_generic_fields()
490 switch (ctrl.len) { in arch_bp_generic_fields()
504 return -EINVAL; in arch_bp_generic_fields()
518 switch (attr->bp_type) { in arch_build_bp_info()
520 hw->ctrl.type = ARM_BREAKPOINT_EXECUTE; in arch_build_bp_info()
523 hw->ctrl.type = ARM_BREAKPOINT_LOAD; in arch_build_bp_info()
526 hw->ctrl.type = ARM_BREAKPOINT_STORE; in arch_build_bp_info()
529 hw->ctrl.type = ARM_BREAKPOINT_LOAD | ARM_BREAKPOINT_STORE; in arch_build_bp_info()
532 return -EINVAL; in arch_build_bp_info()
535 /* Len */ in arch_build_bp_info()
536 switch (attr->bp_len) { in arch_build_bp_info()
538 hw->ctrl.len = ARM_BREAKPOINT_LEN_1; in arch_build_bp_info()
541 hw->ctrl.len = ARM_BREAKPOINT_LEN_2; in arch_build_bp_info()
544 hw->ctrl.len = ARM_BREAKPOINT_LEN_4; in arch_build_bp_info()
547 hw->ctrl.len = ARM_BREAKPOINT_LEN_8; in arch_build_bp_info()
548 if ((hw->ctrl.type != ARM_BREAKPOINT_EXECUTE) in arch_build_bp_info()
553 return -EINVAL; in arch_build_bp_info()
562 if (hw->ctrl.type == ARM_BREAKPOINT_EXECUTE && in arch_build_bp_info()
563 hw->ctrl.len != ARM_BREAKPOINT_LEN_2 && in arch_build_bp_info()
564 hw->ctrl.len != ARM_BREAKPOINT_LEN_4) in arch_build_bp_info()
565 return -EINVAL; in arch_build_bp_info()
568 hw->address = attr->bp_addr; in arch_build_bp_info()
571 hw->ctrl.privilege = ARM_BREAKPOINT_USER; in arch_build_bp_info()
573 hw->ctrl.privilege |= ARM_BREAKPOINT_PRIV; in arch_build_bp_info()
576 hw->ctrl.enabled = !attr->disabled; in arch_build_bp_info()
579 hw->ctrl.mismatch = 0; in arch_build_bp_info()
585 * Validate the arch-specific HW Breakpoint register settings.
596 return -ENODEV; in hw_breakpoint_arch_parse()
604 if (hw->ctrl.len == ARM_BREAKPOINT_LEN_8) in hw_breakpoint_arch_parse()
606 offset = hw->address & alignment_mask; in hw_breakpoint_arch_parse()
614 if (hw->ctrl.len == ARM_BREAKPOINT_LEN_2) in hw_breakpoint_arch_parse()
619 if (hw->ctrl.len == ARM_BREAKPOINT_LEN_1) in hw_breakpoint_arch_parse()
623 ret = -EINVAL; in hw_breakpoint_arch_parse()
627 hw->address &= ~alignment_mask; in hw_breakpoint_arch_parse()
628 hw->ctrl.len <<= offset; in hw_breakpoint_arch_parse()
632 * Mismatch breakpoints are required for single-stepping in hw_breakpoint_arch_parse()
636 return -EINVAL; in hw_breakpoint_arch_parse()
640 return -EPERM; in hw_breakpoint_arch_parse()
643 * Per-cpu breakpoints are not supported by our stepping in hw_breakpoint_arch_parse()
646 if (!bp->hw.target) in hw_breakpoint_arch_parse()
647 return -EINVAL; in hw_breakpoint_arch_parse()
654 (hw->ctrl.type == ARM_BREAKPOINT_LOAD || in hw_breakpoint_arch_parse()
655 hw->ctrl.type == ARM_BREAKPOINT_STORE)) in hw_breakpoint_arch_parse()
656 return -EINVAL; in hw_breakpoint_arch_parse()
664 * Enable/disable single-stepping over the breakpoint bp at address addr.
671 info->step_ctrl.mismatch = 1; in enable_single_step()
672 info->step_ctrl.len = ARM_BREAKPOINT_LEN_4; in enable_single_step()
673 info->step_ctrl.type = ARM_BREAKPOINT_EXECUTE; in enable_single_step()
674 info->step_ctrl.privilege = info->ctrl.privilege; in enable_single_step()
675 info->step_ctrl.enabled = 1; in enable_single_step()
676 info->trigger = addr; in enable_single_step()
683 counter_arch_bp(bp)->step_ctrl.enabled = 0; in disable_single_step()
691 * addresses. There is no straight-forward way, short of disassembling the
703 struct arch_hw_breakpoint_ctrl *ctrl) in get_distance_from_watchpoint() argument
708 lens = __ffs(ctrl->len); in get_distance_from_watchpoint()
709 lene = __fls(ctrl->len); in get_distance_from_watchpoint()
714 return wp_low - addr; in get_distance_from_watchpoint()
716 return addr - wp_high; in get_distance_from_watchpoint()
724 return !user_mode(regs) && info->ctrl.privilege == ARM_BREAKPOINT_USER; in watchpoint_fault_on_uaccess()
731 u32 min_dist = -1, dist; in watchpoint_handler()
735 struct arch_hw_breakpoint_ctrl ctrl; in watchpoint_handler() local
758 info->trigger = wp->attr.bp_addr; in watchpoint_handler()
770 decode_ctrl_reg(ctrl_reg, &ctrl); in watchpoint_handler()
771 dist = get_distance_from_watchpoint(addr, val, &ctrl); in watchpoint_handler()
782 info->trigger = addr; in watchpoint_handler()
785 pr_debug("watchpoint fired: address = 0x%x\n", info->trigger); in watchpoint_handler()
800 * we can single-step over the watchpoint trigger. in watchpoint_handler()
808 if (min_dist > 0 && min_dist != -1) { in watchpoint_handler()
812 info->trigger = addr; in watchpoint_handler()
813 pr_debug("watchpoint fired: address = 0x%x\n", info->trigger); in watchpoint_handler()
839 if (!info->step_ctrl.enabled) in watchpoint_single_step_handler()
844 * single-step. in watchpoint_single_step_handler()
846 if (info->trigger != pc) in watchpoint_single_step_handler()
860 struct arch_hw_breakpoint_ctrl ctrl; in breakpoint_handler() local
865 addr = regs->ARM_pc; in breakpoint_handler()
885 decode_ctrl_reg(ctrl_reg, &ctrl); in breakpoint_handler()
886 if ((1 << (addr & 0x3)) & ctrl.len) { in breakpoint_handler()
887 info->trigger = addr; in breakpoint_handler()
897 if (info->step_ctrl.enabled) in breakpoint_handler()
903 /* Handle any pending watchpoint single-step breakpoints. */ in breakpoint_handler()
921 die("Oops - CFI", regs, 0); in hw_breakpoint_cfi_handler()
997 * One-time initialisation.
1054 * can be maintained across low-power modes without leaving the debug in reset_ctrl_regs()
1067 * Ensure sticky power-down is clear (i.e. debug logic is in reset_ctrl_regs()
1072 err = -EPERM; in reset_ctrl_regs()
1083 err = -EPERM; in reset_ctrl_regs()
1101 * Clear any configured vector-catch events before in reset_ctrl_regs()