在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

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

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

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

微雪電子AIO-3128C主板SPI使用介紹

firefly ? 來(lái)源:微雪電子 ? 作者:微雪電子 ? 2019-12-23 16:14 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

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的組合來(lái)表示當(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 @@ config SPI_ROCKCHIP_TEST bool "ROCKCHIP spi test code" depends on SPI_ROCKCHIP +config SPI_ROCKCHIP_FIREFLY + bool "ROCKCHIP spi firefly code" + depends on SPI_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/aio-3128c.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è)備信息,其定義如下所示:

static const struct of_device_id spidev_dt_ids[] = { { .compatible = "rockchip,spi_firefly" }, {}, };

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

static struct spi_driver spidev_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讀寫(xiě)數(shù)據(jù)過(guò)程

SPI寫(xiě)數(shù)據(jù)

static ssize_t spidev_write(struct file *filp, const char __user *buf, size_t count, loff_t *f_pos) { struct spidev_data *spidev; ssize_t status = 0; unsigned long missing; 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)用寫(xiě)同步函數(shù) } else status = -EFAULT; mutex_unlock(&spidev->buf_lock); return status; }

寫(xiě)同步函數(shù):

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

SPI讀數(shù)據(jù)

在本例所用的模塊中,讀數(shù)據(jù)的過(guò)程為:主機(jī)向模塊寫(xiě)寄存器的地址及讀的指令(如:地址為0xf0,讀指令為0x00)模塊收到讀的指令后,數(shù)據(jù)以頁(yè)的形式發(fā)送主機(jī)設(shè)置讀的模式主機(jī)讀取一頁(yè)數(shù)據(jù)并存儲(chǔ)

