0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Firefly关于SPI简介

firefly 来源:firefly 作者:firefly 2019-11-20 11:21 次阅读
SPI 使用

SPI是一种高速的,全双工,同步串行通信接口,用于连接微控制器传感器、存储设备等。 Firefly-RK3399 开发板提供了 SPI1 (单片选)接口,具体位置如下图:

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的四种工作模式波形图如下:

驱动编写

下面以 W25Q128FV Flash模块为例简单介绍SPI驱动的编写。

硬件连接

Firefly-RK3399 与 W25Q128FV 硬件连接如下表:

编写Makefile/Kconfig

在kernel/drivers/spi/Kconfig中添加对应的驱动文件配置:

config SPI_FIREFLY tristate "Firefly SPI demo support " default y help Select this option if your Firefly board needs to run SPI demo.

在kernel/drivers/spi/Makefile中添加对应的驱动文件名:

obj-$(CONFIG_SPI_FIREFLY) += spi-firefly-demo.o

config中选中所添加的驱动文件,如:

│ Symbol: SPI_FIREFLY [=y] │ Type : tristate │ Prompt: Firefly SPI demo support │ Location: │ -> Device Drivers │ -> SPI support (SPI [=y]) │ Defined at drivers/spi/Kconfig:704 │ Depends on: SPI [=y] && SPI_MASTER [=y]
配置DTS节点

在kernel/arch/arm64/boot/dts/rockchip/rk3399-firefly-demo.dtsi中添加SPI驱动结点描述,如下所示:

