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

Blink LED on button press on nRF9160dk

Hello,

I want to blink the LED by pressing button on nRF9160dk. Is there any sample available to start with?

I started with button demo but on button press there is no output. I saw examples for nRF51/52. But I couldn't find for nRF9160dk.

Thank you

Parents
  • In the Zephyr repository, (/path/to/ncs/zephyr) there are sample applications in the sample folder.

    If you enter the /basic/ folder in /samples/, you can find the blinky and button example applications. If you have a play with both of them individually, you can then combine them to create a callback function that when the button is pressed, it flips the output on the LED. (See the button example for how the interrupt callback works)

    Once you have that down, you can try setting the button GPIO options to detect both falling and rising edges and then you can create an application that can set the LED output based on the current state of the button.

    If you are unsure about any of the functions or what's happening. The Zephyr documentation can help shed light on what everything is doing. docs.zephyrproject.org/.../

Reply
  • In the Zephyr repository, (/path/to/ncs/zephyr) there are sample applications in the sample folder.

    If you enter the /basic/ folder in /samples/, you can find the blinky and button example applications. If you have a play with both of them individually, you can then combine them to create a callback function that when the button is pressed, it flips the output on the LED. (See the button example for how the interrupt callback works)

    Once you have that down, you can try setting the button GPIO options to detect both falling and rising edges and then you can create an application that can set the LED output based on the current state of the button.

    If you are unsure about any of the functions or what's happening. The Zephyr documentation can help shed light on what everything is doing. docs.zephyrproject.org/.../

