This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

How to implement timer interrupt in nRF9160?

Timer interrupt does not work well. In the code below, LED doesn't turn off and "interrupt!" message is not received. I really appreciate it if you give  me any advice.

#include <nrf9160.h>
#include <zephyr.h>
#include <misc/printk.h>
#include <device.h>
#include <nrf_timer.h>
#include <gpio.h>

#define TIEMR_INTERVAL_SEC 5

struct device *dev;

// make 5 sec timer
// f_timer = 16*1000*1000 / (2^PRESCALER)
// sec_interval = (2^PRESCALER) / 16*1000*1000 x NRF_TIMER0_S->CC[0]
void config_timer()
{		
        NRF_TIMER1->TASKS_STOP = 1;
        NRF_TIMER1->MODE = TIMER_MODE_MODE_Timer;
        NRF_TIMER1->TASKS_CLEAR = 1;                             // clear time
        NRF_TIMER1->PRESCALER = 9;                 // value = 0..9
	NRF_TIMER1->BITMODE = TIMER_BITMODE_BITMODE_32Bit;		   //Set counter to 32 bit resolution. MAX count = 2^32-1
	NRF_TIMER1->CC[0] = 156250;  //Set value for TIMER0
        NRF_TIMER1->INTENSET = TIMER_INTENSET_COMPARE1_Enabled << TIMER_INTENSET_COMPARE1_Pos; 
        NVIC_EnableIRQ(TIMER1_IRQn);
        NVIC_SetPriority(TIMER1_IRQn, 0);
}

void TIMER1_IRQHandler(void)
{
	if ((NRF_TIMER1->EVENTS_COMPARE[0] != 0) && ((NRF_TIMER1->INTENSET & TIMER_INTENSET_COMPARE1_Msk) != 0))
        {
                  NRF_TIMER1->EVENTS_COMPARE[0] = 0;           //Clear compare register 0 event	
                  gpio_pin_write(dev, 16, 0);
                  printk("interrupt!\r\n");
        }
}

void main(void)
{
        printk("------------\r\nHello, World!\r\n");
	dev = device_get_binding("GPIO_0");
        gpio_pin_configure(dev, 16, GPIO_DIR_OUT);

        config_timer();
        gpio_pin_write(dev, 16, 1);

        NRF_TIMER1->TASKS_START = 1;
        while(1){
                 printk("Start to sleep\r\n");
                 k_cpu_idle();
                 printk("timer interrupt for %u sec\r\n", TIEMR_INTERVAL_SEC);
        }
}

<>
***** Booting Zephyr OS v1.13.99-ncs2 *****
------------
Hello, World!
Start to sleep
timer interrupt for 5 sec
Start to sleep
// UART communication stops here

