STM32实战:SPI驱动ST7735 TFT屏的初始化与像素填充

Williams lee

1. 硬件连接与引脚定义

第一次接触ST7735驱动的TFT屏时,最让人头疼的就是那一堆密密麻麻的引脚。我刚开始用STM32F103驱动1.44寸屏时,光是接线就折腾了半天。这里分享下我的实际接线经验:

ST7735通常有8个关键引脚需要连接,我用的是常见的7线SPI接法。背光控制(BLK) 这个引脚容易被忽略,其实直接接3.3V就能常亮,如果想省电可以接GPIO控制。RS引脚(也叫DC)特别重要,它决定发送的是命令还是数据,我习惯接在PC6,这个后面代码会体现。

最关键的SPI三兄弟:SCLK接PB13(时钟线)、MOSI接PB15(数据输出)、CS接PB12(片选)。注意STM32的SPI引脚是固定的,不能随便接。有次我把MOSI接到PA7,死活不工作,查了半天手册才发现SPI1和SPI2的引脚位置不同。

复位引脚(RST)建议接GPIO,不要直接接VCC。我遇到过屏幕初始化不稳定的情况,后来改成程序控制复位就解决了。具体接线可以参考这个表格:

屏幕引脚 STM32连接 作用说明
BLK 3.3V 背光控制
RS(DC) PC6 命令/数据切换
RST PC7 硬件复位
CS PB12 片选信号
SCL PB13 SPI时钟
SDA PB15 SPI数据输出
GND GND 地线
VCC 3.3V 电源

实际项目中,我建议先用杜邦线测试,稳定后再焊接。遇到过接触不良导致的花屏问题,用万用表量了半天才发现是CS引脚虚接。

2. SPI配置与初始化陷阱

配置SPI接口时,ST7735有几个参数特别关键。第一次使用时我直接复制了网上例程,结果屏幕显示全是乱码,后来发现是SPI模式没设对。CPOL=1CPHA=1 这个组合(模式3)才是ST7735的正确打开方式。

这里有个坑:不同厂家的ST7735模块可能要求不同。我手上有两个不同批次的屏幕,一个用模式3工作正常,另一个却要用模式0。建议如果遇到显示异常,可以先尝试切换SPI模式:

c复制SPI_InitTypeDef SPI_InitStructure;
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;  // 时钟极性
SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge; // 时钟相位
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;    // 软件控制片选
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_2; // 18MHz
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
SPI_Init(SPI2, &SPI_InitStructure);

时钟分频我一般先用SPI_BaudRatePrescaler_2(18MHz),如果出现数据错误再降低。曾经在长线连接时遇到过干扰,降到SPI_BaudRatePrescaler_8才稳定。

特别注意软件NSS模式,硬件NSS我从来没成功过。还有数据位顺序必须是MSB先行,这个在ST7735手册里有明确要求。

3. 解密ST7735初始化序列

第一次看到ST7735的初始化代码时,我被那一长串命令吓到了。后来发现其实可以分为几个关键部分:

电源配置(0xC0-0xC5)这部分设置内部电压参数,通常直接用厂家推荐值就行。有个技巧:如果屏幕亮度异常,可以调整0xC5命令的VCOM值(我一般用0x0E)。

帧率控制(0xB1-0xB3)这三个命令控制刷新率,参数组合影响屏幕响应速度。普通应用直接用0x01,0x2C,0x2D这个组合就行,追求流畅度可以尝试修改第一个参数。

最麻烦的是Gamma校正(0xE0,0xE1),这两个命令各有16个参数。我对比过不同厂家的初始化代码,发现Gamma值差异很大。后来在ST官方论坛找到一组通用参数:

c复制// 正极性Gamma校正
writeCmdData(0xe0, LCMD);
writeCmdData(0x0f, LDAT); writeCmdData(0x1a, LDAT);
writeCmdData(0x0f, LDAT); writeCmdData(0x18, LDAT);
...
// 负极性Gamma校正  
writeCmdData(0xe1, LCMD);
writeCmdData(0x0f, LDAT); writeCmdData(0x1b, LDAT);
writeCmdData(0x0f, LDAT); writeCmdData(0x17, LDAT);
...

实际使用中,如果发现颜色显示不准,优先检查这部分。我曾经因为漏写一个参数导致红色显示成橙色。

4. 解决屏幕偏移的实战技巧

ST7735手册说支持132x132分辨率,但我们用的屏幕是128x128,这就导致了经典的偏移问题。现象就是绘图时内容显示不全,或者出现错位。