/* Firefly SPI demo */ &spi1 { spi_demo: spi-demo@00{ status = "okay"; compatible = "firefly,rk3399-spi"; reg = <0x00>; spi-max-frequency = <48000000>; /* rk3399 driver support SPI_CPOL | SPI_CPHA | SPI_CS_HIGH */ //spi-cpha; /* SPI mode: CPHA=1 */ //spi-cpol; /* SPI mode: CPOL=1 */ //spi-cs-high; }; }; &spidev0 { status = "disabled"; };
  • status:如果要启用SPI,则设为okay,如不启用,设为disable。

  • spi-demo@00:由于本例子使用CS0,故此处设为00,如果使用CS1,则设为01。

  • compatible:这里的属性必须与驱动中的结构体:of_device_id 中的成员compatible 保持一致。

  • reg:此处与spi-demo@00保持一致,本例设为:0x00。

  • spi-max-frequency:此处设置spi使用的最高频率。Firefly-RK3399最高支持48000000。

  • spi-cpha,spi-cpol:SPI的工作模式在此设置,本例所用的模块SPI工作模式为SPI_MODE_0或者SPI_MODE_3,这里我们选用SPI_MODE_0,如果使用SPI_MODE_3,spi_demo中打开spi-cpha和spi-cpol即可。

  • spidev0: 由于spi_demo与spidev0使用一样的硬件资源,需要把spidev0关掉才能打开spi_demo

定义SPI驱动

在内核源码目录kernel/drivers/spi/中创建新的驱动文件,如:spi-firefly-demo.c 在定义 SPI 驱动之前,用户首先要定义变量 of_device_id 。 of_device_id 用于在驱动中调用dts文件中定义的设备信息,其定义如下所示:

static struct of_device_id firefly_match_table[] = {{ .compatible = "firefly,rk3399-spi",},{},};

此处的compatible与DTS文件中的保持一致。

spi_driver定义如下所示:

static struct spi_driver firefly_spi_driver = { .driver = { .name = "firefly-spi", .owner = THIS_MODULE, .of_match_table = firefly_match_table,}, .probe = firefly_spi_probe,};
注册SPI设备

在初始化函数static int __init spidev_init(void)中向内核注册SPI驱动: spi_register_driver(&firefly_spi_driver);

如果内核启动时匹配成功,则SPI核心会配置SPI的参数(mode、speed等),并调用firefly_spi_probe。

读写 SPI 数据

firefly_spi_probe中使用了两种接口操作读取W25Q128FV的ID: firefly_spi_read_w25x_id_0接口直接使用了spi_transfer和spi_message来传送数据。 firefly_spi_read_w25x_id_1接口则使用SPI接口spi_write_then_read来读写数据。

成功后会打印:

root@rk3399_firefly_box:/ # dmesg | grep firefly-spi [ 1.006235] firefly-spi spi0.0: Firefly SPI demo program [ 1.006246] firefly-spi spi0.0: firefly_spi_probe: setup mode 0, 8 bits/w, 48000000 Hz max [ 1.006298] firefly-spi spi0.0: firefly_spi_read_w25x_id_0: ID = ef 40 18 00 00 [ 1.006361] firefly-spi spi0.0: firefly_spi_read_w25x_id_1: ID = ef 40 18 00 00
打开SPI demo

spi-firefly-demo默认没有打开,如果需要的话可以使用以下补丁打开demo驱动:

--- a/kernel/arch/arm64/boot/dts/rockchip/rk3399-firefly-demo.dtsi +++ b/kernel/arch/arm64/boot/dts/rockchip/rk3399-firefly-demo.dtsi @@ -64,7 +64,7 @@ /* Firefly SPI demo */ &spi1 {spi_demo: spi-demo@00{ - status = "disabled"; + status = "okay"; compatible = "firefly,rk3399-spi"; reg = <0x00>; spi-max-frequency = <48000000>; @@ -76,6 +76,6 @@ }; &spidev0 { - status = "okay"; + status = "disabled"; };

注意:由于spi1_rxd和spi1_txd两个脚可复用为uart4_rx和uart4_tx,所以要留意关闭掉uart4的使用,如下:

kernel/arch/arm64/boot/dts/rockchip/rk3399-firefly-port.dtsi &uart4 { current-speed = <9600>; no-loopback-test; status = "disabled"; };
常用SPI接口

下面是常用的 SPI API 定义:

void spi_message_init(struct spi_message *m); void spi_message_add_tail(struct spi_transfer *t, struct spi_message *m); int spi_sync(struct spi_device *spi, struct spi_message *message) ; int spi_write(struct spi_device *spi, const void *buf, size_t len); int spi_read(struct spi_device *spi, void *buf, size_t len); ssize_t spi_w8r8(struct spi_device *spi, u8 cmd); ssize_t spi_w8r16(struct spi_device *spi, u8 cmd); ssize_t spi_w8r16be(struct spi_device *spi, u8 cmd); int spi_write_then_read(struct spi_device *spi, const void *txbuf, unsigned n_tx, void *rxbuf, unsigned n_rx);
接口使用

Linux提供了一个功能有限的SPI用户接口,如果不需要用到IRQ或者其他内核驱动接口,可以考虑使用接口spidev编写用户层程序控制SPI设备。 在 Firefly-RK3399 开发板中对应的路径为: /dev/spidev0.0

spidev对应的驱动代码: kernel/drivers/spi/spidev.c

内核config需要选上SPI_SPIDEV:

│ Symbol: SPI_SPIDEV [=y] │ Type : tristate │ Prompt: User mode SPI device driver support │ Location: │ -> Device Drivers │ -> SPI support (SPI [=y]) │ Defined at drivers/spi/Kconfig:684 │ Depends on: SPI [=y] && SPI_MASTER [=y]

DTS配置如下:

&spi1 { status = "okay"; max-freq = <48000000>; spidev@00 { compatible = "linux,spidev"; reg = <0x00>; spi-max-frequency = <48000000>; }; };

详细使用说明请参考文档 spidev 。

FAQs
Q1: SPI数据传送异常

A1: 确保 SPI 4个引脚的 IOMUX 配置正确, 确认 TX 送数据时,TX 引脚有正常的波形,CLK 频率正确,CS 信号有拉低,mode 与设备匹配。


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    87

    文章

    10990

    浏览量

    206738
  • 嵌入式主板
    +关注

    关注

    7

    文章

    6064

    浏览量

    34614
  • Firefly
    +关注

    关注

    2

    文章

    534

    浏览量

    6757
收藏 人收藏

    评论

    相关推荐

    请问一下如何对Firefly RK3399的spi驱动进行调试呢

    请问一下如何对Firefly RK3399的spi驱动进行调试呢?
    发表于 03-08 06:36

    Firefly-RK3399 SPI接口使用相关资料分享

    1、SPI接口使用简介SPI 是一种高速的,全双工,同步串行通信接口,用于连接微控制器、传感器、存储设备等。 Firefly-RK3399 开发板提供了
    发表于 06-13 17:06

    SPI、IIC、IIS、UART、SDIO、GPIO 简介

    SPI、IIC、IIS、UART、SDIO、GPIO 简介
    发表于 10-29 17:00 0次下载

    常见总线简介CAN、USART、SPI、SCI等

    常见总线简介,包括CAN、USART、SPI、SCI等
    发表于 11-30 14:08 0次下载

    firefly多窗口显示简介

    Firefly-RK3288平台拥有强大的多线程运算和图像处理能力,基于Firefly-RK3288可以实现多窗口显示的功能,用户可以在Android系统下同时运行不同的应用,也可以对当前多个窗口进行打开、关闭、放大、缩小、最大化、最小化等操作。
    的头像 发表于 11-01 15:13 1294次阅读
    <b class='flag-5'>firefly</b>多窗口显示<b class='flag-5'>简介</b>

    firefly双屏互动简介

    基于Firefly-RK3288高性能开源平台,可以做双屏互动功能,Firefly-RK3288同时接MIPI
    的头像 发表于 11-01 15:31 1293次阅读
    <b class='flag-5'>firefly</b>双屏互动<b class='flag-5'>简介</b>

    Firefly-RK3399--FAQs简介

    Firefly-RK3399 的 HDMI 能自动识别显示的分辨率。
    的头像 发表于 11-19 15:27 2736次阅读
    <b class='flag-5'>Firefly</b>-RK3399--FAQs<b class='flag-5'>简介</b>

    Firefly关于的屏幕模组简介

    Firefly
    的头像 发表于 11-20 10:47 1253次阅读
    <b class='flag-5'>Firefly</b><b class='flag-5'>关于</b>的屏幕模组<b class='flag-5'>简介</b>

    Firefly关于UART接口使用介绍

    Firefly-RK3399 支持五路UART:UART0, UART1, UART2, UART3, UART4,都拥有两个64字节的FIFO缓冲区,用于数据接收和发送。
    的头像 发表于 11-20 11:01 2534次阅读
    <b class='flag-5'>Firefly</b><b class='flag-5'>关于</b>UART接口使用介绍

    Firefly关于-RTC 使用简介

    Firefly-RK3399 开发板上有 一个集成于RK808上的RTC(Real Time Clock),主要功能有时钟,日历,闹钟,周期性中断,双通道32KHz时钟输出。
    的头像 发表于 11-20 11:14 2160次阅读
    <b class='flag-5'>Firefly</b><b class='flag-5'>关于</b>-RTC 使用<b class='flag-5'>简介</b>

    Firefly关于RK3288电源适配器简介

    firefly电源适配器 5V-3A电源适配器
    的头像 发表于 11-25 10:01 2165次阅读
    <b class='flag-5'>Firefly</b><b class='flag-5'>关于</b>RK3288电源适配器<b class='flag-5'>简介</b>

    Firefly-PX3-SE--WIFI模块及以太网简介

    关于Firefly-PX3-SE网络的使用,目前在设置app里面只能使用WiFi以及WIFI热点功能,以太网功能暂时还没添加。
    的头像 发表于 12-02 16:56 1224次阅读

    SPI Nand Flash简介

    1.SPI Nand Flash简介SPI Nand Flash顾名思义就是串行接口的Nand Flash,它和普通并行的Nand Flash相似,比如:SLC Nand Flash。2.SP
    发表于 12-02 10:51 33次下载
    <b class='flag-5'>SPI</b> Nand Flash<b class='flag-5'>简介</b>

    SPI通信简介

    目录(?)[-]SPI简介SPI特点1采用主-从模式Master-Slave 的控制方式2采用同步方式Synchronous传输数据3数据交换Data Exchanges4 SPI有四
    发表于 12-22 19:14 9次下载
    <b class='flag-5'>SPI</b>通信<b class='flag-5'>简介</b>

    FPGA中关于SPI的使用

    FPGA中关于SPI的使用
    的头像 发表于 04-12 10:13 595次阅读