SPI 使用
SPI是一种高速的,全双工,同步串行通信接口,用于连接微控制器、传感器、存储设备等,本文以指纹识别模块为例简单介绍SPI使用。
SPI工作方式
SPI以主从方式工作,这种模式通常有一个主设备和一个或多个从设备,需要至少4根线,分别是:
CS 片选
信号 SCLK
时钟信号 MOSI 主设备数据输出、从设备数据输入 MISO 主设备数据输入,从设备数据输出
Linux内核用CPOL和CPHA的组合来表示当前SPI的四种工作模式:
CPOL=0,CPHA=0 SPI_MODE_0 CPOL=0,CPHA=1 SPI_MODE_1 CPOL=1,CPHA=0 SPI_MODE_2 CPOL=1,CPHA=1 SPI_MODE_3
CPOL:表示时钟信号的初始电平的状态,0为低电平,1为高电平。CPHA:表示在哪个时钟沿采样,0为第一个时钟沿采样,1为第二个时钟沿采样。SPI的四种工作模式波形图如下:
![](http://file.elecfans.com/web1/M00/AB/4C/o4YBAF222kyAUEsPAAEGv54ax4g336.jpg)
在内核添加自己的驱动文件
在内核源码目录kernel/drivers/spi/中创建新的驱动文件,如:spi-rockchip-firefly.c 在驱动文件所在目录下的Kconfig文件添加对应的驱动文件配置,如:
@@-525,6+525,10@@configSPI_ROCKCHIP_
TESTbool"ROCKCHIP spi test code"dependsonSPI_ROCKCHIP+configSPI_ROCKCHIP_FIREFLY+bool"ROCKCHIP spi firefly code"+dependsonSPI_ROCKCHIP+## There are lots of SPI device types, with sens
ors and memory# being probably the most widely used ones.
在驱动文件所在目录下的Makefile文件添加对应的驱动文件名,如:
+obj-$(CONFIG_SPI_ROCKCHIP_FIREFLY) += spi-rockchip-firefly.o
用make menuconfig在内核选项中选中所添加的驱动文件,如:
There is no help av
ailable for this op
tion. │ Symbol: SPI_ROCKCHIP_FIREFLY [=y] │ Type : boolean │ Prompt: ROCKCHIP spi firefly code │ Location: │ -> Device Drivers │ -> SPI support (SPI [=y]) │ -> ROCKCHIP SPI controller core support (SPI_ROCKCHIP_CORE [=y]) │ -> ROCKCHIP SPI interface driver (SPI_ROCKCHIP [=y]) │ Defined at drivers/spi/Kconfig:528 │ Depends on: SPI [=y] && SPI_MASTER [=y] && SPI_ROCKCHIP [=y]
定义和注册SPI设备
在DTS中添加SPI驱动结点描述,如下所示: kernel/arch/arm/boot/dts/firefly-rk3288-aio-3288j.dts
&spi0{status="okay";max-freq=<24000000>;spidev@00{compatible="rockchip,spi_firefly";reg=<0x00>;spi-max-frequency=<14000000>;spi-cpha=<1>;//spi-cpol=<1>;};};
-
status:如果要启用SPI,则设为okay,如不启用,设为disable。
-
spidev@00:由于本例子使用的是SPI0,且使用CS0,故此处设为00,如果使用CS1,则设为01。
-
compatible:这里的属性必须与驱动中的结构体of_device_id 中的成员compatible 保持一致。
-
reg:此处与spidev@00保持一致,本例设为:0x00;
-
spi-max-frequency:此处设置spi使用的最高频率。
-
spi-cpha,spi-cpol:SPI的工作模式在此设置,本例所用的模块SPI工作模式为SPI_MODE_1,故设:spi-cpha = <1>,如果您所用设备工作模式为SPI_MODE0,则需在此把这两个注释掉,如果用SPI_MODE3,则设:spi-cpha = <1>;spi-cpol = <1>。
定义和注册SPI驱动
定义SPI驱动
在定义 SPI 驱动之前,用户首先要定义变量 of_device_id 。 of_device_id 用于在驱动中调用dts文件中定义的设备信息,其定义如下所示:
stat
icconststructof_device_idspidev_dt_ids[]={{.compatible="rockchip,spi_firefly"},{},};
此处的compatible与DTS文件中的保持一致。 定义spi_driver如下所示:
staticstructs
pi_driverspidev_spi_driver={.driver={.name="sile
ad_fp",.owner=THIS_MODULE,.of_match_table=of_match_ptr(spidev_dt_ids),},.probe=spi_gsl_probe,.remove=spi_gsl_remove,};
注册SPI驱动
在初始化函数static int __init spidev_init(void)中创建一个字符设备:
alloc_chrdev_region(&devno,0,255,"sileadfp");
向内核添加该设备:
spidev_major=MAJOR(devno);cdev_init(&spicdev,&spidev_fops);spicdev.owner=THIS_MODULE;status=cdev_add(&spicdev,MKDEV(spidev_major,0),N_SPI_MINO
RS);
创建设备类:
class_create(THIS_MODULE,"spidev");
向内核注册SPI驱动:
spi_register_driver(&spidev_spi_driver);
如果内核启动时匹配成功,则调用该驱动的probe函数。 probe函数如下所示:
static int spi_gsl_probe(struct spi_device *spi) { struct spidev_data *spidev; int status; unsigned long minor; struct gsl_fp_data *fp_data; printk("===============spi_gsl_probe ==============\n"); if(!spi) return -ENOMEM; /* Allocate driver data */ spidev = kzalloc(sizeof(*spidev), GFP_KERNEL); if (!spidev) return -ENOMEM; /* Initialize the driver data */ spidev->spi = spi; spin_lock_init(&spidev->spi_lock);//初始化自旋锁 mutex_init(&spidev->buf_lock);//初始化互斥锁 INIT_LIST_HEAD(&spidev->device_entry);//初始化设备链表 //init fp_data fp_data = kzalloc(sizeof(struct gsl_fp_data), GFP_KERNEL); if(fp_data == NULL){ status = -ENOMEM; return status; } //set fp_data struct value fp_data->spidev = spidev; mutex_lock(&device_list_lock);//上互斥锁 minor = find_first_zero_bit(minors, N_SPI_MINORS);//在内存区中查找第一个值为0的位 if (minor < N_SPI_MINORS) { struct device *dev; spidev->devt = MKDEV(spidev_major, minor); dev = device_create(spidev_class, &spi->dev, spidev->devt, spidev, "silead_fp_dev");创建/dev/下设备结点 status = IS_ERR(dev) ? PTR_ERR(dev) : 0; } else { dev_dbg(&spi->dev, "no minor number available!\n"); status = -ENODEV; } if (status == 0) { set_bit(minor, minors); list_add(&spidev->device_entry, &device_list);//添加进设备链表 } mutex_unlock(&device_list_lock);//解互斥锁 if (status == 0) spi_set_drvdata(spi, spidev); else kfree(spidev); printk("%s:name=%s,bus_num=%d,cs=%d,mode=%d,speed=%d\n",__func__,spi->modalias, spi->master->bus_num, spi->chip_select, spi->mode, spi->max_speed_hz);//打印SPI信息 return status; }
如果注册SPI驱动成功,你可以在/dev/目录下面看你到注册的驱动名称,可以在sys/class/下面看到你注册的驱动设备类。
SPI读写数据过程
SPI写数据
staticssize_tspidev_write(struc
tfile*filp,constchar__user*buf,size_tcount,loff_t*f_pos){structspidev_data*spidev;ssize_tstatus=0;unsignedlongmissing;if(count>bufsiz)return-EMSGSIZE;spidev=filp->private_data;mutex_lock(&spidev->buf_lock);missing=copy_from_user(spidev->buffer,buf,count);//把数据从用户空间传到内核空间if(missing==0){status=spidev_sync_write(spidev,count);//调用写同步函数}elsestatus=-EFAULT;mutex_unlock(&spidev->buf_lock);returnstatus;}
写同步函数:
spidev_sync_write(structspidev_data*spidev,size_tlen){structspi_transfert={.tx_buf=spidev->buffer,//发送缓冲区.len=len,//发送数据长度};structspi_messagem;spi_message_init(&m);//初始化spi_messagespi_message_add_tail(&t,&m);//将新的spi_transfer添加到spi_message队列尾部returnspidev_sync(spidev,&m);//同步读写}
SPI读数据
在本例所用的模块中,读数据的过程为:
staticssize_tspidev_read(structfile*filp,char__user*buf,size_tcount,loff_t*f_pos){structspidev_data*spidev;intstatus=0;inti=0;spidev=filp->private_data;mutex_lock(&spidev->buf_lock);gsl_fp_write(spidev,0x00,0xf0);//读之前先向模块写读的指令及寄存器地址while(1){for(i=0;i<=110*118/128/read_pages;i++){status=gsl_fp_getOneFrame(spidev,0x00);//读1页数据}pos=0;break;}if(status>0){printk("gsl read data success!!!\n");}else{printk("gsl read data fai
led!!!");}mutex_unlock(&spidev->buf_lock);returnstatus;}
staticinlineunsignedintgsl_fp_getOneFrame(structspidev_data*spidev,unsigne
dcharreg_8b){intstatus,i;unsignedcharbuf_d[128*1+3]={0x00,0x00};structspi_transfert;t.tx_buf=buf_d;t.rx_buf=buf_d;t.len=131;status=gsl_spidev_sync_read(spidev,&t);if(status>0){for(i=0;i<128*read_pages;i++)kmalloc_area[pos++]=buf_d[i+3];}returnstatus;}
staticinlinessize_tgsl_spidev_sync_read(structspidev_data*spidev,structspi_transfer*t){structspi_messagem;spi_message_init(&m);t->bits_per_word=8;//每次读的数据长度为8位t->delay_usecs=1;//每次读完延时t->speed_hz=14*1000*1000;//读的速率t->cs_change=1;//CS引脚电平变化spi_message_add_tail(t,&m);returnspidev_sync(spidev,&m);}
注:Firefly的SPI驱动是Linux下通用的驱动,可以参考源码:kernel/drivers/spi/spidev.c
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
相关推荐
SPI是一种高速的,全双工,同步串行通信接口,用于连接微控制器、传感器、存储设备等,本文以指纹识别模块为例简单介绍SPI使用。
发表于 12-16 14:11
•1311次阅读
AIO-3288J 支持SPI桥接/扩展4个增强功能串口(UART)的功能,分别为UART2,RS232(上),RS485,UART3和3个主控自带的串口,分别为UART1,RS232(下)和调试串口。
发表于 12-20 09:40
•1813次阅读
AIO-3288J 开发板上有 4 路 PWM 输出,分别为 PWM0 ~ PWM3, 本章主要描述如何配置 PWM。
发表于 12-20 09:51
•1886次阅读
AIO-3288J开发板有两个版本,一个版本是带有双MIPI 摄像头接口,另一个版本是带单MIPI摄像头接口+HDMIIN,MIPI摄像头图像处理能力达到 4416x3312 像素,支持 4K 视频录制。此外,开发板还支持 USB 摄像头。
发表于 12-20 09:43
•3016次阅读
AIO-3288J 开发板上有 2 个 LED 灯,
发表于 12-20 09:48
•2073次阅读
IR 使用
红外遥控配置
AIO-3288J 开发板上可以接红外收发传感器 IR 实现遥控功能。
发表于 12-20 10:02
•1998次阅读
AIO-3288J 开发板上有 6 个片上 I2C 控制器。
发表于 12-20 10:05
•1420次阅读
firefly
发表于 12-20 10:04
•1398次阅读
接口定义
整机接口定义
发表于 12-25 16:35
•1578次阅读
AIO-3288J HDMI输出及HDMI输入
AIO-3288J底板上有两个HDMI接口,其中一个为HDMI输出接口(在USB口下方),另一个为HDMI输入接口,
发表于 12-25 16:37
•1815次阅读
红外遥控器
12键红外遥控器
产品参数
产品:12键红外遥控器
版本:Firefly定制版
电源:两节7号电池
适配:AIO-3288J
发表于 12-25 16:39
•1784次阅读
AIO-3288J 开发板上的 AD 接口分为:高速 ADC 流接口 (High-speed ADC Stream Interface)、温度传感器 (Temperature Sensor)、
发表于 12-26 14:34
•1874次阅读
编译 Android 固件
准备工作
编译 Android 对机器的配置要求较高:
64 位 CPU
16GB 物理内存+交换内存
30GB 空闲的磁盘空间用于构建,源码树另外占用大约 25GB
发表于 12-26 14:44
•1411次阅读
adb,全称 Android Debug Bridge,是 Android 的命令行调试工具,可以完成多种功能,如跟踪系统日志,上传下载文件,安装应用等。
发表于 12-26 14:48
•1824次阅读
AIO-3288J 有灵活的启动方式。一般情况下,除非硬件损坏,AIO-3288J 开发板是不会变砖的。
发表于 12-26 15:00
•2089次阅读
评论