经过多次测试,我发现最稳定的解决方案是在设置坐标时X轴加2,Y轴加3

c复制void setPos(int sx, int ex, int sy, int ey) {
    // X轴
    writeCmdData(0x2a, LCMD);
    writeCmdData(0x00, LDAT);
    writeCmdData(sx+2, LDAT);  // 关键偏移量
    writeCmdData(0x00, LDAT);
    writeCmdData(ex+2, LDAT);
    
    // Y轴
    writeCmdData(0x2b, LCMD);
    writeCmdData(0x00, LDAT);
    writeCmdData(sy+3, LDAT);  // 关键偏移量
    writeCmdData(0x00, LDAT);
    writeCmdData(ey+3, LDAT);
    
    writeCmdData(0x2c, LCMD); // 开始写入数据
}

这个偏移量不是固定的,不同批次屏幕可能需要微调。我开发过一个自动校准程序:在四个角显示标记点,通过按键调整偏移量直到显示正确。

还有个隐藏问题:内存对齐。STM32的SPI对非对齐访问很敏感,特别是在使用DMA时。建议像素数据按4字节对齐:

c复制__attribute__((aligned(4))) uint16_t frameBuffer[128*128];

5. 像素填充优化之道

最基本的像素填充函数writeAnColor虽然简单,但效率很低。通过示波器测量,发现纯色填充128x128屏幕要花费23ms。经过优化,我总结出几个提速技巧:

批量写入:ST7735支持连续写入,不必每次设置位置。修改后的函数先设置全屏区域,然后循环发送数据:

c复制void fillScreen(uint16_t color) {
    setPos(0, 127, 0, 127);
    uint8_t hi = color >> 8, lo = color & 0xFF;
    LCS_CLR();
    for(uint32_t i=0; i<16384; i++) {
        while(!(SPI2->SR & SPI_I2S_FLAG_TXE));
        SPI2->DR = hi;
        while(!(SPI2->SR & SPI_I2S_FLAG_TXE));
        SPI2->DR = lo;
    }
    LCS_SET();
}

DMA传输:终极提速方案。需要预先构造缓冲区,然后通过DMA发送:

c复制void DMA_Fill(uint16_t color) {
    static uint16_t dmaBuf[128]; // 行缓冲区
    for(int i=0; i<128; i++) dmaBuf[i] = color;
    
    setPos(0, 127, 0, 127);
    DMA1_Channel5->CCR &= ~DMA_CCR_EN;
    DMA1_Channel5->CNDTR = 128*128;
    DMA1_Channel5->CMAR = (uint32_t)dmaBuf;
    DMA1_Channel5->CCR |= DMA_CCR_EN;
}

实测DMA方式仅需2.8ms就能完成全屏填充,比原始方法快8倍。但要注意缓冲区对齐和SPI FIFO设置。

6. 颜色处理的那些坑

ST7735使用RGB565格式,这和常见的RGB888不同。刚开始我直接用24位颜色值,结果显示异常。正确的颜色转换应该是:

c复制#define RGB888_TO_RGB565(r,g,b) (((r>>3)<<11)|((g>>2)<<5)|(b>>3))

实际使用中发现,不同屏幕对颜色的解析有差异。我手头的两个屏幕,一个显示红色为0xF800,另一个却要0xF810才准确。建议重要项目先做颜色校准。

还有个常见问题:颜色反相。这是因为有些模块出厂设置了反相显示,可以通过0x21命令切换:

c复制writeCmdData(0x21, LCMD); // 开启反相
writeCmdData(0x20, LCMD); // 关闭反相

7. 高级技巧:局部刷新与动画优化

做UI动画时,全屏刷新会导致闪烁。通过局部刷新可以大幅提升视觉效果:

c复制void updateArea(int x1, int y1, int x2, int y2, uint16_t* data) {
    setPos(x1, x2, y1, y2);
    LCS_CLR();
    for(int i=0; i<(x2-x1+1)*(y2-y1+1); i++) {
        write2Byte(data[i]);
    }
    LCS_SET();
}

配合双缓冲技术,可以完全消除闪烁。我的做法是:

  1. 在内存中维护两个128x128的缓冲区
  2. 所有绘图操作在后台缓冲区进行
  3. 完成一帧后,通过DMA快速切换缓冲区

对于简单动画,还可以用脏矩形算法,只更新发生变化的部分区域。

8. 常见问题排查指南

花屏问题:先检查电源是否稳定。我用示波器抓取到3.3V上有100mV的纹波,加了100μF电容后问题消失。其次检查SPI时钟相位,这是最常见的问题源。

