当前位置:首页 > 单片机 > 单片机
[导读]本文属于第四部分。7. write,read和ioctl综述 在spi设备驱动层提供了两种数据传输方式。一种是半双工方式,write方法提供了半双工读访问,read方法提供了半双工写访问。另一种就是全双工方式,ioctl调用将同时完成数

本文属于第四部分。

7. write,read和ioctl综述

在spi设备驱动层提供了两种数据传输方式。一种是半双工方式,write方法提供了半双工读访问,read方法提供了半双工写访问。另一种就是全双工方式,ioctl调用将同时完成数据的传送与发送。

在后面的描述中,我们将对write和ioctl方法做出详细的描述,而read方法和write极其相似,将不多做介绍。

接下来首先看看write方法是如何实现的。

8. write方法

8.1 spidev_write

在用户空间执行open打开设备文件以后,就可以执行write系统调用,该系统调用将会执行我们提供的write方法。代码如下:

下列代码位于drivers/spi/spidev.c中。


/*Write-onlymessagewithcurrentdevicesetup*/

staticssize_t

spidev_write(structfile*filp,constchar__user*buf,

size_tcount,loff_t*f_pos)

{

structspidev_data*spidev;

ssize_tstatus=0;

unsignedlongmissing;

/*chipselectonlytogglesatstartorendofoperation*/

if(count>bufsiz)/*数据大于4096字节*/

return-EMSGSIZE;

spidev=filp->private_data;

mutex_lock(&spidev->buf_lock);

/*将用户层的数据拷贝至buffer中,buffer在open方法中分配*/

missing=copy_from_user(spidev->buffer,buf,count);

if(missing==0){

status=spidev_sync_write(spidev,count);

}else

status=-EFAULT;

mutex_unlock(&spidev->buf_lock);

returnstatus;

}

在这里,做的事情很少,主要就是从用户空间将需要发送的数据复制过来。然后调用spidev_sync_write。


8.2 spidev_sync_write

下列代码位于drivers/spi/spidev.c中。


staticinlinessize_t

spidev_sync_write(structspidev_data*spidev,size_tlen)

{

structspi_transfert={

.tx_buf=spidev->buffer,

.len=len,

};

structspi_messagem;

spi_message_init(&m);

spi_message_add_tail(&t,&m);

returnspidev_sync(spidev,&m);

}

staticinlinevoidspi_message_init(structspi_message*m)

{

memset(m,0,sizeof*m);

INIT_LIST_HEAD(&m->transfers);/*初始化链表头*/

}

spi_message_add_tail(structspi_transfer*t,structspi_message*m)

{

list_add_tail(&t->transfer_list,&m->transfers);/*添加transfer_list*/

}

在这里,创建了transfer和message。spi_transfer包含了要发送数据的信息。然后初始化了message中的transfer链表头,并将spi_transfer添加到了transfer链表中。也就是以spi_message的transfers为链表头的链表中,包含了transfer,而transfer正好包含了需要发送的数据。由此可见message其实是对transfer的封装。
最后,调用了spidev_sync,并将创建的spi_message作为参数传入。


8.3 spidev_sync

下列代码位于drivers/spi/spidev.c中。


staticssize_t

spidev_sync(structspidev_data*spidev,structspi_message*message)

{

DECLARE_COMPLETION_ONSTACK(done);/*创建completion*/

intstatus;

message->complete=spidev_complete;/*定义complete方法*/

message->context=&done;/*complete方法的参数*/

spin_lock_irq(&spidev->spi_lock);

if(spidev->spi==NULL)

status=-ESHUTDOWN;

else

status=spi_async(spidev->spi,message);/*异步,用complete来完成同步*/

spin_unlock_irq(&spidev->spi_lock);

if(status==0){

wait_for_completion(&done);/*在bitbang_work中调用complete方法来唤醒*/

status=message->status;

if(status==0)

status=message->actual_length;/*返回发送的字节数*/

}

returnstatus;

}

在这里,初始化了completion,这个东东将实现write系统调用的同步。在后面我们将会看到如何实现的。


随后调用了spi_async,从名字上可以看出该函数是异步的,也就是说该函数返回后,数据并没有被发送出去。因此使用了wait_for_completion来等待数据的发送完成,达到同步的目的。

