CST816S触摸驱动

本文针对淘宝所购买的1.69寸LCD触摸屏驱动进行讲解。

屏幕购买链接如下:

1.69寸全视角IPSTFT彩屏 液晶显示屏SPI位接口ST7789V驱动工业屏-淘宝网 (taobao.com) 

我们先搞一下IIC驱动

	#define CST816_SCL_Clr() 	HAL_GPIO_WritePin(TP_SCL_GPIO_Port,TP_SCL_Pin, GPIO_PIN_RESET)//SCL=SCLK
	#define CST816_SCL_Set() 	HAL_GPIO_WritePin(TP_SCL_GPIO_Port,TP_SCL_Pin, GPIO_PIN_SET)

	#define CST816_SDA_IN() 	GPIO_InitStruct.Pin = TP_SDA_Pin;GPIO_InitStruct.Mode = GPIO_MODE_INPUT;GPIO_InitStruct.Pull = GPIO_PULLUP;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;HAL_GPIO_Init(TP_SDA_GPIO_Port, &GPIO_InitStruct)
	#define CST816_SDA_Get()	HAL_GPIO_ReadPin(TP_SDA_GPIO_Port,TP_SDA_Pin)

	#define CST816_SDA_OUT()	GPIO_InitStruct.Pin = TP_SDA_Pin;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Pull = GPIO_PULLUP;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;HAL_GPIO_Init(TP_SDA_GPIO_Port, &GPIO_InitStruct)
	#define CST816_SDA_Clr() 	HAL_GPIO_WritePin(TP_SDA_GPIO_Port,TP_SDA_Pin, GPIO_PIN_RESET)
	#define CST816_SDA_Set() 	HAL_GPIO_WritePin(TP_SDA_GPIO_Port,TP_SDA_Pin, GPIO_PIN_SET)

由于我所用到的是HAL库,所以这里针对IIC驱动做了宏定义。

用标准库的开发者可以根据需要进行改动。

//模拟IIC部分的函数声明
void CST816_IIC_ACK(void);
void CST816_IIC_NACK(void);
unsigned char CST816_IIC_Wait_ACK(void);
void CST816_IIC_Start(void);
void CST816_IIC_Stop(void);
void CST816_IIC_SendByte(unsigned char byte);
void CST816_IIC_WriteREG(unsigned char reg,unsigned char date);
unsigned char CST816_IIC_ReadREG(unsigned char reg);
unsigned char CST816_IIC_RecvByte(void);

我们IIC的目的是对CST816的寄存器进行读写,所以根据手册内容,我们做出下面的寄存器声明

//CST816寄存器
#define GestureID			0x01		//手势寄存器
#define FingerNum			0x02		//手指数量
#define XposH				0x03		//x高四位
#define XposL				0x04		//x低八位
#define YposH				0x05		//y高四位
#define YposL				0x06		//y低八位
#define ChipID				0xA7		//芯片型号
#define	MotionMask		    0xEC		//触发动作
#define AutoSleepTime	    0xF9		//自动休眠
#define IrqCrl				0xFA		//中断控制
#define AutoReset			0xFB		//无手势休眠
#define LongPressTime	    0xFC		//长按休眠
#define DisAutoSleep	    0xFE		//使能低功耗模式

多数寄存器我们是不需要去打理的,我们最关心的是XposH、XposL、YposH、YposL用于读取手指触摸位置的坐标,以及FingerNum用于读取触摸状态

typedef struct
{
	unsigned char chipID;
	unsigned int X_Pos;			//X坐标
	unsigned int Y_Pos;			//Y坐标
	unsigned char Sta;			//记录触摸状态	
}CST816_Info;

接下来我们创建触摸屏实例,用于触摸屏的信息管理,内容包括芯片ID,X\Y坐标,以及触摸状态等。

下面是函数实现

//IIC驱动实现
#include "CST816.h"
#include "math.h"


CST816_Info	CST816_Instance;			//创建CST816实例。

#define delay_num 4        //宏定义延时时间



GPIO_InitTypeDef GPIO_InitStruct = {0};

//IIC起始信号
void CST816_IIC_Start(void)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};
	
  CST816_SDA_OUT();
	
  CST816_SDA_Set();
  CST816_SCL_Set();
  delay_us(delay_num);	
  CST816_SDA_Clr();
  delay_us(delay_num); 
  CST816_SCL_Clr();
}

//IIC停止信号
void CST816_IIC_Stop(void)
{
	GPIO_InitTypeDef GPIO_InitStruct = {0};
	CST816_SDA_OUT();
	
	CST816_SCL_Clr(); 
	CST816_SDA_Clr();  
	delay_us(delay_num);	
	CST816_SCL_Set();
	CST816_SDA_Set();
	delay_us(delay_num);
}

