 鲜花( 0)  鸡蛋( 0)
|
这是嵌入式中Linux系统触摸屏驱动开发。
6 q8 m0 h& ^6 T- N" W3 s/ M5 U' t: Q
SPI接口的简介
2 M$ o2 u$ A) t# q
# p5 e" L, O" y; @5 B+ ]. Q( [串行外围设备接口SPI总线技术是摩托罗拉公司推出的一种全双工、同步串行接口,它提供了功能强大的四线接口(接收线、传输线、时钟线和从片选线)。
/ V4 y- i t4 g# b
2 N1 p* b) k) KSPI 的从设备和主设备共用一个时钟线,而时钟始终是从主设备里发送出来的。当823e是主模式的时候,片选信号线就停用,如果是从模式的话,它的从片选线低电 平使能。在本例中,823e是主设备,) B0 G1 N0 `% X. u; b# t5 g/ t
3 O6 z' F, U( l3 ]6 @所以我们另外选用了一个823e的GPIO(通用输入输出口)作为从设备的片选信号。大多数同步串行式数据转换器都 很容易与这种接口连接,其硬件功能很强,所以,与SPI有关的软件就相当简单,使CPU
2 j! j% S" |/ k& C$ i" U/ }- k% r0 F( W' _* ]
有更多的时间处理其他事务。
4 D$ L& I/ Y6 k% w8 ]- E
* N' d1 B0 i2 J8 ?9 X触摸屏的硬件
& p5 L9 n3 q( V6 ~$ W) q
% r8 D1 R% B! q触摸屏输入系统由触摸屏、触摸屏控制芯片和数据处理器三部分组成。触摸屏按其技术原理可分为五类:矢量压力传感式、电阻式、电容式、红外线式和表面声波式,其中电阻式触摸屏在嵌入式系统中+ {* U4 d7 T! w& S/ u
7 F5 D. X8 f5 O: o( z
用的较多。 : V4 `& t8 G8 x5 P/ u; \1 r
2 @ N. A6 a- B+ B2 X/ L2 F我 们选用的触摸屏是AMD公司的电阻式触摸屏AMT 9502。触摸屏控制芯片是TI公司的模数转换芯片ADS7846。该芯片支持SPI通信协议,所以我们就用823e的SPI接口与ADS7846芯片通 信,从触摸屏得到的
1 a' b/ Z- y5 A2 ]8 V8 j9 h* }) m+ T
模拟信号经过模数转换器后输入作为数据处理器的823e。
. P5 d' l s' K: \2 x* g5 r" W, Y7 y, c, M: p. n& R4 l
软件程序
4 R6 y" r5 p K9 P+ j+ D2 r) x/ \! d1 S7 c& a4 n5 I+ U3 b3 G3 O3 a) [- l: Z1 a
823e 通过SPI接口与触摸屏控制器通信,所以对触摸屏的控制就是对SPI接口的操作。完成SPI接口驱动的编写之后,就能够与触摸屏控制器建立通信。在 Linux内核运行完毕之后,SPI接口要打开,并且0 ^) u. U% ]+ ?- m: F
6 J1 ~1 u" D& l# N) B/ s
已经分配了一部分内存供它使用。同时,SPI的中断程序已经加入等待队列,一旦SPI接口有中 断,SPI的中断服务程序就被唤醒,开始运行。这部分的工作是在系统启动过程中运行的初始化函数来完成
4 ^2 D: W) P8 ~$ n! l% h' e* C( w3 J+ V
的。下面将结合源代码来讨论初始化函数的编写,其 中,就两点进行重点讨论。 8 \0 r; @$ ?- \/ O; n
. C& O: N+ |; w8 u9 E$ \
因 为SCCx的网络参数空间和SPI的参数空间有冲突,如果要在使用SCCx作为网口的同时还使用SPI驱动的话,就要装载microcode,然后重新定 位SPI的参数空间。而micropatch就是装载microcode的一个
/ ^& E: j( d" v: q7 ]' [% |" X5 E7 M
文件,这个文件里的microcode可以到motorola的官方网站上 去下载。 " |* x! o, a0 d* F8 E
+ p5 E' [: D Z$ X$ _8 h- vCPM包括一部分双向RAM口,称为参数RAM,它包括USB、SCC、SMC、SPI、I2C和IDMA信道操作。其中,SPI和I2C参数区域可以被重新定位到另外的32位的参数区域。仔细阅读完下面的代码,就可以很好的# b+ D2 L$ h1 m
2 J2 x8 Q, L1 Y( j( D; z( o/ ?理解这个过程是如何操作的了:
: _9 p5 w, }5 e! |% E/ H$ r$ w. F" K5 t8 F3 F
spi = (spi_t *)&cp->cp_dparam[PROFF_SPI]; printk(" the spi addr is %p\n",spi); if ((reloc = spi->spi_rpbase)) { spi = (spi_t *)&cp->cp_dpmem[spi->spi_rpbase]; printk(" MICROCODE
' t+ @5 y0 Z w" y8 |7 r5 G
4 u8 B5 `5 Y' J7 ^2 }, fRELOCATION PATCH \n"); } 9 P; k& }8 B. q2 P
7 V b0 s- B; k5 D0 q X" I% y w( N上面这一端代码的作用是:首先查询是否已经使用了microcode,然后取得重新定位后的指针(装载microcode和重新定位的操作在microcode.c里完成)。 l; Q9 ~1 y9 Y- |+ _* j, [
9 z9 S" J2 e" @, ?5 d* a
RAM里的SPI描述符
3 O' n# O2 x. o4 A& ~- u" x: }' a: B; z: A
有 关SPI接口的描述符保存在缓冲区里,缓冲区的地址由双向RAM里SPI缓冲区描述符指定。要发送的数据在发送缓冲区里,接收的数据将被存到发送缓冲区 里。缓冲区描述符环路组成一个环路,帮助逐
' w1 q2 |, u; i, g, u) P
; M* u0 S6 j# c# I$ L步传输(接收)想要发送(接收)的数据。正是由于这些缓冲区描述符,通信处理模块才能够完成通信,并且说明并处 理错误。 d8 d- f/ ^8 _- C
4 O2 m2 N# H1 G
可以通过一段代码来看上面示意图的过程是如何在初始化函数里实现的: ; F' E8 p0 R/ P6 s/ W. x" O
4 W: d4 h+ D" G* D) k0 |8 ~9 k3 h+ [
spi->spi_rbase = r_rbase = dp_addr; spi->spi_tbase = r_tbase = dp_addr + sizeof(cbd_t); /*把RXBD RING的地址写入POINTER TO SPI RX RING 把 TXBD RINT 的地址写入POINTER TO SPI TX 4 S1 e- m0 m' k- a2 M0 M
0 M$ T, F% K5 {- h: l4 zRING*/ spi->spi_rbptr = spi->spi_rbase; spi->spi_tbptr = spi->spi_tbase; /*以上的两句代码必须得写,否则的话就会在读写/ tbdf = (cbd_t *)&cp->cp_dpmem[r_tbase]; rbdf = (cbd_t *)
- G0 r" A. z3 s: Y- a& M9 ^9 @, {2 h/ u F" ?1 c: f. p7 Z
&cp->cp_dpmem[r_rbase]; /*从这句代码里可以看出,RXBDRING的地址是在双向RAM里*/ tbdf->cbd_sc &= ~BD_SC_READY; rbdf->cbd_sc &= ~BD_SC_EMPTY; /*设置RING的状态,发送的RING 设置成非准
5 x4 O$ e3 C5 P7 D; p) M) S4 z; u7 {
+ m9 j% [9 a- O- L备发送状态, 接受的RING设置成非准备接受状态*/ rxbuffer = m8xx_cpm_hostalloc(2); txbuffer = m8xx_cpm_hostalloc(2);/*得到两个空间 */ tbdf->cbd_bufaddr = __pa(txbuffer); rbdf-
" x" W0 T9 D' O4 m( X% a- Z( |/ K C/ L
1 K$ `3 M( L! {8 T/ b>cbd_bufaddr = __pa(rxbuffer); /*内存映射;并把DATA POINTER 设置成RX DATA BUFFER 的地址*/
+ l. K/ G8 G: y' W @& @6 z
+ {$ q; m* H4 Y% t7 [以 上的代码是初始化函数里完成的,一旦初始化函数正确运作,就可以采取正确的步骤进行SPI口通信了。以上初始化完之后,要调用 cpm_install_handler函数,该函数的作用是把中断函数注册进内" H& x/ [; f) v* Z
+ o. g. S$ ?) Z' u* T( D核,一旦SPI口产生硬件中断,就调用中断函数,中断函数的编写可以依据 不同系统的不同需要,在本例中,我们使得一旦调用中断函数,就读取SPI接收到的数据。 , h; F0 M' x, \$ o2 n
: c* ~$ H0 n0 {3 t
接下来以如何发送数据为例,分析如何操作SPI口通信。
C5 j( \4 {. i9 K- S' a, t F6 `! b' s$ S4 X; J, Y6 h
发送数据的步骤
/ `3 }" H3 V0 ?) j# K& g7 g+ h4 k% w3 @" S8 F4 l8 s
在此例中,设SPI接口为主模式。为了开始数据传送过程,内核把要传送的数据写到一个数据缓冲区,然后配置缓冲区描述符,以达到传送的目的。以下给出发送数据的一段代码,通过代码解释传输的过
% ^: D) `% v8 m) E5 A/ h" [: g" ^& o$ p7 C6 e
程。 $ u# E. I3 d+ t" g7 f
1 P& I# O, I2 o4 {% C# t% I: g/ z
memset((void*)txbuffer,0,2);/*清空buffer*/ tbdf->cbd_sc = BD_SC_READY | BD_SC_LAST | BD_SC_WRAP; tbdf->cbd_datlen = 2; /*设置发送缓冲区的状态控制寄存器的值和发送数据的个数*/ ) p- ~# l( T1 t) j2 @. q' ]' X) [
& w( m" ~2 h8 N8 J A+ L* P
rbdf->cbd_sc = BD_SC_EMPTY | BD_SC_WRAP; rbdf->cbd_datlen = 0; /*由于并不打算接受数据,所以个数为0*/ cp->cp_spmode = 0x777f; cp->cp_spie = 0xff; cp->cp_spim = 0x37; /*设置SPI接
( m; h' {! P9 L/ h, i% C5 L! g. ]2 t6 t/ W8 }
口寄存器的值,以便发送数据,设置SPI接口的 主或从模式必须在发送函数里设置,否则的话,不能发送数据*/ cp->cp_spcom |= 0x80;/*开始发送数据*/ udelay(1000);/*必须得等待,否则的话不能
4 Q1 x/ x6 X# P* d9 e( C# |: t0 a# f9 t7 f. t; z
正确读到缓冲区状态控制寄存器的值*/ if((tbdf->cbd_sc & 0x8000)) printk(" spi write error !"); memset((void*)rxbuffer,0,2);
1 U! [' U5 i# C4 I3 E7 s W# |9 ]2 g N- |* ]! f2 D& P
数 据通信过程中,最重要的是时序,正确的时序要通过反复的实验才能得到。图3是在实验过程中得到的逻辑图(安捷伦公司的1672G逻辑分析仪测试结果)。其 中,CS是片选信号,CK是时钟信号,DO
% `0 w$ Z, x+ Z% s3 {+ ~4 y
/ o# o' J Z# w1 g5 |8 l是823e发送的数据。可以使用逻辑分析仪来阅读得到的数据是否和设备发送的数据一致。正确的通信必须经过长时 间的调试才能够取得。 $ ]; O; L3 }2 Z p9 p
% Y+ g1 ?5 |, R& m* F' ]在这强调一下,根据ADS7846的使用手册,驱动程序必须在初始化的时候与ADS7846建立通信。所以,823e首先要向ADS7846发送命令,得到ADS7846的回复后建立通信。驱动程序调用SPI的读写函数来实现对ADS7846的操作。( z& s2 t% B0 h4 \3 A" h% }: j) G# f
转自:http://www.3gtarena.com/ |
|