ArmSom---SPI开发指南

1. 简介

  • RK3588从入门到精通
  • 本⽂主要介绍在Rockchip平台配置spi接口并且使用的方法
  • 开发板:ArmSoM-W3
  • Kernel:5.10.160
  • OS:Debian11

2. SPI接口概述

SPI(Serial Peripheral Interface),即串行外围设备接口,是一种同步的,全双工的,多设备的,多主机的通信协议,用于连接外围设备,如ADC、DAC、数据存储器、定时器、接受器等。

2.1 spi接口的结构

在这里插入图片描述

  1. 主机:主机是SPI总线的控制者,它负责控制数据传输的方向和传输速度。
  2. 从机:从机是SPI总线的被控制者,它根据主机发出的指令,发出或接收数据。
  3. MOSI(Master Out Slave In):主机输出从机输入,用于传输从主机到从机的数据。
  4. MISO(Master In Slave Out):主机输入从机输出,用于传输从从机到主机的数据。
  5. CK(Serial Clock):时钟线,用于同步主机和从机之间的数据传输。
  6. CS(Chip Select):片选线,用于控制主机和从机之间的数据传输。

此外,SPI接口还有其他可选项,如中断线(INT)、复位线(RESET)等。

2.2 SPI接口的工作原理

在SPI接口通信过程中,主机发出一个片选信号,然后在时钟信号的控制下,主机发出一个字节的数据,从机接收到数据之后,也会发出一个字节的数据,主机接收到数据之后,发出一个片选信号,结束一次通信。

SPI接口有两种工作模式:主模式和从模式。主模式下,主机控制从机,从机接收主机发出的指令。从模式下,从机可以接收主机发出的指令,并向主机发送数据。

3.SPI硬件接口

RK3588旗舰芯片上可使用的spi接口有5组,ArmSoM SOM-3588-LGA核心板采用LGA 506引脚封装方式将spi资源全部引出,ArmSoM-W3开发板上40PIN引脚中有两组spi接口:

在这里插入图片描述

4.spi设备配置

spi驱动相关代码路径:

drivers/spi/spi.c spi驱动框架
drivers/spi/spi-rockchip.c rk spi各接口实现
drivers/spi/spidev.c 创建spi设备节点,用户态使用。
drivers/spi/spi-rockchip-test.c spi测试驱动,需要自己手动添加到Makefile编译
Documentation/spi/spidev_test.c 用户态spi测试工具

4.1 配置 DTS 节点

&spi1 {status = "okay";//assigned-clock-rates = <200000000>; //默认不用配置,SPI 设备工作时钟max-freq = <48000000>; /* spi internal clk, don't modify *///dma-names = "tx","rx"; //使能DMA模式//rx-sample-delay-ns = <10>; //默认不用配置,读采样延时spi_dev@0 {compatible = "rockchip,spidev";reg = <0>;spi-max-frequency = <12000000>;spi-lsb-first; //IO 先传输 lsb		};
};
  • status:如果要启用 SPI,则设为 okay,如不启用,设为 disable。

  • spi_dev@0:本例子使用 CS0,故此处设为 0,如果使用 CS1,则设为 1。

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

  • reg:此处与 spi_dev@0 保持一致,这里设为0。

  • spi-max-frequency:此处设置 spi 使用的最高频率,ITX-3588J 最高支持 50000000。

这里使用的驱动是drivers/spi/spidev.c,驱动设备加载注册成功后,会出现类似这个名字的设备:/dev/spidev1.0。

一些常见的SPI驱动API接口:

Linux SPI API:

  • spi_register_driver():注册SPI设备驱动程序。
  • spi_unregister_driver():注销SPI设备驱动程序。
  • spi_setup():设置SPI总线和设备的参数。
  • spi_sync():同步方式进行SPI数据传输。
  • spi_message_init():初始化SPI消息结构。
  • spi_message_add_tail():向SPI消息添加传输操作。
  • spi_sync():同步方式进行SPI数据传输。
  • spi_transfer():进行SPI数据传输。

SPI驱动中常用的ioctl请求值,这些请求值用于设置和读取SPI设备的各种参数,包括通信模式、字长、数据模式和通信速率等。这些请求值通常用于Linux的SPI驱动编程。以下是一些常用的SPI ioctl请求值:

  • SPI_IOC_RD_MODE: 读取SPI设备的通信模式。
  • SPI_IOC_WR_MODE: 设置SPI设备的通信模式。
  • SPI_IOC_RD_MODE32: 读取SPI设备的32位通信模式。
  • SPI_IOC_WR_MODE32: 设置SPI设备的32位通信模式。
  • SPI_IOC_RD_LSB_FIRST: 读取SPI设备的LSB(Least Significant Bit)优先模式。
  • SPI_IOC_WR_LSB_FIRST: 设置SPI设备的LSB优先模式。
  • SPI_IOC_RD_BITS_PER_WORD: 读取SPI设备的字长。
  • SPI_IOC_WR_BITS_PER_WORD: 设置SPI设备的字长。
  • SPI_IOC_RD_MAX_SPEED_HZ: 读取SPI设备的最大通信速率。
  • SPI_IOC_WR_MAX_SPEED_HZ: 设置SPI设备的最大通信速率。
  • SPI_IOC_MESSAGE(N): 一次进行N次双向或多次读写操作。

