Linux系统CH347应用—SPI功能

news/2024/5/18 18:35:36/文章来源:https://blog.csdn.net/WCH_TechGroup/article/details/132173785

 Linux/安卓系统使用CH347转接SPI功能有三种应用方式:

1. 使用CH34X_MPHSI_Master总线驱动为系统扩展原生SPI Master,此方式无需进行单独的应用层编程;

2. 使用CH341PAR_LINUX字符设备驱动,此方式需要配合使用厂商提供的库文件,编程访问设备功能;

3. 使用CH341PAR_ANDROID免驱APP,基于安卓原生USB Host API进行二次开发,此方式无需设备驱动;

基于方式一,可参考如下几篇博客:

基于CH347实现USB扩展SPI/I2C/GPIO Master主机应用方案_扩展spi芯片_PC技术小能手的博客-CSDN博客在安卓/Linux主机上经常会遇到CPU原生SPI/I2C/GPIO Master资源通道不够或者功性能不满足实际产品需求的情况,基于USB2.0高速USB转接芯片CH347,配合厂商提供的USB转MPHSI(Multi Peripheral Serial Line)Master总线驱动(CH34X-MSPI-Master)可轻松实现为系统扩展SPI和I2C总线、GPIO Expander、中断信号等。_扩展spi芯片https://blog.csdn.net/WCH_TechGroup/article/details/130093377CH34X-MPHSI高速Master扩展应用—SPI设备调试_PC技术小能手的博客-CSDN博客本文介绍,基于USB2.0高速USB转接芯片CH347,配合厂商提供的USB转MPHSI(Multi Protocol High-Speed Serial Interface)Master总线驱动(CH34X-MPHSI-Master)为系统扩展SPI总线的用法,除此之外,还可以扩展I2C总线和GPIO等资源。驱动软件正常工作后,会在系统下创建新的SPI Master,拥有独立的bus num,原SPI器件的设备驱动可直接通过DTS配置文件或者sysfs节点挂载到该总线上,原有设备驱动无需任何修改。https://blog.csdn.net/WCH_TechGroup/article/details/133349658本篇博客仅对于方式二做重点介绍:

一、应用框图

二、资源列表

demo:应用软件示例

driver:驱动软件

lib:应用库文件,提供大部分CPU架构的动态库和静态库

三、工作原理

驱动软件正常工作时,会自动在系统/dev目录下创建字符设备,名称为:/dev/ch34x_pis*。基于此节点设备,配合 libch347 动态库,ch347demo应用程序实现对 CH347 芯片的硬件资源的访问。

此方式适用于不需要依赖于原有外设驱动工作的应用场景,使用字符设备实现对于外设的读写功能。类似于串口应用,通过访问 /dev/tty* 设备实现相应设备的:打开、关闭、读写等操作。

四、使用步骤

使用HID驱动模式,直接使用系统自带的 hidraw 驱动即可,使用厂商驱动模式,需要编译使用 CH341PAR_LINUX 资料中driver下的驱动文件,链接地址:

CH341PAR_LINUX.ZIP - 南京沁恒微电子股份有限公司USB转JTAG/SPI/I2C/并口/GPIO等接口的Linux设备驱动程序,支持CH341的USB转SPI/I2C/EPP并口/MEM并口等,支持CH347的480Mbps高速USB转JTAG/SPI/I2C/GPIO等,支持32/64位操作系统。https://www.wch.cn/downloads/CH341PAR_LINUX_ZIP.htmlGitHub - WCHSoftGroup/ch341par_linuxContribute to WCHSoftGroup/ch341par_linux development by creating an account on GitHub.https://github.com/WCHSoftGroup/ch341par_linux驱动使用流程如下:

 (1)驱动加载

1. unzip CH341PAR.ZIP
2. cd driver
3. sudo make install

插入CH347硬件设备,此时会自动在 /dev 目录下创建字符设备:ch34x_pis*,如下所示:

至此,代表驱动程序和芯片工作正常。

(2)拷贝库文件至系统库路径下,此处以X64 CPU的 ch347 动态库为例:

sudo cp lib/x64/dynamic/libch347.so /usr/lib

(3)应用编程—SPI 主机模式 API介绍

