mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
drivers/stmpe811: add spi mode
This commit is contained in:
parent
9a2c880a60
commit
e17fe0aee2
@ -7,5 +7,5 @@ ifneq (,$(filter disp_dev,$(USEMODULE)))
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
ifneq (,$(filter touch_dev,$(USEMODULE)))
|
ifneq (,$(filter touch_dev,$(USEMODULE)))
|
||||||
USEMODULE += stmpe811
|
USEMODULE += stmpe811_i2c
|
||||||
endif
|
endif
|
||||||
|
@ -155,6 +155,10 @@ ifneq (,$(filter slipdev_%,$(USEMODULE)))
|
|||||||
USEMODULE += slipdev
|
USEMODULE += slipdev
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
ifneq (,$(filter stmpe811_%,$(USEMODULE)))
|
||||||
|
USEMODULE += stmpe811
|
||||||
|
endif
|
||||||
|
|
||||||
ifneq (,$(filter sx126%,$(USEMODULE)))
|
ifneq (,$(filter sx126%,$(USEMODULE)))
|
||||||
USEMODULE += sx126x
|
USEMODULE += sx126x
|
||||||
endif
|
endif
|
||||||
|
@ -23,7 +23,11 @@
|
|||||||
|
|
||||||
#include "saul.h"
|
#include "saul.h"
|
||||||
#include "periph/gpio.h"
|
#include "periph/gpio.h"
|
||||||
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
|
#include "periph/spi.h"
|
||||||
|
#else
|
||||||
#include "periph/i2c.h"
|
#include "periph/i2c.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef MODULE_TOUCH_DEV
|
#ifdef MODULE_TOUCH_DEV
|
||||||
#include "touch_dev.h"
|
#include "touch_dev.h"
|
||||||
@ -60,8 +64,17 @@ typedef void (*stmpe811_event_cb_t)(void *arg);
|
|||||||
* @brief Device initialization parameters
|
* @brief Device initialization parameters
|
||||||
*/
|
*/
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
|
/* SPI configuration */
|
||||||
|
spi_t spi; /**< SPI bus */
|
||||||
|
spi_mode_t mode; /**< SPI mode */
|
||||||
|
spi_clk_t clk; /**< clock speed for the SPI bus */
|
||||||
|
gpio_t cs; /**< chip select pin */
|
||||||
|
#else
|
||||||
|
/* I2C details */
|
||||||
i2c_t i2c; /**< I2C device which is used */
|
i2c_t i2c; /**< I2C device which is used */
|
||||||
uint8_t addr; /**< Device I2C address */
|
uint8_t addr; /**< Device I2C address */
|
||||||
|
#endif
|
||||||
gpio_t int_pin; /**< Touch screen interrupt pin */
|
gpio_t int_pin; /**< Touch screen interrupt pin */
|
||||||
uint16_t xmax; /**< Touch screen max X position */
|
uint16_t xmax; /**< Touch screen max X position */
|
||||||
uint16_t ymax; /**< Touch screen max Y position */
|
uint16_t ymax; /**< Touch screen max Y position */
|
||||||
@ -92,7 +105,7 @@ typedef struct {
|
|||||||
* @return 0 on success
|
* @return 0 on success
|
||||||
* @return -ENODEV when no valid device
|
* @return -ENODEV when no valid device
|
||||||
* @return -EIO when software reset failed
|
* @return -EIO when software reset failed
|
||||||
* @return -EPROTO on any I2C error
|
* @return -EPROTO on any bus error
|
||||||
*/
|
*/
|
||||||
int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params,
|
int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params,
|
||||||
stmpe811_event_cb_t cb, void *arg);
|
stmpe811_event_cb_t cb, void *arg);
|
||||||
@ -104,7 +117,7 @@ int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params,
|
|||||||
* @param[out] position Touch position
|
* @param[out] position Touch position
|
||||||
*
|
*
|
||||||
* @return 0 on success
|
* @return 0 on success
|
||||||
* @return -EPROTO on any I2C error
|
* @return -EPROTO on any bus error
|
||||||
*/
|
*/
|
||||||
int stmpe811_read_touch_position(stmpe811_t *dev, stmpe811_touch_position_t *position);
|
int stmpe811_read_touch_position(stmpe811_t *dev, stmpe811_touch_position_t *position);
|
||||||
|
|
||||||
@ -115,7 +128,7 @@ int stmpe811_read_touch_position(stmpe811_t *dev, stmpe811_touch_position_t *pos
|
|||||||
* @param[out] state Touch state
|
* @param[out] state Touch state
|
||||||
*
|
*
|
||||||
* @return 0 on success
|
* @return 0 on success
|
||||||
* @return -EPROTO on any I2C error
|
* @return -EPROTO on any busI2aC error
|
||||||
*/
|
*/
|
||||||
int stmpe811_read_touch_state(const stmpe811_t *dev, stmpe811_touch_state_t *state);
|
int stmpe811_read_touch_state(const stmpe811_t *dev, stmpe811_touch_state_t *state);
|
||||||
|
|
||||||
|
@ -6,12 +6,32 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
config MODULE_STMPE811
|
config MODULE_STMPE811
|
||||||
bool "STMPE811 Touchscreen Controller"
|
bool
|
||||||
depends on HAS_PERIPH_GPIO
|
depends on HAS_PERIPH_GPIO
|
||||||
depends on HAS_PERIPH_GPIO_IRQ
|
depends on HAS_PERIPH_GPIO_IRQ
|
||||||
depends on HAS_PERIPH_I2C
|
|
||||||
depends on TEST_KCONFIG
|
|
||||||
select MODULE_PERIPH_GPIO
|
select MODULE_PERIPH_GPIO
|
||||||
select MODULE_PERIPH_GPIO_IRQ
|
select MODULE_PERIPH_GPIO_IRQ
|
||||||
select MODULE_PERIPH_I2C
|
|
||||||
select MODULE_XTIMER
|
select MODULE_XTIMER
|
||||||
|
depends on TEST_KCONFIG
|
||||||
|
|
||||||
|
choice
|
||||||
|
bool "STMPE811 Touchscreen Controller"
|
||||||
|
optional
|
||||||
|
depends on TEST_KCONFIG
|
||||||
|
help
|
||||||
|
The driver supports both STMPE811 connected either via SPI or
|
||||||
|
I2C bus. Select one combination.
|
||||||
|
|
||||||
|
config MODULE_STMPE811_I2C
|
||||||
|
bool "STMPE811 on I2C"
|
||||||
|
depends on HAS_PERIPH_I2C
|
||||||
|
select MODULE_PERIPH_I2C
|
||||||
|
select MODULE_STMPE811
|
||||||
|
|
||||||
|
config MODULE_STMPE811_SPI
|
||||||
|
bool "STMPE811 on SPI"
|
||||||
|
depends on HAS_PERIPH_SPI
|
||||||
|
select MODULE_PERIPH_SPI
|
||||||
|
select MODULE_STMPE811
|
||||||
|
|
||||||
|
endchoice
|
||||||
|
@ -1,3 +1,11 @@
|
|||||||
FEATURES_REQUIRED += periph_gpio_irq
|
FEATURES_REQUIRED += periph_gpio_irq
|
||||||
FEATURES_REQUIRED += periph_i2c
|
|
||||||
|
ifneq (,$(filter stmpe811_spi,$(USEMODULE)))
|
||||||
|
FEATURES_REQUIRED += periph_spi
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifneq (,$(filter stmpe811_i2c,$(USEMODULE)))
|
||||||
|
FEATURES_REQUIRED += periph_i2c
|
||||||
|
endif
|
||||||
|
|
||||||
USEMODULE += xtimer
|
USEMODULE += xtimer
|
||||||
|
@ -1,2 +1,5 @@
|
|||||||
USEMODULE_INCLUDES_stmpe811 := $(LAST_MAKEFILEDIR)/include
|
USEMODULE_INCLUDES_stmpe811 := $(LAST_MAKEFILEDIR)/include
|
||||||
USEMODULE_INCLUDES += $(USEMODULE_INCLUDES_stmpe811)
|
USEMODULE_INCLUDES += $(USEMODULE_INCLUDES_stmpe811)
|
||||||
|
|
||||||
|
PSEUDOMODULES += stmpe811_i2c
|
||||||
|
PSEUDOMODULES += stmpe811_spi
|
||||||
|
@ -103,6 +103,15 @@ extern "C" {
|
|||||||
#define STMPE811_SYS_CTRL2_TS_OFF (1 << 3) /**< Disable Temperature sensor */
|
#define STMPE811_SYS_CTRL2_TS_OFF (1 << 3) /**< Disable Temperature sensor */
|
||||||
/** @} */
|
/** @} */
|
||||||
|
|
||||||
|
/* @name SPI_CFG register bitfields
|
||||||
|
* @{
|
||||||
|
*/
|
||||||
|
#define STMPE811_SPI_CFG_SPI_CLK_MOD0 (1 << 0) /**< SCAD/A0 pin during power-up reset */
|
||||||
|
#define STMPE811_SPI_CFG_SPI_CLK_MOD1 (1 << 1) /**< SCAD/A0 pin during power-up reset */
|
||||||
|
#define STMPE811_SPI_CFG_AUTO_INCR (1 << 2) /**< SPI transactions internal autoincrement */
|
||||||
|
/** @} */
|
||||||
|
|
||||||
|
|
||||||
/* @name INT_CTRL register bitfields
|
/* @name INT_CTRL register bitfields
|
||||||
* @{
|
* @{
|
||||||
*/
|
*/
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
#ifndef STMPE811_PARAMS_H
|
#ifndef STMPE811_PARAMS_H
|
||||||
#define STMPE811_PARAMS_H
|
#define STMPE811_PARAMS_H
|
||||||
|
|
||||||
|
#include "kernel_defines.h"
|
||||||
#include "board.h"
|
#include "board.h"
|
||||||
#include "stmpe811.h"
|
#include "stmpe811.h"
|
||||||
#include "stmpe811_constants.h"
|
#include "stmpe811_constants.h"
|
||||||
@ -33,12 +34,27 @@ extern "C" {
|
|||||||
*
|
*
|
||||||
* These default values are adapted for the @ref boards_stm32f429i-disc1 board
|
* These default values are adapted for the @ref boards_stm32f429i-disc1 board
|
||||||
*/
|
*/
|
||||||
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
|
/* SPI configuration */
|
||||||
|
#ifndef STMPE811_PARAM_SPI_DEV
|
||||||
|
#define STMPE811_PARAM_SPI_DEV SPI_DEV(0)
|
||||||
|
#endif
|
||||||
|
#ifndef STMPE811_PARAM_CLK
|
||||||
|
#define STMPE811_PARAM_CLK SPI_CLK_1MHZ
|
||||||
|
#endif
|
||||||
|
#ifndef STMPE811_PARAM_CS
|
||||||
|
#define STMPE811_PARAM_CS GPIO_PIN(0, 0)
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
/* I2C configuration */
|
||||||
#ifndef STMPE811_PARAM_I2C_DEV
|
#ifndef STMPE811_PARAM_I2C_DEV
|
||||||
#define STMPE811_PARAM_I2C_DEV I2C_DEV(0)
|
#define STMPE811_PARAM_I2C_DEV I2C_DEV(0)
|
||||||
#endif
|
#endif
|
||||||
#ifndef STMPE811_PARAM_ADDR
|
#ifndef STMPE811_PARAM_ADDR
|
||||||
#define STMPE811_PARAM_ADDR (STMPE811_I2C_ADDR_DEFAULT)
|
#define STMPE811_PARAM_ADDR (STMPE811_I2C_ADDR_DEFAULT)
|
||||||
#endif
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifndef STMPE811_PARAM_INT_PIN
|
#ifndef STMPE811_PARAM_INT_PIN
|
||||||
#define STMPE811_PARAM_INT_PIN GPIO_PIN(0, 15)
|
#define STMPE811_PARAM_INT_PIN GPIO_PIN(0, 15)
|
||||||
#endif
|
#endif
|
||||||
@ -50,6 +66,15 @@ extern "C" {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef STMPE811_PARAMS
|
#ifndef STMPE811_PARAMS
|
||||||
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
|
#define STMPE811_PARAMS { .spi = STMPE811_PARAM_SPI_DEV, \
|
||||||
|
.clk = STMPE811_PARAM_CLK, \
|
||||||
|
.cs = STMPE811_PARAM_CS, \
|
||||||
|
.int_pin = STMPE811_PARAM_INT_PIN, \
|
||||||
|
.xmax = STMPE811_PARAM_XMAX, \
|
||||||
|
.ymax = STMPE811_PARAM_YMAX, \
|
||||||
|
}
|
||||||
|
#else
|
||||||
#define STMPE811_PARAMS { .i2c = STMPE811_PARAM_I2C_DEV, \
|
#define STMPE811_PARAMS { .i2c = STMPE811_PARAM_I2C_DEV, \
|
||||||
.addr = STMPE811_PARAM_ADDR, \
|
.addr = STMPE811_PARAM_ADDR, \
|
||||||
.int_pin = STMPE811_PARAM_INT_PIN, \
|
.int_pin = STMPE811_PARAM_INT_PIN, \
|
||||||
@ -57,6 +82,7 @@ extern "C" {
|
|||||||
.ymax = STMPE811_PARAM_YMAX, \
|
.ymax = STMPE811_PARAM_YMAX, \
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
#endif
|
||||||
/**@}*/
|
/**@}*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -21,7 +21,11 @@
|
|||||||
#include <inttypes.h>
|
#include <inttypes.h>
|
||||||
|
|
||||||
#include "xtimer.h"
|
#include "xtimer.h"
|
||||||
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
|
#include "periph/spi.h"
|
||||||
|
#else
|
||||||
#include "periph/i2c.h"
|
#include "periph/i2c.h"
|
||||||
|
#endif
|
||||||
#include "periph/gpio.h"
|
#include "periph/gpio.h"
|
||||||
|
|
||||||
#include "stmpe811.h"
|
#include "stmpe811.h"
|
||||||
@ -31,12 +35,93 @@
|
|||||||
#define ENABLE_DEBUG 0
|
#define ENABLE_DEBUG 0
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
|
||||||
#define STMPE811_DEV_I2C (dev->params.i2c)
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
#define STMPE811_DEV_ADDR (dev->params.addr)
|
#define BUS (dev->params.spi)
|
||||||
|
#define CS (dev->params.cs)
|
||||||
|
#define CLK (dev->params.clk)
|
||||||
|
#define MODE (dev->params.mode)
|
||||||
|
#define WRITE_MASK (0x7F)
|
||||||
|
#else
|
||||||
|
#define BUS (dev->params.i2c)
|
||||||
|
#define ADDR (dev->params.addr)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
#if IS_USED(MODULE_STMPE811_SPI) /* using SPI mode */
|
||||||
|
static inline void _acquire(const stmpe811_t *dev)
|
||||||
|
{
|
||||||
|
spi_acquire(BUS, CS, MODE, CLK);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void _release(const stmpe811_t *dev)
|
||||||
|
{
|
||||||
|
spi_release(BUS);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _read_reg(const stmpe811_t *dev, uint8_t reg, uint8_t *data)
|
||||||
|
{
|
||||||
|
*data = spi_transfer_reg(BUS, CS, reg | 0x80, 0x00);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _write_reg(const stmpe811_t *dev, uint8_t reg, uint8_t data)
|
||||||
|
{
|
||||||
|
(void)spi_transfer_reg(BUS, CS, (reg & WRITE_MASK), data);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _read_burst(const stmpe811_t *dev, uint8_t reg, void *buf, size_t len)
|
||||||
|
{
|
||||||
|
uint8_t reg_read = reg | 0x80;
|
||||||
|
|
||||||
|
/* since SPI is in auto-increment mode subsequent reads will ignore the
|
||||||
|
content of the reg_read buffer, and itself can't overflow since it matches
|
||||||
|
the amount of data per register */
|
||||||
|
spi_transfer_regs(BUS, CS, reg_read, ®_read, buf, len);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* using I2C mode */
|
||||||
|
|
||||||
|
static inline void _acquire(const stmpe811_t *dev)
|
||||||
|
{
|
||||||
|
i2c_acquire(BUS);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void _release(const stmpe811_t *dev)
|
||||||
|
{
|
||||||
|
i2c_release(BUS);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _read_reg(const stmpe811_t *dev, uint8_t reg, uint8_t *data)
|
||||||
|
{
|
||||||
|
if (i2c_read_reg(BUS, ADDR, reg, data, 0) != 0) {
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _write_reg(const stmpe811_t *dev, uint8_t reg, uint8_t data)
|
||||||
|
{
|
||||||
|
if (i2c_write_reg(BUS, ADDR, reg, data, 0) != 0) {
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _read_burst(const stmpe811_t *dev, uint8_t reg, void *buf, size_t len)
|
||||||
|
{
|
||||||
|
if (i2c_read_regs(BUS, ADDR, reg, buf, len, 0) != 0) {
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif /* bus mode selection */
|
||||||
|
|
||||||
static void _gpio_irq(void *arg)
|
static void _gpio_irq(void *arg)
|
||||||
{
|
{
|
||||||
const stmpe811_t *dev = (const stmpe811_t *)arg;
|
const stmpe811_t *dev = (const stmpe811_t *)arg;
|
||||||
|
|
||||||
assert(dev);
|
assert(dev);
|
||||||
|
|
||||||
if (dev->cb) {
|
if (dev->cb) {
|
||||||
@ -46,15 +131,13 @@ static void _gpio_irq(void *arg)
|
|||||||
|
|
||||||
static int _soft_reset(const stmpe811_t *dev)
|
static int _soft_reset(const stmpe811_t *dev)
|
||||||
{
|
{
|
||||||
if (i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
if (_write_reg(dev, STMPE811_SYS_CTRL1, STMPE811_SYS_CTRL1_SOFT_RESET ) < 0) {
|
||||||
STMPE811_SYS_CTRL1, STMPE811_SYS_CTRL1_SOFT_RESET, 0) < 0) {
|
|
||||||
DEBUG("[stmpe811] soft reset: cannot write soft reset bit to SYS_CTRL1 register\n");
|
DEBUG("[stmpe811] soft reset: cannot write soft reset bit to SYS_CTRL1 register\n");
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
xtimer_msleep(10);
|
xtimer_msleep(10);
|
||||||
|
|
||||||
if (i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
if (_write_reg(dev, STMPE811_SYS_CTRL1, 0) < 0) {
|
||||||
STMPE811_SYS_CTRL1, 0, 0) < 0) {
|
|
||||||
DEBUG("[stmpe811] soft reset: cannot clear SYS_CTRL1 register\n");
|
DEBUG("[stmpe811] soft reset: cannot clear SYS_CTRL1 register\n");
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
@ -65,33 +148,78 @@ static int _soft_reset(const stmpe811_t *dev)
|
|||||||
|
|
||||||
static void _reset_fifo(const stmpe811_t *dev)
|
static void _reset_fifo(const stmpe811_t *dev)
|
||||||
{
|
{
|
||||||
i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
_write_reg(dev, STMPE811_FIFO_CTRL_STA, STMPE811_FIFO_CTRL_STA_RESET);
|
||||||
STMPE811_FIFO_CTRL_STA, STMPE811_FIFO_CTRL_STA_RESET, 0);
|
_write_reg(dev, STMPE811_FIFO_CTRL_STA, 0);
|
||||||
i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
|
||||||
STMPE811_FIFO_CTRL_STA, 0, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _clear_interrupt_status(const stmpe811_t *dev)
|
static void _clear_interrupt_status(const stmpe811_t *dev)
|
||||||
{
|
{
|
||||||
i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR, STMPE811_INT_STA, 0xff, 0);
|
_write_reg(dev, STMPE811_INT_STA, 0xff);
|
||||||
}
|
}
|
||||||
|
|
||||||
int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params, stmpe811_event_cb_t cb, void *arg)
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
|
static int _stmpe811_check_mode(stmpe811_t *dev)
|
||||||
|
{
|
||||||
|
/* can iterate directly through the enum since they might not be
|
||||||
|
monotonically incrementing */
|
||||||
|
uint8_t modes[] = { SPI_MODE_0, SPI_MODE_1, SPI_MODE_2, SPI_MODE_3};
|
||||||
|
uint8_t reg;
|
||||||
|
|
||||||
|
for (uint8_t i = 0; i < sizeof(modes); i++) {
|
||||||
|
DEBUG("[stmpe811] init: set spi mode to 0x%02x ... ", modes[i]);
|
||||||
|
dev->params.mode = modes[i];
|
||||||
|
/* acquire */
|
||||||
|
_acquire(dev);
|
||||||
|
/* configure auto increment SPI */
|
||||||
|
_read_reg(dev, STMPE811_SPI_CFG, ®);
|
||||||
|
reg = STMPE811_SPI_CFG_AUTO_INCR;
|
||||||
|
_write_reg(dev, STMPE811_SPI_CFG, reg);
|
||||||
|
_read_reg(dev, STMPE811_SPI_CFG, ®);
|
||||||
|
if (reg & STMPE811_SPI_CFG_AUTO_INCR) {
|
||||||
|
DEBUG("success\n");
|
||||||
|
_release(dev);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
DEBUG("failed\n");
|
||||||
|
_release(dev);
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t *params, stmpe811_event_cb_t cb,
|
||||||
|
void *arg)
|
||||||
{
|
{
|
||||||
dev->params = *params;
|
dev->params = *params;
|
||||||
dev->cb = cb;
|
dev->cb = cb;
|
||||||
dev->cb_arg = arg;
|
dev->cb_arg = arg;
|
||||||
|
dev->prev_x = 0;
|
||||||
|
dev->prev_y = 0;
|
||||||
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
uint8_t reg;
|
||||||
|
|
||||||
/* Acquire I2C device */
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
i2c_acquire(STMPE811_DEV_I2C);
|
/* configure the chip-select pin */
|
||||||
|
if (spi_init_cs(BUS, CS) != SPI_OK) {
|
||||||
|
DEBUG("[stmpe811] error: unable to configure chip the select pin\n");
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check mode configuration */
|
||||||
|
if(_stmpe811_check_mode(dev) != 0) {
|
||||||
|
DEBUG("[stmpe811] error: couldn't setup SPI\n");
|
||||||
|
return -EIO;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* acquire bus */
|
||||||
|
_acquire(dev);
|
||||||
|
|
||||||
uint16_t device_id;
|
uint16_t device_id;
|
||||||
if (i2c_read_regs(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
if (_read_burst(dev, STMPE811_CHIP_ID, &device_id, 2) < 0) {
|
||||||
STMPE811_CHIP_ID, &device_id, 2, 0) < 0) {
|
|
||||||
DEBUG("[stmpe811] init: cannot read CHIP_ID register\n");
|
DEBUG("[stmpe811] init: cannot read CHIP_ID register\n");
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,7 +227,7 @@ int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params, stmpe811_ev
|
|||||||
if (device_id != STMPE811_CHIP_ID_VALUE) {
|
if (device_id != STMPE811_CHIP_ID_VALUE) {
|
||||||
DEBUG("[stmpe811] init: invalid device id (actual: 0x%04X, expected: 0x%04X)\n",
|
DEBUG("[stmpe811] init: invalid device id (actual: 0x%04X, expected: 0x%04X)\n",
|
||||||
device_id, STMPE811_CHIP_ID_VALUE);
|
device_id, STMPE811_CHIP_ID_VALUE);
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -108,7 +236,7 @@ int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params, stmpe811_ev
|
|||||||
ret = _soft_reset(dev);
|
ret = _soft_reset(dev);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
DEBUG("[stmpe811] init: reset failed\n");
|
DEBUG("[stmpe811] init: reset failed\n");
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -117,58 +245,50 @@ int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params, stmpe811_ev
|
|||||||
/* Initialization sequence */
|
/* Initialization sequence */
|
||||||
|
|
||||||
/* disable temperature sensor and GPIO */
|
/* disable temperature sensor and GPIO */
|
||||||
ret = i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret =
|
||||||
STMPE811_SYS_CTRL2,
|
_write_reg(dev, STMPE811_SYS_CTRL2,
|
||||||
(STMPE811_SYS_CTRL2_TS_OFF | STMPE811_SYS_CTRL2_GPIO_OFF), 0);
|
(STMPE811_SYS_CTRL2_TS_OFF | STMPE811_SYS_CTRL2_GPIO_OFF));
|
||||||
|
|
||||||
/* set to 80 cycles and adc resolution to 12 bit*/
|
/* set to 80 cycles and adc resolution to 12 bit*/
|
||||||
uint8_t reg = ((uint8_t)(STMPE811_ADC_CTRL1_SAMPLE_TIME_80 << STMPE811_ADC_CTRL1_SAMPLE_TIME_POS) |
|
reg =
|
||||||
STMPE811_ADC_CTRL1_MOD_12B);
|
((uint8_t)(STMPE811_ADC_CTRL1_SAMPLE_TIME_80 << STMPE811_ADC_CTRL1_SAMPLE_TIME_POS) |
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
STMPE811_ADC_CTRL1_MOD_12B);
|
||||||
STMPE811_ADC_CTRL1, reg, 0);
|
|
||||||
|
ret += _write_reg(dev, STMPE811_ADC_CTRL1, reg);
|
||||||
|
|
||||||
/* set adc clock speed to 3.25 MHz */
|
/* set adc clock speed to 3.25 MHz */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_ADC_CTRL2, STMPE811_ADC_CTRL2_FREQ_3_25MHZ);
|
||||||
STMPE811_ADC_CTRL2, STMPE811_ADC_CTRL2_FREQ_3_25MHZ, 0);
|
|
||||||
|
|
||||||
/* set GPIO AF to function as ts/adc */
|
/* set GPIO AF to function as ts/adc */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_GPIO_ALT_FUNCTION, 0x00);
|
||||||
STMPE811_GPIO_ALT_FUNCTION, 0x00, 0);
|
|
||||||
|
|
||||||
/* set touchscreen configuration */
|
/* set touchscreen configuration */
|
||||||
reg = ((uint8_t)(STMPE811_TSC_CFG_AVE_CTRL_4 << STMPE811_TSC_CFG_AVE_CTRL_POS) |
|
reg = ((uint8_t)(STMPE811_TSC_CFG_AVE_CTRL_4 << STMPE811_TSC_CFG_AVE_CTRL_POS) |
|
||||||
(uint8_t)(STMPE811_TSC_CFG_TOUCH_DET_DELAY_500US << STMPE811_TSC_CFG_TOUCH_DET_DELAY_POS) |
|
(uint8_t)(STMPE811_TSC_CFG_TOUCH_DET_DELAY_500US <<
|
||||||
(STMPE811_TSC_CFG_SETTLING_500US));
|
STMPE811_TSC_CFG_TOUCH_DET_DELAY_POS) |
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
(STMPE811_TSC_CFG_SETTLING_500US));
|
||||||
STMPE811_TSC_CFG, reg, 0);
|
ret += _write_reg(dev, STMPE811_TSC_CFG, reg);
|
||||||
|
|
||||||
/* set fifo threshold */
|
/* set fifo threshold */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_FIFO_TH, 0x01);
|
||||||
STMPE811_FIFO_TH, 0x01, 0);
|
|
||||||
|
|
||||||
/* reset fifo */
|
/* reset fifo */
|
||||||
_reset_fifo(dev);
|
_reset_fifo(dev);
|
||||||
|
|
||||||
/* set fractional part to 7, whole part to 1 */
|
/* set fractional part to 7, whole part to 1 */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_TSC_FRACTION_Z, STMPE811_TSC_FRACTION_Z_7_1);
|
||||||
STMPE811_TSC_FRACTION_Z, STMPE811_TSC_FRACTION_Z_7_1, 0);
|
|
||||||
|
|
||||||
/* set current limit value to 50 mA */
|
/* set current limit value to 50 mA */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_TSC_I_DRIVE, STMPE811_TSC_I_DRIVE_50MA);
|
||||||
STMPE811_TSC_I_DRIVE, STMPE811_TSC_I_DRIVE_50MA, 0);
|
|
||||||
|
|
||||||
/* enable touchscreen clock */
|
/* enable touchscreen clock */
|
||||||
ret += i2c_read_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _read_reg(dev, STMPE811_SYS_CTRL2, ®);
|
||||||
STMPE811_SYS_CTRL2, ®, 0);
|
|
||||||
reg &= ~STMPE811_SYS_CTRL2_TSC_OFF;
|
reg &= ~STMPE811_SYS_CTRL2_TSC_OFF;
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_SYS_CTRL2, reg);
|
||||||
STMPE811_SYS_CTRL2, reg, 0);
|
|
||||||
|
|
||||||
ret += i2c_read_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _read_reg(dev, STMPE811_TSC_CTRL, ®);
|
||||||
STMPE811_TSC_CTRL, ®, 0);
|
|
||||||
reg |= STMPE811_TSC_CTRL_EN;
|
reg |= STMPE811_TSC_CTRL_EN;
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_TSC_CTRL, reg);
|
||||||
STMPE811_TSC_CTRL, reg, 0);
|
|
||||||
|
|
||||||
/* clear interrupt status */
|
/* clear interrupt status */
|
||||||
_clear_interrupt_status(dev);
|
_clear_interrupt_status(dev);
|
||||||
@ -178,22 +298,21 @@ int stmpe811_init(stmpe811_t *dev, const stmpe811_params_t * params, stmpe811_ev
|
|||||||
gpio_init_int(dev->params.int_pin, GPIO_IN, GPIO_FALLING, _gpio_irq, dev);
|
gpio_init_int(dev->params.int_pin, GPIO_IN, GPIO_FALLING, _gpio_irq, dev);
|
||||||
|
|
||||||
/* Enable touchscreen interrupt */
|
/* Enable touchscreen interrupt */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_INT_EN, STMPE811_INT_EN_TOUCH_DET);
|
||||||
STMPE811_INT_EN, STMPE811_INT_EN_TOUCH_DET, 0);
|
|
||||||
|
|
||||||
/* Enable global interrupt */
|
/* Enable global interrupt */
|
||||||
ret += i2c_write_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
ret += _write_reg(dev, STMPE811_INT_CTRL,
|
||||||
STMPE811_INT_CTRL, STMPE811_INT_CTRL_GLOBAL_INT | STMPE811_INT_CTRL_INT_TYPE, 0);
|
STMPE811_INT_CTRL_GLOBAL_INT | STMPE811_INT_CTRL_INT_TYPE);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
DEBUG("[stmpe811] init: initialization sequence failed\n");
|
DEBUG("[stmpe811] init: initialization sequence failed\n");
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Release I2C device */
|
/* Release I2C device */
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
|
|
||||||
DEBUG("[stmpe811] initialization successful\n");
|
DEBUG("[stmpe811] initialization successful\n");
|
||||||
|
|
||||||
@ -204,31 +323,40 @@ int stmpe811_read_touch_position(stmpe811_t *dev, stmpe811_touch_position_t *pos
|
|||||||
{
|
{
|
||||||
uint16_t tmp_x, tmp_y;
|
uint16_t tmp_x, tmp_y;
|
||||||
|
|
||||||
/* Acquire I2C device */
|
/* Acquire device bus */
|
||||||
i2c_acquire(STMPE811_DEV_I2C);
|
_acquire(dev);
|
||||||
|
|
||||||
/* Ensure there's a least one position measured in the FIFO */
|
/* Ensure there's a least one position measured in the FIFO */
|
||||||
uint8_t fifo_size = 0;
|
uint8_t fifo_size = 0;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
i2c_read_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
_read_reg(dev, STMPE811_FIFO_SIZE, &fifo_size);
|
||||||
STMPE811_FIFO_SIZE, &fifo_size, 0);
|
|
||||||
} while (!fifo_size);
|
} while (!fifo_size);
|
||||||
|
|
||||||
uint8_t xyz[4];
|
uint8_t xyz[4];
|
||||||
uint32_t xyz_ul;
|
uint32_t xyz_ul;
|
||||||
|
|
||||||
if (i2c_read_regs(STMPE811_DEV_I2C, STMPE811_DEV_ADDR,
|
#if IS_USED(MODULE_STMPE811_SPI)
|
||||||
STMPE811_TSC_DATA_NON_INC, &xyz, sizeof(xyz), 0) < 0) {
|
for (uint8_t i = 0; i < sizeof(xyz); i++) {
|
||||||
|
if (_read_reg(dev, STMPE811_TSC_DATA_NON_INC, &xyz[i]) < 0) {
|
||||||
|
DEBUG("[stmpe811] position: cannot read position\n");
|
||||||
|
_release(dev);
|
||||||
|
return -EPROTO;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
if (_read_burst(dev, STMPE811_TSC_DATA_NON_INC, xyz, sizeof(xyz)) < 0) {
|
||||||
DEBUG("[stmpe811] position: cannot read position\n");
|
DEBUG("[stmpe811] position: cannot read position\n");
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Release I2C device */
|
/* Release device bus */
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
|
|
||||||
xyz_ul = ((uint32_t)xyz[0] << 24) | ((uint32_t)xyz[1] << 16) | \
|
xyz_ul = ((uint32_t)xyz[0] << 24) | ((uint32_t)xyz[1] << 16) | \
|
||||||
((uint32_t)xyz[2] << 8) | (xyz[3] << 0);
|
((uint32_t)xyz[2] << 8) | (xyz[3] << 0);
|
||||||
|
|
||||||
tmp_x = (xyz_ul >> 20) & 0xfff;
|
tmp_x = (xyz_ul >> 20) & 0xfff;
|
||||||
tmp_y = (xyz_ul >> 8) & 0xfff;
|
tmp_y = (xyz_ul >> 8) & 0xfff;
|
||||||
@ -272,12 +400,12 @@ int stmpe811_read_touch_state(const stmpe811_t *dev, stmpe811_touch_state_t *sta
|
|||||||
{
|
{
|
||||||
uint8_t val;
|
uint8_t val;
|
||||||
|
|
||||||
/* Acquire I2C device */
|
/* Acquire device bus */
|
||||||
i2c_acquire(STMPE811_DEV_I2C);
|
_acquire(dev);
|
||||||
|
|
||||||
if (i2c_read_reg(STMPE811_DEV_I2C, STMPE811_DEV_ADDR, STMPE811_TSC_CTRL, &val, 0) < 0) {
|
if (_read_reg(dev, STMPE811_TSC_CTRL, &val) < 0) {
|
||||||
DEBUG("[stmpe811] position: cannot read touch state\n");
|
DEBUG("[stmpe811] position: cannot read touch state\n");
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
return -EPROTO;
|
return -EPROTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -292,7 +420,7 @@ int stmpe811_read_touch_state(const stmpe811_t *dev, stmpe811_touch_state_t *sta
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Release I2C device */
|
/* Release I2C device */
|
||||||
i2c_release(STMPE811_DEV_I2C);
|
_release(dev);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ BOARD ?= stm32f429i-disc1
|
|||||||
|
|
||||||
include ../Makefile.tests_common
|
include ../Makefile.tests_common
|
||||||
|
|
||||||
USEMODULE += stmpe811
|
DRIVER ?= stmpe811_i2c
|
||||||
|
USEMODULE += $(DRIVER)
|
||||||
|
|
||||||
include $(RIOTBASE)/Makefile.include
|
include $(RIOTBASE)/Makefile.include
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# this file enables modules defined in Kconfig. Do not use this file for
|
# this file enables modules defined in Kconfig. Do not use this file for
|
||||||
# application configuration. This is only needed during migration.
|
# application configuration. This is only needed during migration.
|
||||||
CONFIG_MODULE_STMPE811=y
|
CONFIG_MODULE_STMPE811_I2C=y
|
||||||
|
Loading…
Reference in New Issue
Block a user