Children
  • Hi,

    Thank you for the response, it is working.

    I am using push button 2 and LED 2. When I press the button, LED glows continuously. I want a function in which when I release the button, LED stops glowing. I have tried it, but it is not working. What should I do?

    I will paste the code.

    #include <zephyr.h>
    #include <device.h>
    #include <gpio.h>
    #include <misc/util.h>
    #include <misc/printk.h>
    
    #define LED_PORT LED1_GPIO_CONTROLLER
    #define LED	LED1_GPIO_PIN
    
    /* change this to use another GPIO port */
    #ifndef SW1_GPIO_CONTROLLER
    #ifdef SW1_GPIO_NAME
    #define SW1_GPIO_CONTROLLER SW1_GPIO_NAME
    #else
    #error SW1_GPIO_NAME or SW1_GPIO_CONTROLLER needs to be set in board.h
    #endif
    #endif
    #define PORT	SW1_GPIO_CONTROLLER
    
    /* change this to use another GPIO pin */
    #ifdef SW1_GPIO_PIN
    #define PIN     SW1_GPIO_PIN
    #else
    #error SW1_GPIO_PIN needs to be set in board.h
    #endif
    
    /* change to use another GPIO pin interrupt config */
    #ifdef SW1_GPIO_FLAGS
    #define EDGE    (SW1_GPIO_FLAGS | GPIO_INT_EDGE)
    #else
    /*
     * If SW0_GPIO_FLAGS not defined used default EDGE value.
     * Change this to use a different interrupt trigger
     */
    #define EDGE    (GPIO_INT_EDGE | GPIO_INT_ACTIVE_LOW)
    #endif
    
    /* change this to enable pull-up/pull-down */
    #ifndef SW1_GPIO_FLAGS
    #ifdef SW1_GPIO_PIN_PUD
    #define SW1_GPIO_FLAGS SW1_GPIO_PIN_PUD
    #else
    #define SW1_GPIO_FLAGS 0
    #endif
    #endif
    #define PULL_UP SW1_GPIO_FLAGS
    
    /* Sleep time */
    #define SLEEP_TIME	500
    
    
    void button_pressed(struct device *gpiob, struct gpio_callback *cb,
    		    u32_t pins)
    {
            printk("Button pressed\n");
            struct device *dev;
    
    	dev = device_get_binding(LED_PORT);
            gpio_pin_configure(dev, LED, GPIO_DIR_OUT);
            if(PIN)
            {
                gpio_pin_write(dev, LED, 1);
            }
            else
            {
                gpio_pin_write(dev, LED, 0);
            }
    }
    
    static struct gpio_callback gpio_cb;
    
    void main(void)
    {
    	struct device *gpiob;
    
    	printk("Press the user defined button on the board\n");
    	gpiob = device_get_binding(PORT);
    	if (!gpiob) {
    		printk("error\n");
    		return;
    	}
    
    
    	gpio_pin_configure(gpiob, PIN,
    			   GPIO_DIR_IN | GPIO_INT |  PULL_UP | EDGE | GPIO_INT_ACTIVE_LOW | GPIO_INT_ACTIVE_HIGH);
    
    	gpio_init_callback(&gpio_cb, button_pressed, BIT(PIN));
    
    	gpio_add_callback(gpiob, &gpio_cb);
    	gpio_pin_enable_callback(gpiob, PIN);
    
    	while (1) {
    		u32_t val = 0U;
    
    		gpio_pin_read(gpiob, PIN, &val);
    		k_sleep(SLEEP_TIME);
    	}
    }
    

  • Hi there,

    I reviewed your code and found a couple of issues.

    Firstly, read the pin at the ISR point and not constantly in main, it's a quick pin level check and can be performed very fast. You only need to spend time reading the pin when something changes or set a boolean flag in the ISR when it is triggered that can activate code in main to read the button state and set the LED.

    Secondly, the pin options used were not correct. To trigger an ISR the rising and falling edge of a pin, you need to use the option GPIO_INT_DOUBLE_EDGE.

    Thirdly, your ISR wasn't actually reading the pin state, your if statement was set to if (BUTTON_PIN) not if (BUTTON_STATE). You were always going to enter the 'true' state of the if statement. I have added in the pin reading elements that are running in main to your ISR.

    A few additional points,

    Consider placing your LED and Button init codes in a seperate function and calling them at the start of main.

    The Struct Device for the LEDs is being called every time you run the ISR, remove it from here and only init the struct once and make it callable by placing the define for the LED device struct outside of main.

    Consider using the option GPIO_INT_DEBOUNCE on buttons to counter the bouncing effect of buttons.

    #include <zephyr.h>
    #include <device.h>
    #include <gpio.h>
    #include <misc/util.h>
    #include <misc/printk.h>
    
    #define LED_PORT LED1_GPIO_CONTROLLER
    #define LED	LED1_GPIO_PIN
    
    /* change this to use another GPIO port */
    #ifndef SW1_GPIO_CONTROLLER
    #ifdef SW1_GPIO_NAME
    #define SW1_GPIO_CONTROLLER SW1_GPIO_NAME
    #else
    #error SW1_GPIO_NAME or SW1_GPIO_CONTROLLER needs to be set in board.h
    #endif
    #endif
    #define PORT	SW1_GPIO_CONTROLLER
    
    /* change this to use another GPIO pin */
    #ifdef SW1_GPIO_PIN
    #define PIN     SW1_GPIO_PIN
    #else
    #error SW1_GPIO_PIN needs to be set in board.h
    #endif
    
    /* change to use another GPIO pin interrupt config */
    #ifdef SW1_GPIO_FLAGS
    #define EDGE    (SW1_GPIO_FLAGS | GPIO_INT_EDGE)
    #else
    /*
     * If SW0_GPIO_FLAGS not defined used default EDGE value.
     * Change this to use a different interrupt trigger
     */
    #define EDGE    (GPIO_INT_EDGE | GPIO_INT_ACTIVE_LOW)
    #endif
    
    /* change this to enable pull-up/pull-down */
    #ifndef SW1_GPIO_FLAGS
    #ifdef SW1_GPIO_PIN_PUD
    #define SW1_GPIO_FLAGS SW1_GPIO_PIN_PUD
    #else
    #define SW1_GPIO_FLAGS 0
    #endif
    #endif
    #define PULL_UP SW1_GPIO_FLAGS
    
    /* Sleep time */
    #define SLEEP_TIME	500
    
    
    void button_pressed(struct device *gpiob, struct gpio_callback *cb,
    		    u32_t pins)
    {
            printk("Button pressed\n");
            struct device *dev;
    
    	dev = device_get_binding(LED_PORT);
            gpio_pin_configure(dev, LED, GPIO_DIR_OUT);
    
    				u32_t val = 0U;
    
    				gpio_pin_read(gpiob, PIN, &val);
    
            if(val == 0)
            {
                gpio_pin_write(dev, LED, 1);
            }
            else if (val == 1)
            {
                gpio_pin_write(dev, LED, 0);
            }
    }
    
    static struct gpio_callback gpio_cb;
    
    void main(void)
    {
    	struct device *gpiob;
    
    	printk("Press the user defined button on the board\n");
    	gpiob = device_get_binding(PORT);
    	if (!gpiob) {
    		printk("error\n");
    		return;
    	}
    
    
    	gpio_pin_configure(gpiob, PIN,
    			   GPIO_DIR_IN | GPIO_INT |  PULL_UP | EDGE | GPIO_INT_DOUBLE_EDGE | GPIO_INT_ACTIVE_LOW | GPIO_INT_ACTIVE_HIGH | GPIO_INT_DEBOUNCE);
    
    	gpio_init_callback(&gpio_cb, button_pressed, BIT(PIN));
    
    	gpio_add_callback(gpiob, &gpio_cb);
    	gpio_pin_enable_callback(gpiob, PIN);
    
    	while (1) {
    		u32_t val = 0U;
    
    		gpio_pin_read(gpiob, PIN, &val);
    		k_sleep(SLEEP_TIME);
    	}
    }
    

Related