显示偏移:除了前面说的+2/+3偏移量,还要注意屏幕的扫描方向。通过0x36命令可以控制显示方向:

c复制writeCmdData(0x36, LCMD);
writeCmdData(0xC8, LDAT); // 竖屏模式

触摸不灵敏:虽然和ST7735无关,但常出现在带触摸的模块上。建议检查触摸屏的采样率和滤波参数。

最后分享一个调试技巧:用逻辑分析仪抓取SPI波形。我通过这种方式发现CS信号有时会提前拉高,导致数据传输不完整。解决方法是在写操作间增加微小延时。

内容推荐

HDR+算法实战:从论文到代码的降噪效果实现(附避坑指南)
本文深入解析HDR+算法在移动摄影中的降噪效果实现,从论文原理到代码落地,详细介绍了多尺度对齐、残差融合等关键技术。特别提供工程实现中的避坑指南,包括内存优化、常见问题解决方案和调试建议,帮助开发者高效实现专业级图像处理效果。
从集合关系到数据结构:偏序、格与Hasse图实战解析
本文深入解析了偏序关系、格与Hasse图的核心概念及其在数据结构中的应用。通过生活实例和Python代码示例,展示了如何将抽象的数学理论转化为实际编程实践,包括偏序关系的验证、Hasse图的绘制以及格结构的判断。特别探讨了这些理论在依赖管理和任务调度等计算机科学领域的实际应用,帮助开发者更好地理解和运用这些高级数据结构概念。
别再死记硬背动态规划了!从‘找茬游戏’到LCS,带你用Python图解算法本质
本文通过‘大家来找茬’游戏生动解析动态规划算法,重点讲解最长公共子序列(LCS)问题的Python实现。从游戏化思维出发,详细展示如何构建决策矩阵、编写状态转移方程,并给出空间优化技巧和编辑距离等实际应用场景,帮助读者直观理解动态规划的核心思想。
BUUCTF:[CISCN2019 华东南赛区]Double Secret 深度解析:RC4加密与SSTI注入的攻防实战
本文深度解析了BUUCTF竞赛中[CISCN2019 华东南赛区]Double Secret题目的攻防实战,重点探讨了RC4加密与SSTI注入的结合利用。通过逆向分析RC4加密流程、构造SSTI注入payload,并组装完整攻击链,最终成功获取flag。文章还分享了实战调试技巧和防御建议,为CTF选手和网络安全爱好者提供了宝贵经验。
链路聚合模式对比:LACP与手动负载均衡在实际网络中的性能差异与选择建议
本文深入对比了链路聚合技术中LACP与手动负载均衡的性能差异,通过实测数据展示了它们在吞吐量、延迟和容错方面的表现。针对企业网络中的不同场景,提供了详细的配置建议和选择指南,帮助网络工程师优化带宽利用和提升网络可靠性。
告别实车路试:用AVL CRUISE M和dSPACE搭建HiL台架,5步搞定ECU极限测试
本文详细介绍了如何利用AVL CRUISE M和dSPACE构建硬件在环(HiL)测试台架,通过5个关键步骤实现ECU极限测试。从仿真模型转换到实时环境配置,再到信号映射和极限测试设计,最后实现自动化测试体系,大幅提升测试效率和覆盖率。这种方案不仅能够模拟极端工况,还能显著降低实车测试成本。
电池供电产品必看:TVS管选型避坑指南(附5V电路实测数据对比)
本文深入解析电池供电产品中TVS管选型的关键要点,特别针对5V电路的漏电流问题提供实测数据对比。通过分析齐纳击穿与雪崩击穿的差异,揭示低压TVS管的三大漏电陷阱,并给出产线可量化的测试方案和选型决策树,帮助工程师优化设计,延长电池寿命。
Native逆向实战(一)——BiliBili Sign算法还原与Frida联动分析
本文详细介绍了BiliBili Sign算法的逆向分析过程,包括Java层定位、Native层动态分析以及算法还原。通过Frida框架和IDA Pro等工具,逐步解析签名生成逻辑,并分享逆向工程中的实用技巧与经验。
从CTF到运维:MySQL HANDLER命令的‘骚操作’实战指南
本文深入探讨了MySQL HANDLER命令在CTF竞赛和运维场景中的实战应用。HANDLER命令作为MySQL特有的功能,能够绕过常规查询限制,提供低开销、逐行访问表数据的能力,适用于安全竞赛中的非预期解和生产环境中的应急处理。文章还详细解析了HANDLER的安全风险与防御措施,帮助开发者高效利用这一强大工具。
CAD - 揭秘 *.dwl 与 *.dwl2:文件锁定的幕后机制与协同设计应用
本文深入解析了CAD设计中的*.dwl与*.dwl2文件锁定机制,揭示了它们在团队协同设计中的关键作用。通过详细的技术原理和实际应用案例,帮助工程师理解如何有效管理这些锁定文件,避免图纸冲突和数据损坏,提升团队协作效率。
【MySQL OCP】从零到一:我的5.7版本通关实战与避坑指南
本文详细分享了MySQL 5.7 OCP认证的备考实战经验与避坑指南。从版本选择、备考资料筛选到考场技巧,全面解析如何高效通过这一含金量高的数据库认证。特别提醒注意考试中的题目陷阱和时间分配,以及考后证书下载的完整流程,助力开发者顺利拿下MySQL OCP证书。
从Apollo源码到独立模块:我是如何把Lattice Planner从CyberRT里‘抠’出来跑在实车上的
本文详细介绍了如何将Apollo平台中的Lattice Planner从CyberRT框架中解耦并部署到实车系统的全流程实战经验。通过数据结构重构、核心算法提取和性能优化,实现了资源占用降低62%的轻量化方案,适合希望复用成熟算法但受限于原有框架的工程团队。
告别版本混乱:pyenv-win在Windows上构建Python多版本开发环境的实战指南
本文详细介绍了如何使用pyenv-win在Windows上管理多版本Python开发环境,解决版本冲突问题。从安装配置到实战技巧,包括镜像加速、项目级版本控制和IDE集成,帮助开发者高效构建隔离的Python工作环境。特别适合需要同时维护多个Python项目的Windows用户。
从零到一:在Kali Linux上利用Docker容器化部署Vulfocus漏洞靶场
本文详细介绍了在Kali Linux上利用Docker容器化部署Vulfocus漏洞靶场的完整流程。从环境准备、镜像拉取到容器配置和运维管理,逐步指导安全人员高效搭建本地漏洞测试环境。通过Docker容器化部署,可节省60%以上磁盘空间,实现漏洞环境的快速切换与隔离测试,是网络安全学习和工具验证的理想解决方案。
告别Mac依赖!Windows电脑也能搞定uni-app云打包成iOS应用(附爱思助手安装指南)
本文详细解析了在Windows环境下使用uni-app进行云打包生成iOS应用的全流程,包括证书准备、描述文件生成、HBuilderX云打包配置以及通过爱思助手实现真机安装。特别适合没有Mac设备的开发者,帮助其高效完成跨平台应用开发,实现ipa文件的生成与测试。
Unity3D WebGL项目发布与IIS部署实战指南
本文详细介绍了Unity3D WebGL项目发布与IIS部署的完整流程,包括关键配置、常见错误排查和局域网访问优化技巧。通过实战经验分享,帮助开发者快速解决部署中的权限、MIME类型和内存分配等问题,提升WebGL项目在IIS上的运行效率。
Android 系统字体家族:从 sans-serif 到 monospace 的样式解析与应用
本文深入解析Android系统内置的13种字体家族,包括sans-serif、serif和monospace等字体样式,及其在移动开发中的实际应用场景。通过具体代码示例和设计建议,帮助开发者合理选择字体,提升UI设计的专业性和可读性。
Cherry Studio 1.6.4升级实战:300+AI助手配置与WebDAV文件管理全攻略
本文详细解析了Cherry Studio 1.6.4版本的升级实战,重点介绍了300+预配置AI助手的分类与应用,以及WebDAV文件管理系统的进阶技巧。通过环境准备、助手配置、工作流优化和性能调优的全方位指南,帮助用户高效利用这一生产力工具,提升数字化工作效率。
LaTeX算法排版避坑指南:从Undefined control sequence到完美排版
本文详细解析了LaTeX算法排版中常见的'Undefined control sequence'报错问题,提供了从宏包缺失到期刊格式适配的全面解决方案。通过典型错误速查表、正确环境配置及高级排错技巧,帮助科研人员快速实现算法完美排版,特别适合Elsevier、Springer等期刊投稿需求。
从MATLAB实践到视觉直觉:揭秘图像傅里叶变换与频率中心化的必要性
本文深入探讨了图像傅里叶变换在MATLAB中的实践应用,重点解析了频率中心化(fftshift)的必要性及其在图像处理中的关键作用。通过实际代码示例,展示了如何将低频分量移至频谱中心,便于设计滤波器和分析图像信息分布,从而提升视觉直觉和操作效率。
已经到底了哦
精选内容
热门内容
最新内容
云服务器硬盘消失?三步搞定Windows Server 2019磁盘初始化(附GPT分区选择指南)
本文详细介绍了在Windows Server 2019云服务器中解决硬盘不可见问题的三步操作指南,包括磁盘初始化、分区选择(GPT或MBR)及格式化。特别针对GPT分区的优势进行了分析,帮助用户根据需求选择合适的分区方案,确保数据存储的高效与安全。
微机原理避坑指南:SRAM、DRAM、Flash,三大存储器接口设计到底有啥不同?
本文深入解析SRAM、DRAM和Flash三大存储器在微机原理中的接口设计差异,提供ARM Cortex-M平台下的硬件连接方案和软件调试技巧。从地址线处理、时序参数匹配到PCB信号完整性设计,全面剖析常见设计陷阱,帮助工程师规避存储器接口开发中的典型错误,提升嵌入式系统稳定性。
别再只盯着PRI和UNI了!MySQL里这个‘MUL’标记,才是外键和一对多关系的幕后功臣
本文深入解析MySQL中的MUL标记,揭示其作为外键和一对多关系实现的核心机制。通过对比PRI、UNI和MUL的特性,详细阐述MUL在数据模型构建、完整性维护及查询优化中的关键作用,并提供实战分析和性能优化建议,帮助开发者更好地理解和应用这一重要数据库特性。
华为鲲鹏/飞腾ARM服务器上,手把手解决Kettle ETL部署的4个典型报错
本文详细解析了在华为鲲鹏/飞腾ARM服务器上部署Kettle ETL时常见的4个典型报错及其解决方案。从平台兼容性报错、SWT组件加载失败到GTK库缺失和跨平台路径问题,提供了从诊断到修复的完整指南,帮助工程师高效完成国产化替代环境下的ETL部署工作。
不只是安装:用GEM5在Ubuntu 22.04上跑通你的第一个CPU模拟(从Hello World到自定义脚本)
本文详细介绍了如何在Ubuntu 22.04系统上使用GEM5进行CPU模拟,从基础的Hello World程序验证到自定义脚本编写。通过分步指导和实用示例,帮助读者掌握GEM5的配置、运行和结果分析技巧,特别适合计算机体系结构研究者和开发者。
Bench2Drive:解锁端到端自动驾驶闭环评估的44种交互场景挑战
本文深入解析Bench2Drive平台如何通过44种交互场景实现端到端自动驾驶闭环评估。该平台采用短路程专项测试设计,精准定位算法弱点,覆盖紧急制动、无保护左转等高频高危场景,并引入效率分数和舒适度分数等创新指标。相比传统开环测试,Bench2Drive的闭环特性更能模拟真实驾驶中的因果链和蝴蝶效应,为开发者提供可解释、可复现的评估结果。
RoboMaster电控新手避坑:用STM32CubeMX配置大疆C板CAN总线驱动GM6020电机(附完整代码)
本文详细介绍了如何使用STM32CubeMX配置大疆C型开发板的CAN总线驱动GM6020电机,涵盖硬件准备、CubeMX关键配置、代码实现及常见问题排查。特别针对RoboMaster电控新手常见的CAN总线配置错误和筛选器初始化问题提供了解决方案,并附完整代码示例,帮助开发者快速掌握电机驱动技术。
FastAdmin前后端分离项目单点登录实战:一个关键文件的改造
本文详细介绍了如何改造FastAdmin的单点登录机制,特别针对前后端分离项目中的Token管理问题。通过修改Auth.php文件,添加Token清除逻辑,确保同一账号只能在一个设备登录,提升系统安全性。文章还提供了完整的对接方案和性能优化建议,帮助开发者高效实现单点登录功能。
从零构建:基于STC89c51与ESP8266的物联网环境监测系统实战
本文详细介绍了如何从零构建基于STC89c51与ESP8266的物联网环境监测系统,涵盖硬件选型、电路设计、软件开发、数据采集与上传等关键步骤。通过DHT11温湿度传感器和MQ-135气体传感器实现环境数据采集,并利用ESP8266模块实现数据上传至云端,最终通过手机APP展示实时数据。项目成本低、上手快,适合物联网初学者。
Rancher V2.9.0 Docker离线安装与集群配置实战
本文详细介绍了Rancher V2.9.0在Docker离线环境下的安装与集群配置实战,包括离线镜像包准备、关键容器启动配置、私有仓库深度优化以及集群网络调优等关键步骤。特别针对企业内网环境中的常见问题提供了解决方案,帮助用户高效完成Rancher部署与集群管理。