Files
luban-lite/kernel/rt-thread/components/drivers/touch/touch.c
刘可亮 11c97ef399 v1.2.1
2025-07-22 11:15:46 +08:00

382 lines
8.9 KiB
C

/*
* Copyright (c) 2006-2021, RT-Thread Development Team
*
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date Author Notes
* 2019-05-20 tyustli the first version
*/
#include "touch.h"
#include <string.h>
#define DBG_TAG "touch"
#define DBG_LVL DBG_INFO
#include <rtdbg.h>
static uint16_t g_touch_angle = 0;
static uint8_t g_touch_dynamic_enable = 0;
static uint8_t g_osd_enable = 0;
static void aic_set_dynamic_touch_rotation(uint16_t angle)
{
g_touch_angle = angle;
g_touch_dynamic_enable = 1;
}
static uint16_t aic_get_touch_dynamic_rotation(uint16_t angle)
{
return g_touch_angle;
}
static uint8_t aic_get_touch_dynamic_rotation_enabled(uint8_t flag)
{
return g_touch_dynamic_enable;
}
static void aic_set_osd_rotation_flag(uint8_t flag)
{
g_osd_enable = flag;
}
static uint8_t aic_get_osd_rotation_flag(uint8_t flag)
{
return g_osd_enable;
}
/* ISR for touch interrupt */
void rt_hw_touch_isr(rt_touch_t touch)
{
RT_ASSERT(touch);
if (touch->parent.rx_indicate == RT_NULL)
{
return;
}
if (touch->irq_handle != RT_NULL)
{
touch->irq_handle(touch);
}
touch->parent.rx_indicate(&touch->parent, 1);
}
#ifdef RT_TOUCH_PIN_IRQ
static void touch_irq_callback(void *param)
{
rt_hw_touch_isr((rt_touch_t)param);
}
#endif
/* touch interrupt initialization function */
static rt_err_t rt_touch_irq_init(rt_touch_t touch)
{
#ifdef RT_TOUCH_PIN_IRQ
if (touch->config.irq_pin.pin == RT_PIN_NONE)
{
return -RT_EINVAL;
}
rt_pin_mode(touch->config.irq_pin.pin, touch->config.irq_pin.mode);
if (touch->config.irq_pin.mode == PIN_MODE_INPUT_PULLDOWN)
{
rt_pin_attach_irq(touch->config.irq_pin.pin, PIN_IRQ_MODE_RISING, touch_irq_callback, (void *)touch);
}
else if (touch->config.irq_pin.mode == PIN_MODE_INPUT_PULLUP)
{
rt_pin_attach_irq(touch->config.irq_pin.pin, PIN_IRQ_MODE_FALLING, touch_irq_callback, (void *)touch);
}
else if (touch->config.irq_pin.mode == PIN_MODE_INPUT)
{
rt_pin_attach_irq(touch->config.irq_pin.pin, PIN_IRQ_MODE_RISING_FALLING, touch_irq_callback, (void *)touch);
}
rt_pin_irq_enable(touch->config.irq_pin.pin, PIN_IRQ_ENABLE);
#endif
return RT_EOK;
}
/* touch interrupt enable */
static void rt_touch_irq_enable(rt_touch_t touch)
{
#ifdef RT_TOUCH_PIN_IRQ
if (touch->config.irq_pin.pin != RT_PIN_NONE)
{
rt_pin_irq_enable(touch->config.irq_pin.pin, RT_TRUE);
}
#else
touch->ops->touch_control(touch, RT_TOUCH_CTRL_ENABLE_INT, RT_NULL);
#endif
}
/* touch interrupt disable */
static void rt_touch_irq_disable(rt_touch_t touch)
{
#ifdef RT_TOUCH_PIN_IRQ
if (touch->config.irq_pin.pin != RT_PIN_NONE)
{
rt_pin_irq_enable(touch->config.irq_pin.pin, RT_FALSE);
}
#else
touch->ops->touch_control(touch, RT_TOUCH_CTRL_DISABLE_INT, RT_NULL);
#endif
}
static rt_err_t rt_touch_open(rt_device_t dev, rt_uint16_t oflag)
{
rt_touch_t touch;
RT_ASSERT(dev != RT_NULL);
touch = (rt_touch_t)dev;
if (oflag & RT_DEVICE_FLAG_INT_RX && dev->flag & RT_DEVICE_FLAG_INT_RX)
{
/* Initialization touch interrupt */
rt_touch_irq_init(touch);
}
return RT_EOK;
}
static rt_err_t rt_touch_close(rt_device_t dev)
{
rt_touch_t touch;
RT_ASSERT(dev != RT_NULL);
touch = (rt_touch_t)dev;
/* touch disable interrupt */
rt_touch_irq_disable(touch);
return RT_EOK;
}
static rt_size_t rt_touch_read(rt_device_t dev, rt_off_t pos, void *buf, rt_size_t len)
{
rt_touch_t touch;
rt_size_t result = 0;
RT_ASSERT(dev != RT_NULL);
touch = (rt_touch_t)dev;
if (buf == NULL || len == 0)
{
return 0;
}
result = touch->ops->touch_readpoint(touch, buf, len);
return result;
}
static rt_err_t rt_touch_control(rt_device_t dev, int cmd, void *args)
{
rt_touch_t touch;
rt_err_t result = RT_EOK;
RT_ASSERT(dev != RT_NULL);
touch = (rt_touch_t)dev;
rt_uint16_t angle = 0;
rt_uint8_t flag = 0;
switch (cmd)
{
case RT_TOUCH_CTRL_SET_MODE:
result = touch->ops->touch_control(touch, RT_TOUCH_CTRL_SET_MODE, args);
if (result == RT_EOK)
{
rt_uint16_t mode;
mode = *(rt_uint16_t*)args;
if (mode == RT_DEVICE_FLAG_INT_RX)
{
rt_touch_irq_enable(touch); /* enable interrupt */
}
}
break;
case RT_TOUCH_CTRL_SET_X_RANGE:
result = touch->ops->touch_control(touch, RT_TOUCH_CTRL_SET_X_RANGE, args);
if (result == RT_EOK)
{
touch->info.range_x = *(rt_int32_t *)args;
LOG_D("set x coordinate range :%d\n", touch->info.range_x);
}
break;
case RT_TOUCH_CTRL_SET_Y_RANGE:
result = touch->ops->touch_control(touch, RT_TOUCH_CTRL_SET_Y_RANGE, args);
if (result == RT_EOK)
{
touch->info.range_y = *(rt_uint32_t *)args;
LOG_D("set y coordinate range :%d \n", touch->info.range_x);
}
break;
case RT_TOUCH_CTRL_SET_DYNAMIC_ROTATE:
if (args) {
angle = *(rt_uint16_t *)args;
aic_set_dynamic_touch_rotation(angle);
} else {
result = -RT_EINVAL;
}
break;
case RT_TOUCH_CTRL_GET_DYNAMIC_ROTATE:
if (args)
*(rt_uint16_t *)args = aic_get_touch_dynamic_rotation(angle);
else
result = -RT_EINVAL;
break;
case RT_TOUCH_CTRL_GET_DYNAMIC_FLAG:
if (args)
*(rt_uint8_t *)args = aic_get_touch_dynamic_rotation_enabled(flag);
else
result = -RT_EINVAL;
break;
case RT_TOUCH_CTRL_SET_OSD_FLAG:
if (args) {
flag = *(rt_uint8_t *)args;
aic_set_osd_rotation_flag(flag);
} else {
result = -RT_EINVAL;
}
break;
case RT_TOUCH_CTRL_GET_OSD_FLAG:
if (args)
*(rt_uint8_t *)args = aic_get_osd_rotation_flag(flag);
else
result = -RT_EINVAL;
break;
case RT_TOUCH_CTRL_DISABLE_INT:
rt_touch_irq_disable(touch);
break;
case RT_TOUCH_CTRL_ENABLE_INT:
rt_touch_irq_enable(touch);
break;
case RT_TOUCH_CTRL_GET_ID:
case RT_TOUCH_CTRL_GET_INFO:
default:
return touch->ops->touch_control(touch, cmd, args);
}
return result;
}
#ifdef RT_USING_DEVICE_OPS
const static struct rt_device_ops rt_touch_ops =
{
RT_NULL,
rt_touch_open,
rt_touch_close,
rt_touch_read,
RT_NULL,
rt_touch_control
};
#endif
#ifdef RT_USING_PM
static int aic_touch_suspend(const struct rt_device *device, rt_uint8_t mode)
{
rt_touch_t touch;
RT_ASSERT(device != RT_NULL);
touch = (rt_touch_t)device;
switch (mode)
{
case PM_SLEEP_MODE_IDLE:
break;
case PM_SLEEP_MODE_LIGHT:
case PM_SLEEP_MODE_DEEP:
case PM_SLEEP_MODE_STANDBY:
touch->ops->touch_control(touch, RT_TOUCH_CTRL_POWER_OFF, NULL);
break;
default:
break;
}
return 0;
}
static void aic_touch_resume(const struct rt_device *device, rt_uint8_t mode)
{
rt_touch_t touch;
RT_ASSERT(device != RT_NULL);
touch = (rt_touch_t)device;
switch (mode)
{
case PM_SLEEP_MODE_IDLE:
break;
case PM_SLEEP_MODE_LIGHT:
case PM_SLEEP_MODE_DEEP:
case PM_SLEEP_MODE_STANDBY:
touch->ops->touch_control(touch, RT_TOUCH_CTRL_POWER_ON, NULL);
break;
default:
break;
}
}
static struct rt_device_pm_ops aic_touch_pm_ops =
{
SET_DEVICE_PM_OPS(aic_touch_suspend, aic_touch_resume)
NULL,
};
#endif
/*
* touch register
*/
int rt_hw_touch_register(rt_touch_t touch,
const char *name,
rt_uint32_t flag,
void *data)
{
rt_int8_t result;
rt_device_t device;
RT_ASSERT(touch != RT_NULL);
device = &touch->parent;
#ifdef RT_USING_DEVICE_OPS
device->ops = &rt_touch_ops;
#else
device->init = RT_NULL;
device->open = rt_touch_open;
device->close = rt_touch_close;
device->read = rt_touch_read;
device->write = RT_NULL;
device->control = rt_touch_control;
#endif
device->type = RT_Device_Class_Touch;
device->rx_indicate = RT_NULL;
device->tx_complete = RT_NULL;
device->user_data = data;
result = rt_device_register(device, name, flag | RT_DEVICE_FLAG_STANDALONE);
if (result != RT_EOK)
{
LOG_E("rt_touch register err code: %d", result);
return result;
}
#ifdef RT_USING_PM
#ifdef AIC_TOUCH_PANEL_WAKE_UP
rt_pm_set_pin_wakeup_source(touch->config.irq_pin.pin);
rt_device_wakeup_enable(device, RT_TRUE);
#endif
rt_pm_device_register(device, &aic_touch_pm_ops);
#endif
LOG_I("rt_touch init success");
return RT_EOK;
}