source: mainline/kernel/arch/mips32/src/interrupt.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: 3.6 KB
Line 
1/*
2 * Copyright (c) 2003-2004 Jakub Jermar
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
29/** @addtogroup kernel_mips32_interrupt
30 * @{
31 */
32/** @file
33 */
34
35#include <interrupt.h>
36#include <arch/interrupt.h>
37#include <typedefs.h>
38#include <arch.h>
39#include <arch/cp0.h>
40#include <time/clock.h>
41#include <ipc/sysipc.h>
42
43// TODO: This is SMP unsafe!!!
44
45uint32_t count_hi = 0;
46static unsigned long nextcount;
47static unsigned long lastcount;
48
49/** Table of interrupt handlers. */
50int_handler_t int_handler[INTERRUPTS] = {};
51
52/** Disable interrupts.
53 *
54 * @return Old interrupt priority level.
55 */
56ipl_t interrupts_disable(void)
57{
58 ipl_t ipl = (ipl_t) cp0_status_read();
59 cp0_status_write(ipl & ~cp0_status_ie_enabled_bit);
60 return ipl;
61}
62
63/** Enable interrupts.
64 *
65 * @return Old interrupt priority level.
66 */
67ipl_t interrupts_enable(void)
68{
69 ipl_t ipl = (ipl_t) cp0_status_read();
70 cp0_status_write(ipl | cp0_status_ie_enabled_bit);
71 return ipl;
72}
73
74/** Restore interrupt priority level.
75 *
76 * @param ipl Saved interrupt priority level.
77 */
78void interrupts_restore(ipl_t ipl)
79{
80 cp0_status_write(cp0_status_read() | (ipl & cp0_status_ie_enabled_bit));
81}
82
83/** Read interrupt priority level.
84 *
85 * @return Current interrupt priority level.
86 */
87ipl_t interrupts_read(void)
88{
89 return cp0_status_read();
90}
91
92/** Check interrupts state.
93 *
94 * @return True if interrupts are disabled.
95 *
96 */
97bool interrupts_disabled(void)
98{
99 return !(cp0_status_read() & cp0_status_ie_enabled_bit);
100}
101
102/** Start hardware clock
103 *
104 */
105static void timer_start(void)
106{
107 lastcount = cp0_count_read();
108 nextcount = cp0_compare_value + cp0_count_read();
109 cp0_compare_write(nextcount);
110}
111
112static void timer_interrupt_handler(unsigned int intr)
113{
114 if (cp0_count_read() < lastcount)
115 /* Count overflow detected */
116 count_hi++;
117
118 lastcount = cp0_count_read();
119
120 unsigned long drift = cp0_count_read() - nextcount;
121 while (drift > cp0_compare_value) {
122 drift -= cp0_compare_value;
123 CPU->missed_clock_ticks++;
124 }
125
126 nextcount = cp0_count_read() + cp0_compare_value - drift;
127 cp0_compare_write(nextcount);
128
129 clock();
130}
131
132/* Initialize basic tables for exception dispatching */
133void interrupt_init(void)
134{
135 int_handler[INT_TIMER] = timer_interrupt_handler;
136
137 timer_start();
138 cp0_unmask_int(INT_TIMER);
139}
140
141/** @}
142 */
Note: See TracBrowser for help on using the repository browser.