interrupt.c 4.8 KB
Newer Older
B
Bernard Xiong 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * File      : interrupt.c
 * This file is part of RT-Thread RTOS
 * COPYRIGHT (C) 2006 - 2011, RT-Thread Development Team
 *
 * The license and distribution terms for this file may be
 * found in the file LICENSE in this distribution or at
 * http://www.rt-thread.org/license/LICENSE
 *
 * Change Logs:
 * Date           Author       Notes
 * 2010-10-15     Bernard      first version
 * 2010-10-15     lgnq         modified for LS1B
14
 * 2013-03-29     aozima       Modify the interrupt interface implementations.
B
Bernard Xiong 已提交
15 16 17
 */

#include <rtthread.h>
18
#include <rthw.h>
B
Bernard Xiong 已提交
19 20 21 22 23 24 25 26 27
#include "ls1b.h"

#define MAX_INTR 32

extern rt_uint32_t rt_interrupt_nest;
rt_uint32_t rt_interrupt_from_thread;
rt_uint32_t rt_interrupt_to_thread;
rt_uint32_t rt_thread_switch_interrupt_flag;

28
static struct rt_irq_desc irq_handle_table[MAX_INTR];
B
Bernard Xiong 已提交
29 30 31 32
void rt_interrupt_dispatch(void *ptreg);
void rt_hw_timer_handler();

static struct ls1b_intc_regs volatile *ls1b_hw0_icregs
G
Grissiom 已提交
33
= (struct ls1b_intc_regs volatile *)(LS1B_INTREG_BASE);
B
Bernard Xiong 已提交
34 35 36 37 38 39 40

/**
 * @addtogroup Loongson LS1B
 */

/*@{*/

41
static void rt_hw_interrupt_handler(int vector, void *param)
B
Bernard Xiong 已提交
42
{
G
Grissiom 已提交
43
    rt_kprintf("Unhandled interrupt %d occured!!!\n", vector);
B
Bernard Xiong 已提交
44 45 46 47 48 49 50
}

/**
 * This function will initialize hardware interrupt
 */
void rt_hw_interrupt_init(void)
{
G
Grissiom 已提交
51
    rt_int32_t idx;
B
Bernard Xiong 已提交
52

G
Grissiom 已提交
53 54 55 56 57 58
    /* pci active low */
    ls1b_hw0_icregs->int_pol = -1;	   //must be done here 20110802 lgnq
    /* make all interrupts level triggered */
    (ls1b_hw0_icregs+0)->int_edge = 0x0000e000;
    /* mask all interrupts */
    (ls1b_hw0_icregs+0)->int_clr = 0xffffffff;
B
Bernard Xiong 已提交
59

60
    rt_memset(irq_handle_table, 0x00, sizeof(irq_handle_table));
G
Grissiom 已提交
61 62 63 64 65 66 67 68 69 70
    for (idx = 0; idx < MAX_INTR; idx ++)
    {
        irq_handle_table[idx].handler = rt_hw_interrupt_handler;
    }

    /* init interrupt nest, and context in thread sp */
    rt_interrupt_nest = 0;
    rt_interrupt_from_thread = 0;
    rt_interrupt_to_thread = 0;
    rt_thread_switch_interrupt_flag = 0;
B
Bernard Xiong 已提交
71 72 73 74 75 76 77 78
}

/**
 * This function will mask a interrupt.
 * @param vector the interrupt number
 */
void rt_hw_interrupt_mask(int vector)
{
G
Grissiom 已提交
79 80
    /* mask interrupt */
    (ls1b_hw0_icregs+(vector>>5))->int_en &= ~(1 << (vector&0x1f));
B
Bernard Xiong 已提交
81 82 83 84 85 86 87 88
}

/**
 * This function will un-mask a interrupt.
 * @param vector the interrupt number
 */
void rt_hw_interrupt_umask(int vector)
{
G
Grissiom 已提交
89
    (ls1b_hw0_icregs+(vector>>5))->int_en |= (1 << (vector&0x1f));
B
Bernard Xiong 已提交
90 91 92 93 94 95 96 97
}

/**
 * This function will install a interrupt service routine to a interrupt.
 * @param vector the interrupt number
 * @param new_handler the interrupt service routine to be installed
 * @param old_handler the old interrupt service routine
 */
98
rt_isr_handler_t rt_hw_interrupt_install(int vector, rt_isr_handler_t handler,
armink_ztl's avatar
armink_ztl 已提交
99
                                         void *param, const char *name)
B
Bernard Xiong 已提交
100
{
101 102
    rt_isr_handler_t old_handler = RT_NULL;

G
Grissiom 已提交
103 104
    if (vector >= 0 && vector < MAX_INTR)
    {
105 106 107
        old_handler = irq_handle_table[vector].handler;

#ifdef RT_USING_INTERRUPT_INFO
108
        rt_strncpy(irq_handle_table[vector].name, name, RT_NAME_MAX);
109
#endif /* RT_USING_INTERRUPT_INFO */
110 111
        irq_handle_table[vector].handler = handler;
        irq_handle_table[vector].param = param;
112 113 114
    }

    return old_handler;
B
Bernard Xiong 已提交
115 116 117 118
}

void rt_interrupt_dispatch(void *ptreg)
{
119 120
    int irq;
    void *param;
G
Grissiom 已提交
121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153
    rt_isr_handler_t irq_func;
    static rt_uint32_t status = 0;
    rt_uint32_t c0_status;
    rt_uint32_t c0_cause;
    volatile rt_uint32_t cause_im;
    volatile rt_uint32_t status_im;
    rt_uint32_t pending_im;

    /* check os timer */
    c0_status = read_c0_status();
    c0_cause = read_c0_cause();

    cause_im = c0_cause & ST0_IM;
    status_im = c0_status & ST0_IM;
    pending_im = cause_im & status_im;

    if (pending_im & CAUSEF_IP7)
    {
        rt_hw_timer_handler();
    }

    if (pending_im & CAUSEF_IP2)
    {
        /* the hardware interrupt */
        status = ls1b_hw0_icregs->int_isr;
        if (!status)
            return;

        for (irq = MAX_INTR; irq > 0; --irq)
        {
            if ((status & (1 << irq)))
            {
                status &= ~(1 << irq);
154 155 156

                irq_func = irq_handle_table[irq].handler;
                param = irq_handle_table[irq].param;
B
Bernard Xiong 已提交
157

G
Grissiom 已提交
158
                /* do interrupt */
159
                irq_func(irq, param);
160 161 162 163

#ifdef RT_USING_INTERRUPT_INFO
                irq_handle_table[irq].counter++;
#endif /* RT_USING_INTERRUPT_INFO */
B
Bernard Xiong 已提交
164

G
Grissiom 已提交
165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185
                /* ack interrupt */
                ls1b_hw0_icregs->int_clr |= (1 << irq);
            }
        }
    }
    else if (pending_im & CAUSEF_IP3)
    {
        rt_kprintf("%s %d\r\n", __FUNCTION__, __LINE__);
    }
    else if (pending_im & CAUSEF_IP4)
    {
        rt_kprintf("%s %d\r\n", __FUNCTION__, __LINE__);
    }
    else if (pending_im & CAUSEF_IP5)
    {
        rt_kprintf("%s %d\r\n", __FUNCTION__, __LINE__);
    }
    else if (pending_im & CAUSEF_IP6)
    {
        rt_kprintf("%s %d\r\n", __FUNCTION__, __LINE__);
    }
B
Bernard Xiong 已提交
186 187 188
}

/*@}*/