关注、星标嵌入式客栈,精彩及时送达

[导读] 最近使用ZYNQ做一个高速数据采集,需要访问一个ADI的高速模数采样芯片,该芯片是利用三线制实现读以及写的功能。三线制实现写通信或许大家都经常会这样用,三线制实现读/写或许有的朋友就未曾这样用过。今天就分享一下利用现成IP不写任何代码怎么实现三线制SPI。

背景

ADI很多芯片都采用三线制SPI进行控制,以AD9467为例,AD9467是一款 pipeline架构16位高速ADC芯片,采样率高达250MSPS。在一些复杂系统中其应用领域比较广泛:

  • 多载波,多模式蜂窝接收机

  • 天线阵列定位

  • 功率放大器线性化

  • 无线宽带通信系统

  • 雷达系统

  • 红外成像系统

  • 通讯仪表系统等

从芯片框图,大致可以看出,该芯片主要由以下部分组成:

  • 三线制SPI通信接口,实现芯片的寄存器读写控制。主要用于芯片模式配置。

  • LVDS接口:则负责数据的对外传输,遵循ANSI-644 标准。

  • CLK+/CLK-:为输入时钟,时钟之于数字芯片相当于心脏之于人,一切的动作都是由时钟驱动的。

  • VIN+/VIN- :差分输入,模拟信号输入通道。

对于芯片的其他部分,不是本文介绍的重点,这里来看看其SPI的通信时序图:

结合SPI模式时序图:

  • 在上升沿采样

  • 下一位数据在CLK低期间变换

故,CPOL=0,CPHA=0.

另外,第一个bit用于标识本次报文你发起的是读还是写操作,这种设计是不是有点类似I2C标准中的读写位?

柳暗花明

那么问题来了,我们需要做的SPI通信需要实现三线制SPI进行读以及写:

  • 如果用单片机编程IO口去翻比较容易,但是要实现高速AD数据传输,常规的单片机就捉襟见肘了。LVDS接口的数据吞吐率很难做到。

  • 如果使用ZYNQ内置的SPI外设也很容易,该外设很容易配成三线制模式。很不幸,外设引脚基本用掉了。不过可以考虑用EMIO把相应的脚从PL端拉出去。

  • 如果利用ZYNQ PS端的GPIO也可以做到,也很不幸,做的板子PS端GPIO所剩无几。

  • 利用赛灵思的AXI Quad SPI  IP在PL端去实现。折腾一段时间,发现这个IP貌似不支持三线制SPI。

  • 自己用verilog HDL写个IP挂在AXI总线上,实现Linux设备驱动,这个方案可以。可惜,比较懒,不想重新造轮子!

  • .......

经过一番折腾,在ADI官方发现了一个宝藏:

https://wiki.analog.com/resources/fpga/peripherals/spi_engine

官方实现了SPI engine IP 框架:

  • 执行模块 Execution Module:主模块,用于处理SPI引擎命令流并实现SPI总线接口逻辑

  • AXI接口模块:内存映射软件可访问SPI引擎命令流和/或卸载核心的接口

  • Offload模块:存储SPI引擎命令流,由外部事件触发执行

  • 互连Interconnect 模块:将多个SPI Engine命令流连接到SPI Engine执行模块

其verilog HDL代码库位于:

https://github.com/analogdevicesinc/hdl.git

PS/PL设计

下好hdl库,按照向导将库make,执行对应的tcl脚本,生成了hdl库相应所需文件。然后按照需要设计以下block设计:

  • 将PS端的DDR以及PL所需的时钟FCLK_CLK0配置好,这里输出100MHz

  • 从ip库里拉出来axi_spi_engine_v1_0以及spi_engine_execution_v1_0,按照上面图连好线

  • 连好AXI接口,以及相应的复位、时钟信号等

  • 设置需要几个片选信号,可根据需要几个从芯片可以设置多个片选信号,比如我设置2个,这样在linux设备树上就对应挂载两个设备。

然后在顶层设计文件进行例化,这里问题来了,spi_1还是4个脚,如果就这样拉出到PL端的引脚上,还是四线制,那么该怎么改呢?