/*** CH347OpenDevice - open device* @pathname: device path in /dev directory** The function return positive file descriptor if successful, others if fail.*/
extern int CH347OpenDevice(const char *pathname);/*** CH347CloseDevice - close device* @fd: file descriptor of device** The function return true if successful, false if fail.*/
extern bool CH347CloseDevice(int fd);/*** CH347SPI_SetDataBits - SPI data bits setting* @fd: file descriptor of device* @iDataBits: 0: 8bit, 1: 16bit ** The function return true if successful, false if fail.*/
extern bool CH347SPI_SetDataBits(int fd, uint8_t iDataBits);/*** CH347SPI_Init - SPI interface initialization* @fd: file descriptor of device* @SpiCfg: pointer to SPI configuration, SPI frequency could be set by SpiCfg->iClock or CH347SPI_SetFrequency API, the latter is preferred** The function return true if successful, false if fail.*/
extern bool CH347SPI_Init(int fd, mSpiCfgS *SpiCfg);/*** CH347SPI_SetChipSelect - SPI chip selection initialization* @fd: file descriptor of device* @iEnableSelect: low 8 bits: CS1, high 8 bits: CS2, byte value -> 1: set CS, 0: ignore CS setting* @iChipSelect: low 8 bits: CS1, high 8 bits: CS2, CS output, byte value -> 1: set CS, 0: cancel CS* @iIsAutoDeativeCS: low 16 bits: CS1, high 16 bits: CS2, automatically undo the CS after operation completed* @iActiveDelay: low 16 bits: CS1, high 16 bits: CS2, delay time of read and write operation after setting CS, unit: us* @iDelayDeactive: low 16 bits: CS1, high 16 bits: CS2,, delay time of read and write operation after canceling CS, unit: us** The function return true if successful, false if fail.*/
extern bool CH347SPI_SetChipSelect(int fd, uint16_t iEnableSelect, uint16_t iChipSelect, int iIsAutoDeativeCS, int iActiveDelay, int iDelayDeactive);/*** CH347SPI_ChangeCS - SPI CS setting, must call CH347SPI_Init first* @fd: file descriptor of device* @iStatus: 0: cancel CS, 1: set CS** The function return true if successful, false if fail.*/
extern bool CH347SPI_ChangeCS(int fd, uint8_t iStatus);/*** CH347SPI_Write - write SPI data* @fd: file descriptor of device* @ignoreCS: ignore SPI chip select while true, else auto set CS* @iChipSelect: SPI chip select, ignore while BIT7 is 0, valid while BIT7 is 1* @iLength: length to write* @iWriteStep: per write length* @ioBuffer: pointer to write buffer** The function return true if successful, false if fail.*/
extern bool CH347SPI_Write(int fd, bool ignoreCS, int iChipSelect, int iLength, int iWriteStep, void *ioBuffer);/*** CH347SPI_Read - read SPI data* @fd: file descriptor of device* @ignoreCS: ignore SPI chip select while true, else auto set CS* @iChipSelect: SPI chip select, ignore while BIT7 is 0, valid while BIT7 is 1* @iLength: length to write* @oLength: pointer to read length* @ioBuffer: pointer to buffer, store data to be written from MOSI, and return data to be read from MISO** The function return true if successful, false if fail.*/
extern bool CH347SPI_Read(int fd, bool ignoreCS, int iChipSelect, int iLength, uint32_t *oLength, void *ioBuffer);/*** CH347SPI_WriteRead - write then read SPI data* @fd: file descriptor of device* @ignoreCS: ignore SPI chip select while true, else auto set CS* @iChipSelect: SPI chip select, ignore while BIT7 is 0, valid while BIT7 is 1* @iLength: data length to xfer* @ioBuffer: pointer to buffer, store data to be written from MOSI, and return data to be read from MISO** The function return true if successful, false if fail.*/
extern bool CH347SPI_WriteRead(int fd, bool ignoreCS, int iChipSelect, int iLength, void *ioBuffer);

如上API接口函数,根据不同的业务场景可以选用不同的函数。

CH347SPI_Write: 适用于单次SPI通讯长度固定,或者类似SPI FLASH有固定页/扇区大小的器件,可以提升效率。

CH347SPI_Read:适用于单次SPI通讯长度固定,或者类似SPI FLASH有固定页/扇区大小的器件,可以提升效率。

CH347SPI_WriteRead:适用于所有类型的SPI通讯,交换传输,数据从MOSI输出,同时从MISO上采集数据保存到 ioBuffer 缓冲区中

CH347 SPI主机模式操作流程:

注:

1、CH347SPI_SetDataBits 用于启用/禁用 SPI 16位读写功能,此API为可选项。

2、CH347SPI_ChangeCS 用于手动控制芯片的 CS 片选引脚,此API为可选项。

CH347SPI_WriteRead 函数说明

ignoreCS:此次xfer传输是否忽略片选设置,true:传输开始和结束不操作片选信号,false:传输开始和结束自动使能和失能片选信号(当 iChipSelect 设定有效时)

