source: mainline/kernel/arch/mips32/src/exception.c@ 124bc22

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 124bc22 was 124bc22, checked in by Jakub Jermar <jakub@…>, 6 years ago

Reorganize interrupt and IRQ handling on mips32

This allows msim to use MIPS CPU interrupt numbers as IRQ numbers and
Malta to use ISA IRQ numbers as IRQ numbers. Common code can still
register MIPS CPU interrupts by their respective numbers.

  • Property mode set to 100644
File size: 6.9 KB
RevLine 
[f761f1eb]1/*
[df4ed85]2 * Copyright (c) 2003-2004 Jakub Jermar
[f761f1eb]3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
[c5429fe]29/** @addtogroup kernel_mips32
[b45c443]30 * @{
31 */
32/** @file
33 */
34
[f761f1eb]35#include <arch/exception.h>
[9c0a9b3]36#include <arch/interrupt.h>
[b3f8fb7]37#include <arch/mm/tlb.h>
[f761f1eb]38#include <panic.h>
39#include <arch/cp0.h>
40#include <arch.h>
[63e27ef]41#include <assert.h>
[1084a784]42#include <proc/thread.h>
[7a8c866a]43#include <interrupt.h>
[b2e121a]44#include <halt.h>
[7688b5d]45#include <ddi/irq.h>
[5bb8e45]46#include <arch/debugger.h>
[e2b762ec]47#include <symtab.h>
[b2fa1204]48#include <log.h>
[124bc22]49#include <arch/machine_func.h>
[e2b762ec]50
[a000878c]51static const char *exctable[] = {
[2f40fe4]52 "Interrupt",
53 "TLB Modified",
54 "TLB Invalid",
55 "TLB Invalid Store",
56 "Address Error - load/instr. fetch",
57 "Address Error - store",
58 "Bus Error - fetch instruction",
59 "Bus Error - data reference",
60 "Syscall",
61 "BreakPoint",
62 "Reserved Instruction",
63 "Coprocessor Unusable",
64 "Arithmetic Overflow",
65 "Trap",
66 "Virtual Coherency - instruction",
67 "Floating Point",
68 NULL, NULL, NULL, NULL, NULL, NULL, NULL,
[da1bafb]69 "WatchHi/WatchLo", /* 23 */
[2f40fe4]70 NULL, NULL, NULL, NULL, NULL, NULL, NULL,
71 "Virtual Coherency - data",
[7a8c866a]72};
73
[22a28a69]74void istate_decode(istate_t *istate)
[7a8c866a]75{
[b2fa1204]76 log_printf("epc=%#010" PRIx32 "\tsta=%#010" PRIx32 "\t"
[41a7f62]77 "lo =%#010" PRIx32 "\thi =%#010" PRIx32 "\n",
[8469c53]78 istate->epc, istate->status, istate->lo, istate->hi);
[a35b458]79
[b2fa1204]80 log_printf("a0 =%#010" PRIx32 "\ta1 =%#010" PRIx32 "\t"
[41a7f62]81 "a2 =%#010" PRIx32 "\ta3 =%#010" PRIx32 "\n",
[0c61955]82 istate->a0, istate->a1, istate->a2, istate->a3);
[a35b458]83
[b2fa1204]84 log_printf("t0 =%#010" PRIx32 "\tt1 =%#010" PRIx32 "\t"
[41a7f62]85 "t2 =%#010" PRIx32 "\tt3 =%#010" PRIx32 "\n",
[0c61955]86 istate->t0, istate->t1, istate->t2, istate->t3);
[a35b458]87
[b2fa1204]88 log_printf("t4 =%#010" PRIx32 "\tt5 =%#010" PRIx32 "\t"
[41a7f62]89 "t6 =%#010" PRIx32 "\tt7 =%#010" PRIx32 "\n",
[0c61955]90 istate->t4, istate->t5, istate->t6, istate->t7);
[a35b458]91
[b2fa1204]92 log_printf("t8 =%#010" PRIx32 "\tt9 =%#010" PRIx32 "\t"
[41a7f62]93 "v0 =%#010" PRIx32 "\tv1 =%#010" PRIx32 "\n",
[0c61955]94 istate->t8, istate->t9, istate->v0, istate->v1);
[a35b458]95
[b2fa1204]96 log_printf("s0 =%#010" PRIx32 "\ts1 =%#010" PRIx32 "\t"
[41a7f62]97 "s2 =%#010" PRIx32 "\ts3 =%#010" PRIx32 "\n",
[0c61955]98 istate->s0, istate->s1, istate->s2, istate->s3);
[a35b458]99
[b2fa1204]100 log_printf("s4 =%#010" PRIx32 "\ts5 =%#010" PRIx32 "\t"
[41a7f62]101 "s6 =%#010" PRIx32 "\ts7 =%#010" PRIx32 "\n",
[0c61955]102 istate->s4, istate->s5, istate->s6, istate->s7);
[a35b458]103
[b2fa1204]104 log_printf("s8 =%#010" PRIx32 "\tat =%#010" PRIx32 "\t"
[41a7f62]105 "kt0=%#010" PRIx32 "\tkt1=%#010" PRIx32 "\n",
[0c61955]106 istate->s8, istate->at, istate->kt0, istate->kt1);
[a35b458]107
[b2fa1204]108 log_printf("sp =%#010" PRIx32 "\tra =%#010" PRIx32 "\t"
[8469c53]109 "gp =%#010" PRIx32 "\n",
110 istate->sp, istate->ra, istate->gp);
[7a8c866a]111}
112
[214ec25c]113static void unhandled_exception(unsigned int n, istate_t *istate)
[7a8c866a]114{
[f651e80]115 fault_if_from_uspace(istate, "Unhandled exception %s.", exctable[n]);
[ac11ac7]116 panic_badtrap(istate, n, "Unhandled exception %s.", exctable[n]);
[7a8c866a]117}
118
[214ec25c]119static void reserved_instr_exception(unsigned int n, istate_t *istate)
[3b712407]120{
[a000878c]121 if (*((uint32_t *) istate->epc) == 0x7c03e83b) {
[63e27ef]122 assert(THREAD);
[3b712407]123 istate->epc += 4;
[ce890ec9]124 istate->v1 = istate->kt1;
[a000878c]125 } else
[5201199]126 unhandled_exception(n, istate);
[3b712407]127}
128
[214ec25c]129static void breakpoint_exception(unsigned int n, istate_t *istate)
[7a8c866a]130{
[5bb8e45]131#ifdef CONFIG_DEBUG
[25d7709]132 debugger_bpoint(istate);
[5bb8e45]133#else
[7c3fb9b]134 /*
135 * It is necessary to not re-execute BREAK instruction after
136 * returning from Exception handler
137 * (see page 138 in R4000 Manual for more information)
138 */
[25d7709]139 istate->epc += 4;
[5bb8e45]140#endif
[7a8c866a]141}
142
[214ec25c]143static void tlbmod_exception(unsigned int n, istate_t *istate)
[7a8c866a]144{
[25d7709]145 tlb_modified(istate);
[7a8c866a]146}
147
[214ec25c]148static void tlbinv_exception(unsigned int n, istate_t *istate)
[7a8c866a]149{
[25d7709]150 tlb_invalid(istate);
[7a8c866a]151}
152
[5a95b25]153#ifdef CONFIG_FPU_LAZY
[214ec25c]154static void cpuns_exception(unsigned int n, istate_t *istate)
[7a8c866a]155{
156 if (cp0_cause_coperr(cp0_cause_read()) == fpu_cop_id)
157 scheduler_fpu_lazy_request();
[874621f]158 else {
[ac11ac7]159 fault_if_from_uspace(istate,
160 "Unhandled Coprocessor Unusable Exception.");
161 panic_badtrap(istate, n,
162 "Unhandled Coprocessor Unusable Exception.");
[874621f]163 }
[7a8c866a]164}
[5a95b25]165#endif
[7a8c866a]166
[214ec25c]167static void interrupt_exception(unsigned int n, istate_t *istate)
[7a8c866a]168{
[7be6379]169 uint32_t ip;
170 uint32_t im;
171
[da1bafb]172 /* Decode interrupt number and process the interrupt */
[7be6379]173 ip = (cp0_cause_read() & cp0_cause_ip_mask) >> cp0_cause_ip_shift;
174 im = (cp0_status_read() & cp0_status_im_mask) >> cp0_status_im_shift;
[a35b458]175
[da1bafb]176 unsigned int i;
[124bc22]177 for (i = 0; i < INTERRUPTS; i++) {
[7be6379]178
179 /*
180 * The interrupt could only occur if it is unmasked in the
181 * status register. On the other hand, an interrupt can be
182 * apparently pending even if it is masked, so we need to
183 * check both the masked and pending interrupts.
184 */
185 if (im & ip & (1 << i)) {
[124bc22]186 if (int_handler[i])
187 int_handler[i](i);
188 else {
[7688b5d]189 /*
190 * Spurious interrupt.
191 */
192#ifdef CONFIG_DEBUG
[b2fa1204]193 log(LF_ARCH, LVL_DEBUG,
194 "cpu%u: spurious interrupt (inum=%u)",
[6cd9aa6]195 CPU->id, i);
[7688b5d]196#endif
197 }
198 }
199 }
[7a8c866a]200}
201
[1b109cb]202/** Handle syscall userspace call */
[214ec25c]203static void syscall_exception(unsigned int n, istate_t *istate)
[f761f1eb]204{
[ac11ac7]205 fault_if_from_uspace(istate, "Syscall is handled through shortcut.");
[f761f1eb]206}
[7a8c866a]207
208void exception_init(void)
209{
[b3b7e14a]210 unsigned int i;
[a35b458]211
[7a8c866a]212 /* Clear exception table */
[7688b5d]213 for (i = 0; i < IVT_ITEMS; i++)
[b3b7e14a]214 exc_register(i, "undef", false,
215 (iroutine_t) unhandled_exception);
[a35b458]216
[b3b7e14a]217 exc_register(EXC_Bp, "bkpoint", true,
218 (iroutine_t) breakpoint_exception);
219 exc_register(EXC_RI, "resinstr", true,
220 (iroutine_t) reserved_instr_exception);
221 exc_register(EXC_Mod, "tlb_mod", true,
222 (iroutine_t) tlbmod_exception);
223 exc_register(EXC_TLBL, "tlbinvl", true,
224 (iroutine_t) tlbinv_exception);
225 exc_register(EXC_TLBS, "tlbinvl", true,
226 (iroutine_t) tlbinv_exception);
227 exc_register(EXC_Int, "interrupt", true,
228 (iroutine_t) interrupt_exception);
[a35b458]229
[7a8c866a]230#ifdef CONFIG_FPU_LAZY
[b3b7e14a]231 exc_register(EXC_CpU, "cpunus", true,
232 (iroutine_t) cpuns_exception);
[7a8c866a]233#endif
[a35b458]234
[b3b7e14a]235 exc_register(EXC_Sys, "syscall", true,
236 (iroutine_t) syscall_exception);
[7a8c866a]237}
[b45c443]238
[3c5006a0]239/** @}
[b45c443]240 */
Note: See TracBrowser for help on using the repository browser.