看看wiki中图以及描述,发现需要还需要在转一下:

  • 如果是三线模式时,three_wire会变成1,这个通过AXI总线命令传过来。

  • sdo_t则可以控制sdo内部信号是否输出,如果门控关断则mosi脚变成高阻,可以采样外部信号,从而传入可以通过2路选择器传入sdi转而为读信号。

从而添加如下代码在顶层文件:

assign phy_sclk = spi_sclk;
assign phy_cs = spi_cs;
assign phy_mosi = spi_sdo_t ? 1'bz : spi_sdo;
assign spi_sdi = spi_three_wire ? phy_mosi : phy_miso;

比如,我是这样写的:

`timescale 1ns / 100ps
//顶层设计文件
module system_top (
//DDR信号
inout   [14:0]  ddr_addr,
inout   [ 2:0]  ddr_ba,
inout           ddr_cas_n,
inout           ddr_ck_n,
inout           ddr_ck_p,
inout           ddr_cke,
inout           ddr_cs_n,
inout   [ 3:0]  ddr_dm,
inout   [31:0]  ddr_dq,
inout   [ 3:0]  ddr_dqs_n,
inout   [ 3:0]  ddr_dqs_p,
inout           ddr_odt,
inout           ddr_ras_n,
inout           ddr_reset_n,
inout           ddr_we_n,
//必须的一些PS信号
inout           fixed_io_ddr_vrn,
inout           fixed_io_ddr_vrp,
//54个PS MIO引脚
inout   [53:0]  fixed_io_mio,
//PS时钟引脚
inout           fixed_io_ps_clk,
//PS上电复位信号
inout           fixed_io_ps_porb,
//PS SRSTB信号
inout           fixed_io_ps_srstb,
output [1:0]       spi_0_cs,
output             spi_0_sclk,
input              spi_0_sdi,
output             spi_0_sdo,
);wire ip_spi_0_cs;
wire ip_spi_0_sclk;
wire ip_spi_0_sdi;
wire ip_spi_0_sdo;
wire ip_spi_0_three_wire;
wire ip_spi_0_sdo_t;assign spi_0_cs = ip_spi_0_cs;
assign spi_0_sclk = ip_spi_0_sclk;
//如此处理,这样引出的SPI可以兼容3线制以及4线制SPI
assign spi_0_sdo = ip_spi_0_sdo_t ? 1'bz : ip_spi_0_sdo;
assign ip_spi_0_sdi = ip_spi_0_three_wire ? spi_0_sdo : spi_0_sdi;//例化block设计
ip_block_wrapper i_system_wrapper (//DDR以及常规MIO、时钟、复位等信号.DDR_addr(ddr_addr),.DDR_ba(ddr_ba),.DDR_cas_n(ddr_cas_n),.DDR_ck_n(ddr_ck_n),.DDR_ck_p(ddr_ck_p),.DDR_cke(ddr_cke),.DDR_cs_n(ddr_cs_n),.DDR_dm(ddr_dm),.DDR_dq(ddr_dq),.DDR_dqs_n(ddr_dqs_n),.DDR_dqs_p(ddr_dqs_p),.DDR_odt(ddr_odt),.DDR_ras_n(ddr_ras_n),.DDR_reset_n(ddr_reset_n),.DDR_we_n(ddr_we_n),.FIXED_IO_ddr_vrn(fixed_io_ddr_vrn),.FIXED_IO_ddr_vrp(fixed_io_ddr_vrp),.FIXED_IO_mio(fixed_io_mio),.FIXED_IO_ps_clk(fixed_io_ps_clk),.FIXED_IO_ps_porb(fixed_io_ps_porb),.FIXED_IO_ps_srstb(fixed_io_ps_srstb),       //连至顶层.spi_0_cs(ip_spi_0_cs),.spi_0_sclk(ip_spi_0_sclk),.spi_0_sdi(ip_spi_0_sdi),.spi_0_sdo(ip_spi_0_sdo),.spi_0_sdo_t(ip_spi_0_sdo_t),.spi_0_three_wire(ip_spi_0_three_wire)     );
endmodule

Linux端配置

首先需要配置设备树:

&axi_spi_engine_0 {status = "okay";//配置SPI控制器匹配字段,这样会自动编译ADI 提供的SPI 控制器驱动compatible = "adi,axi-spi-engine-1.00.a";spi-rx-bus-width = <1>;spi-tx-bus-width = <1>; bits-per-word = <8>;interrupt-parent = <&intc>;interrupts = <0 30 4>;      num-cs = <4>;#address-cells = <0x1>;#size-cells = <0x0>;    ad9467_0: ad9467@0 {compatible = "adi,ad9467";reg = <0>;spi-max-frequency = <500000>;#address-cells = <1>;#size-cells = <1>;spi-rx-bus-width = <1>; spi-tx-bus-width = <1>;bits-per-word = <8>;spi-3wire;        };ad9467_1: ad9467@1 {compatible = "spidev";reg = <1>;spi-max-frequency = <500000>;#address-cells = <1>;#size-cells = <1>;        spi-rx-bus-width = <1>; spi-tx-bus-width = <1>;bits-per-word = <8>;//这个字段需要使能,表示该设备是三线制spi-3wire;           };
};

基于ADI提高的Linux代码库:

https://github.com/analogdevicesinc/Linux

配置相应的SPI控制器驱动,然后编译部署。由于该demo涉及些项目其他的技术细节,这里就不描述了,来看看疗效:

看这个波形或许会有朋友问:为啥数据发送结束,SDO/SDI复用脚波形有一个上升的渐变暂态过程,这是由于此时从端芯片从输出态转为高阻态,而主端芯片此时也是高阻态,由于线路电容效应故而会有这样一个变化过程。

总结一下

利用ADI提高的IP库,不用敲一行代码可以很容易就实现了三线制SPI,香吧?该方案可以同时兼容三线制/四线制SPI,是一个成熟稳定的方案。为什么ZYNQ芯片这样一款SOC芯片以及Linux会被人喜欢,由此可见一斑。因为有大量成熟的轮子可供使用,而不必自己去造轮子。从而加速产品的研发进度,使用户可以专注于自己需要解决的应用问题。这里有一个tips拿走不谢:

在做应用开发时,首先梳理出需求,要干什么?去往哪里?但别急着撸代码,先搜搜看看有没有现成的轮子,拒绝重新造轮子!一定会加速开发进程。但也需要注意一下开源资源是否可以免费商用,注意知识产权IP问题!~

END

往期精彩推荐,点击即可阅读

▲Linux驱动相关专辑 

手把手教信号处理专辑

片机相关专辑

分享 ????  点赞 ????  在看 ❤️

以“三连”行动支持优质内容!

啥? 一行代码不敲就构建三线制SPI驱动?相关推荐

  1. “Talk is cheap, show me the code”你一行代码有多少漏洞?

    受访者 | 梁宇宁 记者 | 伍杏玲 出品 | CSDN(ID:CSDNnews) 平常程序员喜欢说"Talk is cheap, show me the code"这句话,可是你 ...

  2. 不写一行代码,也能玩转Kaggle竞赛?

    整理 | Jane 出品 | AI科技大本营(ID:rgznai100) [导读]AI科技大本营会给大家分享一些 Kaggle 上的资源,如 Kaggle 开放的数据集,也会分享一些好的竞赛方案或有意 ...

  3. PyTorch Hub发布获Yann LeCun强推!一行代码调用经典模型

    作者 | Team PyTorch 译者 | Monanfei 责编 | 夕颜 出品 | AI科技大本营(ID: rgznai100) 导读:6月11日,Facebook PyTorch 团队推出了全 ...

  4. 一行代码不用写,就可以训练、测试、使用模型,这个star量1.5k的项目帮你做到...

    机器之心报道 机器之心编辑部 igel 是 GitHub 上的一个热门工具,基于 scikit-learn 构建,支持 sklearn 的所有机器学习功能,如回归.分类和聚类.用户无需编写一行代码即可 ...

  5. pytorch 获取模型参数_剑指TensorFlow,PyTorch Hub官方模型库一行代码复现主流模型...

    选自PyTorch 机器之心编译 参与:思源.一鸣 经典预训练模型.新型前沿研究模型是不是比较难调用?PyTorch 团队今天发布了模型调用神器 PyTorch Hub,只需一行代码,BERT.GPT ...

  6. 一行代码搞定 R 语言模型输出!(使用 stargazer 包)

    引言 使用stargazer包可以将 R 构建的模型结果以LATEX.HTML和ASCII格式输出,方便我们生成标准格式的表格. 再结合rmarkdown,你就可以轻轻松松输出一篇优雅的文章啦~ 本文 ...

  7. PyTorch Hub发布!一行代码调用最潮模型,图灵奖得主强推

    文章来源:量子位 原文地址:https://mp.weixin.qq.com/s/lS3YiXzYyY6-XNTFyH_GHg 如有兴趣可以**点击加入极市CV专业微信群**,获取更多高质量干货 为了 ...

  8. 牛气!GitHub 标星 119K+!这些神器仅需一行代码即可下载全网视频!

    作者 | JackTian 来源 | 杰哥的IT之旅(ID:Jake_Internet) 今天这篇文章的内容对于经常浏览各大视频网站的同学来说,是一大神器工具.当你看到自己目前所需的视频时想尽各种办法 ...

  9. 一行代码不用写,就可以训练、测试、使用模型,这个 star 量 1.5k 的项目帮你做到...

    公众号关注 "小詹学Python" 设为"星标",第一时间知晓最新干货~ 转自 | 机器之心 igel 是 GitHub 上的一个热门工具,基于 scikit- ...

最新文章

  1. php对连接加密后传,PHP 加密解密和解决URL传输问题
  2. 制作rhel的yum仓库
  3. golang中的plugin包
  4. 【数据结构与算法】之深入解析“奇怪的打印机”的求解思路与算法示例
  5. Vue基础之Vue实例
  6. 暗备用的运行状态_瞧:我利用“无偏二极管”发明设计出了【宇宙“暗物质”、“暗能量”探测器】...
  7. java-抽象类和接口对区别
  8. [安卓] 18、一个简单的例子做自定义动画按钮和自定义Actionbar
  9. Oracle中shrink space命令详解
  10. HEVC里面CU与TU打印到屏幕及提取到txt文本
  11. python3 ftplib_python3从零学习-5.10.9、ftplib—FTP 协议客户端
  12. 刚接触mybatis,很容易分不清#{id}和#{param1}的使用情况,本篇文章详解了两者的作用以及使用情况
  13. javac编译java_使用javac编译java文件
  14. 手机电子名片html,利用JavaScript的AngularJS库制作电子名片的方法
  15. quartus频率计 时钟设置_基于QuartusII的两种数字频率计的设计与比较
  16. dart语言hello world
  17. 怎么将pdf转换成word 三个简单妙招通通管用
  18. licenses.licx文件
  19. 失落世界服务器国庆宝箱位置,失落的斯菲尔 图文攻略 全剧情任务流程全宝箱收集...
  20. Centos下docker/docker-compose离线安装

热门文章

  1. mvc html.listbox,asp.net-mvc – 为什么ListBoxFor不选择项目,但ListBox是?
  2. 重命名文件名 | 一键导出表格,让您的文件快速整理归档
  3. 高效光伏硅片制造中的水基超声化学清洗
  4. 基于PLC的大小球分拣系统控制设计物料分拣组态
  5. ASP.NET----模块和处理程序
  6. c语言如何删除链表中内容,C语言删除双向链表中数据项程序
  7. 音视频开发之旅(32)-音视频学习资料
  8. Matlab统计参数
  9. python制作图片贴纸,python - 如何将剪贴画放在matplotlib中的绘制数据后面 - SO中文参考 - www.soinside.com...
  10. linux系统如何配置ssh链接方式【以kali linux系统为例】【有一个坑,网上很多配置找不到PermitRootLogin选项】