iChipSelect:SPI 片选设置,BIT7为1,片选有效,BIT7为0,撤销片选

iLength:此次xfer数据传输的长度

ioBuffer:需要xfer传输的缓冲区,由于SPI是exchange交换传输,因此该缓冲区内容会先经过MOSI信号线对外输出,然后该API成功返回后,其内容是从MISO信号线上采集的数据。

示例:

uint8_t ioBuffer[2] = {0x11, 0x22};CH347SPI_WriteRead(fd, false, 0x80, 2, ioBuffer);

 此时,MOSI输出2个字节数据 0x11 和 0x22,然后返回 MISO数据。

图1:MISO悬空(高电平)

图2:MISO接GND(低电平)

如上为CH347的SPI功能使用说明,其他平台上Linux和Android系统上接口函数均保持类似,可直接参考移植。

 (4)应用编程—SPI 从机模式 API介绍

/*** CH347OpenDevice - open device* @pathname: device path in /dev directory** The function return positive file descriptor if successful, others if fail.*/
extern int CH347OpenDevice(const char *pathname);/*** CH347CloseDevice - close device* @fd: file descriptor of device** The function return true if successful, false if fail.*/
extern bool CH347CloseDevice(int fd);/*** CH347SPI_Init - SPI interface initialization* @fd: file descriptor of device* @SpiCfg: pointer to SPI configuration, SPI frequency could be set by SpiCfg->iClock or CH347SPI_SetFrequency API, the latter is preferred** The function return true if successful, false if fail.*/
extern bool CH347SPI_Init(int fd, mSpiCfgS *SpiCfg);/*** CH347SPI_Slave_Control - switch of reading SPI data from master * @fd: file descriptor of device* @enable: true: start reading continuously, false: stop reading** The function return true if successful, false if fail.*/
extern bool CH347SPI_Slave_Control(int fd, bool enable);/*** CH347SPI_Slave_QweryData - get spi data length* @fd: file descriptor of device* @oLength: pointer to read length** The function return true if successful, false if fail.*/
extern bool CH347SPI_Slave_QweryData(int fd, uint32_t *oLength);/*** CH347SPI_Slave_FIFOReset - reset spi data fifo* @fd: file descriptor of device** The function return true if successful, false if fail.*/
extern bool CH347SPI_Slave_FIFOReset(int fd);/*** CH347SPI_Slave_ReadData - read spi data in slave mode* @fd: file descriptor of device* @oReadBuffer: pointer to read buffer* @oReadLength: pointer to read length** The function return true if successful, false if fail.*/
extern bool CH347SPI_Slave_ReadData(int fd, void *oReadBuffer, uint32_t *oReadLength);

CH347SPI_Slave_Control: 将芯片的SPI接口切换到从机接收模式,并且驱动程序也会进入专用的从机接收模式,此时对于芯片的设置和其他功能均无法使用,直至再次调用此接口关闭从机接收模式。

CH347SPI_Slave_QweryData:查询从机模式SPI接口接收的数据总量。

CH347SPI_Slave_FIFOReset:复位SPI从机FIFO缓冲区

CH347SPI_Slave_ReadData:从SPI从机的FIFO缓冲区取出数据

CH347 SPI从机模式操作流程:

示例:

bool CH347_SPI_Slave_Init()
{bool ret;mSpiCfgS SpiCfg = { 0 };/* set spi interface in spi slave mode, [mode3] & [MSB] & output [0xFF] by default */SpiCfg.iByteOrder = 1;SpiCfg.iSpiOutDefaultData = 0xFF;SpiCfg.iMode = 0x83;/* init spi interface */ret = CH347SPI_Init(ch347device.fd, &SpiCfg);if (!ret) {printf("Failed to init SPI interface.\n");return false;} else {printf("SPI init slave ok.\n");}return true;
}void ch34x_demo_spi_slave_operate(bool enable)
{bool ret = false;uint8_t oBuffer[SPI_SLAVE_MAX_LENGTH];uint32_t oLength;int i;if (enable) {ret = CH347_SPI_Slave_Init();if (ret == false) {printf("Failed to init CH347 SPI interface.\n");return;}printf("CH347 SPI interface init succeed.\n");ret = CH347SPI_Slave_FIFOReset(ch347device.fd);if (ret == false) {printf("Failed to reset SPI slave fifo.\n");return;}printf("CH347 SPI slave fifo reset succeed.\n");ret = CH347SPI_Slave_Control(ch347device.fd, true);if (!ret)return;printf("Begin read data in slave mode.\n");while (1) {ret = CH347SPI_Slave_QweryData(ch347device.fd, &oLength);if (!ret) {printf("CH347SPI_Slave_QweryData failed.\n");goto exit;}if (oLength == 0) {usleep(10 * 1000);continue;}ret = CH347SPI_Slave_ReadData(ch347device.fd, oBuffer, &oLength);if (!ret) {printf("CH347SPI_Slave_ReadData failed.\n");goto exit;}printf("\nRead Slave data, len: %d, contents:\n", oLength);for (i = 0; i < oLength; i++) {printf("%02x ", oBuffer[i]);if (((i + 1) % 20) == 0)putchar(20);}putchar(20);}} elseret = CH347SPI_Slave_Control(ch347device.fd, false);return;exit:CH347SPI_Slave_Control(ch347device.fd, false);
}

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

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

