rt-thread/bsp/stm32f429-disco/drivers/drv_touch.c

255 lines
7.2 KiB
C

/*
* File : drv_touch.c
* This file is part of RT-Thread RTOS
* COPYRIGHT (C) 2018 RT-Thread Develop Team
*
* The license and distribution terms for this file may be
* found in the file LICENSE in this distribution or at
* http://www.rt-thread.org/license/LICENSE
*
* Change Logs:
* Date Author Notes
* 2018-10-03 xuzhuoyi first implementation.
*/
#include "drv_touch.h"
#include "drivers/i2c.h"
#ifdef PKG_USING_LITTLEVGL2RTT
#include "littlevgl2rtt.h"
#endif
#define TSC_I2C_ADDR 0x41 /* 7-bit I2C address */
static struct rt_i2c_bus_device *stmpe811_i2c_bus;
/**
\fn int32_t touch_readRegister (uint8_t reg, uint8_t *val)
\brief Read register value from Touchscreen controller
\param[in] reg Register address
\param[out] val Pointer where data will be read from register
\returns
- \b 0: function succeeded
- \b -1: function failed
*/
static int32_t touch_read (uint8_t reg, uint8_t *val)
{
struct rt_i2c_msg msgs[2];
msgs[0].addr = TSC_I2C_ADDR;
msgs[0].flags = RT_I2C_WR;
msgs[0].buf = ®
msgs[0].len = 1;
msgs[1].addr = TSC_I2C_ADDR;
msgs[1].flags = RT_I2C_RD;
msgs[1].buf = val;
msgs[1].len = 1;
if (rt_i2c_transfer(stmpe811_i2c_bus, msgs, 2) == 2)
{
return RT_EOK;
}
else
{
return -RT_ERROR;
}
}
/**
\fn int32_t touch_writeData (uint8_t reg, const uint8_t *val)
\brief Write value to Touchscreen controller register
\param[in] reg Register address
\param[in] val Pointer with data to write to register
\returns
- \b 0: function succeeded
- \b -1: function failed
*/
static int32_t touch_write (uint8_t reg, uint8_t val)
{
struct rt_i2c_msg msgs;
rt_uint8_t buf[2] = {reg, val};
msgs.addr = TSC_I2C_ADDR;
msgs.flags = RT_I2C_WR;
msgs.buf = buf;
msgs.len = 2;
if (rt_i2c_transfer(stmpe811_i2c_bus, &msgs, 1) == 1)
{
return RT_EOK;
}
else
{
return -RT_ERROR;
}
}
/**
\fn int32_t Touch_Initialize (void)
\brief Initialize touchscreen
\returns
- \b 0: function succeeded
- \b -1: function failed
*/
static rt_err_t stmpe811_touch_init(rt_device_t dev)
{
stmpe811_i2c_bus = rt_i2c_bus_device_find(STMPE811_I2CBUS_NAME);
// ptrI2C->Initialize (NULL);
// ptrI2C->PowerControl(ARM_POWER_FULL);
// ptrI2C->Control (ARM_I2C_BUS_SPEED, ARM_I2C_BUS_SPEED_FAST);
// ptrI2C->Control (ARM_I2C_BUS_CLEAR, 0);
touch_write(STMPE811_SYS_CTRL1, 0x02); /* Reset Touch-screen controller */
rt_thread_mdelay(10); /* Wait 10ms */
touch_write(STMPE811_SYS_CTRL2, 0x0C); /* Enable TSC and ADC */
touch_write(STMPE811_ADC_CTRL1, 0x68); /* Set sample time , 12-bit mode */
rt_thread_mdelay(1); /* Wait 1ms */
touch_write(STMPE811_ADC_CTRL2, 0x01); /* ADC frequency 3.25 MHz */
touch_write(STMPE811_TSC_CFG, 0xC2); /* Detect delay 10us,
Settle time 500us */
touch_write(STMPE811_FIFO_TH, 0x01); /* Threshold for FIFO */
touch_write(STMPE811_FIFO_STA, 0x01); /* FIFO reset */
touch_write(STMPE811_FIFO_STA, 0x00); /* FIFO not reset */
touch_write(STMPE811_TSC_FRACTION_Z, 0x07); /* Fraction z */
touch_write(STMPE811_TSC_I_DRIVE, 0x01); /* Drive 50 mA typical */
touch_write(STMPE811_GPIO_AF, 0x00); /* Pins are used for touchscreen */
touch_write(STMPE811_TSC_CTRL, 0x01); /* Enable TSC */
return 0;
}
/**
\fn int32_t Touch_Uninitialize (void)
\brief De-initialize touchscreen
\returns
- \b 0: function succeeded
- \b -1: function failed
*/
int32_t touch_uninitialize (void) {
touch_write(STMPE811_SYS_CTRL1, 0x02); /* Reset Touch-screen controller */
return 0;
}
/**
\fn int32_t Touch_GetState (TOUCH_STATE *pState)
\brief Get touchscreen state
\param[out] pState pointer to TOUCH_STATE structure
\returns
- \b 0: function succeeded
- \b -1: function failed
*/
int32_t touch_get_state(struct touch_state *pState)
{
uint8_t val;
uint8_t num;
uint8_t xyz[4];
int32_t res;
struct rt_i2c_msg msgs[2];
/* Read touch status */
res = touch_read(STMPE811_TSC_CTRL, &val);
if (res < 0) return -1;
pState->pressed = (val & (1 << 7)) ? 1 : 0;
if (pState->pressed)
{
val = STMPE811_TSC_DATA;
/* If FIFO overflow, discard all samples except the last one */
res = touch_read(STMPE811_FIFO_SIZE, &num);
if (res < 0 || num == 0) return -1;
while (num--)
{
msgs[0].addr = TSC_I2C_ADDR;
msgs[0].flags = RT_I2C_WR;
msgs[0].buf = &val;
msgs[0].len = 1;
//rt_i2c_transfer(stmpe811_i2c_bus, &msgs, 1);
//ptrI2C->MasterTransmit (TSC_I2C_ADDR, &val, 1, true);
//while (ptrI2C->GetStatus().busy);
msgs[1].addr = TSC_I2C_ADDR;
msgs[1].flags = RT_I2C_RD;
msgs[1].buf = xyz;
msgs[1].len = 4;
rt_i2c_transfer(stmpe811_i2c_bus, msgs, 2);
//ptrI2C->MasterReceive (TSC_I2C_ADDR, xyz, 4, false);
//while (ptrI2C->GetStatus().busy);
}
pState->x = (int16_t)((xyz[0] << 4) | ((xyz[1] & 0xF0) >> 4));
pState->y = (int16_t) (xyz[2] | ((xyz[1] & 0x0F) << 8));
}
else
{
/* Clear all data in FIFO */
touch_write(STMPE811_FIFO_STA, 0x1);
touch_write(STMPE811_FIFO_STA, 0x0);
}
return 0;
}
void touch_show_state()
{
int16_t x;
int16_t y;
struct touch_state ts;
touch_get_state(&ts);
x = (3706 - ts.x) / 14;
y = (-461 + ts.y) / 10.5;
rt_kprintf("[drv_touch] touch_show_state, x: %d, y: %d, pressed: %d, padding: %d\n", ts.x , ts.y, ts.pressed, ts.padding);
rt_kprintf("[drv_touch] touch_show_state, phy x: %d, phy y: %d\n", x , y);
}
MSH_CMD_EXPORT(touch_show_state, show screen coordinate in touching);
static void touch_timer(void *parameter)
{
int16_t x;
int16_t y;
struct touch_state ts;
touch_get_state(&ts);
#ifdef PKG_USING_LITTLEVGL2RTT
if(ts.pressed)
{
x = (3706 - ts.x) / 14;
y = (-461 + ts.y) / 10.5;
littlevgl2rtt_send_input_event(x, y, LITTLEVGL2RTT_INPUT_DOWN);
}
else
littlevgl2rtt_send_input_event(-1, -1, LITTLEVGL2RTT_INPUT_UP);
#endif
}
static int rt_hw_touch_init(void)
{
static struct rt_device touch;
/* init device structure */
touch.type = RT_Device_Class_Unknown;
touch.init = stmpe811_touch_init;
/* create 1/8 second timer */
touch.user_data = rt_timer_create("touch", touch_timer, RT_NULL,
RT_TICK_PER_SECOND/8, RT_TIMER_FLAG_PERIODIC);
/* register touch device to RT-Thread */
rt_device_register(&touch, "touch", RT_DEVICE_FLAG_RDWR);
return RT_EOK;
}
INIT_BOARD_EXPORT(rt_hw_touch_init);