From cb4b5b7a39bdc5b9feb14881e1b8c26de4ab0ade Mon Sep 17 00:00:00 2001 From: lephe Date: Sun, 30 Apr 2017 18:24:35 +0200 Subject: [PATCH] Untested INTC module in /include/modules/interrupts.h --- include/modules/interrupts.h | 200 ++++++++++++++++++++++++++++++++++- src/core/modules.c | 16 ++- version | 2 +- 3 files changed, 212 insertions(+), 6 deletions(-) diff --git a/include/modules/interrupts.h b/include/modules/interrupts.h index 47a0129..41e6bae 100644 --- a/include/modules/interrupts.h +++ b/include/modules/interrupts.h @@ -464,6 +464,178 @@ typedef mod_intc_irq_7305_t mod_intc_intreq00_7305_t; typedef mod_intc_irq_7305_t mod_intc_intmsk00_7305_t; typedef mod_intc_irq_7305_t mod_intc_intmskclr00_7305_t; +/* + mod_intc_masks_7305_t + A set of bits to mask individual interrupts. + - Masking interrupts is realized by writing 1 to IMRs ; + - Clearing masks is realized by writing 1 to IMCRs ; + Using the wrong register set, such as writing 0 to IMRs to clear a + mask, will be ignored and have no effect. Reading from IMCRs yields an + undefined value. +*/ +typedef struct +{ + byte_union(IMR0, + uint :1; + uint TUNI1_2 :1; /* TMU1 overflow interrupts */ + uint TUNI1_1 :1; + uint TUNI1_0 :1; + uint SDHII3 :1; /* SD Card Host 1 interrupts */ + uint SDHII2 :1; + uint SDHII1 :1; + uint SDHII0 :1; + ); + pad(3); + + byte_union(IMR1, + uint :4; + uint DEI3 :1; /* DMAC0A interrupts */ + uint DEI2 :1; + uint DEI1 :1; + uint DEI0 :1; + ); + pad(3); + + byte_union(IMR2, + uint :7; + uint SCIFA0 :1; /* Asynchronous Serial interrupts */ + ); + pad(3); + + byte_union(IMR3, + uint DEI3 :1; /* DMAC1A interrupts */ + uint DEI2 :1; + uint DEI1 :1; + uint DEI0 :1; + uint :4; + ); + pad(3); + + byte_union(IMR4, + uint :1; + uint TUNI0_2 :1; /* TMU0 overflow interrupts */ + uint TUNI0_1 :1; + uint TUNI0_0 :1; + uint :3; + uint LCDCI :1; /* LCD Controller Interrupt */ + ); + pad(3); + + byte_union(IMR5, + uint KEYI :1; /* Key Interface */ + uint DADERR :1; /* DMAC0B interrupts */ + uint DEI5 :1; + uint DEI4 :1; + uint :1; + uint SCIF2 :1; /* Serial Communication Interface */ + uint SCIF1 :1; + uint SCIF0 :1; + ); + pad(3); + + byte_union(IMR6, + uint :2; + uint :1; + uint SCIFA4 :1; /* SCIFA4 interrupt */ + uint :1; + uint :1; + uint MSIOFI0 :1; /* Clock-synchronized SCIF channel 0 */ + uint MSIOFI1 :1; /* Clock-synchronized SCIF channel 1 */ + ); + pad(3); + + uint8_t IMR7; + pad(3); + + byte_union(IMR8, + uint SDHII3 :1; /* SD Card Host 0 interrupts */ + uint SDHII2 :1; + uint SDHII1 :1; + uint SDHII0 :1; + uint :2; + uint SCFIA5 :1; /* SCIFA5 interrupt */ + uint FSI :1; /* FIFO-Buffered Serial Interface */ + ); + pad(3); + + byte_union(IMR9, + uint :3; + uint CMTI :1; /* Compare Match Timer Interrupt */ + uint :1; + uint USI1 :1; /* USB1 */ + uint USI0 :1; /* USB0 */ + uint :1; + ); + pad(3); + + byte_union(IMR10, + uint :1; + uint DADERR :1; /* DMAC1B interrupts */ + uint DEI5 :1; + uint DEI4 :1; + uint :1; + uint ATI :1; /* RTC Alarm interrupt */ + uint PRI :1; /* RTC Periodic interrupt */ + uint CUI :1; /* RTC Carry interrupt */ + ); + pad(3); + + byte_union(IMR11, + uint :5; + uint TPUI :1; /* Timer-Pulse Unit */ + uint :2; + ); + pad(3); + + uint8_t IMR12; + char gap2[15]; + +} __attribute__((packed)) mod_intc_masks_7305_t; + +/* + mod_intc_userimask_7305_t + User Interrupt Mask: Specifies the minimum required level for + interrupts to be accepted. + + WARNING: Writing to this register requires the eight upper bits of the + operand (ie. the new value of USERIMASK) to be 0xa5; otherwise, the + write is ignored. To modify the value of this register, do not access + the bit field directly, backup the variable and modify it: + + void set_user_imask(int new_level) + { + mod_intc_userimask_7305_t mask = *(INTC._7305.USERIMASK); + mask._a5 = 0xa5; + mask.UIMASK = new_level & 0x0f; + *(INTC._7305.USERIMASK) = mask; + } +*/ +typedef struct +{ + lword_union(, + uint _a5 :8; /* Always set to 0xa5 before writing */ + uint :16; + uint UIMASK :4; /* User Interrupt Mask Level */ + uint :4; + ); + +} __attribute__((packed, aligned(4))) mod_intc_userimask_7305_t; + +/* + mod_intc_nmifcr_7305_t + NMI Flag Control Register: Indicates the state of the NMI pin and the + NMI interrupt request. +*/ +typedef struct +{ + byte_union(, + uint const NMIL :1; /* NMI Interupt Level */ + uint :14; + uint NMIFL :1; /* NMI Interrupt Request Flag */ + ); + +} __attribute__((packed, aligned(2))) mod_intc_nmifcr_7305_t; + /* mod_intc_7305_t Finally the whole interrupt controller. @@ -480,10 +652,34 @@ typedef struct /* Interrupt mask & mask clear registers */ volatile mod_intc_intmsk00_7305_t *INTMSK00; -// volatile mod_intc_masks_7305_t *masks; + volatile mod_intc_masks_7305_t *masks; volatile mod_intc_intmskclr00_7305_t *INTMSKCLR00; -// volatile mod_intc_masks_clear_7305_t *masks_clear; + volatile mod_intc_masks_7305_t *masks_clear; + + /* Other registers */ + volatile mod_intc_intreq00_7305_t *INTREQ00; + volatile mod_intc_userimask_7305_t *USERIMASK; + volatile mod_intc_nmifcr_7305_t *NMIFCR; } mod_intc_7305_t; + + +//--- +// Platform-independent structures. +// Unfortunately there is nothing here. Users willing to manage interrupts +// using the INTC register will have to handle explicitely both platforms. +//--- + +/* + mod_intc_t + Interrupt Controller. +*/ +typedef union +{ + mod_intc_7705_t _7705; + mod_intc_7305_t _7305; + +} __attribute__((packed)) mod_intc_t; + #endif // _MODULE_INTERRUPTS diff --git a/src/core/modules.c b/src/core/modules.c index 827110c..9e5c429 100644 --- a/src/core/modules.c +++ b/src/core/modules.c @@ -1,5 +1,6 @@ #include #include +#include #include #include @@ -15,6 +16,7 @@ mod_tmu_t TMU; mod_rtc_t RTC; +mod_intc_t INTC; @@ -24,7 +26,6 @@ mod_rtc_t RTC; static void mod_init_7705(void) { -/* INTC._7705.iprs.IPRA = (void *)0xfffffee2; INTC._7705.iprs.IPRB = (void *)0xfffffee4; INTC._7705.iprs.IPRC = (void *)0xa4000016; @@ -41,7 +42,6 @@ static void mod_init_7705(void) INTC._7705.IRR0 = (void *)0xa4000004; INTC._7705.IRR1 = (void *)0xa4000006; INTC._7705.IRR2 = (void *)0xa4000008; -*/ TMU.timers[0] = (void *)0xfffffe94; TMU.timers[1] = (void *)0xfffffea0; @@ -56,7 +56,17 @@ static void mod_init_7705(void) static void mod_init_7305(void) { -// INTC._7305.iprs = (void *)0xa4080000; + INTC._7305.ICR0 = (void *)0xa4140000; + INTC._7305.ICR1 = (void *)0xa414001c; + INTC._7305.INTPRI00 = (void *)0xa4140010; + INTC._7305.iprs = (void *)0xa4080000; + INTC._7305.INTMSK00 = (void *)0xa4140044; + INTC._7305.masks = (void *)0xa4080080; + INTC._7305.INTMSKCLR00 = (void *)0xa4140064; + INTC._7305.masks_clear = (void *)0xa40800c0; + INTC._7305.INTREQ00 = (void *)0xa4140024; + INTC._7305.USERIMASK = (void *)0xa4700000; + INTC._7305.NMIFCR = (void *)0xa41400c0; TMU.timers[0] = (void *)0xa4490008; TMU.timers[1] = (void *)0xa4490014; diff --git a/version b/version index 7409007..405a713 100644 --- a/version +++ b/version @@ -1 +1 @@ -beta-0.9-352 +beta-0.9-354