static ssize_t spidev_read(struct file *filp, char __user *buf, size_t count, loff_t *f_pos) { struct spidev_data *spidev; int status = 0; int i = 0; spidev = filp->private_data; mutex_lock(&spidev->buf_lock); gsl_fp_write(spidev, 0x00, 0xf0);//讀之前先向模塊寫(xiě)讀的指令及寄存器地址 while(1){ for(i=0;i <= 110*118/128/read_pages;i++){ status = gsl_fp_getOneFrame(spidev,0x00);//讀1頁(yè)數(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); return status; } static inline unsigned int gsl_fp_getOneFrame(struct spidev_data *spidev,unsigned char reg_8b) { int status,i; unsigned char buf_d[128*1+3] = {0x00,0x00}; struct spi_transfer t; 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]; } return status; } static inline ssize_t gsl_spidev_sync_read(struct spidev_data *spidev,struct spi_transfer *t) { struct spi_message m; 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); return spidev_sync(spidev, &m); }

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

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

    關(guān)注

    87

    文章

    11497

    瀏覽量

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

    關(guān)注

    7

    文章

    6099

    瀏覽量

    36229
  • Firefly
    +關(guān)注

    關(guān)注

    2

    文章

    542

    瀏覽量

    7472
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    超高性價(jià)比AIO-3128C開(kāi)源主板

    `Firefly AIO-3128C開(kāi)源主板擁有超高性價(jià)比處理核心四核ARM Cortex-A7,全面低于市面主流A7處理器的價(jià)格。主頻高達(dá)1.3GHz,集成Mali-400 MP2圖形處理器,內(nèi)嵌2D加速器,擁有強(qiáng)大的圖形與視頻處理能力。豐富的外部擴(kuò)展接口,可以直接應(yīng)用
    發(fā)表于 05-16 16:27

    AIO-3128C一體板的特點(diǎn)及其功能分別有哪些呢

    AIO-3128C一體板的特點(diǎn)及其功能分別有哪些呢?
    發(fā)表于 01-14 06:28

    firefly RK3128開(kāi)發(fā)板 AIO-3128C介紹

    采用了基于Cortex-A7構(gòu)架的四核處理器RK3128,集成Mali-400 MP2圖形處理器,支持OpenGL ES1.1/2.0,可以實(shí)現(xiàn)1080P、H.265硬件解碼和1080P、H.264視頻編碼。AIO-3128C引出全面的開(kāi)發(fā)接口,可快速應(yīng)用于各種行業(yè)產(chǎn)品中
    的頭像 發(fā)表于 11-13 14:25 ?2997次閱讀
    firefly RK<b class='flag-5'>3128</b>開(kāi)發(fā)板 <b class='flag-5'>AIO-3128C</b><b class='flag-5'>介紹</b>

    電子AIO-3128C主板U-Boot使用介紹

    RK U-Boot 基于開(kāi)源的 U-Boot 進(jìn)行開(kāi)發(fā),工作模式有啟動(dòng)加載模式和下載模式。
    的頭像 發(fā)表于 12-23 16:08 ?1567次閱讀
    <b class='flag-5'>微</b><b class='flag-5'>雪</b><b class='flag-5'>電子</b><b class='flag-5'>AIO-3128C</b><b class='flag-5'>主板</b>U-Boot使用<b class='flag-5'>介紹</b>

    電子AIO-3128C主板屏幕模組簡(jiǎn)介

    屏幕模組 10.1寸LVDS顯示模組
    的頭像 發(fā)表于 12-23 16:11 ?1271次閱讀
    <b class='flag-5'>微</b><b class='flag-5'>雪</b><b class='flag-5'>電子</b><b class='flag-5'>AIO-3128C</b><b class='flag-5'>主板</b>屏幕模組簡(jiǎn)介

    電子AIO-3128C主板UART使用介紹

    AIO-3128C 開(kāi)發(fā)板內(nèi)置 3 路 UART,分別為 uart0,uart1,uart2。
    的頭像 發(fā)表于 12-23 16:10 ?2558次閱讀
    <b class='flag-5'>微</b><b class='flag-5'>雪</b><b class='flag-5'>電子</b><b class='flag-5'>AIO-3128C</b><b class='flag-5'>主板</b>UART使用<b class='flag-5'>介紹</b>

    電子AIO-3128C主板PWM使用介紹

    AIO-3128C 開(kāi)發(fā)板上有 4 路 PWM 輸出,分別為 PWM0 ~ PWM3,
    的頭像 發(fā)表于 12-23 16:15 ?2770次閱讀

    電子AIO-3128C主板MIPI DSI使用介紹

    MIPI DSI 使用 Config配置
    的頭像 發(fā)表于 12-23 16:16 ?1.4w次閱讀
    <b class='flag-5'>微</b><b class='flag-5'>雪</b><b class='flag-5'>電子</b><b class='flag-5'>AIO-3128C</b><b class='flag-5'>主板</b>MIPI DSI使用<b class='flag-5'>介紹</b>

    電子AIO-3128C主板LED使用簡(jiǎn)介

    AIO-3128C 開(kāi)發(fā)板上有 2 個(gè) LED 燈
    的頭像 發(fā)表于 12-23 16:17 ?3085次閱讀
    <b class='flag-5'>微</b><b class='flag-5'>雪</b><b class='flag-5'>電子</b><b class='flag-5'>AIO-3128C</b><b class='flag-5'>主板</b>LED使用簡(jiǎn)介

    fireflyAIO-3128C主板啟動(dòng)模式介紹

    AIO-3128C 有靈活的啟動(dòng)方式。一般情況下,除非硬件損壞,AIO-3128C 開(kāi)發(fā)板是不會(huì)變磚的。
    的頭像 發(fā)表于 12-24 09:44 ?1082次閱讀

    fireflyAIO-3128C主板串口調(diào)試介紹

    AIO-3128C 開(kāi)發(fā)板的調(diào)試串口與 TF 卡接口有信號(hào)引腳復(fù)用,因此無(wú)法同時(shí)使用,即: 使用調(diào)試串口時(shí)必須拔掉 TF 卡,而使用 TF 卡時(shí)不要連接調(diào)試串口。
    的頭像 發(fā)表于 12-24 09:50 ?1738次閱讀
    fireflyAIO-<b class='flag-5'>3128C</b><b class='flag-5'>主板</b>串口調(diào)試<b class='flag-5'>介紹</b>

    電子BB Black卡片式電腦介紹

    電子
    的頭像 發(fā)表于 12-27 09:28 ?1695次閱讀
    <b class='flag-5'>微</b><b class='flag-5'>雪</b><b class='flag-5'>電子</b>BB Black卡片式電腦<b class='flag-5'>介紹</b>

    AIO-3128C 開(kāi)發(fā)板防雷防靜電明細(xì)

    AIO-3128C 開(kāi)發(fā)板防雷防靜電明細(xì) AIO-3128C 開(kāi)發(fā)板防雷防靜電明細(xì) 1. 產(chǎn)品概述 AIO-3128C 一體板采用了瑞芯(Rockchip)四核處理器 RK
    發(fā)表于 03-16 14:55 ?1211次閱讀

    瑞芯AIO-3399C(AI)主板的主要特性是什么

    AIO-3399C(AI)主板采用 瑞芯Rockchip 全新一代 64 位六核處理器,擁有強(qiáng)大的硬解碼能力,支持 4K 硬解,多種顯示輸出接口,可直接驅(qū)動(dòng)多種分辨率 eDP 接口液晶屏
    發(fā)表于 01-20 14:27 ?1306次閱讀
    瑞芯<b class='flag-5'>微</b><b class='flag-5'>AIO-3399C</b>(AI)<b class='flag-5'>主板</b>的主要特性是什么

    AIO-3128C一體板產(chǎn)品規(guī)格書(shū)

    電子發(fā)燒友網(wǎng)站提供《AIO-3128C一體板產(chǎn)品規(guī)格書(shū).pdf》資料免費(fèi)下載
    發(fā)表于 03-07 14:09 ?0次下載
    主站蜘蛛池模板: 午夜在线视频网站 | 黄色午夜视频 | 国产美女特级嫩嫩嫩bbb | 一女被两男吃奶玩乳尖口述 | 欧美性受xxxx极品 | 亚洲无卡| 久久久久国产精品免费网站 | 日本不卡视频在线观看 | 欧美午夜视频在线观看 | 丁香花五月婷婷 | 操亚洲| 一区二区三区精品视频 | 91中文字幕视频 | 国产精品777 | 激情亚洲色图 | 亚洲天堂导航 | 视频免费在线观看 | 韩国理伦片在线观看2828 | 综合激情网站 | 天天干天天要 | 永久黄网站色视频免费观看 | 日本一二区视频 | 91极品女神私人尤物在线播放 | 李老汉和小花的性生生活 | 亚洲狠狠婷婷综合久久久久图片 | 日韩黄色成人 | 亚洲深夜 | 色秀视频免费高清网站 | 日本妞xxxxxxxxx69 | 久久噜国产精品拍拍拍拍 | 熊出没之环球大冒险旧版免费观看 | 午夜爽爽性刺激一区二区视频 | 男人都懂的网址在线看片 | 人人爽天天碰天天躁夜夜躁 | 国产人成精品免费视频 | 特级无码毛片免费视频尤物 | 一级毛片在线 | 2021最新久久久视精品爱 | 天天干天天操天天添 | 国产精品久久久久久免费播放 | 一级片+国产 |