一、FSMC外扩内存SRAM FSMC(Flexible Static Memory Controller,灵活的静态存储控制器)是STM32系列采用的一种新型的存储器扩展技术,使用FSMC外设来管理扩展的存储器。在外部存储器扩展方面具有独特的优势,可根据系统的应用需要,方便地进行不同类型大容量静态存储器的扩展。FSMC可以用于驱动包括SRAM、NOR FLASH以及NAND FLSAH类型的存储器,不能驱动如SDRAM这种动态的存储器。而在STM32F429系列的控制器中,它具有FMC外设,支持控制SDRAM存储器。 ) Y" Y0 w0 R; q$ z+ `" W5 G$ F
FSMC是STM32系列中内部集成256 KB以上FlaSh,后缀为xC、xD和xE的高存储密度微控制器特有的存储控制机制。之所以称为“可变”,是由于通过对特殊功能寄存器的设置,FSMC能够根据不同的外部存储器类型,发出相应的数据/地址/控制信号类型以匹配信号的速度,从而使得STM32系列微控制器不仅能够应用各种不同类型、不同速度的外部静态存储器,而且能够在不增加外部器件的情况下同时扩展多种不同类型的静态存储器,满足系统设计对存储容量、产品体积以及成本的综合要求。 1 B% M* s% U% h e
常用存储器介绍
" _4 }! Q/ p& Y, l; o
4 f5 |& [2 U) J
存储器的种类:存储器是计算机结构的重要组成部分。存储器是用来存储程序代码和数据的部件,有了存储器计算机才具有记忆功能。
8 Y1 y" M# i8 p. [4 o
5 [2 j5 Y$ J% ]1 t9 {
1.RAM存储器:RAM是“Random Access Memory”的缩写,被译为随机存储器。所谓“随机存取”,指的是当存储器中的消息被读取或写入时,所需要的时间与这段信息所在的位置无关。这个词的由来是因为早期计算机曾使用磁鼓作为存储器,磁鼓是顺序读写设备,而RAM可随读取其内部任意地址的数据,时间都是相同的,因此得名。 + w7 m6 \3 Q4 E, d5 Q k/ S
实际上现在RAM已经专门用于指代作为计算机内存的易失性半导体存储器。根据RAM的存储机制,又分为动态随机存储器DRAM(Dynamic RAM)以及静态随机存储器SRAM(Static RAM)两种。 % e. A. _. B6 Z% o9 F
8 C! ]+ b' U, k
2.DRAM存储器:动态随机存储器 DRAM的存储单元以电容的电荷来表示数据,有电荷代表 1,无电荷代表 0,,代表 1 的电容会放电,代表 0的电容会吸收电荷,因此它需要定期刷新操作,这就是“动态(Dynamic)”。 6 s+ X# m' L5 v8 q" ?
根据 DRAM的通讯方式,又分为同步和异步两种,这两种方式根据通讯时是否需要使用时钟信号来区分。由于使用时钟同步的通讯速度更快,所以同步 DRAM 使用更为广泛,这种DRAM 被称为 SDRAM(Synchronous DRAM)。常见的DRAM大多是SDRAM。
# M1 C2 |- Q4 n) J% F
& b! X' ^. S7 S8 j9 [5 E( X
为了进一步提高 SDRAM的通讯速度,人们设计了DDR SDRAM (Double DataRate SDRAM)存储器。DDR SDRAM在时钟的上升沿及下降沿各表示一个数据,也就是说在 1 个时钟周期内可以表示 2数据,在时钟频率同样的情况下,提高了一倍的速度。 ]$ J, n! q4 N
DDRII和 DDRIII,它们的通讯方式并没有区别主要是通讯同步时钟的频率提高了。
3 L/ G1 A6 j3 p7 _4 v
' `/ b* h# Z) k. ~- L( E% d1 |
3.SRAM存储器:静态随机存储器 SRAM的存储单元以锁存器来存储数据,见图。这种电路结构不需要定时刷新充电,就能保持状态(当然,如果断电了,数据还是会丢失的),所以这种存储器被称为“静态(Static)”RAM。
8 z; S e& y# D) E! k( P
同样地,SRAM 根据其通讯方式也分为同步SRAM和异步 SRAM,相对来说,异步SRAM用得较多。
4 j9 y! d9 Z: Q$ G
所以在实际应用场合中,SRAM一般只用于CPU内部的高速缓存(Cache),而外部扩展的内存一般使用DRAM。DRAM和SRAM的特性对比如下:
1 }- K8 p. V% X/ a" Y4 N3 J
8 L. P& N* i; A' r* i- a
4.非易失性存储器:非易失性存储器种类非常多,半导体类的有 ROM 和 FLASH,而其它的则包括光盘、软盘及机械硬盘。
" h: Y& h% T/ m
ROM是“Read Only Memory”的缩写,意为只能读的存储器。由于技术的发展,后来设计出了可以方便写入数据的ROM,而这个“Read Only Memory”的名称被沿用下来了,现在一般用于指代非易失性半导体存储器,包括FLASH存储器,有些人也把它归到ROM类里边。 ( A0 n+ P: q) w& V: P" e/ o
' B( t0 F4 `% p7 ^
FLASH存储器又称为闪存,它也是可重复擦写的储器,部分书籍会把FLASH存储器称为FLASH ROM,但它的容量一般比EEPROM大得多,且在擦除时,一般以多个字节为单位。根据存储单元电路的不同,FLASH存储器又分为NOR FLASH和NAND FLASH。
2 w7 Q( X# W+ w. {) ^$ |
6 |8 ~! f4 |3 X/ z2 L8 D
SRAM控制原理
/ M0 l1 w, V$ l2 d4 M3 w
5 `% n% d3 o$ R D% W
STM32控制器芯片内部有一定大小的SRAM及FLASH作为内存和程序存储空间,但当程序较大,内存和程序空间不足时,就需要在STM32芯片的外部扩展存储器了。STM32F407系列芯片可以扩展外部SRAM用作内存。9 [% ?& V: b) }/ \( p( h( Z/ f
v1 P. d/ Y7 d7 N0 w; z! L8 R# \( q+ S
给STM32芯片扩展内存与给PC扩展内存的原理是一样的,只是PC上一般以内存条的形式扩展,而且内存条实质是由多个内存颗粒(即SDRAM芯片)组成的通用标准模块,而STM32扩展时,直接与SRAM芯片连接。 ! }2 l8 t# c/ v6 @8 ~
SRAM外观与内部框图
T: G+ w A; e* c; I& j! s' w
1 W$ }5 F0 s. _) M
, q7 {7 G1 N7 c/ e% k: P/ q; v4 T
①地址数据接口:A0-A17用作地址线,I/O0-I/O15用于传输数据,一次性传输16位数据,并行传输
" z6 o- H$ m8 |/ R1 q
②存储矩阵:每个存储单元16位,SRAM内部包含的存储阵列,可以把它理解成一张表格,数据就填在这张表格上。和表格查找一样,指定一个行地址和列地址,就可以精确地找到目标单元格而这样的表则被称为存储矩阵。
9 c: b- T. _. q: H% n7 [% ~
; D' o& u0 v6 H( w
当选中一个数据单元后,可以通过UB#或 LB#其中一个设置为低电平,I/O会对应输出该地址的高 8位和低 8位数据,因此它们被称为数据掩码信号。 2 t0 e/ K9 U. F" O. H. m+ r
③控制电路:控制电路主要包含了片选、读写使能以及上面提到的宽度控制信号UB#和LB#。利用CE#片选信号,可以把多个SRAM芯片组成一个大容量的内存条。OE#和WE#可以控制读写使能,防止误操作。整个数据的传输都是在控制电路的控制下完成的,当想要读数据时,使用到OE线,把它拉低;想要写数据时应当把WE线拉低;只想读存储单元的高8位,则拉低UB线;只想读存储单元的低8位,则拉低LB线;一次性读26位都拉低即可。 / Y$ n3 n3 x3 |5 K5 Q5 `+ w
SRAM的控制比较简单,只要控制信号线使能了访问,从地址线输入要访问的地址,即可从I/O数据线写入或读出数据。
3 ]; ~3 [) c7 A2 w2 E' N E c" n
& e/ `% f; D& Y8 s
SRAM的读时序
6 ?! u5 b' z% w, n3 {7 {( B1 N
; T N* l8 w* e$ ~. ], C1 y# o/ W
$ ?+ i- D: _4 _8 M0 o: A7 G
对什么时候使能读信号、什么时候拉低片选、地址线号线维持的时间等都有一定的要求,重点时序包括:读周期时间(tRC)、地址建立时间(tAA)、OE建立时间(tDOE),这些具体的时间都可在SRAM芯片的数据手册上找到。
% ^# {4 v1 b; ^$ ^6 j- H
SRAM的写时序
' h1 A% }1 y; q, I& i: Y2 W( r3 ~4 P
5 @! r& ]- d) ^% L( n4 ]
% h& J0 f4 ~; d
重点时序:写周期时间(tWC)、地址建立时间(tSA)、WE脉宽(tPWE) ( o% a q* V& ^5 r; _/ P
SRAM的读写流程 Z0 f; ?* a9 m; k; S# t
8 ]- F2 U' q4 ^$ t% D6 [0 E& G6 e
读写时序的流程很类似,过程如下: (1) 主机使用地址信号线发出要访问的存储器目标地址; (2) 控制片选信号CE#使能存储器芯片; (3) 若是要进行读操作,则控制读使能信号OE#表示要读数据,若进行写操作则控制写使能信号WE#表示要写数据; (4) 使用掩码信号LB#与UB#指示要访问目标地址的高、低字节部分; (5) 若是读取过程,存储器会通过数据线向主机输出目标数据,若是写入过程,主要使用数据线向存储器传输目标数据。 $ {7 t9 m; a* M, K) y! j9 b
STM32-FSMC控制器框图分析
# ^8 J8 p1 ?) e; L0 p% T
. S* o; I! ?* g
; ?; m, E. o8 z( C3 ~; G- X0 l
①通讯引脚:由于控制不同类型存储器的时候会有一些不同的引脚,看起来有非常多,其中地址线FSMC_A和数据线FSMC_D是所有控制器都共用的。
* j1 v$ }4 u0 V! N. ?# N8 \
' B# I8 [9 V$ b: Y( T
注:其中比较特殊的FSMC_NE是用于控制SRAM芯片的控制信号线,STM32具有FSMC_NE1/2/3/4号引脚,不同的引脚对应STM32内部不同的地址区域。 - T4 ]4 J( U. X; q* F
②存储器控制器:上面不同类型的引脚是连接到FSMC内部对应的存储控制器中NOR/PSRAM/SRAM设备使用相同的控制器,NAND/PC卡设备使用相同的控制器,不同的控制器有专用的寄存器用于配置其工作模式。 $ f" Z/ Q7 d8 |5 F& s
控制SRAM的有FSMC_ BCR 、FSMC_ BTR以及FSMC_BWTR寄存器。每种寄存器都有4个,分别对应于4个不同的存储区域,各种寄存器介绍如下:
/ U: J) Q4 Q; r0 A0 u4 Q
a.FSMC_BCR控制寄存器:可配置要控制的存储器类型、数据线宽度以及信号有效极性能 参数。 b.FMC_BTR时序寄存器:用于配置SRAM访问时的各种时间延迟,如数据保持时间、地址保持时间等。 c.FMC_BWTR写时序寄存器:与FMC_BTR寄存器控制的参数类似,它专门用于控制写时序的时间参数。
" i4 O+ i9 m" Q, J
③时钟控制逻辑:FSMC外设挂载在AHB总线上,时钟信号来自于HCLK(默认168MHz),控制器的同步时钟输出就是由它分频得到。 2 E" Q7 n. j+ e
例如,NOR控制器的FSMC_CLK引脚输出的时钟,它可用于与同步类型的SRAM芯片进行同步通讯,它的时钟频率可通过FSMC_BTR寄存器的CLKDIV位配置,可以配置为HCLK的1/2或1/3,也就是说,若它与同步类型的SRAM通讯时,同步时钟最高频率为84MHz。
5 k( \6 X ^) \6 q: W$ i
后面示例中的SRAM为异步类型的存储器,不使用同步时钟信号,所以时钟分频配置不起作用。 " }3 U( |8 Q$ F0 b! G# m8 Z
FSMC地址映射
$ [6 U( ~ C. I- h" f# ^0 B. D
/ |: q: i, u* F$ ]: C0 `# Z1 K
FSMC连接好外部的存储器并初始化后,就可以直接通过访问地址来读写数据。 8 k9 d0 F1 b" P/ _! ~
FSMC访问存储器的方式与I2C EEPROM、SPI FLASH的不一样,后两种方式都需要控制I2C或SPI总线给存储器发送地址,然后获取数据;在程序里,这个地址和数据都需要分开使用不同的变量存储,并且访问时还需要使用代码控制发送读写命令。
* G+ [$ u! |: a' M \
而使用FSMC外接存储器时,其存储单元是映射到STM32的内部寻址空间的;在程序里,定义一个指向这些地址的指针,然后就可以通过指针直接修改该存储单元的内容,FSMC外设会自动完成数据访问过程,读写命令之类的操作不需要程序控制。 & q6 z2 J+ U- z& X2 |
3 b( f9 _0 C- y
FSMC的NOR/PSRAM/SRAM/NAND FLASH以及PC卡的地址被映射到了External RAM地址空间内,使得访问FSMC控制的存储器时,就跟访问STM32的片上外设寄存器一样。
/ N" {7 m3 Z" w
FSMC把整个External RAM存储区域分成了4个Bank区域,并分配了地址范围及适用的存储器类型,如NOR及SRAM存储器只能使用Bank1的地址。 g# z- y9 D% }9 P. a% [& F% H9 F! D
在NOR及SRAM区域,每个Bank的内部又分成了4个小块,每个小块有相应的控制引脚用于连接片选信号,如FSMC_NE[4:1]信号线可用于选择BANK1内部的4小块地址区域,当STM32访问0x68000000-0x6BFFFFFF地址空间时,会访问到Bank1的第3小块区域,相应的FSMC_NE3信号线会输出控制信号。 " t; E4 z1 H0 Q; A! W( ^2 a9 e! {, Y
% b* l% d$ }1 Z
FSMC读时序如下:
7 ?, A) ~! o5 s
6 ~* n6 W- w( x8 c$ f8 h( a
FSMC写时序如下: , `) [/ ^( x/ c: _% s1 K/ l
9 W: Z; d" ]0 t4 x$ p ~8 Z
FSMC外扩SRAM实例
5 k: g7 u% _! O
5 a' n+ d2 g7 v: K8 G+ |( N
实验要求:配置STM32F407的FSMC以支持驱动IS61LV25616 SRAM芯片的读写。
6 m- C( Y5 Z! D
SRAM电器接线图如下
" T8 f" c. h0 I5 n% |
; d- V) ~9 q' z$ Z( j/ S& l
实验步骤:
4 }) k- q9 t& l7 m" e, L' c
1.配置RCC 8 ^9 H: B% ^0 W9 L
2.配置FSMC管脚
$ V: \3 N: ~% [) H' h0 I
+ Y0 A1 J6 O: y7 R& J- W# ?6 s
1 l7 S& c( Q" G3 J5 ?! z
3.配置FSMC协议
0 X6 {$ w. G" ~5 ^2 a7 L; Y2 ]. [7 q
* Y" f5 q- J+ i
tRC =ADDSET + DATAST,tRC是SRAM芯片的一个需求时间,可以通过查询芯片手册得到tRC的最小满足时间。IS61LV25616的型号有IS61LV25616-8、IS61LV25616-10、IS61LV25616-12、IS61LV25616-15,分别代表高速访问时间为8,10,12,15ns,本实验中以-10为例可知tRC最小时间为10ns ,tDOE最大时间为5ns,在配置是应当将tRC的值设置为大于10ns 4 ~, U8 p. Q Y3 S
6 F$ ~/ H2 F# x+ D s* K
tDOE =DATASET,tDOE就是读使能信号拉低之后到得到有效数据之间的时间。 ADDSET一般是地址发出后到读使能信号开始拉低的时间,对于SRAM未作要求,大于0即可。FSMC的主频为168M,一个周期约等于6ns
7 c/ Z3 y9 u& F# ]' z7 v
3 K% \: n9 | H0 W
4.编写代码 - //main.c
) o) G9 @& Q/ C) y1 C8 T! ]% I
+ a0 `; O8 s. c( M! K" n- #define SRAM_BANK_ADDR ((uint32_t)0x68000000)//外扩存储器首地址
9 l0 r; a# K+ w3 V" o: u# W- m - # z, f6 w5 _ F; \- b0 e
- int main(void)
% w7 w" w+ D; ^ - {
" |8 G7 c! j0 U* ?' r - uint8_t *p = (uint8_t *)SRAM_BANK_ADDR ;
3 R( {( L4 Q: z" M1 `7 x+ D: ? - uint8_t i;
8 w% T1 c8 V. D' ? - HAL_Init();) J, F1 p: ?1 t
- 3 D+ `( V2 I. v, p+ C
- , }: f. r5 v! R, t! m
- SystemClock_Config();9 F9 w' k0 g q! D6 o* j
- MX_GPIO_Init();
: T8 t% ]( U- @! u5 {0 w# W. U - MX_USART1_UART_Init();+ ]( J" x. y& c- D8 I
- MX_FSMC_Init();
, J$ P Z8 N4 v) A5 ]4 w$ ]- H8 q* M - 7 B! o; h v* P3 r* ]
( J; i% J) w* Q- Q+ q6 _- printf("this is fsmc sram test\n");
! }( J4 l2 A! O) N9 Y* M+ Q
" W0 |! ]+ n! u- //写内存
! v+ b; E& s$ ~4 N" O0 f - for(i=1; i<=10; i++)! f3 D: u- T. d0 T$ X1 f& r
- {
2 s0 l; U0 X0 n6 s$ ~ - p[i] = i;
3 M1 t0 n5 c' w, E( `% { - }
4 O- p* E. r9 _" D3 O' Z
- s2 D# Q( H6 ?8 z0 j- //读内存: F) ^" p( l; U$ p
- for(i=1; i<=10; i++)9 a9 g% Y8 ^0 I5 }! Y
- {* P6 I5 L$ M% G8 d! `
- printf("p[%d] = %d\n",i, p[i] );/ C" C% h& e+ x5 u- n
- }! t/ I8 p4 G/ o3 q0 ~' t. r
- while (1)
# V( Y! F' p9 v5 e* h/ i - {
& ~5 Q2 a" ?) W5 [# M+ ` - Remote_Infrared_KeyDeCode();
) | |6 ?9 z) a - }& h9 p+ D! t3 T7 s) g, m
- }
复制代码9 x- r( w4 v3 ?* x
二、触摸屏专题讲解
3 _& F- e( Q- p0 s: e0 s" \4 H
触摸屏简介 9 r+ Y2 @2 v, r& r; J$ t
; u* C3 C4 M, t/ E5 ~/ k \
触摸屏又称触控面板,它是一种把触摸位置转化成坐标数据的输入设备,可以响应用户点击的位置(对点击事件进行响应),是典型的人交互设备,使得人机交互更加人性化。根据触摸屏的检测原理,主要分为电阻式触摸屏和电容式触摸屏。 * G" V3 K7 i. B( b: T3 j% q
电阻触摸屏:电阻屏造价便宜,能适应较恶劣的环境,但它只支持单点触控(一次只能检测面板上的一个触摸位置,不管有多少触点同时被触控,一次只检测一个),触摸时需要一定的压力,使用久了容易造成表面磨损,影响屏幕寿命。 % @; w. _9 [- j0 D$ f/ {! [+ k
电容触摸屏:具有支持多点触控(如玩手机游戏时可以同时有多个控键发生作用)、检测精度高的特点,电容屏通过与导电物体产生的电容效应来检测触摸动作,只能感应导电物体的触摸,湿度较大或屏幕表面有水珠时会影响电容屏的检测效果。 4 H( ~; k0 c3 y
目前电容式触摸屏被大部分应用在智能手机、平板电脑等电子设备中,而在汽车导航、工控机等设备中电阻式触摸屏仍占主流。
C5 \ @3 \) n5 P, ]4 h- A
7 |2 Y0 H' A8 u2 n2 X9 ?+ Z/ u! c
触摸屏与显示屏的关系:触摸屏实际上是在液晶屏上面贴了一层大小相等的透明的薄膜,这个薄膜能够感知触碰,根据薄膜反馈的触摸位置,我们就能知道用户触碰在屏幕的什么位地方了。为了美观,电路一般都铺设在显示屏的边框上进行流通,最终通过fpc排线流出,所以绝对的无边框屏幕时不存在的。
4 E1 S2 N, V$ A( c% Q! Z7 Z3 v
: }4 O5 {9 I( M2 c* h
电容触摸屏检测原理 1 G$ `2 s3 |9 v. m& X
1 F% r* J7 L3 {4 d3 B2 T) \; _
与电阻式触摸屏不同,电容式触摸屏不需要通过压力使触点变形。它的基本原理是利用充电时间检测电容大小,若手指触摸屏幕,会影响触摸点附近两个电极之间的耦合,从而改变两个电极之间的电容量,若检测到某电容的电容量发生了改变,即可获知该电容处有触摸动作从而通过检测出电容值的变化来获知触摸信号。 ) Z) y7 v0 n0 F
简单理解为么一个电容容量大小固定,充电时间不变。但是手指触摸时,由于手指的导电性,会有一部分电流流过手指使得电容的充电速度变慢,充电时间变长,由充电时间发生变化可检测到触点的位置。 : Q0 q& e5 d" t* _$ o1 W* z; Z
5 J0 r% W Z9 `1 z4 H
电阻触摸屏检测原理
, @5 O6 T; j% K, X+ g+ \# e# m
% }) m2 p" i7 N% q+ k
电阻式的触摸屏结构主要由表面硬涂层、两个ITO层、间隔点以及玻璃底层构成,这些结构层都是透明的,整个触摸屏覆盖在液晶面板上,透过触摸屏可看到液晶面板。表面涂层起到保护作用,玻璃底层起承载的作用,而两个ITO层是触摸屏的关键结构,它们是涂有铟锡金属氧化物的导电层。两个ITO层之间使用间隔点使两层分开,当触摸屏表面受到压力时,表面弯曲使得上层ITO与下层ITO接触,在触点处连通电路。 I X8 V4 h( ?3 E2 ]& }( [
+ y8 Y+ e! ~* }/ Q& X! c
我们可以把上面的ITO层看作X层,下面的ITO层看作Y层,每一层都会加上一个电压,ITO层是均匀的导电物质,那么整个一层的电阻是呈线性排布,不同地方的电阻大小不同。流过ITO层的电流恒定不变,假设对X层和Y层都加上一个5V电压,当有触控时,上下两层检测到触点的电压都为2.5V,那么理想状态下触点为中心,假设上层检测到触点的电压3V,那么上层触点在顺着电压方向屏幕的3/5处,下层检测到触点的电压2V,下层触点在顺着电压方向屏幕的2/5处,这样就相当于确定了一个坐标,从而确定触点的位置。
! O; {) Z/ p+ u* ^
两个ITO涂层的两端分别引出X-、X+、Y-、Y+四个电极,这是电阻屏最常见的四线结构,通过这些电极,外部电路向这两个涂层可以施加匀强电场或检测电压。 . A2 H6 k! n8 U3 b, f) [1 M& r" h+ O! M
) Q# d4 }- K5 d z
电阻触摸屏校准 - Y l7 ?) w# K* R: V
, }- r/ s5 D* S
电阻触摸屏的校准就是确定位置和对应的电压值之间的比例关系。假设添加X的场电压为5V,理想状体下,我们希望检测x方向最左侧为0V,最右侧为5V,但是实际上最左侧可能为大于0V,最右侧小于5V,但是ITO层在电压方向的电压变化仍是线性关系,那么最左侧的电压值称为偏移值,线性关系称为斜率。Y层同理。
0 f6 `: ?! w" A0 w. R9 H$ h
/ s6 T+ N9 N$ q5 q3 e7 J
X0 = xfac*ADC_X + xoff,xfac是斜率, xoff是偏移值 Y0 = yfac*ADC_Y + yoff,yfac是斜率, yoff是偏移值 6 |0 | N. V; ]/ C Z* _
触摸控制芯片XPT2406 - Q: M# d8 A, ^/ m3 {* F+ @7 A
3 Z* K' m- k P* n
3 B- ]! `& ~+ w+ J% m
该芯片将触点的电压值数据通过SPI的方式传输到MCU,可以看到,红框内部由时钟线、片选线、数据输入、数据输出,这是典型的SPI传输结构。除此之外还有一个中断管脚PENIRQ,当触控一个点后,中断管脚发生作用,告诉MCU内部的转换数据已经准备好,让MCU进行读数据。
8 }4 s8 a1 s+ i; t2 J7 `
触摸屏操作实例
* a, t" l/ d3 W! e5 n
$ Y6 E+ X8 u* S. ?; P1 r4 d
实验内容:在触点处显示圆圈(本次涉及到的液晶屏代码将在后续讲解,本次内容很少而且简单,不影响阅读) & ^6 `; C5 F2 G ? H" w
说明:当触摸屏有触点按下时,PENIRQ引脚会输出低电平,说明数据已经准备好。直到没有触摸的时候,它才会输出高电平。因此可以设置低电平中断,在中断处理过程中向控制芯片发送读数据的命令并读取数据。 $ \2 p: O* A+ L/ L" f
硬件连线如下,其中INT管脚与中断管脚及逆行连接,用于触发中断。
* E& @3 ? h4 i
XPT2406命令字(控制字节)如下: 8 @9 v5 j4 Y9 d& q; ? \/ F
, a* X2 ?" N0 E9 O. p
位3选择0,说明读取的数值位12位分辨率,即读取的电压值位12位。发送命令后应当连续发送两个空命令(一个命令8位)来获得数据。至于为什么要继续发送两个空命令,具体原因见本公众号SPI文章传输原理部分。
# q" R6 W3 a9 h0 K
# ^* h5 u5 q& A9 b
步骤:(部分配置省略,只进行重要配置说明)
, e3 d0 I; A4 X
1.配置RCC + N7 k" x1 p+ t, B f
2.配置iSPI2
$ Q5 m6 }# L. g% A3 M! r( z( d5 q; _9 |/ m4 A+ D5 H% s3 l) Y; W
' Y6 d+ n7 ?2 @4 F
3 f: i1 w1 D+ [6 ~( n1 U
3.配置中断:将PG7管脚配置为中断模式 : _: i3 T" o$ i/ X, J: i
" s0 B6 Y9 {. ?. i1 b- Q
0 M" U* `; S# n3 [9 ^& R) D) ~
t2 G+ L5 l! J2 {# [, A C
4.编写代码 - //main.c3 C" s" F; @( y# T5 ~
5 S) ^6 a) |, j- #include "spi.h"
]9 S9 z, b- u3 ^% T8 |% ?$ @, s) V - #include "gpio.h"4 ]) ^/ `" U" Z" T
- #include "Touch.h"
6 o' R/ G" ` A' ` - #include "lcd.h"" {2 w* h1 L+ d6 e
- 6 `/ ?# A/ @+ @8 {
- //校验参数(计算得出)
# C7 E2 M5 v/ c, A/ @5 h - Pen_Holder Pen_Point ={
% j7 r& o# b- [, s - .xfac = 0.259067,
7 }. C8 I: p6 w" D5 [6 c1 d/ ]: [; b - .xoff = -20,
- G. H k' j' Q5 B - .yfac = 0.158228,! N9 u4 p$ {7 J# g
- .yoff = -38, 8 _6 e7 D. j& ?: I. K
- };/ _4 k# Z: G; \
- ) z0 q5 N/ ]) m- D0 R4 M+ [
- void touch_adjust()
+ f! U+ h2 S# G - {
* Q* i. ~3 H6 x! k! S - Draw_Circle(50, 50, 5,Yellow) ; 7 o" J' C( R% Y
- Draw_Circle(100, 100, 5,Yellow) ;
# D) D1 J- @8 U( J0 M: G - }
: r( O! e& i7 |: d) ?; P
* Y ^! `+ M/ n- int main(void)
& b$ [+ S5 P3 f- H - {/ y! W7 A2 z! n1 ]
- HAL_Init();5 Y4 R$ y1 R) H2 O+ H0 T0 j4 o
- SystemClock_Config();
6 o! C) r4 v) }1 m - MX_GPIO_Init();
+ J* }! E' ~ k - MX_FSMC_Init();8 c% \1 f! K9 n- a% D5 b* J* Y% b
- MX_SPI2_Init();
1 }; H0 P8 Q; X8 m% L - MX_USART1_UART_Init();
1 ?+ ^4 s; w3 X -
, \. B, n3 m! Y) B* A - printf("this is touch lcd test\n");$ H: r `" U$ {6 R; W5 x2 [% k
- lcd_init();
( F3 J" ? M" p5 u1 C$ Z - lcd_clear(Blue);' t) s0 A9 N) d1 k
- //touch_adjust();
% t- n M1 i: R8 i& N# X - while (1)
$ M7 k' d$ X& p1 \ - {. P) a/ S7 ?' o: A f" V- Y7 r3 P% i
* i! o6 v6 }8 s& t9 c0 q6 r- if( Pen_Point.Key_Sta==Key_Down ) //触摸屏被按下
~; T. g8 O' o' U, j - {
# a6 j5 s' ? T& R* T - HAL_Delay(20); //按下之后会有抖动,因此延时; c1 {: q; B0 Z/ B
- if(!HAL_GPIO_ReadPin(GPIOG,GPIO_PIN_7)) //确实按下
0 v4 y( ~. O& P- H( g# X# t# P- X - {
9 q8 d) r7 G6 F0 z- Q" j! G* X; G - HAL_NVIC_DisableIRQ(EXTI9_5_IRQn);//关闭中断
8 d& v; ]. d4 V2 {8 T0 Z - //如果不关可能会因为一些毛刺而再次触发中断
( K- P* e+ R$ I n - printf("++++++touch+++++\r\n");
* C" B+ N/ [/ D9 x7 E, H - if( Convert_Pos() ) 8 A9 k2 q' J7 C* s' ~9 l" E
- //存放点击的X,Y坐标值,成功返回1失败返回0
3 N( H2 F( E+ n1 X9 W+ o3 r - { ' F0 a( R) j1 M$ N
- printf("x = %d,y = %d\r\n",Pen_Point.X0,Pen_Point.Y0); 7 v5 s& v- F: u8 ?# b S
- //画一个以X0,Y0为圆心,半径为10,颜色为黄色的圆, w) E: V7 z2 e& @& D7 n
- Draw_Circle(Pen_Point.X0, Pen_Point.Y0, 10,Yellow) ;
$ k8 u2 w6 q+ V$ r9 o - //恢复为0,以便存放下次触发位置 , U) C% I) R( s
- Pen_Point.X0 = 0;: f. |3 ~! e, e7 J- p
- Pen_Point.Y0 = 0;3 a/ o; Y7 i2 q0 r( } ]/ I* O
- }1 G7 x0 X6 v+ D( a8 ]5 V
- HAL_Delay(200);//跳出毛刺
8 p$ A" X8 L0 L" N) O - //再次开启中断以便下次触发 9 Z; m, Q( X% d* B h Z
- HAL_NVIC_EnableIRQ(EXTI9_5_IRQn);
5 M- `+ v# k. F7 U0 D6 |. |) W - }
4 j2 D4 G# i3 ~0 J - Pen_Point.Key_Sta=Key_Up; 5 l, g4 E$ }2 F) a: i# F9 @
- }
6 c+ P$ p+ O2 R! v$ o; j' v3 m - }
& P2 k" T4 s6 W8 I. a4 u; I/ j2 J7 c - }
复制代码 + p) M6 i1 Y" c4 H$ u
说明:关于程序中的校验参数怎么得到。我们可以看到程序中有touch_adjust()这样一个函数,这个函数的功能就是在屏幕上分别画以(50,50)(100,100)为中心,半径为5的圆,在程序初始化时就调用此函数,之后用触笔分别点击这两个圆,将每个圆点击后检测到的X和Y电压值带入二元一次方程。解方程即可得参数。
6 [' r* |5 P" g# d. L3 l' I5 x1 E0 _
; O B$ Q7 E4 t0 ?7 H- i6 S% w- 6 n l. M. a* d1 T- i
- //gpio.c% m8 u M7 L" h n8 \; ]
- //中断处理函数
. H0 p, P; W4 s5 q$ {* W - extern Pen_Holder Pen_Point;6 |5 Z) B y9 m" w
- 4 u5 j6 V* q$ o; V
- void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin), c- M7 p, {7 M/ N2 t$ h
- {
( ^1 V: q4 S8 f# _! u - if(GPIO_Pin == GPIO_PIN_7)& f# V1 i/ X9 q+ Y7 e2 B
- {$ H9 d! Y* W/ T+ x# s1 q0 X
- Pen_Point.Key_Sta = Key_Down;8 [* [2 z# e9 t( |5 [$ }* H9 b4 O
- }& S r% [7 b' u; \$ p; m3 X
- }
复制代码- //Touch.h
. s7 W) m& n1 M6 ^ b! Q. R - ' S: v5 D9 C/ S) |
- #ifndef _TOUCH_H_1 }/ f k6 Y5 S5 j- h. V" c
- #define _TOUCH_H_
$ p$ r1 l% w4 P; c - ' f/ w, d- o8 N# U% J) U' J0 I
- #include "stm32f4xx.h"0 D/ l( D l1 O6 i# p" W
- # P9 a& h2 P2 E2 @- a+ b, a
- //ADS7843/7846/UH7843/7846/XPT2046/TSC2046 指令集
$ t0 M# x6 z7 J* t: r4 ? - #define CMD_RDY 0X90 //0B10010000即用差分方式读Y坐标
. G1 ~8 X9 M: k# U" t! x+ x8 o - #define CMD_RDX0XD0 //0B11010000即用差分方式读X坐标% G6 a7 g2 E) L2 q. k# f" D
. ^! d6 z% K; E# U- % L" W! ~- C/ Y# q# O2 I0 d' Z; s
- /*按键状态*/ $ n7 e* v0 p2 ^/ Q; p
- #define Key_Down 0x01
0 j9 w, ]& i+ q9 q - #define Key_Up 0x00
' V7 x1 d* z7 n5 Z' {8 a
, g6 n0 z5 Q% J- /* 笔杆结构体 */" J+ z8 n6 h7 j: I. [3 J& g
- typedef struct 2 A5 b* q$ b" G9 y
- {) T: I/ m6 ?1 U% f
- uint16_t X0; //原始坐标- J: e7 G9 J4 r% D! x9 D, }* s) x
- uint16_t Y0;: \# Y5 Z' _6 B4 A, ~
- uint16_t ADC_X; //X方向的电压值5 h8 K. v7 E9 _
- uint16_t ADC_Y;//Y方向的电压值 # J) e6 u9 y' s0 z
- uint8_t Key_Sta;//笔的状态 0 i6 c+ }# R6 c9 J' m" R6 K, v' z0 {
- //触摸屏校准参数
8 ]* r% i: S. F' Y9 j' Y - float xfac;
+ `8 ~# S/ _* t - short xoff; //X方向的电压值偏移值' t7 T) F8 D# i C+ M
-
! C' K/ q, q, {- W - float yfac;
% U( V3 x; y1 B T. S' R - short yoff;
6 y5 ? L9 g2 t6 w6 T2 I/ J - }Pen_Holder;
$ Q. l7 B" f9 E. _ - ' ]5 {# [& D6 ^- L: W) F% L
- //宏定义,用于片选
! I1 R6 w5 {, z4 q k - #define SPI_TOUCH_CS_LOW() HAL_GPIO_WritePin(GPIOG, GPIO_PIN_15, GPIO_PIN_RESET)
, J7 b* j: B# T; J - #define SPI_TOUCH_CS_HIGH() HAL_GPIO_WritePin(GPIOG, GPIO_PIN_15, GPIO_PIN_SET)
) K# r f2 f: J$ o7 `4 n: ?0 {2 i - 0 l& w2 G2 x3 k& r- }6 l
- uint8_t Convert_Pos(void);5 X1 c: H' W m# s0 x7 X5 @0 i
- #endif
复制代码- : p* w( Z" `# k8 H' y1 p2 D
- //Touch.c( @. [( F: l( T+ U7 W# a
& ?% M \3 r; B0 G$ U- #include "stm32f4xx.h"/ Y6 E% j% Z4 b! y9 u3 l
- #include "Touch.h"
, [4 C z ?1 E+ V/ F, A% d - #include "spi.h"! ~1 z5 h6 Y, \% O; e; i, _
/ c6 {7 l. O4 c9 y8 b+ ~: s- extern Pen_Holder Pen_Point; /* 定义笔实体 全局*/
' n' i$ o2 j0 N4 r+ ]/ x
5 H8 M: q1 \3 z7 ?" e3 ~- {% |8 D- extern SPI_HandleTypeDef hspi2;
- u A3 @) z: T" L5 \ - /*封装了一个发送命令的函数*/7 b( E3 n4 k. s- _8 l; R- [
- static uint8_t WR_Cmd(uint8_t cmd)0 y7 n; [/ e$ e& {
- {
% c: s; c1 h3 I1 c' R - uint8_t Tx_DATA[1] = {cmd};//将命令放到Tx_DATA中* y. Q& L6 }0 s- h. b- G- l
- uint8_t Rx_DATA[1] = {0};//Rx_DATA用来接收返回值
1 f2 H4 W" S5 |0 L( ^& X - HAL_SPI_TransmitReceive( &hspi2,Tx_DATA,Rx_DATA,1,0x1000);
0 K4 I2 F/ p# D6 o, c2 N5 [; n$ d - return Rx_DATA[0];
6 b3 `7 R" a' f7 T" Y - }; }* a2 G2 L: G6 {
- " E- u" S3 F; _- E4 }, D( T C
- //从SPI读数据
+ N! Q7 v7 m; [4 K - //从7846/7843/XPT2046/UH7843/UH7846读取adcÖ值
f2 y. S1 Z. _+ h
# Z K: C$ m3 O' H+ B- /*************************************************/* T- _' x! @) C% b$ ?5 Z* a
- /* 功 能:读取X轴或Y轴的ADC值*/
+ H* K/ Y2 [% Z/ M, l6 z m - /* 入口参数:CMD命令 */
+ A, A% k% y0 s: b - /*************************************************/* A5 N4 e% |/ J# Q5 z( Y/ ]2 N& i0 X
- uint16_t XPT_Read_AD(uint8_t CMD)
& A1 J3 s! e6 W$ a' Q) T/ y! H - {
' g8 x: }" e9 f4 j u' m - uint16_t NUMH,NUML;
9 z$ Q- H6 x2 F p - uint16_t Num = 0; " y' N9 k# b9 |
- SPI_TOUCH_CS_LOW(); // CS=0 开始SPI通信 $ M% P* @: l6 h
- WR_Cmd(CMD);
" i! h. X* G- s - HAL_Delay(1); //延时等待转换完成, u3 |8 N/ U5 V9 c
-
8 m. m, t; @0 \& _* F0 O - NUMH=WR_Cmd(0x00);
: a/ K# g B- n' V6 H - NUML=WR_Cmd(0x00); //发送两次空命令来获取电压值* Q! H1 P2 D' m1 r8 O8 w
- Num=((NUMH)<<8)+NUML;
" V% U" Y9 \) G( ]% Z - Num>>=4; //只有高12位有效
: I2 {3 y y! }4 X o9 w. Q - + _0 [& C; y( |4 X7 L0 T& r
- SPI_TOUCH_CS_HIGH(); // CS=1 结束通信 6 e6 J# d u4 @2 d* \6 j: h0 m
-
: A1 H# T3 O' q - //printf("num = %d\n", Num);1 y) f4 A; I3 Y0 q& l3 X
- return(Num);
h3 g& p# l0 L - }
1 V: D8 b3 I. O1 Q8 P) X - ' T c" N2 T; g( @! I
- 2 H" ?' [5 L0 |0 K& T* j' d
- #define READ_TIMES 10 //读取次数 E; N2 t( V# }* }- \' \
- #define LOST_VAL 4 //丢弃值
: ^, I4 [1 p" ~1 {! l9 X- v* h - /*************************************************/
' z/ @! X- z4 s& m) b4 q - /*功 能:读取X轴或Y轴的ADC值 */
" s+ M$ P- n) l& x: a - /*入口参数:CMD_RDX:读取X的ADC值 */ , W4 N7 p: U8 c
- /* CMD_RDY:读取Y的ADC值 */
( U$ V Y/ b8 q' k3 g - /*说 明:与上一个函数相比,这个带有滤波,多读几个数据,去掉一些后求平均*/% C4 b s1 T3 G( H' [6 P
- /*************************************************/0 d6 W# U8 q" A; T+ ~3 t, y
- uint16_t XPT_Read_XY(uint8_t xy)0 v7 ~2 B, N" q+ G
- {
3 r1 V0 k# F$ M - uint16_t i, j;. Y. O( F6 W1 B
- uint16_t buf[READ_TIMES];2 S2 }' b: x% k8 r$ O
- uint16_t sum=0; J' X7 H$ y+ f8 O n
- uint16_t temp;! i& M$ e2 T! C; K) X/ ~0 [& \
- for(i=0;i<READ_TIMES;i++)
, \1 j/ \1 u9 L* V# }6 o$ R, V - { . T9 f( o& s- Z2 Z# }
- buf[i]=XPT_Read_AD(xy);
7 b( F4 Q- V6 w, G. i' k - } 6 D' C1 P, A* p; P
- for(i=0;i<READ_TIMES-1; i++)//排序
7 [9 t: S$ ^9 a& i7 M# p - {
7 x) Q7 W6 @, d# h* o; Y |, | - for(j=i+1;j<READ_TIMES;j++)8 d( ?* U+ Q" K- {2 a' v# Q$ W
- { {& h# `% v* M4 D& Q9 k% P
- if(buf[i]>buf[j])//升序排列
0 D' ~" \+ C! g" ?( g' ]1 T - {
/ D! J6 i1 i- G4 V8 g( Q6 o - temp=buf[i];
9 |5 u7 z% s9 q - buf[i]=buf[j];! f" t9 l( U3 N7 ^, W6 D% @
- buf[j]=temp;7 M6 q6 k/ a6 `1 S
- }/ [ Q0 ^4 c- c8 Z
- }' X) G+ {! B) e4 j/ ^/ }
- } # n! r% U, P5 F4 o/ i
- sum=0;6 H+ ~; ^ B) I; H$ i! b6 P5 W
- for(i=LOST_VAL;i<READ_TIMES-LOST_VAL;i++)
% B6 ?7 W) m" U: ~1 m3 Q - sum+=buf[i];
u1 [( L: g! {6 c7 B# G - temp=sum/(READ_TIMES-2*LOST_VAL);
3 x. _, `) L( k1 E - return temp; 0 r8 {6 O4 f( E
- }
8 m. {' x) G; |* M3 G+ _ - /*************************************************/# \# r- f# c) Q- s
- /*功 能:读取X轴和Y轴的ADC值 */
3 ]( h6 J' ^( J+ @ - /*入口参数:Pen_Point.X_ADC,&Pen_Point.Y_ADC */0 O: h# @6 C% ~' p
- /*出口参数:0:成功(返回的X,Y_ADC有效) */
2 D% O9 L6 u' L1 D0 v, { - /* 1: 失败(返回的X,Y_ADC无效) */ 9 E' O- g! f; v5 N7 y
- /*************************************************/* f! ^2 g' F$ m
- uint8_t Read_XPT(uint16_t *x,uint16_t *y)5 y# b, h. r2 }! Z0 W6 J
- {! E1 I7 a" R- ^1 a3 k. l
- uint16_t xtemp,ytemp;
3 F: h8 |) ]1 r - xtemp=XPT_Read_XY(CMD_RDX);
+ p& t: W: c/ d C+ T, t* [, K. f1 b - ytemp=XPT_Read_XY(CMD_RDY); 2 K. B n% O" a; t! U9 d
- if(xtemp<100||ytemp<100)
$ M, ?( m: r: V2 r- | - return 1;//读取失败1 L4 p P: _# u h- j5 L3 n$ h" N
- *x=xtemp;
6 j2 R7 `' \" a$ |# B0 h - *y=ytemp;+ F1 e1 K% v; R/ @8 S
- return 0;//读取成功" X; u3 H+ K$ P7 c; E
- }8 s# n; n0 k8 F/ @5 {0 w: C
- /*************************************************/" J% E3 O& J# i$ g& D7 r, C {
- /*功 能:连续两次读取ADC值 */
: @5 m0 G4 D5 L' M4 \2 N - /*原 理: 把两次读取的值作比较,在误差范围内可取 */ ( `- q' }4 R3 s' P. b/ y
- /*入口参数:Pen_Point.X_ADC,&Pen_Point.Y_ADC */9 q8 \' ?# T! j& [
- /*出口参数:0:成功(返回的X,Y_ADC值有效) */
! ~! u! x1 }; D$ N& ~& s5 J$ V3 v - /* 1: 失败(返回的X,Y_ADC值无效) */
3 p$ `0 ?! D$ w# P7 @$ j' @ - /*************************************************/
/ g# g; {$ |" J) k+ t0 u9 i5 k - #define ERR_RANGE 50 //误差范围 9 q& P3 b0 L! h- S9 Z2 C
- uint8_t Read_XPT2(uint16_t*x,uint16_t *y)
, X% d+ J, [; M2 u# O/ [0 _ - {
: J: m( z" U5 T- j8 v6 W" q% j - uint16_t x1,y1;
J% F% C: i4 b: [ - uint16_t x2,y2;
8 G+ m7 W. z4 ~+ { - uint8_t res;
4 t4 `+ t. t8 ~% K# l$ {2 t- A3 v - res=Read_XPT(&x1,&y1); //第一次读取ADC值 0 y# E0 }8 ^" y E3 m" u. w. G* z
- if(res==1)
% T! w+ b. E, Q0 S ]$ M - return(1);//读取失败返回1$ _& o+ k: [( x- _
- HAL_Delay(3);- Q" ^, p$ A7 E7 D" M
- res=Read_XPT(&x2,&y2);//第二次读取ADC值
3 Z* a5 C/ n/ G* A9 r5 c m) q - if(res==1)+ X9 O, r( I& r/ _ \
- return(1); //读取失败返回1
2 ]9 J! h( J4 p: p, T- H1 u/ A" M - //前后两次采样在±50内2 l& e% X4 V* w+ k- q
- if(((x2<=x1&&x1<x2+ERR_RANGE)||(x1<=x2&&x2<x1+ERR_RANGE)) ( \2 a& E s! P& u8 y* q
- &&((y2<=y1&&y1<y2+ERR_RANGE)||(y1<=y2&&y2<y1+ERR_RANGE)))
2 b4 u; E6 Z$ k/ i7 E: s/ q; \ - {
; y7 G$ F+ y0 P2 X1 [$ O! m - *x=(x1+x2)/2;
# u& `7 l7 M H - *y=(y1+y2)/2;: y; z! _" A0 x
- return 0; //正确读取,返回0# E" o7 S- v# M$ \ C; R
- }else return 1; //误差太大,读取错误
; T/ j9 h: H/ D4 Q4 R @5 G) l - }
/ P2 T: {0 a: c$ ^/ b+ l - 8 S0 `: {8 X* q" q: X
- //转换结果
7 j5 D0 z' Y( Q/ Z" ]! y - //根据触摸屏的校准参数来决定转换后的结果,保存在X0,Y0中
3 a7 W! ?4 f1 y E u - uint8_t Convert_Pos(void)2 y+ ?. l5 T7 m8 r3 b7 d
- {
! N) k6 p: [. d& I4 ]8 ? - HAL_Delay(8);
0 i" Q# [: B4 I f( `: c - if(Read_XPT2(&Pen_Point.ADC_X,&Pen_Point.ADC_Y) ==0 )
$ |0 q* I7 y' Z+ Z+ U/ J - {
1 u) m. e& w; l1 ^. t, G - //printf("adc_x= %d, adc_y = %d\n",Pen_Point.ADC_X,Pen_Point.ADC_Y);
- x; @- r1 P; \# w2 u - Pen_Point.X0=Pen_Point.xfac*Pen_Point.ADC_X+Pen_Point.xoff;; T4 T) `+ e& b) O" d% S
- Pen_Point.Y0=Pen_Point.yfac*Pen_Point.ADC_Y+Pen_Point.yoff; - S* ^3 d: q5 G* d1 n4 A: p
- return 1;
' F- x7 f6 v3 a4 _2 L9 a - }) O" `9 f, C4 }+ N6 r& |1 Z
- else return 0;
/ T8 w" M/ q" J* H9 b1 S' o8 }; i - }
复制代码 0 @3 a6 A8 g, j+ A7 J# e
转载自: 骆驼听海 如有侵权请联系删除0 M* a- ? w; o% f! T
* E6 _( ?% P7 H1 u
|