Parents Reply Children
  • Hi, Hakon. Thank you for your reply. I tested the following code, but got fail. Do you know why?

    <main.c>
    #include <nrf9160.h>
    #include <zephyr.h>
    #include <misc/printk.h>
    #include <device.h>
    #include <nrf_timer.h>
    #include <gpio.h>
    
    #define TIEMR_INTERVAL_SEC 5
    
    struct device *dev;
    
    // make 5 sec timer
    // f_timer = 16*1000*1000 / (2^PRESCALER)
    // sec_interval = (2^PRESCALER) / 16*1000*1000 x NRF_TIMER0_S->CC[0]
    void config_timer()
    {		
            NRF_TIMER1->TASKS_STOP = 1;
            NRF_TIMER1->MODE = TIMER_MODE_MODE_Timer;
            NRF_TIMER1->TASKS_CLEAR = 1;                             // clear time
            NRF_TIMER1->PRESCALER = 9;                 // value = 0..9
    	NRF_TIMER1->BITMODE = TIMER_BITMODE_BITMODE_32Bit;		   //Set counter to 32 bit resolution. MAX count = 2^32-1
    	NRF_TIMER1->CC[0] = 156250;  //Set value for TIMER0 156250
            NRF_TIMER1->SHORTS = TIMER_SHORTS_COMPARE0_CLEAR_Enabled << TIMER_SHORTS_COMPARE0_CLEAR_Pos;
            NRF_TIMER1->INTENSET = TIMER_INTENSET_COMPARE0_Enabled << TIMER_INTENSET_COMPARE0_Pos; 
            NVIC_SetPriority(TIMER1_IRQn, 1);
            NVIC_EnableIRQ(TIMER1_IRQn);
            NRF_TIMER1->TASKS_START = 1;               // Start TIMER0
    }
    
    void TIMER1_IRQHandler(void)
    {
            gpio_pin_write(dev, 16, 0);
            printk("interrupt!\r\n");
    	if ((NRF_TIMER1->EVENTS_COMPARE[0] != 0) && ((NRF_TIMER1->INTENSET & TIMER_INTENSET_COMPARE0_Msk) != 0))
            {
                      NRF_TIMER1->EVENTS_COMPARE[0] = 0;           //Clear compare register 0 event	
                      gpio_pin_write(dev, 16, 0);
                      printk("interrupt!\r\n");
            }
    }
    
    void main(void)
    {
            printk("------------\r\nHello, World!\r\n");
    	dev = device_get_binding("GPIO_0");
            gpio_pin_configure(dev, 16, GPIO_DIR_OUT);
    
            config_timer();
            gpio_pin_write(dev, 16, 1);
    
            while(1){
                     printk("Start to sleep\r\n");
                     k_cpu_idle();
                     printk("timer interrupt for %u sec\r\n", TIEMR_INTERVAL_SEC);
            }
    }
    
    

    ***** Booting Zephyr OS v1.13.99-ncs2 *****
    ------------
    Hello, World!
    Start to sleep
    timer interrupt for 5 sec
    Start to sleep
    ***** Hardware exception *****
    Current thread ID = 0x2000009c
    Faulting instruction address = 0xf4240
    Fatal fault in ISR! Spinning...

  • Hi,

     

    I suspect that this is caused by the OS itself, as the interrupt vector should be declared through the zephyr rtos API, similar to what is done here for the libbsd port file:

    https://github.com/NordicPlayground/fw-nrfconnect-nrf/blob/master/lib/bsdlib/bsd_os.c#L284

    Here's the documentation for this: https://docs.zephyrproject.org/1.13.0/api/kernel_api.html#c.IRQ_DIRECT_CONNECT

    Kind regards,

    Håkon

  • Hi, Hakon. Thank you for your advice, which is very helpful.

    I'm new to zephyr.

    Do you mean that zephyr has some bug in timer interrupt, or I'm doing a wrong thing?

    I'm referring to the code below as well as nRF9160-SICA-R_Nordic_Semiconductor.pdf.

    https://github.com/NordicPlayground/nrf51-TIMER-examples/blob/master/timer_example_timer_mode/main.c

  • Hi,

     

    Since zephyr rtos owns the interrupts, we need to instruct it that we want the interrupt in our application, similar to this:

    timer.zip

    I would instead of doing the above, especially for longer delays, recommend that you look into the kernel timers instead (RTC based):

    https://docs.zephyrproject.org/1.13.0/kernel/timing/timers.html

    Kind regards,

    Håkon

  • Thanks a lot! The code below works!

    #include <nrf9160.h>
    #include <zephyr.h>
    #include <misc/printk.h>
    #include <device.h>
    #include <gpio.h>
    
    #define TIEMR_INTERVAL_SEC 5
    
    struct device *dev;
    struct k_timer my_timer;
    uint8_t toggle = 0;
    
    void my_expiry_function(struct k_timer *timer_id){
            ++toggle;
            gpio_pin_write(dev, 16, toggle % 2);
            printk("interrupt!\r\n");
    }
    
    void main(void)
    {
            printk("------------\r\nHello, World!\r\n");
            k_timer_init(&my_timer, my_expiry_function, NULL);
    
    	dev = device_get_binding("GPIO_0");
            gpio_pin_configure(dev, 16, GPIO_DIR_OUT);
            gpio_pin_write(dev, 16, toggle);
            k_timer_start(&my_timer, K_SECONDS(TIEMR_INTERVAL_SEC), K_SECONDS(TIEMR_INTERVAL_SEC));
            
            while(1){
                     printk("Start to sleep\r\n");
                     k_cpu_idle();
                     printk("timer interrupt for %u sec\r\n", TIEMR_INTERVAL_SEC);
            }
    }
    

Related