//发送ACK应答
void CST816_IIC_ACK(void)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};
  CST816_SDA_OUT();
	
  CST816_SCL_Clr();
  CST816_SDA_Clr();
  delay_us(delay_num);     
  CST816_SCL_Set();
  delay_us(delay_num);                  
  CST816_SCL_Clr();   
}

//发送NACK不应答
void CST816_IIC_NACK(void)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};
  CST816_SDA_OUT();
	
  CST816_SCL_Clr();
  CST816_SDA_Set();
  delay_us(delay_num);     
  CST816_SCL_Set();
  delay_us(delay_num);                  
  CST816_SCL_Clr(); 
}

//等待IIC应答信号
unsigned char CST816_IIC_Wait_ACK(void)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};
  unsigned char t = 0;

	CST816_SDA_IN();
	CST816_SDA_Set();
	delay_us(delay_num); 
	CST816_SCL_Set();
	delay_us(delay_num); 
	while(CST816_SDA_Get())
	{
		t++;
		if(t>250)
		{
			CST816_IIC_Stop();
			return 1;
		}
	}
  CST816_SCL_Clr();              
  return 0;	
}


//IIC发送一字节数据
void CST816_IIC_SendByte(unsigned char byte)
{
	GPIO_InitTypeDef GPIO_InitStruct = {0};
	unsigned char  BitCnt;
	CST816_SDA_OUT();
	CST816_SCL_Clr();
	for(BitCnt=0;BitCnt<8;BitCnt++)//要传送的数据长度为8位
	{
		if(byte&0x80) CST816_SDA_Set();//判断发送位
		else CST816_SDA_Clr(); 
		byte<<=1;
		delay_us(delay_num); 
		CST816_SCL_Set();
		delay_us(delay_num);
		CST816_SCL_Clr();
		delay_us(delay_num);
	}
}

//IIC接收一字节数据
unsigned char CST816_IIC_RecvByte(void)
{
  GPIO_InitTypeDef GPIO_InitStruct = {0};
  unsigned char retc;
  unsigned char BitCnt;
  retc=0; 
  CST816_SDA_IN();//置数据线为输入方式                
  for(BitCnt=0;BitCnt<8;BitCnt++)
  {        
		CST816_SCL_Clr();
		delay_us(delay_num);
		CST816_SCL_Set();//置时钟线为高使数据线上数据有效                
		retc=retc<<1;
		if(CST816_SDA_Get()) retc++;//读数据位,接收的数据位放入retc中 
		delay_us(delay_num);
  }
  return(retc);
}

//IIC写入指定寄存器数据
void CST816_IIC_WriteREG(unsigned char reg,unsigned char date)
{
	CST816_IIC_Start();
	CST816_IIC_SendByte(0x2A);
	CST816_IIC_Wait_ACK();
	CST816_IIC_SendByte(reg);
	CST816_IIC_Wait_ACK();
	CST816_IIC_SendByte(date);
	CST816_IIC_Wait_ACK();
	CST816_IIC_Stop();
	HAL_Delay(10);
}

//IIC读取指定寄存器数据
unsigned char CST816_IIC_ReadREG(unsigned char reg)
{
	unsigned char date;
	CST816_IIC_Start();
	CST816_IIC_SendByte(0x2A);
	CST816_IIC_Wait_ACK();
	CST816_IIC_SendByte(reg);
	CST816_IIC_Wait_ACK();
	CST816_IIC_Start();
//	delay_us(4);
	CST816_IIC_SendByte(0x2B);
	CST816_IIC_Wait_ACK();
	date=CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	CST816_IIC_Stop();
//	delay_us(5);
	return date;
}

 IIC协议部分不展开介绍,我们来看有关CST816寄存器的读取

void CST816_Init(void)
{
    //我自己不需要做任何操作,由于习惯就写了Init函数。
    //可根据自己需求进行修改
}
unsigned char CST816_Get_ChipID()
{
	return CST816_IIC_ReadREG(ChipID);    //读取芯片ID
}


void CST816_Get_XY()
{

    //这个函数非常值得注意,为什么我没有使用上面提到的直接读取指定寄存器的函数,
    //因为这个CST816芯片的原因,当读取过XY坐标四个寄存器之中的任意一个之后,其他寄存器会被清空,得到的数据不准确。
    //因此必须连续读取,这样也可以学习下IIC总线的读取逻辑嘛对不对。

	unsigned char temp[4];
	unsigned int x,y;
	
	CST816_IIC_Start();
	CST816_IIC_SendByte(0x2A);
	CST816_IIC_Wait_ACK();
	CST816_IIC_SendByte(0x03);
	CST816_IIC_Wait_ACK();
	CST816_IIC_Start();
//	delay_us(4);
	CST816_IIC_SendByte(0x2B);
	CST816_IIC_Wait_ACK();
	temp[0]=CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	temp[1]=CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	temp[2]=CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	temp[3]=CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	CST816_IIC_Stop();
	
	x=(unsigned int)((temp[0]&0x0F)<<8)|temp[1];//(temp[0]&0X0F)<<4|
	y=(unsigned int)((temp[2]&0x0F)<<8)|temp[3];//(temp[2]&0X0F)<<4|
	if(x<240&&y<280)
	{
		CST816_Instance.X_Pos	=	x;
		CST816_Instance.Y_Pos	=	y;
	}
}



