2015-06-05 01:46:25 +00:00
|
|
|
/** esp_iomux.h
|
|
|
|
*
|
|
|
|
* GPIO functions.
|
|
|
|
*
|
|
|
|
* Part of esp-open-rtos
|
|
|
|
* Copyright (C) 2015 Superhouse Automation Pty Ltd
|
|
|
|
* BSD Licensed as described in the file LICENSE
|
|
|
|
*/
|
|
|
|
#ifndef _ESP_GPIO_H
|
2015-06-17 23:00:02 +00:00
|
|
|
#define _ESP_GPIO_H
|
2015-06-05 01:46:25 +00:00
|
|
|
#include <stdbool.h>
|
2015-08-19 05:46:25 +00:00
|
|
|
#include "esp/gpio_regs.h"
|
2015-06-05 01:46:25 +00:00
|
|
|
#include "esp/iomux.h"
|
2015-09-23 22:58:34 +00:00
|
|
|
#include "esp/interrupts.h"
|
2015-06-05 01:46:25 +00:00
|
|
|
|
|
|
|
typedef enum {
|
2015-06-08 08:09:06 +00:00
|
|
|
GPIO_INPUT,
|
|
|
|
GPIO_OUTPUT, /* "Standard" push-pull output */
|
|
|
|
GPIO_OUT_OPEN_DRAIN, /* Open drain output */
|
2015-06-05 01:46:25 +00:00
|
|
|
} gpio_direction_t;
|
|
|
|
|
2016-02-22 02:34:11 +00:00
|
|
|
/* Enable GPIO on the specified pin, and set it to input or output mode
|
2015-06-05 01:46:25 +00:00
|
|
|
*/
|
2015-11-28 07:01:03 +00:00
|
|
|
void gpio_enable(const uint8_t gpio_num, const gpio_direction_t direction);
|
2015-06-05 01:46:25 +00:00
|
|
|
|
2016-02-22 02:34:11 +00:00
|
|
|
/* Enable/disable internal pullup resistor for a particular GPIO
|
|
|
|
*
|
|
|
|
* Note: According to Espressif, pullup resistor values are between 30K and
|
|
|
|
* 100K ohms (see http://bbs.espressif.com/viewtopic.php?t=1079#p4097)
|
|
|
|
* However, measured values suggest that the actual value is likely to be close
|
|
|
|
* to 47K in reality.
|
|
|
|
*/
|
|
|
|
void gpio_set_pullup(uint8_t gpio_num, bool enabled, bool enabled_during_sleep);
|
|
|
|
|
2015-06-05 01:46:25 +00:00
|
|
|
/* Disable GPIO on the specified pin, and set it Hi-Z.
|
|
|
|
*
|
|
|
|
* If later muxing this pin to a different function, make sure to set
|
2015-08-19 00:38:31 +00:00
|
|
|
* IOMUX_PIN_OUTPUT_ENABLE if necessary to enable the output buffer.
|
2015-06-05 01:46:25 +00:00
|
|
|
*/
|
2015-11-28 07:01:03 +00:00
|
|
|
static inline void gpio_disable(const uint8_t gpio_num)
|
2015-06-05 01:46:25 +00:00
|
|
|
{
|
2015-08-19 05:46:25 +00:00
|
|
|
GPIO.ENABLE_OUT_CLEAR = BIT(gpio_num);
|
2015-08-19 00:38:31 +00:00
|
|
|
*gpio_iomux_reg(gpio_num) &= ~IOMUX_PIN_OUTPUT_ENABLE;
|
2015-06-05 01:46:25 +00:00
|
|
|
}
|
|
|
|
|
2016-02-22 02:34:11 +00:00
|
|
|
/* Set whether the specified pin continues to drive its output when the ESP8266
|
|
|
|
* goes into sleep mode. Note that this setting is reset to off whenever
|
|
|
|
* gpio_enable is called, so this must be called after calling that function.
|
|
|
|
*/
|
|
|
|
static inline void gpio_set_output_on_sleep(const uint8_t gpio_num, bool enabled)
|
|
|
|
{
|
|
|
|
if (enabled) {
|
|
|
|
IOMUX.PIN[gpio_to_iomux(gpio_num)] |= IOMUX_PIN_OUTPUT_ENABLE_SLEEP;
|
|
|
|
} else {
|
|
|
|
IOMUX.PIN[gpio_to_iomux(gpio_num)] &= ~IOMUX_PIN_OUTPUT_ENABLE_SLEEP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-05 01:46:25 +00:00
|
|
|
/* Set output of a pin high or low.
|
|
|
|
*
|
|
|
|
* Only works if pin has been set to GPIO_OUTPUT via gpio_enable()
|
|
|
|
*/
|
2015-11-28 07:01:03 +00:00
|
|
|
static inline void gpio_write(const uint8_t gpio_num, const bool set)
|
2015-06-05 01:46:25 +00:00
|
|
|
{
|
|
|
|
if(set)
|
2015-08-19 05:46:25 +00:00
|
|
|
GPIO.OUT_SET = BIT(gpio_num);
|
2015-06-05 01:46:25 +00:00
|
|
|
else
|
2015-08-19 05:46:25 +00:00
|
|
|
GPIO.OUT_CLEAR = BIT(gpio_num);
|
2015-06-05 01:46:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Toggle output of a pin
|
|
|
|
*
|
|
|
|
* Only works if pin has been set to GPIO_OUTPUT via gpio_enable()
|
|
|
|
*/
|
2015-11-28 07:01:03 +00:00
|
|
|
static inline void gpio_toggle(const uint8_t gpio_num)
|
2015-06-05 01:46:25 +00:00
|
|
|
{
|
|
|
|
/* Why implement like this instead of GPIO_OUT_REG ^= xxx?
|
|
|
|
Concurrency. If an interrupt or higher priority task writes to
|
|
|
|
GPIO_OUT between reading and writing, only the gpio_num pin can
|
|
|
|
get an invalid value. Prevents one task from clobbering another
|
|
|
|
task's pins, without needing to disable/enable interrupts.
|
|
|
|
*/
|
2015-08-19 05:46:25 +00:00
|
|
|
if(GPIO.OUT & BIT(gpio_num))
|
|
|
|
GPIO.OUT_CLEAR = BIT(gpio_num);
|
2015-06-05 01:46:25 +00:00
|
|
|
else
|
2015-08-19 05:46:25 +00:00
|
|
|
GPIO.OUT_SET = BIT(gpio_num);
|
2015-06-05 01:46:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Read input value of a GPIO pin.
|
|
|
|
*
|
|
|
|
* If pin is set as an input, this reads the value on the pin.
|
|
|
|
* If pin is set as an output, this reads the last value written to the pin.
|
|
|
|
*/
|
2015-11-28 07:01:03 +00:00
|
|
|
static inline bool gpio_read(const uint8_t gpio_num)
|
2015-06-05 01:46:25 +00:00
|
|
|
{
|
2015-08-19 05:46:25 +00:00
|
|
|
return GPIO.IN & BIT(gpio_num);
|
2015-06-05 01:46:25 +00:00
|
|
|
}
|
|
|
|
|
2015-06-08 08:09:06 +00:00
|
|
|
extern void gpio_interrupt_handler(void);
|
|
|
|
|
|
|
|
/* Set the interrupt type for a given pin
|
2015-06-17 22:59:33 +00:00
|
|
|
*
|
2015-08-19 05:46:25 +00:00
|
|
|
* If int_type is not GPIO_INTTYPE_NONE, the gpio_interrupt_handler will be attached and unmasked.
|
2015-06-08 08:09:06 +00:00
|
|
|
*/
|
2015-11-28 07:01:03 +00:00
|
|
|
static inline void gpio_set_interrupt(const uint8_t gpio_num, const gpio_inttype_t int_type)
|
2015-06-08 08:09:06 +00:00
|
|
|
{
|
2015-08-19 05:46:25 +00:00
|
|
|
GPIO.CONF[gpio_num] = SET_FIELD(GPIO.CONF[gpio_num], GPIO_CONF_INTTYPE, int_type);
|
|
|
|
if(int_type != GPIO_INTTYPE_NONE) {
|
2015-06-08 23:00:32 +00:00
|
|
|
_xt_isr_attach(INUM_GPIO, gpio_interrupt_handler);
|
|
|
|
_xt_isr_unmask(1<<INUM_GPIO);
|
2015-06-08 08:09:06 +00:00
|
|
|
}
|
|
|
|
}
|
2015-06-05 01:46:25 +00:00
|
|
|
|
2015-06-17 22:59:33 +00:00
|
|
|
/* Return the interrupt type set for a pin */
|
2015-11-28 07:01:03 +00:00
|
|
|
static inline gpio_inttype_t gpio_get_interrupt(const uint8_t gpio_num)
|
2015-06-17 22:59:33 +00:00
|
|
|
{
|
2015-08-26 00:49:00 +00:00
|
|
|
return (gpio_inttype_t)FIELD2VAL(GPIO_CONF_INTTYPE, GPIO.CONF[gpio_num]);
|
2015-06-17 22:59:33 +00:00
|
|
|
}
|
|
|
|
|
2015-06-05 01:46:25 +00:00
|
|
|
#endif
|