// SPDX-License-Identifier: GPL-2.0 #include #include #include #include "dwarf-regs.h" #include "perf_regs.h" #include "util/sample.h" #include "debug.h" int perf_sdt_arg_parse_op(uint16_t e_machine, char *old_op, char **new_op) { int ret = SDT_ARG_SKIP; switch (e_machine) { case EM_AARCH64: ret = __perf_sdt_arg_parse_op_arm64(old_op, new_op); break; case EM_PPC: case EM_PPC64: ret = __perf_sdt_arg_parse_op_powerpc(old_op, new_op); break; case EM_386: case EM_X86_64: ret = __perf_sdt_arg_parse_op_x86(old_op, new_op); break; default: pr_debug("Unknown ELF machine %d, standard arguments parse will be skipped.\n", e_machine); break; } return ret; } uint64_t perf_intr_reg_mask(uint16_t e_machine) { uint64_t mask = 0; switch (e_machine) { case EM_ARM: mask = __perf_reg_mask_arm(/*intr=*/true); break; case EM_AARCH64: mask = __perf_reg_mask_arm64(/*intr=*/true); break; case EM_CSKY: mask = __perf_reg_mask_csky(/*intr=*/true); break; case EM_LOONGARCH: mask = __perf_reg_mask_loongarch(/*intr=*/true); break; case EM_MIPS: mask = __perf_reg_mask_mips(/*intr=*/true); break; case EM_PPC: case EM_PPC64: mask = __perf_reg_mask_powerpc(/*intr=*/true); break; case EM_RISCV: mask = __perf_reg_mask_riscv(/*intr=*/true); break; case EM_S390: mask = __perf_reg_mask_s390(/*intr=*/true); break; case EM_386: case EM_X86_64: mask = __perf_reg_mask_x86(/*intr=*/true); break; default: pr_debug("Unknown ELF machine %d, interrupt sampling register mask will be empty.\n", e_machine); break; } return mask; } uint64_t perf_user_reg_mask(uint16_t e_machine) { uint64_t mask = 0; switch (e_machine) { case EM_ARM: mask = __perf_reg_mask_arm(/*intr=*/false); break; case EM_AARCH64: mask = __perf_reg_mask_arm64(/*intr=*/false); break; case EM_CSKY: mask = __perf_reg_mask_csky(/*intr=*/false); break; case EM_LOONGARCH: mask = __perf_reg_mask_loongarch(/*intr=*/false); break; case EM_MIPS: mask = __perf_reg_mask_mips(/*intr=*/false); break; case EM_PPC: case EM_PPC64: mask = __perf_reg_mask_powerpc(/*intr=*/false); break; case EM_RISCV: mask = __perf_reg_mask_riscv(/*intr=*/false); break; case EM_S390: mask = __perf_reg_mask_s390(/*intr=*/false); break; case EM_386: case EM_X86_64: mask = __perf_reg_mask_x86(/*intr=*/false); break; default: pr_debug("Unknown ELF machine %d, user sampling register mask will be empty.\n", e_machine); break; } return mask; } const char *perf_reg_name(int id, uint16_t e_machine, uint32_t e_flags) { const char *reg_name = NULL; switch (e_machine) { case EM_ARM: reg_name = __perf_reg_name_arm(id); break; case EM_AARCH64: reg_name = __perf_reg_name_arm64(id); break; case EM_CSKY: reg_name = __perf_reg_name_csky(id, e_flags); break; case EM_LOONGARCH: reg_name = __perf_reg_name_loongarch(id); break; case EM_MIPS: reg_name = __perf_reg_name_mips(id); break; case EM_PPC: case EM_PPC64: reg_name = __perf_reg_name_powerpc(id); break; case EM_RISCV: reg_name = __perf_reg_name_riscv(id); break; case EM_S390: reg_name = __perf_reg_name_s390(id); break; case EM_386: case EM_X86_64: reg_name = __perf_reg_name_x86(id); break; default: break; } if (reg_name) return reg_name; pr_debug("Failed to find register %d for ELF machine type %u\n", id, e_machine); return "unknown"; } int perf_reg_value(u64 *valp, struct regs_dump *regs, int id) { int i, idx = 0; u64 mask = regs->mask; if ((u64)id >= PERF_SAMPLE_REGS_CACHE_SIZE) return -EINVAL; if (regs->cache_mask & (1ULL << id)) goto out; if (!(mask & (1ULL << id))) return -EINVAL; for (i = 0; i < id; i++) { if (mask & (1ULL << i)) idx++; } regs->cache_mask |= (1ULL << id); regs->cache_regs[id] = regs->regs[idx]; out: *valp = regs->cache_regs[id]; return 0; } uint64_t perf_arch_reg_ip(uint16_t e_machine) { switch (e_machine) { case EM_ARM: return __perf_reg_ip_arm(); case EM_AARCH64: return __perf_reg_ip_arm64(); case EM_CSKY: return __perf_reg_ip_csky(); case EM_LOONGARCH: return __perf_reg_ip_loongarch(); case EM_MIPS: return __perf_reg_ip_mips(); case EM_PPC: case EM_PPC64: return __perf_reg_ip_powerpc(); case EM_RISCV: return __perf_reg_ip_riscv(); case EM_S390: return __perf_reg_ip_s390(); case EM_386: case EM_X86_64: return __perf_reg_ip_x86(); default: pr_err("Failed to find IP register for ELF machine type %u\n", e_machine); return 0; } } uint64_t perf_arch_reg_sp(uint16_t e_machine) { switch (e_machine) { case EM_ARM: return __perf_reg_sp_arm(); case EM_AARCH64: return __perf_reg_sp_arm64(); case EM_CSKY: return __perf_reg_sp_csky(); case EM_LOONGARCH: return __perf_reg_sp_loongarch(); case EM_MIPS: return __perf_reg_sp_mips(); case EM_PPC: case EM_PPC64: return __perf_reg_sp_powerpc(); case EM_RISCV: return __perf_reg_sp_riscv(); case EM_S390: return __perf_reg_sp_s390(); case EM_386: case EM_X86_64: return __perf_reg_sp_x86(); default: pr_err("Failed to find SP register for ELF machine type %u\n", e_machine); return 0; } }