unsigned char CST816_Get_Sta()
{
	unsigned char sta;
	
	CST816_IIC_Start();
	CST816_IIC_SendByte(0x2A);
	CST816_IIC_Wait_ACK();
	CST816_IIC_SendByte(0x01);
	CST816_IIC_Wait_ACK();
	CST816_IIC_Start();
//	delay_us(4);
	CST816_IIC_SendByte(0x2B);
	CST816_IIC_Wait_ACK();
	CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	sta=CST816_IIC_RecvByte();
	CST816_IIC_ACK();
	CST816_IIC_Stop();
	if(sta!=255&sta!=0)return 1;
	else return 0;     
}

剩下的寄存器用到的情况很少,如果有需要可以自己去定义一个函数。

类似这样,仅做展示,作者并没有测试是否可行,就不帖出来喽~

unsigned char CST816_Get_Sta(void);			//更新触摸状态
void CST816_Set_MotionMask(unsigned char Motion);
void CST816_Set_IrqCtrl(unsigned char IRQCtrl);
void CST816_Set_AutoRST(unsigned char time);				//设置多长时间无触摸自动复位,默认5s,写0则禁用功能
void CST816_Set_LongPressRST(unsigned char time);		//设置长按多长时间复位,默认为10s,写0禁用功能
void CST816_Set_AutoSleep(unsigned char en);				//是否使能自动进入低功耗模式

unsigned char CST816_Get_Gesture(void);

 

### 关于CST816T触摸驱动芯片的信息 CST816系列是一款广泛应用于电容式触摸屏的驱动芯片,具有高灵敏度、低功耗以及支持多点触控的特点。对于具体型号CST816T的相关资料获取,可以从以下几个方面入手: #### 数据手册下载 目前公开可用的数据手册资源主要集中在开源文档库中。例如,在GitCode平台上有一个专门针对CST816D的数据手册仓库[^1],虽然其名称标注为CST816D,但由于CST816系列内部架构相似,该手册可以作为初步参考资料。如果需要更精确匹配CST816T的手册,则建议进一步查找特定版本。 - **项目地址**: [https://gitcode.com/Open-source-documentation-tutorial/cd171](https://gitcode.com/Open-source-documentation-tutorial/cd171) 此外,还有关于CST816S寄存器配置的详细说明文档可供参考[^3],尽管这是另一个子型号,但它提供了深入理解整个家族产品工作原理的重要线索。 #### 开发环境集成实例 在实际应用案例中,已有开发者成功实现了基于LVGL图形库框架下对CST816T的支持[^2]。此实现过程涉及到了如何适配具体的硬件接口标准(I²C/SPI),并调整坐标映射逻辑来满足不同显示屏尺寸需求。通过研究这些现有方案能够加速新项目的开发进度。 特别值得注意的是,在ESP-IDF环境下移植显示与触摸功能时,通常会将相关驱动程序代码放置于指定目录结构之下以便管理维护方便[^4]。下面给出一段简化版初始化函数示例用于演示基本流程: ```c #include "driver/i2c.h" #define I2C_MASTER_SCL_IO 22 /*!< GPIO number used for I2C master clock */ #define I2C_MASTER_SDA_IO 21 /*!< GPIO number used for I2C data */ void cst816t_init() { i2c_config_t conf; memset(&conf, 0, sizeof(i2c_config_t)); conf.mode = I2C_MODE_MASTER; conf.sda_io_num = I2C_MASTER_SDA_IO; conf.scl_io_num = I2C_MASTER_SCL_IO; conf.master.clk_speed = 100000; // Set speed to 100kHz i2c_param_config(I2C_NUM_0, &conf); i2c_driver_install(I2C_NUM_0, conf.mode, 0, 0, 0); uint8_t buffer[2]; buffer[0] = 0x00; // Register address buffer[1] = 0xFF; // Value to write into register i2c_master_write_to_device(I2C_NUM_0, CST816T_ADDR, buffer, 2, pdMS_TO_TICKS(100)); } ``` 上述代码片段展示了如何利用Espressif SDK中的`i2c`模块完成向目标设备发送命令字节序列的操作。 ---
评论 44
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Distance_90

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值