8.4 spi_async

下列代码位于drivers/spi/spi.h中。


/**

*spi_async-asynchronousSPItransfer

*@spi:devicewithwhichdatawillbeexchanged

*@message:describesthedatatransfers,includingcompletioncallback

*Context:any(irqsmaybeblocked,etc)

*

*Thiscallmaybeusedin_irqandothercontextswhichcan'tsleep,

*aswellasfromtaskcontextswhichcansleep.

*

*Thecompletioncallbackisinvokedinacontextwhichcan'tsleep.

*Beforethatinvocation,thevalueofmessage->statusisundefined.

*Whenthecallbackisissued,message->statusholdseitherzero(to

*indicatecompletesuccess)oranegativeerrorcode.Afterthat

*callbackreturns,thedriverwhichissuedthetransferrequestmay

*deallocatetheassociatedmemory;it'snolongerinusebyanySPI

*coreorcontrollerdrivercode.

*

*Notethatalthoughallmessagestoaspi_devicearehandledin

*FIFOorder,messagesmaygotodifferentdevicesinotherorders.

*Somedevicemightbehigherpriority,orhavevarious"hard"access

*timerequirements,forexample.

*

*Ondetectionofanyfaultduringthetransfer,processingof

*theentiremessageisaborted,andthedeviceisdeselected.

*Untilreturningfromtheassociatedmessagecompletioncallback,

*nootherspi_messagequeuedtothatdevicewillbeprocessed.

*(Thisruleappliesequallytoallthesynchronoustransfercalls,

*whicharewrappersaroundthiscoreasynchronousprimitive.)

*/

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

SPI 是英语Serial Peripheral interface的缩写,顾名思义就是串行外围设备接口。是Motorola(摩托罗拉)首先在其MC68HCXX系列处理器上定义的。SPI是一种单主机、高速的,全双工,同步...

关键字: spi spi通信原理

为什么要加锁在SMP系统中,如果仅仅是需要串行地增加一个变量的值,那么使用原子操作的函数(API)就可以了。但现实中更多的场景并不会那么简单,比如需要将一个结构体A中的数据提取出来,然后格式化、解析,再添加到另一个结构体...

关键字: spi ic ck

为什么要加锁在SMP系统中,如果仅仅是需要串行地增加一个变量的值,那么使用原子操作的函数(API)就可以了。但现实中更多的场景并不会那么简单,比如需要将一个结构体A中的数据提取出来,然后格式化、解析,再添加到另一个结构体...

关键字: spi ic ck

SPI接口通信原理

关键字: spi 通讯

点击上方名片关注我们朱老师推荐语:此岗位为AIoT终身成长大会员同学提供的自己公司的岗位内推,总部在深圳,是一家专业从事闭路电视监控设备、会议摄像机的研发、制造、销售的高科技企业,有学过嵌入式课程或者海思项目的同学,想换...

关键字: 开发工程师 linux驱动 驱动开发

在嵌入式系统开发中,经常通过键盘来实现人机交互。本文介绍了一种直接利用ARM的I/O口扩展矩阵键盘的方法。同时以TQ2440开发板为例,对硬件电路连接和相应的linux驱动设计方法都作了详细说明。

关键字: ARM 矩阵键盘 linux驱动

通常在以往接触的Linux驱动,没遇到使用电池供电的情况,因此几乎没关注电源的管理。然而实际中,不少使用电池供电的硬件平台,例如手机、POS机等,就需要对电源进行管理,比如在不使用设备的时候,休眠屏幕省电。

关键字: 电源管理 嵌入式基础 linux驱动

  本文根据网络视频采集的需要,将网络传输与视频采集相结合,设计了以S3C2440为核心的USB摄像头视频采集和嵌入式Linux系统下的视频服务器,从而实现了远程网络视频信息采集。   

关键字: s3c2440 视频采集 usb摄像头

         之前在提起自动化或是智能化时,人们会不自觉的想到工业生产,这是因为自动化这个字眼进入中国,确实是以工业

关键字: 嵌入式 Linux s3c2440 视频采集

Linux 点击上方蓝字 记得关注我们哦! 内核里已经提供spi接口小屏的设备驱动,在内核的配置选项: make menuconfig ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf...

关键字: spi 内核
关闭
关闭