2014-02-03 19:31:04 +01:00
|
|
|
/*
|
2015-06-03 18:14:16 +02:00
|
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
2014-02-03 19:31:04 +01:00
|
|
|
*
|
2014-08-27 18:47:31 +02:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2015-11-19 15:54:52 +01:00
|
|
|
* @defgroup drivers_periph_gpio GPIO
|
|
|
|
* @ingroup drivers_periph
|
2014-02-04 18:46:02 +01:00
|
|
|
* @brief Low-level GPIO peripheral driver
|
2014-02-03 19:31:04 +01:00
|
|
|
*
|
2015-12-18 13:10:36 +01:00
|
|
|
* This is a basic GPIO (General-purpose input/output) interface to allow
|
|
|
|
* platform independent access to a MCU's input/output pins. This interface is
|
|
|
|
* intentionally designed to be as simple as possible, to allow for easy
|
|
|
|
* implementation and maximum portability.
|
|
|
|
*
|
|
|
|
* The interface provides capabilities to initialize a pin as output-,
|
|
|
|
* input- and interrupt pin. With the API you can basically set/clear/toggle the
|
|
|
|
* digital signal at the hardware pin when in output mode. Configured as input you can
|
|
|
|
* read a digital value that is being applied to the pin externally. When initializing
|
|
|
|
* an external interrupt pin, you can register a callback function that is executed
|
|
|
|
* in interrupt context once the interrupt condition applies to the pin. Usually you
|
|
|
|
* can react to rising or falling signal flanks (or both).
|
|
|
|
*
|
|
|
|
* In addition the API provides to set standard input/output circuit modes such as
|
|
|
|
* e.g. internal push-pull configurations.
|
|
|
|
*
|
|
|
|
* All modern micro controllers organize their GPIOs in some form of ports,
|
|
|
|
* often named 'PA', 'PB', 'PC'..., or 'P0', 'P1', 'P2'..., or similar. Each of
|
|
|
|
* these ports is then assigned a number of pins, often 8, 16, or 32. A hardware
|
|
|
|
* pin can thus be described by its port/pin tuple. To access a pin, the
|
|
|
|
* @p GPIO_PIN(port, pin) macro should be used. For example: If your platform has
|
|
|
|
* a pin PB22, it will be port=1 and pin=22. The @p GPIO_PIN macro should be
|
|
|
|
* overridden by a MCU, to allow for efficient encoding of the the port/pin tuple.
|
|
|
|
* For example, on many platforms it is possible to `OR` the pin number with the
|
|
|
|
* corresponding ports base register address. This allows for efficient decoding
|
|
|
|
* of pin number and base address without the need of any address lookup.
|
|
|
|
*
|
|
|
|
* In case the driver does not define it, the below macro definition is used to
|
|
|
|
* simply map the port/pin tuple to the pin value. In that case, predefined GPIO
|
|
|
|
* definitions in `RIOT/boards/ * /include/periph_conf.h` will define the selected
|
|
|
|
* GPIO pin.
|
|
|
|
*
|
2014-12-04 10:03:15 +01:00
|
|
|
* @{
|
2014-10-25 15:37:04 +02:00
|
|
|
* @file
|
2014-02-04 18:46:02 +01:00
|
|
|
* @brief Low-level GPIO peripheral driver interface definitions
|
2014-02-03 19:31:04 +01:00
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
2015-03-23 21:07:50 +01:00
|
|
|
#ifndef GPIO_H
|
|
|
|
#define GPIO_H
|
2014-02-03 19:31:04 +01:00
|
|
|
|
2016-02-17 10:27:10 +01:00
|
|
|
#include <limits.h>
|
|
|
|
|
2015-06-03 18:14:16 +02:00
|
|
|
#include "periph_cpu.h"
|
2015-10-27 13:29:37 +01:00
|
|
|
#include "periph_conf.h"
|
|
|
|
/* TODO: remove once all platforms are ported to this interface */
|
|
|
|
#include "periph/dev_enums.h"
|
2014-02-03 19:31:04 +01:00
|
|
|
|
2014-10-13 15:49:17 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2014-02-03 19:31:04 +01:00
|
|
|
/**
|
2015-06-14 16:12:20 +02:00
|
|
|
* @brief Default GPIO macro maps port-pin tuples to the pin value
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-09-15 11:46:49 +02:00
|
|
|
#ifndef GPIO_PIN
|
|
|
|
#define GPIO_PIN(x,y) ((x & 0) | y)
|
2014-10-01 18:42:34 +02:00
|
|
|
#endif
|
2015-06-03 18:14:16 +02:00
|
|
|
|
2015-06-14 16:12:20 +02:00
|
|
|
/**
|
|
|
|
* @brief Define global value for GPIO not defined
|
|
|
|
*/
|
|
|
|
#ifndef GPIO_UNDEF
|
2016-02-17 10:27:10 +01:00
|
|
|
#define GPIO_UNDEF (UINT_MAX)
|
2015-06-14 16:12:20 +02:00
|
|
|
#endif
|
|
|
|
|
2015-06-03 18:14:16 +02:00
|
|
|
/**
|
|
|
|
* @brief Define the default GPIO type identifier
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_GPIO_T
|
2016-02-17 10:27:10 +01:00
|
|
|
typedef unsigned int gpio_t;
|
2014-10-01 18:42:34 +02:00
|
|
|
#endif
|
2015-06-03 18:14:16 +02:00
|
|
|
|
|
|
|
/**
|
2016-02-20 16:48:24 +01:00
|
|
|
* @brief Available pin modes
|
|
|
|
*
|
|
|
|
* Generally, a pin can be configured to be input or output. In output mode, a
|
|
|
|
* pin can further be put into push-pull or open drain configuration. Though
|
|
|
|
* this is supported by most platforms, this is not always the case, so driver
|
|
|
|
* implementations may return an error code if a mode is not supported.
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2016-02-20 16:48:24 +01:00
|
|
|
#ifndef HAVE_GPIO_MODE_T
|
2014-02-03 19:31:04 +01:00
|
|
|
typedef enum {
|
2016-02-20 16:48:24 +01:00
|
|
|
GPIO_IN , /**< configure as input without pull resistor */
|
|
|
|
GPIO_IN_PD, /**< configure as input with pull-down resistor */
|
|
|
|
GPIO_IN_PU, /**< configure as input with pull-up resistor */
|
|
|
|
GPIO_OUT, /**< configure as output in push-pull mode */
|
|
|
|
GPIO_OD, /**< configure as output in open-drain mode without
|
|
|
|
* pull resistor */
|
|
|
|
GPIO_OD_PU /**< configure as output in open-drain mode with
|
|
|
|
* pull resistor enabled */
|
|
|
|
} gpio_mode_t;
|
2015-06-03 18:14:16 +02:00
|
|
|
#endif
|
2014-02-03 19:31:04 +01:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Definition of possible active flanks for external interrupt mode
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
#ifndef HAVE_GPIO_FLANK_T
|
2014-02-03 19:31:04 +01:00
|
|
|
typedef enum {
|
2014-04-03 17:44:49 +02:00
|
|
|
GPIO_FALLING = 0, /**< emit interrupt on falling flank */
|
|
|
|
GPIO_RISING = 1, /**< emit interrupt on rising flank */
|
|
|
|
GPIO_BOTH = 2 /**< emit interrupt on both flanks */
|
2014-02-03 19:31:04 +01:00
|
|
|
} gpio_flank_t;
|
2015-06-03 18:14:16 +02:00
|
|
|
#endif
|
2014-02-03 19:31:04 +01:00
|
|
|
|
2014-07-28 16:21:30 +02:00
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Signature of event callback functions triggered from interrupts
|
2014-07-28 16:21:30 +02:00
|
|
|
*
|
|
|
|
* @param[in] arg optional context for the callback
|
|
|
|
*/
|
|
|
|
typedef void (*gpio_cb_t)(void *arg);
|
|
|
|
|
2016-01-27 17:00:23 +01:00
|
|
|
/**
|
|
|
|
* @brief Default interrupt context for GPIO pins
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_GPIO_ISR_CTX_T
|
|
|
|
typedef struct {
|
|
|
|
gpio_cb_t cb; /**< interrupt callback */
|
|
|
|
void *arg; /**< optional argument */
|
|
|
|
} gpio_isr_ctx_t;
|
|
|
|
#endif
|
|
|
|
/** @} */
|
|
|
|
|
2014-02-03 19:31:04 +01:00
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Initialize the given pin as general purpose input or output
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin pin to initialize
|
|
|
|
* @param[in] dir direction for the pin, input or output
|
|
|
|
* @param[in] pullup define what pull resistors should be used
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2014-04-03 17:44:49 +02:00
|
|
|
* @return 0 on success
|
2015-06-03 18:14:16 +02:00
|
|
|
* @return -1 on error
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2016-02-20 16:48:24 +01:00
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Initialize a GPIO pin for external interrupt usage
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* The registered callback function will be called in interrupt context every
|
|
|
|
* time the defined flank(s) are detected.
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2014-02-03 19:31:04 +01:00
|
|
|
* The interrupt is activated automatically after the initialization.
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin pin to initialize
|
|
|
|
* @param[in] pullup define what pull resistors should be enabled
|
2014-04-03 17:44:49 +02:00
|
|
|
* @param[in] flank define the active flank(s)
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] cb callback that is called from interrupt context
|
2014-07-28 16:21:30 +02:00
|
|
|
* @param[in] arg optional argument passed to the callback
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2014-04-03 17:44:49 +02:00
|
|
|
* @return 0 on success
|
2015-06-03 18:14:16 +02:00
|
|
|
* @return -1 on error
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2016-02-20 16:48:24 +01:00
|
|
|
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
|
|
|
gpio_cb_t cb, void *arg);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
2014-07-03 21:51:37 +02:00
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Enable pin's interrupt if configured as interrupt source
|
2014-07-03 21:51:37 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to enable the interrupt for
|
2014-07-03 21:51:37 +02:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
void gpio_irq_enable(gpio_t pin);
|
2014-07-03 21:51:37 +02:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Disable the pin's interrupt if configured as interrupt source
|
2014-07-03 21:51:37 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to disable the interrupt for
|
2014-07-03 21:51:37 +02:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
void gpio_irq_disable(gpio_t pin);
|
2014-07-03 21:51:37 +02:00
|
|
|
|
2014-02-03 19:31:04 +01:00
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Get the current value of the given pin
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to read
|
2014-07-28 16:21:30 +02:00
|
|
|
* @return the current value, 0 for LOW, != 0 for HIGH
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2014-04-03 17:44:49 +02:00
|
|
|
* @return 0 when pin is LOW
|
|
|
|
* @return greater 0 for HIGH
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
int gpio_read(gpio_t pin);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Set the given pin to HIGH
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to set
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
void gpio_set(gpio_t pin);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Set the given pin to LOW
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to clear
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
void gpio_clear(gpio_t pin);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Toggle the value of the given pin
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to toggle
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
void gpio_toggle(gpio_t pin);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
|
|
|
/**
|
2015-06-03 18:14:16 +02:00
|
|
|
* @brief Set the given pin to the given value
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-06-03 18:14:16 +02:00
|
|
|
* @param[in] pin the pin to set
|
2014-04-03 17:44:49 +02:00
|
|
|
* @param[in] value value to set the pin to, 0 for LOW, HIGH otherwise
|
2014-02-03 19:31:04 +01:00
|
|
|
*/
|
2015-06-03 18:14:16 +02:00
|
|
|
void gpio_write(gpio_t pin, int value);
|
2014-02-03 19:31:04 +01:00
|
|
|
|
2014-10-13 15:49:17 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-03-23 21:07:50 +01:00
|
|
|
#endif /* GPIO_H */
|
2014-02-03 19:31:04 +01:00
|
|
|
/** @} */
|