相关文章

【springcloud-config】配置中心客户端导入依赖spring-cloud-config-server后,maven一直爆红问题解决

问题描述 配置中心客户端导入了 spring-cloud-config-server 后&#xff0c;导入依赖爆红&#xff1b; 解决办法&#xff1a; 参考官网中文文档&#xff1a;spring-cloud -config 配置中心 中文文档 补充导入 spring-config-starter-config 配置即可 <!--springcloud-c…

跨境商城源码可以支持多种营销推广方式吗?

一、多种营销推广方式的重要性 跨境商城源码作为现代电商领域的重要工具&#xff0c;其支持多种营销推广方式对于吸引用户、增加销量以及提升品牌影响力都至关重要。通过采用多种营销推广方式&#xff0c;商家可以全方位地宣传和推广产品&#xff0c;吸引更多的潜在顾客&#x…

JS多选答题时,选项互斥时的情况

在做答题类的项目时&#xff0c;应该会比较常见多选题选相互斥的问题&#xff0c;例如&#xff1a; 你喜欢什么颜色&#xff1f;&#xff08;&#xff09;A、红色B、紫色C、蓝色D、灰色E、均无如该题&#xff0c;当选择选项E时&#xff0c;明显与其他选项互斥。这个时候经常会…

SAP 公司间销售

一、 概述 很多项目中&#xff0c;特别是集团型公司&#xff0c;生产总部在某地&#xff0c;但是在各个省会城市&#xff0c;乃至国外都有相应的贸易公司&#xff0c;特别是国外&#xff0c;此时贸易公司接到客户采购订单&#xff0c;但是贸易公司没有库存&#xff0c;甚至没有…

Zoho Mail荣登福布斯小型企业企业邮箱排行榜

在过去的数十载里&#xff0c;电子邮件已成为电子通信领域中不可或缺的一环&#xff0c;而在未来的岁月里&#xff0c;它有望继续在全球范围内普及应用。尽管如今市场上有许多免费的企业邮箱供用户和企业选用&#xff0c;但其中许多产品在特定场景下的专业化功能尚显不足&#…

selenium多窗口、多iframe切换、alert切换

多标签/多窗口之间的切换 场景&#xff1a; 在页面操作过程中有时候点击某个链接会弹出新的窗口&#xff0c;这时就需要切换到新打开的窗口上进行操作。这种情况下&#xff0c;需要识别多标签或窗口的情况。 操作方法&#xff1a; switch_to.window()方法&#xff1a;切换窗口…

如何高效的开展app的性能测试?

APP性能测试是什么 从网上查了一下&#xff0c;貌似也没什么特别的定义&#xff0c;我这边根据自己的经验给出一个自己的定义&#xff0c;如有巧合纯属雷同。 客户端性能测试就是&#xff0c;从业务和用户的角度出发&#xff0c;设计合理且有效的性能测试场景&#xff0c;制定…

js给一段话,遇到的第一个括号处加上换行符

