日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > 單片機 > 單片機
[導讀]本文屬于第四部分。7. write,read和ioctl綜述 在spi設備驅動層提供了兩種數據傳輸方式。一種是半雙工方式,write方法提供了半雙工讀訪問,read方法提供了半雙工寫訪問。另一種就是全雙工方式,ioctl調用將同時完成數

本文屬于第四部分。

7. write,read和ioctl綜述

在spi設備驅動層提供了兩種數據傳輸方式。一種是半雙工方式,write方法提供了半雙工讀訪問,read方法提供了半雙工寫訪問。另一種就是全雙工方式,ioctl調用將同時完成數據的傳送與發(fā)送。

在后面的描述中,我們將對write和ioctl方法做出詳細的描述,而read方法和write極其相似,將不多做介紹。

接下來首先看看write方法是如何實現的。

8. write方法

8.1 spidev_write

在用戶空間執(zhí)行open打開設備文件以后,就可以執(zhí)行write系統(tǒng)調用,該系統(tǒng)調用將會執(zhí)行我們提供的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字節(jié)*/

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;

}

在這里,做的事情很少,主要就是從用戶空間將需要發(fā)送的數據復制過來。然后調用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*/

}

在這里,創(chuàng)建了transfer和message。spi_transfer包含了要發(fā)送數據的信息。然后初始化了message中的transfer鏈表頭,并將spi_transfer添加到了transfer鏈表中。也就是以spi_message的transfers為鏈表頭的鏈表中,包含了transfer,而transfer正好包含了需要發(fā)送的數據。由此可見message其實是對transfer的封裝。
最后,調用了spidev_sync,并將創(chuàng)建的spi_message作為參數傳入。


8.3 spidev_sync

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


staticssize_t

spidev_sync(structspidev_data*spidev,structspi_message*message)

{

DECLARE_COMPLETION_ONSTACK(done);/*創(chuàng)建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;/*返回發(fā)送的字節(jié)數*/

}

returnstatus;

}

在這里,初始化了completion,這個東東將實現write系統(tǒng)調用的同步。在后面我們將會看到如何實現的。


隨后調用了spi_async,從名字上可以看出該函數是異步的,也就是說該函數返回后,數據并沒有被發(fā)送出去。因此使用了wait_for_completion來等待數據的發(fā)送完成,達到同步的目的。

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.)

*/

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯系該專欄作者,如若文章內容侵犯您的權益,請及時聯系本站刪除。
換一批
延伸閱讀

SPI 是英語Serial Peripheral interface的縮寫,顧名思義就是串行外圍設備接口。是Motorola(摩托羅拉)首先在其MC68HCXX系列處理器上定義的。SPI是一種單主機、高速的,全雙工,同步...

關鍵字: spi spi通信原理

為什么要加鎖在SMP系統(tǒng)中,如果僅僅是需要串行地增加一個變量的值,那么使用原子操作的函數(API)就可以了。但現實中更多的場景并不會那么簡單,比如需要將一個結構體A中的數據提取出來,然后格式化、解析,再添加到另一個結構體...

關鍵字: spi ic ck

為什么要加鎖在SMP系統(tǒng)中,如果僅僅是需要串行地增加一個變量的值,那么使用原子操作的函數(API)就可以了。但現實中更多的場景并不會那么簡單,比如需要將一個結構體A中的數據提取出來,然后格式化、解析,再添加到另一個結構體...

關鍵字: spi ic ck

SPI接口通信原理

關鍵字: spi 通訊

點擊上方名片關注我們朱老師推薦語:此崗位為AIoT終身成長大會員同學提供的自己公司的崗位內推,總部在深圳,是一家專業(yè)從事閉路電視監(jiān)控設備、會議攝像機的研發(fā)、制造、銷售的高科技企業(yè),有學過嵌入式課程或者海思項目的同學,想換...

關鍵字: 開發(fā)工程師 linux驅動 驅動開發(fā)

在嵌入式系統(tǒng)開發(fā)中,經常通過鍵盤來實現人機交互。本文介紹了一種直接利用ARM的I/O口擴展矩陣鍵盤的方法。同時以TQ2440開發(fā)板為例,對硬件電路連接和相應的linux驅動設計方法都作了詳細說明。

關鍵字: ARM 矩陣鍵盤 linux驅動

通常在以往接觸的Linux驅動,沒遇到使用電池供電的情況,因此幾乎沒關注電源的管理。然而實際中,不少使用電池供電的硬件平臺,例如手機、POS機等,就需要對電源進行管理,比如在不使用設備的時候,休眠屏幕省電。

關鍵字: 電源管理 嵌入式基礎 linux驅動

  本文根據網絡視頻采集的需要,將網絡傳輸與視頻采集相結合,設計了以S3C2440為核心的USB攝像頭視頻采集和嵌入式Linux系統(tǒng)下的視頻服務器,從而實現了遠程網絡視頻信息采集。   

關鍵字: s3c2440 視頻采集 usb攝像頭

         之前在提起自動化或是智能化時,人們會不自覺的想到工業(yè)生產,這是因為自動化這個字眼進入中國,確實是以工業(yè)

關鍵字: 嵌入式 Linux s3c2440 視頻采集

Linux 點擊上方藍字 記得關注我們哦! 內核里已經提供spi接口小屏的設備驅動,在內核的配置選項: make?menuconfig?ARCH=arm?CROSS_COMPILE=arm-linux-gnueabihf...

關鍵字: spi 內核
關閉