4.2 SPI测试实验

将上述设备树以及驱动修改之后更新板卡的内核,通过SPI设备文件来判断spi驱动是否加载成功

这里测试使用的是spi1这一组spi相关接口,检查spi设备:

root@linaro-alip:~# ls /dev/spi*
/dev/spidev1.0	
root@linaro-alip:~# 

测试SPI接口,使用/dev/spidev1.0设备节点,实现短接MOSI和MISO线路以自发自收数据,同时在未短接时报告错误

4.2.1 硬件连接

将SPI1的 MIOS与MOSI引脚(板卡上的7和8)短接即可,如下图所示
在这里插入图片描述

4.2.2 测试程序

用户应用层使用spidev驱动的步骤如下:

  1. 打开SPI设备文件:用户可以通过打开/dev/spidevX.Y文件来访问SPI设备,其中X是SPI控制器的编号,Y是SPI设备的编号。

  2. 配置SPI参数:用户可以使用ioctl命令SPI_IOC_WR_MODE、SPI_IOC_WR_BITS_PER_WORD和SPI_IOC_WR_MAX_SPEED_HZ来设置SPI模式、数据位数和时钟速度等参数。

  3. 发送和接收数据:用户可以使用read和write系统调用来发送和接收SPI数据。写入的数据将被传输到SPI设备,而从设备读取的数据将被存储在用户提供的缓冲区中。

  4. 关闭SPI设备文件:当不再需要与SPI设备通信时,用户应该关闭SPI设备文件。

总结起来,spidev驱动提供了一种简单而灵活的方式来与SPI设备进行通信,使得用户可以轻松地在Linux系统上开发和控制SPI设备。

armsom团队 的测试程序spi_test.c如下:

#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <linux/spi/spidev.h>#define SPI_DEV_PATH "/dev/spidev1.0"int fd;
static unsigned mode = SPI_MODE_0;
static uint8_t bits = 8;
static uint32_t speed = 1000000; // 设置SPI速度为1MHz
static uint16_t delay;void transfer(int fd, uint8_t const *tx, uint8_t *rx, size_t len)
{int ret;struct spi_ioc_transfer tr = {.tx_buf = (unsigned long)tx,.rx_buf = (unsigned long)rx,.len = len,.delay_usecs = delay,.speed_hz = speed,.bits_per_word = bits,.cs_change = 0, // 设置为1以在每次传输前切换片选,这里不切换片选};ret = ioctl(fd, SPI_IOC_MESSAGE(1), &tr);if (ret < 1) {perror("SPI transfer failed");}
}void spi_init(void)
{int ret;// 打开 SPI 设备fd = open(SPI_DEV_PATH, O_RDWR);if (fd < 0) {perror("Can't open SPI device");exit(1);}// 设置 SPI 工作模式ret = ioctl(fd, SPI_IOC_WR_MODE, &mode);if (ret == -1) {perror("Can't set SPI mode");exit(1);}// 设置位数ret = ioctl(fd, SPI_IOC_WR_BITS_PER_WORD, &bits);if (ret == -1) {perror("Can't set bits per word");exit(1);}// 设置SPI速度ret = ioctl(fd, SPI_IOC_WR_MAX_SPEED_HZ, &speed);if (ret == -1) {perror("Can't set max speed");exit(1);}// 打印设置printf("SPI mode: 0x%x\n", mode);printf("Bits per word: %d\n", bits);printf("Max speed: %d Hz\n", speed);
}int main(int argc, char *argv[])
{if (argc != 2) {printf("Usage: %s <string_to_send>\n", argv[0]);return 1;}char *tx_buffer = argv[1]; // 获取要发送的字符串作为命令行参数// 初始化SPI接口spi_init();// 设置要接收数据的缓冲区unsigned char rx_buffer[strlen(tx_buffer) + 1];// 执行SPI数据传输transfer(fd, tx_buffer, rx_buffer, strlen(tx_buffer));// 打印发送和接收的数据printf("Sent: %s\n", tx_buffer);printf("Received: %s\n", rx_buffer);// 关闭SPI设备close(fd);return 0;
}

4.2.3 编译&运行

开发板上下载gcc编译器:

apt upgrade
apt update
apt install gcc

编译:

gcc spi_test.c -o spi

短接与不短接的运行情况如下:

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/162359.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深入详解高性能消息队列中间件 RabbitMQ

目录 1、引言 2、什么是 RabbitMQ &#xff1f; 3、RabbitMQ 优势 4、RabbitMQ 整体架构剖析 4.1、发送消息流程 4.2、消费消息流程 5、RabbitMQ 应用 5.1、广播 5.2、RPC VC常用功能开发汇总&#xff08;专栏文章列表&#xff0c;欢迎订阅&#xff0c;持续更新...&am…

火山引擎云原生存储加速实践

在火山引擎相关的业务中绝大部分的机器学习和数据湖的算力都运行在云原生 K8s 平台上。云原生架构下存算分离和弹性伸缩的计算场景&#xff0c;极大的推动了存储加速这个领域的发展&#xff0c;目前业界也衍生出了多种存储加速服务。但是面对计算和客户场景的多样性&#xff0c…

大数据毕业设计选题推荐-旅游景点游客数据分析-Hadoop-Spark-Hive

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…

linux三种软件安装方式

文章目录 前言一、安装jdk(使用rpm安装)二、Tomcat安装(解压缩方式)三、安装Docker(yum安装) 推荐四、其实我们也可以宝塔安装(傻瓜式一键安装) 前言 之前已经讲解了一些linux的基本使用&#xff0c;现在我们来配置java环境以及安装docker linux讲解链接 一、安装jdk(使用rpm安…

第26期 | GPTSecurity周报

GPTSecurity是一个涵盖了前沿学术研究和实践经验分享的社区&#xff0c;集成了生成预训练Transformer&#xff08;GPT&#xff09;、人工智能生成内容&#xff08;AIGC&#xff09;以及大型语言模型&#xff08;LLM&#xff09;等安全领域应用的知识。在这里&#xff0c;您可以…

golang工程中间件——redis常用结构及应用(string, hash, list)

Redis 命令中心 【golang工程中间件——redisxxxxx】这些篇文章专门以应用为主&#xff0c;原理性的后续博主复习到的时候再详细阐述 string结构以及应用 字符数组&#xff0c;redis字符串是二进制安全字符串&#xff0c;可以存储图片等二进制数据&#xff0c;同时也可以存…

微服务架构——笔记(3)Eureka

微服务架构——笔记&#xff08;3&#xff09; 基于分布式的微服务架构 本次笔记为 此次项目的记录&#xff0c;便于整理思路&#xff0c;仅供参考&#xff0c;笔者也将会让程序更加完善 内容包括&#xff1a;1.支付模块、2.消费者订单模块、支付微服务入驻Eureka、Eureka集群…

基于单片机的商场防盗防火系统设计

收藏和点赞&#xff0c;您的关注是我创作的动力 文章目录 概要 一、系统分析二、系统总设计2.1基于单片机的商场防火防盗系统的总体功能2.2系统的组成 三 软件设计4.1软件设计思路4.2软件的实现4.2.1主控模块实物 四、 结论五、 文章目录 概要 本课题设计一种商场防火防盗报警…

Win10 + VS017 编译SQLite3.12.2源码

参考&#xff1a; [1] WIN10 VS2019下编译GDAL3.0PROJ6SQLite_gdal 3 win10编译-CSDN博客 [2] 如何编译SQLite-How To Compile SQLite-CSDN博客 如何生成静态库&#xff1a; 参考&#xff1a; WIN10 VS2019下编译GDAL3.0PROJ6SQLite_gdal 3 win10编译-CSDN博客 如何生成exe:…

3D高斯泼溅(Splatting)简明教程

在线工具推荐&#xff1a; Three.js AI纹理开发包 - YOLO合成数据生成器 - GLTF/GLB在线编辑 - 3D模型格式在线转换 - 3D场景编辑器 3D 高斯泼溅&#xff08;Splatting&#xff09;是用于实时辐射场渲染的 3D 高斯分布描述的一种光栅化技术&#xff0c;它允许实时渲染从小图像样…

15种稳定扩散模型的技术示例

推荐Stable Diffusion自动纹理工具&#xff1a; DreamTexture.js自动纹理化开发包 什么是稳定扩散模型&#xff1f; 潜在扩散模型 &#xff08;LDM&#xff09; 是一种图像生成技术&#xff0c;其工作原理是在潜在表示空间中迭代“去噪”数据&#xff0c;然后将表示解码为完整…

Android Framework学习之Activity启动原理

Android Activity启动原理 Android 13.0 Activity启动原理逻辑流程图如下&#xff1a;