123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367 |
- /*
- * Copyright (c) 2021, Shenzhen Academy of Aerospace Technology
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2021-11-16 Dystopia the first version
- */
- #include "trap.h"
- #include "c66xx.h"
- #include <rthw.h>
- #include <rtdef.h>
- #include <rtthread.h>
- #define RT_SYS_STACK_SIZE 4096
- rt_uint8_t rt_system_stack[RT_SYS_STACK_SIZE];
- rt_uint8_t *rt_system_stack_top;
- void rt_trap_init(void)
- {
- rt_system_stack_top = &rt_system_stack[RT_SYS_STACK_SIZE-1];
- rt_system_stack_top = (rt_uint8_t *)RT_ALIGN_DOWN((rt_uint32_t)rt_system_stack_top, 8);
- ack_exception(EXCEPT_TYPE_NXF);
- ack_exception(EXCEPT_TYPE_EXC);
- ack_exception(EXCEPT_TYPE_IXF);
- ack_exception(EXCEPT_TYPE_SXF);
- rt_hw_enable_exception();
- }
- void show_regs(struct rt_hw_exp_stack_register *regs)
- {
- rt_kprintf("\n");
- rt_kprintf("PC: %08lx SP: %08lx\n",
- regs->pc, regs->hw_register.sp);
- rt_kprintf("Status: %08lx ORIG_A4: %08lx\n",
- regs->csr, regs->orig_a4);
- rt_kprintf("A0: %08lx B0: %08lx\n",
- regs->hw_register.a0, regs->hw_register.b0);
- rt_kprintf("A1: %08lx B1: %08lx\n",
- regs->hw_register.a1, regs->hw_register.b1);
- rt_kprintf("A2: %08lx B2: %08lx\n",
- regs->hw_register.a2, regs->hw_register.b2);
- rt_kprintf("A3: %08lx B3: %08lx\n",
- regs->hw_register.a3, regs->hw_register.b3);
- rt_kprintf("A4: %08lx B4: %08lx\n",
- regs->hw_register.a4, regs->hw_register.b4);
- rt_kprintf("A5: %08lx B5: %08lx\n",
- regs->hw_register.a5, regs->hw_register.b5);
- rt_kprintf("A6: %08lx B6: %08lx\n",
- regs->hw_register.a6, regs->hw_register.b6);
- rt_kprintf("A7: %08lx B7: %08lx\n",
- regs->hw_register.a7, regs->hw_register.b7);
- rt_kprintf("A8: %08lx B8: %08lx\n",
- regs->hw_register.a8, regs->hw_register.b8);
- rt_kprintf("A9: %08lx B9: %08lx\n",
- regs->hw_register.a9, regs->hw_register.b9);
- rt_kprintf("A10: %08lx B10: %08lx\n",
- regs->hw_register.a10, regs->hw_register.b10);
- rt_kprintf("A11: %08lx B11: %08lx\n",
- regs->hw_register.a11, regs->hw_register.b11);
- rt_kprintf("A12: %08lx B12: %08lx\n",
- regs->hw_register.a12, regs->hw_register.b12);
- rt_kprintf("A13: %08lx B13: %08lx\n",
- regs->hw_register.a13, regs->hw_register.b13);
- rt_kprintf("A14: %08lx B14: %08lx\n",
- regs->hw_register.a14, regs->hw_register.dp);
- rt_kprintf("A15: %08lx B15: %08lx\n",
- regs->hw_register.a15, regs->hw_register.sp);
- rt_kprintf("A16: %08lx B16: %08lx\n",
- regs->hw_register.a16, regs->hw_register.b16);
- rt_kprintf("A17: %08lx B17: %08lx\n",
- regs->hw_register.a17, regs->hw_register.b17);
- rt_kprintf("A18: %08lx B18: %08lx\n",
- regs->hw_register.a18, regs->hw_register.b18);
- rt_kprintf("A19: %08lx B19: %08lx\n",
- regs->hw_register.a19, regs->hw_register.b19);
- rt_kprintf("A20: %08lx B20: %08lx\n",
- regs->hw_register.a20, regs->hw_register.b20);
- rt_kprintf("A21: %08lx B21: %08lx\n",
- regs->hw_register.a21, regs->hw_register.b21);
- rt_kprintf("A22: %08lx B22: %08lx\n",
- regs->hw_register.a22, regs->hw_register.b22);
- rt_kprintf("A23: %08lx B23: %08lx\n",
- regs->hw_register.a23, regs->hw_register.b23);
- rt_kprintf("A24: %08lx B24: %08lx\n",
- regs->hw_register.a24, regs->hw_register.b24);
- rt_kprintf("A25: %08lx B25: %08lx\n",
- regs->hw_register.a25, regs->hw_register.b25);
- rt_kprintf("A26: %08lx B26: %08lx\n",
- regs->hw_register.a26, regs->hw_register.b26);
- rt_kprintf("A27: %08lx B27: %08lx\n",
- regs->hw_register.a27, regs->hw_register.b27);
- rt_kprintf("A28: %08lx B28: %08lx\n",
- regs->hw_register.a28, regs->hw_register.b28);
- rt_kprintf("A29: %08lx B29: %08lx\n",
- regs->hw_register.a29, regs->hw_register.b29);
- rt_kprintf("A30: %08lx B30: %08lx\n",
- regs->hw_register.a30, regs->hw_register.b30);
- rt_kprintf("A31: %08lx B31: %08lx\n",
- regs->hw_register.a31, regs->hw_register.b31);
- }
- void do_trap(struct rt_exception_info *except_info, struct rt_hw_exp_stack_register *regs)
- {
- rt_kprintf("Enter exception: %s\n", except_info->kernel_str);
- show_regs(regs);
- for(;;){}
- }
- static struct rt_exception_info iexcept_table[10] = {
- { " - instruction fetch", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - fetch packet", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - execute packet", ABORT_TYPE_UNDDEF, ABORT_OPCODE_ILL },
- { " - undefined instruction", ABORT_TYPE_UNDDEF, ABORT_OPCODE_ILL },
- { " - resource conflict", ABORT_TYPE_UNDDEF, ABORT_OPCODE_ILL },
- { " - resource access", ABORT_TYPE_UNDDEF, ABORT_PRVREG_ILL },
- { " - privilege", ABORT_TYPE_UNDDEF, ABORT_PRVOPC_ILL },
- { " - loops buffer", ABORT_TYPE_UNDDEF, ABORT_PRVOPC_ILL },
- { " - software exception", ABORT_TYPE_UNDDEF, ABORT_ILLTRP_ILL },
- { " - unknown exception", ABORT_TYPE_UNDDEF, ABORT_PRVOPC_ILL }
- };
- /*
- * Process an internal exception (non maskable)
- */
- static int process_iexcept(struct rt_hw_exp_stack_register *regs)
- {
- unsigned int iexcept_report = get_iexcept();
- unsigned int iexcept_num = 0;
- ack_exception(EXCEPT_TYPE_IXF);
- while(iexcept_report)
- {
- iexcept_num = ffs(iexcept_report);
- iexcept_report &= ~(1 << iexcept_num);
- set_iexcept(iexcept_report);
- if (*(unsigned int *)regs->pc == BKPT_OPCODE)
- {
- /* This is a breakpoint */
- struct rt_exception_info bkpt_exception = \
- { " - undefined instruction",\
- ABORT_TYPE_UNDDEF, ABORT_BRKPT_ILL };
- do_trap(&bkpt_exception, regs);
- iexcept_report &= ~(0xFF);
- set_iexcept(iexcept_report);
- continue;
- }
- do_trap(&iexcept_table[iexcept_num], regs);
- }
- return 0;
- }
- static struct rt_exception_info eexcept_table[128] = {
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - external exception", ABORT_TYPE_BUS, ABORT_BUS_ADDRERR },
- { " - CPU memory protection fault", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - CPU memory protection fault in L1P", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - DMA memory protection fault in L1P", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - CPU memory protection fault in L1D", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - DMA memory protection fault in L1D", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - CPU memory protection fault in L2", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - DMA memory protection fault in L2", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - EMC CPU memory protection fault", ABORT_TYPE_MAP, ABORT_BUS_ACCERR },
- { " - EMC bus error", ABORT_TYPE_MAP, ABORT_BUS_ADDRERR }
- };
- /*
- * Process an external exception (maskable)
- */
- static void process_eexcept(struct rt_hw_exp_stack_register *regs)
- {
- int except_num = 0;
- int bank = 0;
- int i = 0;
- for (i = 0; i <= 3; i++)
- {
- while (INTC_MEXPMASK[i])
- {
- __dint();
- except_num = ffs(INTC_MEXPMASK[i]);
- INTC_MEXPMASK[i] &= ~(1 << except_num); /* ack the external exception */
- __rint();
- do_trap(&eexcept_table[except_num + (bank << 5)], regs);
- }
- bank++;
- }
- ack_exception(EXCEPT_TYPE_EXC);
- }
- extern void hw_nmi_handler(struct rt_hw_exp_stack_register *regs);
- /*
- * Main exception processing
- */
- int rt_hw_process_exception(struct rt_hw_exp_stack_register *regs)
- {
- int type = 0;
- int type_num = 0;
- int ie_num = 9; /* default is unknown exception */
- while ((type = get_except_type()) != 0) {
- type_num = fls(type) - 1;
- switch(type_num) {
- case EXCEPT_TYPE_NXF: /* NMI exception */
- ack_exception(EXCEPT_TYPE_NXF); /* clear exception */
- if (hw_nmi_handler != RT_NULL)
- {
- hw_nmi_handler(regs);
- }
- break;
- case EXCEPT_TYPE_IXF: /* internal exception */
- if (process_iexcept(regs))
- {
- return 1;
- }
- break;
- case EXCEPT_TYPE_EXC: /* external exception */
- process_eexcept(regs);
- break;
- case EXCEPT_TYPE_SXF: /* software exception */
- ie_num = 8;
- ack_exception(type_num);
- break;
- default: /* clear exception */
- ack_exception(type_num);
- do_trap(&iexcept_table[ie_num], regs);
- break;
- }
- }
- return 0;
- }
|