| /* |
| * processor control and status functions |
| * |
| * Copyright (C) 2014, Red Hat Inc, Andrew Jones <drjones@redhat.com> |
| * |
| * This work is licensed under the terms of the GNU LGPL, version 2. |
| */ |
| #include <libcflat.h> |
| #include <asm/ptrace.h> |
| #include <asm/processor.h> |
| #include <asm/thread_info.h> |
| |
| static const char *vector_names[] = { |
| "el1t_sync", |
| "el1t_irq", |
| "el1t_fiq", |
| "el1t_error", |
| "el1h_sync", |
| "el1h_irq", |
| "el1h_fiq", |
| "el1h_error", |
| "el0_sync_64", |
| "el0_irq_64", |
| "el0_fiq_64", |
| "el0_error_64", |
| "el0_sync_32", |
| "el0_irq_32", |
| "el0_fiq_32", |
| "el0_error_32", |
| }; |
| |
| static const char *ec_names[EC_MAX] = { |
| [ESR_EL1_EC_UNKNOWN] = "UNKNOWN", |
| [ESR_EL1_EC_WFI] = "WFI", |
| [ESR_EL1_EC_CP15_32] = "CP15_32", |
| [ESR_EL1_EC_CP15_64] = "CP15_64", |
| [ESR_EL1_EC_CP14_MR] = "CP14_MR", |
| [ESR_EL1_EC_CP14_LS] = "CP14_LS", |
| [ESR_EL1_EC_FP_ASIMD] = "FP_ASMID", |
| [ESR_EL1_EC_CP10_ID] = "CP10_ID", |
| [ESR_EL1_EC_CP14_64] = "CP14_64", |
| [ESR_EL1_EC_ILL_ISS] = "ILL_ISS", |
| [ESR_EL1_EC_SVC32] = "SVC32", |
| [ESR_EL1_EC_SVC64] = "SVC64", |
| [ESR_EL1_EC_SYS64] = "SYS64", |
| [ESR_EL1_EC_IABT_EL0] = "IABT_EL0", |
| [ESR_EL1_EC_IABT_EL1] = "IABT_EL1", |
| [ESR_EL1_EC_PC_ALIGN] = "PC_ALIGN", |
| [ESR_EL1_EC_DABT_EL0] = "DABT_EL0", |
| [ESR_EL1_EC_DABT_EL1] = "DABT_EL1", |
| [ESR_EL1_EC_SP_ALIGN] = "SP_ALIGN", |
| [ESR_EL1_EC_FP_EXC32] = "FP_EXC32", |
| [ESR_EL1_EC_FP_EXC64] = "FP_EXC64", |
| [ESR_EL1_EC_SERROR] = "SERROR", |
| [ESR_EL1_EC_BREAKPT_EL0] = "BREAKPT_EL0", |
| [ESR_EL1_EC_BREAKPT_EL1] = "BREAKPT_EL1", |
| [ESR_EL1_EC_SOFTSTP_EL0] = "SOFTSTP_EL0", |
| [ESR_EL1_EC_SOFTSTP_EL1] = "SOFTSTP_EL1", |
| [ESR_EL1_EC_WATCHPT_EL0] = "WATCHPT_EL0", |
| [ESR_EL1_EC_WATCHPT_EL1] = "WATCHPT_EL1", |
| [ESR_EL1_EC_BKPT32] = "BKPT32", |
| [ESR_EL1_EC_BRK64] = "BRK64", |
| }; |
| |
| void show_regs(struct pt_regs *regs) |
| { |
| int i; |
| |
| printf("pc : [<%016lx>] lr : [<%016lx>] pstate: %08lx\n", |
| regs->pc, regs->regs[30], regs->pstate); |
| printf("sp : %016lx\n", regs->sp); |
| |
| for (i = 29; i >= 0; --i) { |
| printf("x%-2d: %016lx ", i, regs->regs[i]); |
| if (i % 2 == 0) |
| printf("\n"); |
| } |
| printf("\n"); |
| } |
| |
| bool get_far(unsigned int esr, unsigned long *far) |
| { |
| unsigned int ec = esr >> ESR_EL1_EC_SHIFT; |
| |
| asm volatile("mrs %0, far_el1": "=r" (*far)); |
| |
| switch (ec) { |
| case ESR_EL1_EC_IABT_EL0: |
| case ESR_EL1_EC_IABT_EL1: |
| case ESR_EL1_EC_PC_ALIGN: |
| case ESR_EL1_EC_DABT_EL0: |
| case ESR_EL1_EC_DABT_EL1: |
| case ESR_EL1_EC_WATCHPT_EL0: |
| case ESR_EL1_EC_WATCHPT_EL1: |
| if ((esr & 0x3f /* DFSC */) != 0x10 |
| || !(esr & 0x400 /* FnV */)) |
| return true; |
| } |
| return false; |
| } |
| |
| extern unsigned long _text; |
| |
| static void bad_exception(enum vector v, struct pt_regs *regs, |
| unsigned int esr, bool esr_valid, bool bad_vector) |
| { |
| unsigned long far; |
| bool far_valid = get_far(esr, &far); |
| unsigned int ec = esr >> ESR_EL1_EC_SHIFT; |
| uintptr_t text = (uintptr_t)&_text; |
| |
| printf("Load address: %" PRIxPTR "\n", text); |
| printf("PC: %" PRIxPTR " PC offset: %" PRIxPTR "\n", |
| (uintptr_t)regs->pc, (uintptr_t)regs->pc - text); |
| |
| if (bad_vector) { |
| if (v < VECTOR_MAX) |
| printf("Unhandled vector %d (%s)\n", v, |
| vector_names[v]); |
| else |
| printf("Got bad vector=%d\n", v); |
| } else if (esr_valid) { |
| if (ec_names[ec]) |
| printf("Unhandled exception ec=%#x (%s)\n", ec, |
| ec_names[ec]); |
| else |
| printf("Got bad ec=%#x\n", ec); |
| } |
| |
| printf("Vector: %d (%s)\n", v, vector_names[v]); |
| printf("ESR_EL1: %8s%08x, ec=%#x (%s)\n", "", esr, ec, ec_names[ec]); |
| printf("FAR_EL1: %016lx (%svalid)\n", far, far_valid ? "" : "not "); |
| dump_stack(); |
| printf("Exception frame registers:\n"); |
| show_regs(regs); |
| abort(); |
| } |
| |
| void install_exception_handler(enum vector v, unsigned int ec, exception_fn fn) |
| { |
| struct thread_info *ti = current_thread_info(); |
| |
| if (v < VECTOR_MAX && ec < EC_MAX) |
| ti->exception_handlers[v][ec] = fn; |
| } |
| |
| void install_irq_handler(enum vector v, irq_handler_fn fn) |
| { |
| struct thread_info *ti = current_thread_info(); |
| |
| if (v < VECTOR_MAX) |
| ti->exception_handlers[v][0] = (exception_fn)fn; |
| } |
| |
| void default_vector_sync_handler(enum vector v, struct pt_regs *regs, |
| unsigned int esr) |
| { |
| struct thread_info *ti = thread_info_sp(regs->sp); |
| unsigned int ec = esr >> ESR_EL1_EC_SHIFT; |
| |
| if (ti->flags & TIF_USER_MODE) { |
| if (ec < EC_MAX && ti->exception_handlers[v][ec]) { |
| ti->exception_handlers[v][ec](regs, esr); |
| return; |
| } |
| ti = current_thread_info(); |
| } |
| |
| if (ec < EC_MAX && ti->exception_handlers[v][ec]) |
| ti->exception_handlers[v][ec](regs, esr); |
| else |
| bad_exception(v, regs, esr, true, false); |
| } |
| |
| void default_vector_irq_handler(enum vector v, struct pt_regs *regs, |
| unsigned int esr) |
| { |
| struct thread_info *ti = thread_info_sp(regs->sp); |
| irq_handler_fn irq_handler = |
| (irq_handler_fn)ti->exception_handlers[v][0]; |
| |
| if (ti->flags & TIF_USER_MODE) { |
| if (irq_handler) { |
| irq_handler(regs); |
| return; |
| } |
| ti = current_thread_info(); |
| irq_handler = (irq_handler_fn)ti->exception_handlers[v][0]; |
| } |
| |
| if (irq_handler) |
| irq_handler(regs); |
| else |
| bad_exception(v, regs, esr, false, false); |
| } |
| |
| void vector_handlers_default_init(vector_fn *handlers) |
| { |
| handlers[EL1H_SYNC] = default_vector_sync_handler; |
| handlers[EL1H_IRQ] = default_vector_irq_handler; |
| handlers[EL0_SYNC_64] = default_vector_sync_handler; |
| handlers[EL0_IRQ_64] = default_vector_irq_handler; |
| } |
| |
| /* Needed to compile with -Wmissing-prototypes */ |
| void do_handle_exception(enum vector v, struct pt_regs *regs, unsigned int esr); |
| |
| void do_handle_exception(enum vector v, struct pt_regs *regs, unsigned int esr) |
| { |
| struct thread_info *ti = thread_info_sp(regs->sp); |
| |
| if (ti->flags & TIF_USER_MODE) { |
| if (v < VECTOR_MAX && ti->vector_handlers[v]) { |
| ti->vector_handlers[v](v, regs, esr); |
| return; |
| } |
| ti = current_thread_info(); |
| } |
| |
| if (v < VECTOR_MAX && ti->vector_handlers[v]) |
| ti->vector_handlers[v](v, regs, esr); |
| else |
| bad_exception(v, regs, esr, true, true); |
| } |
| |
| void install_vector_handler(enum vector v, vector_fn fn) |
| { |
| struct thread_info *ti = current_thread_info(); |
| |
| if (v < VECTOR_MAX) |
| ti->vector_handlers[v] = fn; |
| } |
| |
| static void __thread_info_init(struct thread_info *ti, unsigned int flags) |
| { |
| memset(ti, 0, sizeof(struct thread_info)); |
| ti->cpu = mpidr_to_cpu(get_mpidr()); |
| ti->flags = flags; |
| } |
| |
| void thread_info_init(struct thread_info *ti, unsigned int flags) |
| { |
| __thread_info_init(ti, flags); |
| vector_handlers_default_init(ti->vector_handlers); |
| } |
| |
| void start_usr(void (*func)(void *arg), void *arg, unsigned long sp_usr) |
| { |
| sp_usr &= (~15UL); /* stack ptr needs 16-byte alignment */ |
| |
| __thread_info_init(thread_info_sp(sp_usr), TIF_USER_MODE); |
| thread_info_sp(sp_usr)->pgtable = current_thread_info()->pgtable; |
| |
| asm volatile( |
| "mov x0, %0\n" |
| "msr sp_el0, %1\n" |
| "msr elr_el1, %2\n" |
| "mov x3, xzr\n" /* clear and "set" PSR_MODE_EL0t */ |
| "msr spsr_el1, x3\n" |
| "eret\n" |
| :: "r" (arg), "r" (sp_usr), "r" (func) : "x0", "x3"); |
| } |
| |
| bool is_user(void) |
| { |
| return current_thread_info()->flags & TIF_USER_MODE; |
| } |
| |
| bool __mmu_enabled(void) |
| { |
| return read_sysreg(sctlr_el1) & SCTLR_EL1_M; |
| } |