0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

fireflyAIO-3288J主板SPI使用介紹

firefly ? 來源:firefly ? 作者:firefly ? 2019-12-20 09:41 ? 次閱讀
SPI 使用

SPI是一種高速的,全雙工,同步串行通信接口,用于連接微控制器傳感器、存儲(chǔ)設(shè)備等,本文以指紋識(shí)別模塊為例簡(jiǎn)單介紹SPI使用。

SPI工作方式

SPI以主從方式工作,這種模式通常有一個(gè)主設(shè)備和一個(gè)或多個(gè)從設(shè)備,需要至少4根線,分別是:

CS 片選信號(hào) SCLK 時(shí)鐘信號(hào) MOSI 主設(shè)備數(shù)據(jù)輸出、從設(shè)備數(shù)據(jù)輸入 MISO 主設(shè)備數(shù)據(jù)輸入,從設(shè)備數(shù)據(jù)輸出

Linux內(nèi)核用CPOL和CPHA的組合來表示當(dāng)前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:表示時(shí)鐘信號(hào)的初始電平的狀態(tài),0為低電平,1為高電平。CPHA:表示在哪個(gè)時(shí)鐘沿采樣,0為第一個(gè)時(shí)鐘沿采樣,1為第二個(gè)時(shí)鐘沿采樣。SPI的四種工作模式波形圖如下:

在內(nèi)核添加自己的驅(qū)動(dòng)文件

在內(nèi)核源碼目錄kernel/drivers/spi/中創(chuàng)建新的驅(qū)動(dòng)文件,如:spi-rockchip-firefly.c 在驅(qū)動(dòng)文件所在目錄下的Kconfig文件添加對(duì)應(yīng)的驅(qū)動(dòng)文件配置,如:

@@-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 sensors and memory# being probably the most widely used ones.

在驅(qū)動(dòng)文件所在目錄下的Makefile文件添加對(duì)應(yīng)的驅(qū)動(dòng)文件名,如:

+obj-$(CONFIG_SPI_ROCKCHIP_FIREFLY) += spi-rockchip-firefly.o

用make menuconfig在內(nèi)核選項(xiàng)中選中所添加的驅(qū)動(dòng)文件,如:

There is no help available for this option. │ 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]
定義和注冊(cè)SPI設(shè)備

在DTS中添加SPI驅(qū)動(dòng)結(jié)點(diǎn)描述,如下所示: 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,則設(shè)為okay,如不啟用,設(shè)為disable。

  • spidev@00:由于本例子使用的是SPI0,且使用CS0,故此處設(shè)為00,如果使用CS1,則設(shè)為01。

  • compatible:這里的屬性必須與驅(qū)動(dòng)中的結(jié)構(gòu)體of_device_id 中的成員compatible 保持一致。

  • reg:此處與spidev@00保持一致,本例設(shè)為:0x00;

  • spi-max-frequency:此處設(shè)置spi使用的最高頻率。

  • spi-cpha,spi-cpol:SPI的工作模式在此設(shè)置,本例所用的模塊SPI工作模式為SPI_MODE_1,故設(shè):spi-cpha = <1>,如果您所用設(shè)備工作模式為SPI_MODE0,則需在此把這兩個(gè)注釋掉,如果用SPI_MODE3,則設(shè):spi-cpha = <1>;spi-cpol = <1>。

定義和注冊(cè)SPI驅(qū)動(dòng)
定義SPI驅(qū)動(dòng)

在定義 SPI 驅(qū)動(dòng)之前,用戶首先要定義變量 of_device_id 。 of_device_id 用于在驅(qū)動(dòng)中調(diào)用dts文件中定義的設(shè)備信息,其定義如下所示:

staticconststructof_device_idspidev_dt_ids[]={{.compatible="rockchip,spi_firefly"},{},};

此處的compatible與DTS文件中的保持一致。 定義spi_driver如下所示:

staticstructspi_driverspidev_spi_driver={.driver={.name="silead_fp",.owner=THIS_MODULE,.of_match_table=of_match_ptr(spidev_dt_ids),},.probe=spi_gsl_probe,.remove=spi_gsl_remove,};
注冊(cè)SPI驅(qū)動(dòng)

在初始化函數(shù)static int __init spidev_init(void)中創(chuàng)建一個(gè)字符設(shè)備:

alloc_chrdev_region(&devno,0,255,"sileadfp");

向內(nèi)核添加該設(shè)備:

spidev_major=MAJOR(devno);cdev_init(&spicdev,&spidev_fops);spicdev.owner=THIS_MODULE;status=cdev_add(&spicdev,MKDEV(spidev_major,0),N_SPI_MINORS);

創(chuàng)建設(shè)備類:

class_create(THIS_MODULE,"spidev");

向內(nèi)核注冊(cè)SPI驅(qū)動(dòng):

spi_register_driver(&spidev_spi_driver);

如果內(nèi)核啟動(dòng)時(shí)匹配成功,則調(diào)用該驅(qū)動(dòng)的probe函數(shù)。 probe函數(shù)如下所示:

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);//初始化設(shè)備鏈表 //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);//在內(nèi)存區(qū)中查找第一個(gè)值為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");創(chuàng)建/dev/下設(shè)備結(jié)點(diǎn) 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);//添加進(jìn)設(shè)備鏈表 } 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; }

如果注冊(cè)SPI驅(qū)動(dòng)成功,你可以在/dev/目錄下面看你到注冊(cè)的驅(qū)動(dòng)名稱,可以在sys/class/下面看到你注冊(cè)的驅(qū)動(dòng)設(shè)備類。

SPI讀寫數(shù)據(jù)過程
SPI寫數(shù)據(jù)
staticssize_tspidev_write(structfile*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);//把數(shù)據(jù)從用戶空間傳到內(nèi)核空間if(missing==0){status=spidev_sync_write(spidev,count);//調(diào)用寫同步函數(shù)}elsestatus=-EFAULT;mutex_unlock(&spidev->buf_lock);returnstatus;}

寫同步函數(shù):

