/**************************************************************************** * arch/arm/src/s32k1xx/s32k1xx_pinirq.c * * Copyright (C) 2019 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name NuttX nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include "up_arch.h" #include "up_internal.h" #include "s32k1xx.h" #include "hardware/s32k1xx_port.h" #ifdef CONFIG_S32K1XX_GPIOIRQ /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ /* Configuration ************************************************************/ /* The S32K1xx port interrupt logic is very flexible and will program * interrupts on most all pin events. In order to keep the memory usage to * a minimum, the NuttX port supports enabling interrupts on a per-port * basis. */ #if defined (CONFIG_S32K1XX_PORTAINTS) || defined (CONFIG_S32K1XX_PORTBINTS) || \ defined (CONFIG_S32K1XX_PORTCINTS) || defined (CONFIG_S32K1XX_PORTDINTS) || \ defined (CONFIG_S32K1XX_PORTEINTS) # define HAVE_PORTINTS 1 #endif /**************************************************************************** * Private Types ****************************************************************************/ struct s32k1xx_pinirq_s { xcpt_t handler; void *arg; }; /**************************************************************************** * Private Data ****************************************************************************/ /* Per pin port interrupt vectors. NOTE: Not all pins in each port * correspond to externally available GPIOs. However, I believe that the * Kinesis will support interrupts even if the pin is not available as * a GPIO. Hence, we need to support all 32 pins for each port. To keep the * memory usage at a minimum, the logic may be configure per port. */ #ifdef CONFIG_S32K1XX_PORTAINTS static struct s32k1xx_pinirq_s g_portaisrs[32]; #endif #ifdef CONFIG_S32K1XX_PORTBINTS static struct s32k1xx_pinirq_s g_portbisrs[32]; #endif #ifdef CONFIG_S32K1XX_PORTCINTS static struct s32k1xx_pinirq_s g_portcisrs[32]; #endif #ifdef CONFIG_S32K1XX_PORTDINTS static struct s32k1xx_pinirq_s g_portdisrs[32]; #endif #ifdef CONFIG_S32K1XX_PORTEINTS static struct s32k1xx_pinirq_s g_porteisrs[32]; #endif /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: s32k1xx_portinterrupt * * Description: * Common port interrupt handling. * ****************************************************************************/ #ifdef HAVE_PORTINTS static int s32k1xx_portinterrupt(int irq, FAR void *context, uintptr_t addr, struct s32k1xx_pinirq_s *isrtab) { uint32_t isfr = getreg32(addr); int i; /* Examine each pin in the port */ for (i = 0; i < 32 && isfr != 0; i++) { /* A bit set in the ISR means that an interrupt is pending for this * pin. If the pin is programmed for level sensitive inputs, then * the interrupt handling logic MUST disable the interrupt (or cause * the level to change) to prevent infinite interrupts. */ uint32_t bit = (1 << i); if ((isfr & bit) != 0) { /* I think that bits may be set in the ISFR for DMA activities * well. So, no error is declared if there is no registered * interrupt handler for the pin. */ if (isrtab[i].handler != NULL) { xcpt_t handler = isrtab[i].handler; void *arg = isrtab[i].arg; /* There is a registered interrupt handler... invoke it */ handler(irq, context, arg); } /* Writing a one to the ISFR register will clear the pending * interrupt. If pin is configured to generate a DMA request * then the ISFR bit will be cleared automatically at the * completion of the requested DMA transfer. If configured for * a level sensitive interrupt and the pin remains asserted and * the bit will set again immediately after it is cleared. */ isfr &= ~bit; putreg32(bit, addr); } } return OK; } #endif /**************************************************************************** * Name: s32k1xx_portXinterrupt * * Description: * Handle interrupts arriving on individual ports * ****************************************************************************/ #ifdef CONFIG_S32K1XX_PORTAINTS static int s32k1xx_portainterrupt(int irq, FAR void *context, FAR void *arg) { return s32k1xx_portinterrupt(irq, context, S32K1XX_PORTA_ISFR, g_portaisrs); } #endif #ifdef CONFIG_S32K1XX_PORTBINTS static int s32k1xx_portbinterrupt(int irq, FAR void *context, FAR void *arg) { return s32k1xx_portinterrupt(irq, context, S32K1XX_PORTB_ISFR, g_portbisrs); } #endif #ifdef CONFIG_S32K1XX_PORTCINTS static int s32k1xx_portcinterrupt(int irq, FAR void *context, FAR void *arg) { return s32k1xx_portinterrupt(irq, context, S32K1XX_PORTC_ISFR, g_portcisrs); } #endif #ifdef CONFIG_S32K1XX_PORTDINTS static int s32k1xx_portdinterrupt(int irq, FAR void *context, FAR void *arg) { return s32k1xx_portinterrupt(irq, context, S32K1XX_PORTD_ISFR, g_portdisrs); } #endif #ifdef CONFIG_S32K1XX_PORTEINTS static int s32k1xx_porteinterrupt(int irq, FAR void *context, FAR void *arg) { return s32k1xx_portinterrupt(irq, context, S32K1XX_PORTE_ISFR, g_porteisrs); } #endif /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: s32k1xx_pinirq_initialize * * Description: * Initialize logic to support a second level of interrupt decoding for * GPIO pins. * ****************************************************************************/ void s32k1xx_pinirq_initialize(void) { #ifdef CONFIG_S32K1XX_PORTAINTS irq_attach(S32K1XX_IRQ_PORTA, s32k1xx_portainterrupt, NULL); putreg32(0xffffffff, S32K1XX_PORTA_ISFR); up_enable_irq(S32K1XX_IRQ_PORTA); #endif #ifdef CONFIG_S32K1XX_PORTBINTS irq_attach(S32K1XX_IRQ_PORTB, s32k1xx_portbinterrupt, NULL); putreg32(0xffffffff, S32K1XX_PORTB_ISFR); up_enable_irq(S32K1XX_IRQ_PORTB); #endif #ifdef CONFIG_S32K1XX_PORTCINTS irq_attach(S32K1XX_IRQ_PORTC, s32k1xx_portcinterrupt, NULL); putreg32(0xffffffff, S32K1XX_PORTC_ISFR); up_enable_irq(S32K1XX_IRQ_PORTC); #endif #ifdef CONFIG_S32K1XX_PORTDINTS irq_attach(S32K1XX_IRQ_PORTD, s32k1xx_portdinterrupt, NULL); putreg32(0xffffffff, S32K1XX_PORTD_ISFR); up_enable_irq(S32K1XX_IRQ_PORTD); #endif #ifdef CONFIG_S32K1XX_PORTEINTS irq_attach(S32K1XX_IRQ_PORTE, s32k1xx_porteinterrupt, NULL); putreg32(0xffffffff, S32K1XX_PORTE_ISFR); up_enable_irq(S32K1XX_IRQ_PORTE); #endif } /**************************************************************************** * Name: s32k1xx_pinirqattach * * Description: * Attach a pin interrupt handler. The normal initialization sequence is: * * 1. Call s32k1xx_pinconfig() to configure the interrupting pin (pin * interrupts will be disabled. * 2. Call s32k1xx_pinirqattach() to attach the pin interrupt handling * function. * 3. Call s32k1xx_pinirqenable() to enable interrupts on the pin. * * Input Parameters: * pinset - Pin configuration * pinisr - Pin interrupt service routine * arg - An argument that will be provided to the interrupt service * routine. * * Returned Value: * Zero (OK) is returned on success; a negated errno value is returned on * any failure to indicate the nature of the failure. * *******************************************************************************/ int s32k1xx_pinirqattach(uint32_t pinset, xcpt_t pinisr, void *arg) { #ifdef HAVE_PORTINTS struct s32k1xx_pinirq_s *isrtab; irqstate_t flags; unsigned int port; unsigned int pin; /* It only makes sense to call this function for input pins that are configured * as interrupts. */ DEBUGASSERT((pinset & _PIN_INTDMA_MASK) == _PIN_INTERRUPT); DEBUGASSERT((pinset & _PIN_IO_MASK) == _PIN_INPUT); /* Get the port number and pin number */ port = (pinset & _PIN_PORT_MASK) >> _PIN_PORT_SHIFT; pin = (pinset & _PIN_MASK) >> _PIN_SHIFT; /* Get the table associated with this port */ DEBUGASSERT(port < S32K1XX_NPORTS); flags = enter_critical_section(); switch (port) { #ifdef CONFIG_S32K1XX_PORTAINTS case S32K1XX_PORTA : isrtab = g_portaisrs; break; #endif #ifdef CONFIG_S32K1XX_PORTBINTS case S32K1XX_PORTB : isrtab = g_portbisrs; break; #endif #ifdef CONFIG_S32K1XX_PORTCINTS case S32K1XX_PORTC : isrtab = g_portcisrs; break; #endif #ifdef CONFIG_S32K1XX_PORTDINTS case S32K1XX_PORTD : isrtab = g_portdisrs; break; #endif #ifdef CONFIG_S32K1XX_PORTEINTS case S32K1XX_PORTE : isrtab = g_porteisrs; break; #endif default: leave_critical_section(flags); return -EINVAL; } /* Get the old PIN ISR and set the new PIN ISR */ isrtab[pin].handler = pinisr; isrtab[pin].arg = arg; /* And return the old PIN isr address */ leave_critical_section(flags); return OK; #else return -ENOSYS; #endif /* HAVE_PORTINTS */ } /************************************************************************************ * Name: s32k1xx_pinirqenable * * Description: * Enable the interrupt for specified pin IRQ * ************************************************************************************/ void s32k1xx_pinirqenable(uint32_t pinset) { #ifdef HAVE_PORTINTS uintptr_t base; uint32_t regval; unsigned int port; unsigned int pin; /* Get the port number and pin number */ port = (pinset & _PIN_PORT_MASK) >> _PIN_PORT_SHIFT; pin = (pinset & _PIN_MASK) >> _PIN_SHIFT; DEBUGASSERT(port < S32K1XX_NPORTS); if (port < S32K1XX_NPORTS) { /* Get the base address of PORT block for this port */ base = S32K1XX_PORT_BASE(port); /* Modify the IRQC field of the port PCR register in order to enable * the interrupt. */ regval = getreg32(base + S32K1XX_PORT_PCR_OFFSET(pin)); regval &= ~PORT_PCR_IRQC_MASK; switch (pinset & _PIN_INT_MASK) { case PIN_INT_ZERO : /* Interrupt when logic zero */ regval |= PORT_PCR_IRQC_ZERO; break; case PIN_INT_RISING : /* Interrupt on rising edge */ regval |= PORT_PCR_IRQC_RISING; break; case PIN_INT_FALLING : /* Interrupt on falling edge */ regval |= PORT_PCR_IRQC_FALLING; break; case PIN_INT_BOTH : /* Interrupt on either edge */ regval |= PORT_PCR_IRQC_BOTH; break; case PIN_INT_ONE : /* Interrupt when logic one */ regval |= PORT_PCR_IRQC_ONE; break; case PIN_DMA_RISING : /* DMA on rising edge */ regval |= PORT_PCR_IRQC_DMARISING; break; case PIN_DMA_FALLING : /* DMA on falling edge */ regval |= PORT_PCR_IRQC_DMAFALLING; break; case PIN_DMA_BOTH : /* DMA on either edge */ regval |= PORT_PCR_IRQC_DMABOTH; break; default: return; } putreg32(regval, base + S32K1XX_PORT_PCR_OFFSET(pin)); } #endif /* HAVE_PORTINTS */ } /************************************************************************************ * Name: s32k1xx_pinirqdisable * * Description: * Disable the interrupt for specified pin * ************************************************************************************/ void s32k1xx_pinirqdisable(uint32_t pinset) { #ifdef HAVE_PORTINTS uintptr_t base; uint32_t regval; unsigned int port; unsigned int pin; /* Get the port number and pin number */ port = (pinset & _PIN_PORT_MASK) >> _PIN_PORT_SHIFT; pin = (pinset & _PIN_MASK) >> _PIN_SHIFT; DEBUGASSERT(port < S32K1XX_NPORTS); if (port < S32K1XX_NPORTS) { /* Get the base address of PORT block for this port */ base = S32K1XX_PORT_BASE(port); /* Clear the IRQC field of the port PCR register in order to disable * the interrupt. */ regval = getreg32(base + S32K1XX_PORT_PCR_OFFSET(pin)); regval &= ~PORT_PCR_IRQC_MASK; putreg32(regval, base + S32K1XX_PORT_PCR_OFFSET(pin)); } #endif /* HAVE_PORTINTS */ } #endif /* CONFIG_S32K1XX_GPIOIRQ */