list.forEach((item,index0)>{const productName item.name;const index productName.indexOf(&#xff08;);if (index -1) {return productName;}const before productName.slice(0, index);const after productName.slice(index);item.namebefore \n after;});

吃透Spring源码分析专题

想说的话 本人在互联网摸爬滚打至今(23年)6年了&#xff0c;平时有写博客的习惯&#xff0c;这个习惯是从大学的时候开始的&#xff0c;目前主要关注java领域相关的技术&#xff0c;python也有涉及&#xff0c;写Spring专题是因为Spring确实很重要&#xff0c;在目前这个开发模…

【C++】二叉树进阶 -- 详解

一、二叉搜索树概念 二叉搜索树 又称二叉排序树&#xff0c;它或者是一棵空树&#xff0c;或者是具有以下性质的二叉树&#xff1a; 若它的左子树不为空&#xff0c;则左子树上所有节点的值都小于根节点的值 若它的右子树不为空&#xff0c;则右子树上所有节点的值都大于根节点…

AI驱动的图纸数据提取

推荐&#xff1a;用 NSDT编辑器 快速搭建可编程3D场景 你是否曾经需要组合来自两个不同来源&#xff08;例如图像和文本&#xff09;的对象数据&#xff1f; 我们在工作的过程中经常面临这样的挑战。 在这里&#xff0c;我们展示了技术绘图领域的一个示例。 此类图纸用于许多领…

Python数据挖掘 | 升级版自动查核酸

&#x1f4d5;作者简介&#xff1a;热爱跑步的恒川&#xff0c;致力于C/C、Java、Python等多编程语言&#xff0c;热爱跑步&#xff0c;喜爱音乐的一位博主。 &#x1f4d7;本文收录于恒川的日常汇报系列&#xff0c;大家有兴趣的可以看一看 &#x1f4d8;相关专栏C语言初阶、C…

Mac电脑怎么在Dock窗口预览,Dock窗口预览工具DockView功能介绍

DockView是一款Mac电脑上的软件&#xff0c;它可以增强Dock的功能&#xff0c;让用户更方便地管理和切换应用程序。 DockView的主要功能是在 DockQ&#xff0c;栏上显示每个窗口的缩略图&#xff0c;并提供了一些相关的操作选项。当用户将鼠标悬停在Dock栏上的应用程序图标上时…

硬件信息查看工具 EtreCheckpro mac中文版功能介绍

etrecheckpro mac中文版是一款专业的硬件信息查看工具&#xff0c;它能够快速的检测Mac电脑的软硬件信息&#xff0c;加强用户对自己计算机的了解&#xff0c;EtreCheckPro for Mac下载首先会对电脑的软硬件信息进行扫描收集&#xff0c;之后才会显示出来。EtreCheck Mac版报告…

解决:无法打开Zotero数据库

在实验室电脑上面下载了Zotero和坚果云&#xff0c;一系列操作下来&#xff0c;我的笔记本上无法打开Zotero数据库了&#xff01;显示下面的界面&#xff1a; 于是网上找解决方法 1.https://www.zhihu.com/question/519740718 2.https://devpress.csdn.net/awstech/64e7311b…

服务器数据恢复-raid0硬盘故障导致服务器崩溃的数据恢复案例

服务器故障&分析&#xff1a; 某公司一台服务器&#xff0c;共2块磁盘组建了一组raid0磁盘阵列。RAID0阵列中有1块硬盘出现故障离线&#xff0c;导致服务器崩溃。 通过和用户方的沟通&#xff0c;北亚企安数据恢复工程师初步判断&#xff1a;故障服务器中出现故障离线的硬盘…

C语言里的static变量其他语言是看不上还是学不去?

C语言里的static变量其他语言是看不上还是学不去? static变量在C语言中被用于具有静态存储期的局部变量或全局变量。它有以下几个特点&#xff1a; 1. 静态存储期&#xff1a;static变量在程序执行时分配内存&#xff0c;直到程序结束才会释放&#xff0c;其生命周期与程序的…

短视频矩阵系统源码/源头搭建技术交付

一、短视频矩阵系统&#xff0c;短视频矩阵源码技术 1、抖音开放平台申请账号&#xff0c;快手平台申请账号&#xff1b;阿里云混剪接口。 2、系统总台支持OEM代理&#xff0c;可以按点数管理。 3、代理功能。包括是否允许再次开二级代理、是否允许OEM等。 4、可支持一条龙…

小白也会的校园网宽带拨号自动重连设置

开始菜单搜索“任务计划程序”&#xff1a;我这个开始菜单和你们的不太一样&#xff0c;用了StartAllBack设置的&#xff0c;总之能找到这个程序就行了 提示&#xff1a;可以按下“Win R”&#xff0c;打开“运行”&#xff0c;输入taskschd.msc来打开任务计划程序 点击“任务…

基于卷积神经网络的乳腺癌分类 深度学习 医学图像 计算机竞赛

文章目录 1 前言2 前言3 数据集3.1 良性样本3.2 病变样本 4 开发环境5 代码实现5.1 实现流程5.2 部分代码实现5.2.1 导入库5.2.2 图像加载5.2.3 标记5.2.4 分组5.2.5 构建模型训练 6 分析指标6.1 精度&#xff0c;召回率和F1度量6.2 混淆矩阵 7 结果和结论8 最后 1 前言 &…