spidev_sync_write(structspidev_data*spidev,size_tlen){structspi_transfert={.tx_buf=spidev->buffer,//發(fā)送緩沖區(qū).len=len,//發(fā)送數(shù)據(jù)長(zhǎng)度};structspi_messagem;spi_message_init(&m);//初始化spi_messagespi_message_add_tail(&t,&m);//將新的spi_transfer添加到spi_message隊(duì)列尾部returnspidev_sync(spidev,&m);//同步讀寫}
SPI讀數(shù)據(jù)

在本例所用的模塊中,讀數(shù)據(jù)的過程為:

  • 主機(jī)向模塊寫寄存器的地址及讀的指令(如:地址為0xf0,讀指令為0x00)

  • 模塊收到讀的指令后,數(shù)據(jù)以頁的形式發(fā)送

  • 主機(jī)設(shè)置讀的模式

  • 主機(jī)讀取一頁數(shù)據(jù)并存儲(chǔ)

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頁數(shù)據(jù)}pos=0;break;}if(status>0){printk("gsl read data success!!!\n");}else{printk("gsl read data failed!!!");}mutex_unlock(&spidev->buf_lock);returnstatus;}
staticinlineunsignedintgsl_fp_getOneFrame(structspidev_data*spidev,unsignedcharreg_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;//每次讀的數(shù)據(jù)長(zhǎng)度為8位t->delay_usecs=1;//每次讀完延時(shí)t->speed_hz=14*1000*1000;//讀的速率t->cs_change=1;//CS引腳電平變化spi_message_add_tail(t,&m);returnspidev_sync(spidev,&m);}

注:Firefly的SPI驅(qū)動(dòng)是Linux下通用的驅(qū)動(dòng),可以參考源碼:kernel/drivers/spi/spidev.c


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11171

    瀏覽量

    208475
  • 嵌入式主板
    +關(guān)注

    關(guān)注

    7

    文章

    6083

    瀏覽量

    35098
  • Firefly
    +關(guān)注

    關(guān)注

    2

    文章

    538

    瀏覽量

    6946
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    fireflyAIO-3288C主板SPI接口簡(jiǎn)介

    SPI是一種高速的,全雙工,同步串行通信接口,用于連接微控制器、傳感器、存儲(chǔ)設(shè)備等,本文以指紋識(shí)別模塊為例簡(jiǎn)單介紹SPI使用。
    的頭像 發(fā)表于 12-16 14:11 ?1187次閱讀
    <b class='flag-5'>fireflyAIO-3288</b>C<b class='flag-5'>主板</b><b class='flag-5'>SPI</b>接口簡(jiǎn)介

    fireflyAIO-3288J主板UART使用簡(jiǎn)介

    AIO-3288J 支持SPI橋接/擴(kuò)展4個(gè)增強(qiáng)功能串口(UART)的功能,分別為UART2,RS232(上),RS485,UART3和3個(gè)主控自帶的串口,分別為UART1,RS232(下)和調(diào)試串口。
    的頭像 發(fā)表于 12-20 09:40 ?1630次閱讀
    <b class='flag-5'>fireflyAIO-3288J</b><b class='flag-5'>主板</b>UART使用簡(jiǎn)介

    fireflyAIO-3288J主板PWM輸出簡(jiǎn)介

    AIO-3288J 開發(fā)板上有 4 路 PWM 輸出,分別為 PWM0 ~ PWM3, 本章主要描述如何配置 PWM。
    的頭像 發(fā)表于 12-20 09:51 ?1771次閱讀

    fireflyAIO-3288J主板MIPI CSI介紹

    AIO-3288J開發(fā)板有兩個(gè)版本,一個(gè)版本是帶有雙MIPI 攝像頭接口,另一個(gè)版本是帶單MIPI攝像頭接口+HDMIIN,MIPI攝像頭圖像處理能力達(dá)到 4416x3312 像素,支持 4K 視頻錄制。此外,開發(fā)板還支持 USB 攝像頭。
    的頭像 發(fā)表于 12-20 09:43 ?2796次閱讀
    <b class='flag-5'>fireflyAIO-3288J</b><b class='flag-5'>主板</b>MIPI CSI<b class='flag-5'>介紹</b>

    fireflyAIO-3288J主板LED介紹

    AIO-3288J 開發(fā)板上有 2 個(gè) LED 燈,
    的頭像 發(fā)表于 12-20 09:48 ?1957次閱讀

    fireflyAIO-3288J主板IR使用介紹

    IR 使用 紅外遙控配置 AIO-3288J 開發(fā)板上可以接紅外收發(fā)傳感器 IR 實(shí)現(xiàn)遙控功能。
    的頭像 發(fā)表于 12-20 10:02 ?1807次閱讀
    <b class='flag-5'>fireflyAIO-3288J</b><b class='flag-5'>主板</b>IR使用<b class='flag-5'>介紹</b>

    fireflyAIO-3288J主板I2C簡(jiǎn)介

    AIO-3288J 開發(fā)板上有 6 個(gè)片上 I2C 控制器。
    的頭像 發(fā)表于 12-20 10:05 ?1287次閱讀

    fireflyAIO-3288J主板GPIO使用介紹

    firefly
    的頭像 發(fā)表于 12-20 10:04 ?1302次閱讀
    <b class='flag-5'>fireflyAIO-3288J</b><b class='flag-5'>主板</b>GPIO使用<b class='flag-5'>介紹</b>

    fireflyAIO-3288J接口定義介紹

    接口定義 整機(jī)接口定義
    的頭像 發(fā)表于 12-25 16:35 ?1470次閱讀
    <b class='flag-5'>fireflyAIO-3288J</b>接口定義<b class='flag-5'>介紹</b>

    fireflyAIO-3288J方案

    AIO-3288J HDMI輸出及HDMI輸入 AIO-3288J底板上有兩個(gè)HDMI接口,其中一個(gè)為HDMI輸出接口(在USB口下方),另一個(gè)為HDMI輸入接口,
    的頭像 發(fā)表于 12-25 16:37 ?1660次閱讀
    <b class='flag-5'>fireflyAIO-3288J</b>方案

    fireflyAIO-3288J紅外遙控器介紹

    紅外遙控器 12鍵紅外遙控器 產(chǎn)品參數(shù) 產(chǎn)品:12鍵紅外遙控器 版本:Firefly定制版 電源:兩節(jié)7號(hào)電池 適配:AIO-3288J
    的頭像 發(fā)表于 12-25 16:39 ?1667次閱讀

    fireflyAIO-3288J主板ADC使用簡(jiǎn)介

    AIO-3288J 開發(fā)板上的 AD 接口分為:高速 ADC 流接口 (High-speed ADC Stream Interface)、溫度傳感器 (Temperature Sensor)、
    的頭像 發(fā)表于 12-26 14:34 ?1712次閱讀

    fireflyAIO-3288J主板編譯Android固件簡(jiǎn)介

    編譯 Android 固件 準(zhǔn)備工作 編譯 Android 對(duì)機(jī)器的配置要求較高: 64 位 CPU 16GB 物理內(nèi)存+交換內(nèi)存 30GB 空閑的磁盤空間用于構(gòu)建,源碼樹另外占用大約 25GB
    的頭像 發(fā)表于 12-26 14:44 ?1317次閱讀

    fireflyAIO-3288J主板ADB使用介紹

    adb,全稱 Android Debug Bridge,是 Android 的命令行調(diào)試工具,可以完成多種功能,如跟蹤系統(tǒng)日志,上傳下載文件,安裝應(yīng)用等。
    的頭像 發(fā)表于 12-26 14:48 ?1696次閱讀

    fireflyAIO-3288J主板啟動(dòng)模式介紹

    AIO-3288J 有靈活的啟動(dòng)方式。一般情況下,除非硬件損壞,AIO-3288J 開發(fā)板是不會(huì)變磚的。
    的頭像 發(fā)表于 12-26 15:00 ?1957次閱讀