当前位置: 首页 > news >正文

spi 驱动-数据发送流程分析

总结

        核心函数是spi_sync, 设备驱动->核心函数-> 控制器驱动

 实例分析

(gdb) c
Continuing.Thread 115 hit Breakpoint 1, bcm2835_spi_transfer_one (master=0xffffffc07b8e6000, spi=0xffffffc07b911800, tfr=0xffffff8009f53c40) at drivers/spi/spi-bcm2835.c:534
534	{
(gdb) bt
#0  bcm2835_spi_transfer_one (master=0xffffffc07b8e6000, spi=0xffffffc07b911800, tfr=0xffffff8009f53c40) at drivers/spi/spi-bcm2835.c:534
#1  0xffffff80086cbe8c in spi_transfer_one_message (ctlr=0xffffffc07b8e6000, msg=0xffffff8009f53ca0) at drivers/spi/spi.c:1031
#2  0xffffff80086cc990 in __spi_pump_messages (ctlr=0xffffffc07b8e6000, in_kthread=<optimized out>) at drivers/spi/spi.c:1265
#3  0xffffff80086cccb8 in __spi_sync (spi=0xffffffc07b911800, message=0xffffff8009f53ca0) at drivers/spi/spi.c:3129
#4  0xffffff80086ccd04 in spi_sync (spi=0xffffffc07b911800, message=0xffffff8009f53ca0) at drivers/spi/spi.c:3165
#5  0xffffff80086ce14c in spidev_sync (spidev=<optimized out>, message=0xffffff8009f53ca0) at drivers/spi/spidev.c:112
#6  0xffffff80086ce2f0 in spidev_sync_write (len=<optimized out>, spidev=<optimized out>) at drivers/spi/spidev.c:132
#7  spidev_write (filp=<optimized out>, buf=<optimized out>, count=4, f_pos=<optimized out>) at drivers/spi/spidev.c:199
#8  0xffffff80082c919c in __vfs_write (file=0xffffffc072ea2500, p=0x5587f7cb60 "555\n/dev/spidev0.0\n", count=4, pos=0xffffff8009f53db0) at fs/read_write.c:485
#9  0xffffff80082c93e0 in vfs_write (file=0xffffffc072ea2500, buf=0x5587f7cb60 "555\n/dev/spidev0.0\n", count=4, pos=0xffffff8009f53db0) at fs/read_write.c:549
#10 0xffffff80082c9768 in ksys_write (fd=<optimized out>, buf=0x5587f7cb60 "555\n/dev/spidev0.0\n", count=4) at fs/read_write.c:599
#11 0xffffff80082c9810 in __do_sys_write (count=<optimized out>, buf=<optimized out>, fd=<optimized out>) at fs/read_write.c:611
#12 __se_sys_write (fd=1, buf=367353383776, count=4) at fs/read_write.c:608
#13 0xffffff80082c984c in __arm64_sys_write (regs=0xffffff8009f53ec0) at fs/read_write.c:608
#14 0xffffff8008099254 in __invoke_syscall (regs=0xffffff8009f53ec0, syscall_fn=0xffffff80082c9820 <__arm64_sys_write>) at arch/arm64/kernel/syscall.c:36
#15 0xffffff80080993a4 in invoke_syscall (regs=0xffffff8009f53ec0, scno=<optimized out>, sc_nr=<optimized out>, syscall_table=0xffffff80089c0778 <sys_call_table>) at arch/arm64/kernel/syscall.c:48
#16 0xffffff800809943c in el0_svc_common (regs=0xffffff8009f53ec0, scno=64, sc_nr=294, syscall_table=0xffffff80089c0778 <sys_call_table>) at arch/arm64/kernel/syscall.c:114
#17 0xffffff8008099524 in el0_svc_handler (regs=0xffffff8009f53ec0) at arch/arm64/kernel/syscall.c:160
#18 0xffffff8008084088 in el0_svc () at arch/arm64/kernel/entry.S:917

源码分析

1. 注册字符设备并提供操作函数 spidev_fops

static int __init spidev_init(void)status = register_chrdev(SPIDEV_MAJOR, "spi", &spidev_fops);
}static const struct file_operations spidev_fops = {.....write =    spidev_write,....
};


2. 分析spidev_write 调用spi_sync

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;spidev = filp->private_data;missing = copy_from_user(spidev->tx_buffer, buf, count);if (missing == 0)status = spidev_sync_write(spidev, count);....}
static inline ssize_t
spidev_sync_write(struct spidev_data *spidev, size_t len)
{struct spi_transfer    t = {.tx_buf        = spidev->tx_buffer,.len        = len,.speed_hz    = spidev->speed_hz,};struct spi_message    m;spi_message_init(&m);spi_message_add_tail(&t, &m);return spidev_sync(spidev, &m);
}static ssize_t
spidev_sync(struct spidev_data *spidev, struct spi_message *message)
{int status;struct spi_device *spi;spi = spidev->spi;status = spi_sync(spi, message);return status;
}


3. 分析 spi_sync 调用 ctlr->transfer_one_message

int spi_sync(struct spi_device *spi, struct spi_message *message)
{ret = __spi_sync(spi, message);
}static int __spi_sync(struct spi_device *spi, struct spi_message *message)
{DECLARE_COMPLETION_ONSTACK(done);int status;struct spi_controller *ctlr = spi->controller;unsigned long flags;status = __spi_validate(spi, message);if (status != 0)return status;message->complete = spi_complete;message->context = &done;message->spi = spi;SPI_STATISTICS_INCREMENT_FIELD(&ctlr->statistics, spi_sync);SPI_STATISTICS_INCREMENT_FIELD(&spi->statistics, spi_sync);if (ctlr->transfer == spi_queued_transfer) {   //新方法status = __spi_queued_transfer(spi, message, false);} else {                    //legacy method //老方法status = spi_async_locked(spi, message);}if (status == 0) {/* Push out the messages in the calling context if we* can.*/if (ctlr->transfer == spi_queued_transfer) {SPI_STATISTICS_INCREMENT_FIELD(&ctlr->statistics, spi_sync_immediate);SPI_STATISTICS_INCREMENT_FIELD(&spi->statistics, spi_sync_immediate);__spi_pump_messages(ctlr, false); //处理spi 消息}wait_for_completion(&done);status = message->status;}message->context = NULL;return status;
}
static int __spi_queued_transfer(struct spi_device *spi,struct spi_message *msg,bool need_pump)
{struct spi_controller *ctlr = spi->controller;unsigned long flags;...list_add_tail(&msg->queue, &ctlr->queue); 将消息队列添加到队列的尾部if (!ctlr->busy && need_pump)kthread_queue_work(&ctlr->kworker, &ctlr->pump_messages);....return 0;
}
static void __spi_pump_messages(struct spi_controller *ctlr, bool in_kthread)
{unsigned long flags;bool was_busy = false;int ret;/* Lock queue */spin_lock_irqsave(&ctlr->queue_lock, flags);/* Make sure we are not already running a message */if (ctlr->cur_msg) {spin_unlock_irqrestore(&ctlr->queue_lock, flags);return;}/* If another context is idling the device then defer */if (ctlr->idling) {kthread_queue_work(&ctlr->kworker, &ctlr->pump_messages);spin_unlock_irqrestore(&ctlr->queue_lock, flags);return;}/* Check if the queue is idle */if (list_empty(&ctlr->queue) || !ctlr->running) {if (!ctlr->busy) {spin_unlock_irqrestore(&ctlr->queue_lock, flags);return;}/* Only do teardown in the thread */if (!in_kthread) {kthread_queue_work(&ctlr->kworker,&ctlr->pump_messages);spin_unlock_irqrestore(&ctlr->queue_lock, flags);return;}ctlr->busy = false;ctlr->idling = true;spin_unlock_irqrestore(&ctlr->queue_lock, flags);kfree(ctlr->dummy_rx);ctlr->dummy_rx = NULL;kfree(ctlr->dummy_tx);ctlr->dummy_tx = NULL;if (ctlr->unprepare_transfer_hardware &&ctlr->unprepare_transfer_hardware(ctlr))dev_err(&ctlr->dev,"failed to unprepare transfer hardware\n");if (ctlr->auto_runtime_pm) {pm_runtime_mark_last_busy(ctlr->dev.parent);pm_runtime_put_autosuspend(ctlr->dev.parent);}trace_spi_controller_idle(ctlr);spin_lock_irqsave(&ctlr->queue_lock, flags);ctlr->idling = false;spin_unlock_irqrestore(&ctlr->queue_lock, flags);return;}/* Extract head of queue */ctlr->cur_msg =list_first_entry(&ctlr->queue, struct spi_message, queue);list_del_init(&ctlr->cur_msg->queue);if (ctlr->busy)was_busy = true;elsectlr->busy = true;spin_unlock_irqrestore(&ctlr->queue_lock, flags);mutex_lock(&ctlr->io_mutex);if (!was_busy && ctlr->auto_runtime_pm) {ret = pm_runtime_get_sync(ctlr->dev.parent);if (ret < 0) {pm_runtime_put_noidle(ctlr->dev.parent);dev_err(&ctlr->dev, "Failed to power device: %d\n",ret);mutex_unlock(&ctlr->io_mutex);return;}}if (!was_busy)trace_spi_controller_busy(ctlr);if (!was_busy && ctlr->prepare_transfer_hardware) {ret = ctlr->prepare_transfer_hardware(ctlr);if (ret) {dev_err(&ctlr->dev,"failed to prepare transfer hardware\n");if (ctlr->auto_runtime_pm)pm_runtime_put(ctlr->dev.parent);mutex_unlock(&ctlr->io_mutex);return;}}trace_spi_message_start(ctlr->cur_msg);if (ctlr->prepare_message) {ret = ctlr->prepare_message(ctlr, ctlr->cur_msg);if (ret) {dev_err(&ctlr->dev, "failed to prepare message: %d\n",ret);ctlr->cur_msg->status = ret;spi_finalize_current_message(ctlr);goto out;}ctlr->cur_msg_prepared = true;}ret = spi_map_msg(ctlr, ctlr->cur_msg);if (ret) {ctlr->cur_msg->status = ret;spi_finalize_current_message(ctlr);goto out;}ret = ctlr->transfer_one_message(ctlr, ctlr->cur_msg); 调用控制器的传输函数if (ret) {dev_err(&ctlr->dev,"failed to transfer one message from queue\n");goto out;}out:mutex_unlock(&ctlr->io_mutex);/* Prod the scheduler in case transfer_one() was busy waiting */if (!ret)cond_resched();
}

4.  spi controler "传输函数" 之间的相互调用


        # ctlr->transfer =           spi_queued_transfer;  // 调用  ctlr->transfer_one_message
        ctlr->transfer_one_message = spi_transfer_one_message; //ctlr->transfer_one

        ctrl->transfer_one = bcm2835_spi_transfer_one;

相关文章:

spi 驱动-数据发送流程分析

总结 核心函数是spi_sync&#xff0c; 设备驱动->核心函数-> 控制器驱动 实例分析 (gdb) c Continuing.Thread 115 hit Breakpoint 1, bcm2835_spi_transfer_one (master0xffffffc07b8e6000, spi0xffffffc07b911800, tfr0xffffff8009f53c40) at drivers/spi/spi-bcm2835…...

平面分割--------PCL

平面分割 bool PclTool::planeSegmentation(pcl::PointCloud<pcl::PointXYZ>::Ptr cloud, pcl::ModelCoefficients::Ptr coefficients, pcl::PointIndices::Ptr inliers) {std::cout << "Point cloud data: " << cloud->points.size() <<…...

前端之深拷贝

前提&#xff1a; 就是在实际开发中&#xff0c;我有一个编辑的弹窗&#xff0c;可以查看和编辑&#xff0c;因为弹窗里面是一个步骤条&#xff0c;点击下一步就要向对应的接口发送请求&#xff0c;考虑到就比如我点击下一步&#xff0c;此次表箱信息其实不需要修改&#xff0…...

2024年 Java 面试八股文——SpringCloud篇

目录 1.Spring Cloud Alibaba 中的 Nacos 是如何进行服务注册和发现的&#xff1f; 2.Spring Cloud Alibaba Sentinel 的流量控制规则有哪些&#xff1f; 3.Spring Cloud Alibaba 中如何实现分布式配置管理&#xff1f; 4.Spring Cloud Alibaba RocketMQ 的主要特点有哪些&…...

linux C语言Makefile

ChatGPT 在Linux中使用Makefile来自动化C语言项目的构建过程是很普遍的实践。Makefile是一个包含了一系列构建目标及如何构建这些目标的依赖和规则的文本文件。 一个基本的Makefile例子可能会像这样&#xff1a; # 定义编译器 CCgcc# 定义编译选项 CFLAGS-I.# 定义可执行文件…...

pgvector扩展在IvorySQL Oracle兼容模式下的应用实践

向量数据库是生成式人工智能(GenAI)的关键组成部分。作为PostgreSQL的重要扩展&#xff0c;pgvector支持高达16000维的向量计算能力&#xff0c;使得PostgreSQL能够直接转化为高效的向量数据库。 IvorySQL基于PostgreSQL开发&#xff0c;因此它同样支持添加pgvector扩展。在Ora…...

c++ 线程概述

C中的线程是并发编程的重要组成部分&#xff0c;它允许程序同时执行多个任务。以下是对C线程的概述&#xff1a; 基本概念&#xff1a; 并发&#xff1a;意味着两个或多个任务同时执行。在单核CPU上&#xff0c;由于只有一个CPU&#xff0c;某一时刻只能执行一个任务&#xff0…...

纯血鸿蒙APP实战开发——短视频切换实现案例

短视频切换实现案例 介绍 短视频切换在应用开发中是一种常见场景&#xff0c;上下滑动可以切换视频&#xff0c;十分方便。本模块基于Swiper组件和Video组件实现短视频切换功能。 效果图预览 使用说明 上下滑动可以切换视频。点击屏幕暂停视频&#xff0c;再次点击继续播放…...

36.Docker-Dockerfile自定义镜像

镜像结构 镜像是将应用程序及其需要的系统函数库、环境、配置、依赖打包而成。 镜像是分层机构&#xff0c;每一层都是一个layer BaseImage层&#xff1a;包含基本的系统函数库、环境变量、文件系统 EntryPoint:入口&#xff0c;是镜像中应用启动的命令 其他&#xff1a;在…...

【webrtc】MessageHandler 4: 基于线程的消息处理:以Fake 收发包模拟为例

G:\CDN\rtcCli\m98\src\media\base\fake_network_interface.h// Fake NetworkInterface that sends/receives RTP/RTCP packets.虚假的网络接口,用于模拟发送包、接收包单纯仅是处理一个ST_RTP包 消息的id就是ST_RTP 类型,– 然后给到目的地:mediachannel处理: 最后消息消…...

C#运算符“/”使用方法

C#中&#xff0c;当需要对两个整数进行除法运算时&#xff0c;结果会被截断为整数部分&#xff0c;即使结果本应是一个小数。这是因为整数除法会丢弃小数部分&#xff0c;只保留整数部分。 要想保留小数部分&#xff0c;需要将至少其中一个操作数转换为float、double或者 deci…...

虚拟机网络桥接模式无法通信,获取到的ip为169.254.X.X

原因&#xff1a;VMware自动选择的网卡可能不对 解决&#xff1a;编辑-虚拟网络编辑器-更改桥接模式-选择宿主机物理网卡&#xff0c;断开虚拟机网络连接后重新连接即可...

【数据结构】初识数据结构

引入&#xff1a; 哈喽大家好&#xff0c;我是野生的编程萌新&#xff0c;首先感谢大家的观看。数据结构的学习者大多有这样的想法&#xff1a;数据结构很重要&#xff0c;一定要学好&#xff0c;但数据结构比较抽象&#xff0c;有些算法理解起来很困难&#xff0c;学的很累。我…...

相机知识的补充

一&#xff1a;镜头 1.1MP的概念 相机中MP的意思是指百万像素。MP是mega pixel的缩写。mega意为一百万&#xff0c;mega pixel 指意为100万像素。“像素”是相机感光器件上的感光最小单位。就像是光学相机的感光胶片的银粒一样&#xff0c;记忆在数码相机的“胶片”&#xff…...

在Linux操作系统中实现磁盘开机自动挂载

当一个分区创建好&#xff0c;然后文件系统创建完毕之后&#xff0c; 需要使用mount命令将分区挂载到空目录上&#xff0c;这个挂载关系是临时的&#xff0c;也就是说当重启机器的时候&#xff0c;硬盘分区于空目录之间的挂载关系就会解除。 磁盘于目录之间的挂载关系断开意味…...

单片机编程实例400例大全(100-200)

今天继续分享单片机编程实例第100-200例。 今天的实例会比前面100复杂一些&#xff0c;我大概看了下&#xff0c;很多都具备实际产品的参考价值。 今天继续分享单片机编程实例第100-200例。 今天的实例会比前面100复杂一些&#xff0c;我大概看了下&#xff0c;很多都具备实际…...

新兴游戏引擎Godot vs. 主流游戏引擎Unity和虚幻引擎,以及版本控制工具Perforce Helix Core如何与其高效集成

游戏行业出现一个新生事物——Godot&#xff0c;一个免费且开源的2D和3D游戏引擎。曾经由Unity和虚幻引擎&#xff08;Unreal Engine&#xff09;等巨头主导的领域如今迎来了竞争对手。随着最近“独特”定价模式的变化&#xff0c;越来越多的独立开发者和小型开发团队倾向于选择…...

Leetcode—1652. 拆炸弹【简单】

2024每日刷题&#xff08;127&#xff09; Leetcode—1652. 拆炸弹 实现代码 class Solution { public:vector<int> decrypt(vector<int>& code, int k) {int codeSize code.size();vector<int> ans(codeSize, 0);if(k 0) {return ans;}if(k > 0)…...

JAVASE---抽象类相关

instanceof 和类型转换 System.out.println(X instanceof Y );主要看X与Y之间是否存在父子&#xff08;继承&#xff09;关系&#xff0c;如果存在则编译可完成&#xff0c;否则无法 进行编译。 1.父类引用指向子类的对象 2.把子类转换为父类&#xff0c;向上转型; 3.把父类转…...

深入理解C++中的inline函数

在C编程中&#xff0c;我们经常会遇到inline关键字&#xff0c;它用于修饰函数&#xff0c;以建议编译器将该函数的调用替换为函数体的直接拷贝。这就是inline函数的基本概念。然而&#xff0c;inline函数并非真正意义上的函数&#xff0c;而只是一种"在调用点插入函数体&…...

CSS如何制作响应式图片集布局_利用object-fit填充空间

object-fit 应选 cover、contain 或 scale-down&#xff1a;cover 等比缩放裁剪填满&#xff0c;适合封面&#xff1b;contain 等比缩放留白显全图&#xff0c;适合证件照&#xff1b;scale-down 仅大图缩放防模糊。图片容器宽高固定时&#xff0c;object-fit 怎么选固定尺寸容…...

互联网大厂Java求职者面试全流程解析(含技术点详解)

互联网大厂Java求职者面试全流程解析&#xff08;含技术点详解&#xff09; 文章标签 Java,Spring Boot,面试,互联网大厂,技术详解,微服务,缓存,消息队列 文章简述 本文模拟了互联网大厂Java岗位的面试过程&#xff0c;采用严肃面试官与搞笑程序员谢飞机的故事方式展开。文章涵…...

SketchUp动态组件实战:用onClick函数制作一个能自动开关的门(附完整公式)

SketchUp动态组件实战&#xff1a;用onClick函数制作自动开关门 在建筑可视化与室内设计领域&#xff0c;交互式组件能显著提升方案展示的专业度与趣味性。想象一下&#xff0c;当客户点击你模型中的门扇时&#xff0c;它能像真实物体一样旋转开合——这种动态演示效果远比静态…...

如何3分钟破解网盘限速:八大平台直链下载助手完整指南

如何3分钟破解网盘限速&#xff1a;八大平台直链下载助手完整指南 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 &#xff0c;支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼…...

别再手动登录了!用VBS脚本自动打开Chrome并填写表单(附完整代码)

解放双手&#xff1a;用VBS脚本实现Chrome自动化表单填写全攻略 每次打开浏览器、输入网址、填写账号密码、点击登录...这些重复性操作是否让你感到厌倦&#xff1f;对于测试工程师、运维人员或经常需要处理批量表单的行政人员来说&#xff0c;这类机械操作不仅耗时耗力&#x…...

【卷卷观察】GitHub Star 造假产业链,以及它是怎么变成 AI 圈融资货币的

结论先说&#xff1a;GitHub Trending 上的高星项目&#xff0c;有相当比例的星是买来的。这不是小范围的作弊&#xff0c;是一个成熟的、面向 VC 融资市场的地下经济。而且 AI 项目是重灾区。上周看到一条 Twitter&#xff0c;大意是&#xff1a;"GitHub Trending 上的项…...

Flink Watermark 设计分析

Flink Watermark 演进分析 1. 核心痛点&#xff1a;如何衡量事件时间进度&#xff1f; 在乱序流中&#xff0c;直接使用“当前看到的最大时间戳”作为进度会导致窗口过早关闭。系统需要一种机制来声明&#xff1a;“我认为这个时间点之前的数据已经全部到齐”。 Watermark 就…...

从.NET 8到.NET 9 Preview 5:C# 14 AOT编译Dify客户端的兼容性断层分析,3大Breaking Change已致17家客户生产环境回滚

第一章&#xff1a;从.NET 8到.NET 9 Preview 5的AOT演进全景图.NET 平台的 AOT&#xff08;Ahead-of-Time&#xff09;编译能力在 .NET 8 中首次成为生产就绪特性&#xff0c;而 .NET 9 Preview 5 则标志着其成熟度与工程化落地的重大跃迁。这一演进不仅体现在性能提升与二进制…...

不止于仿真:如何将Prescan十字路口碰撞结果导入Python进行数据分析与可视化(附代码)

从仿真到决策&#xff1a;Prescan十字路口碰撞数据的Python深度分析实战 在自动驾驶系统开发过程中&#xff0c;仿真测试是不可或缺的关键环节。Prescan与Simulink的强强联合为工程师们提供了高效的虚拟测试环境&#xff0c;但仿真结束后的数据分析往往被忽视。本文将带您突破传…...

JASP:零成本实现专业级统计分析的完全免费开源工具

JASP&#xff1a;零成本实现专业级统计分析的完全免费开源工具 【免费下载链接】jasp-desktop JASP aims to be a complete statistical package for both Bayesian and Frequentist statistical methods, that is easy to use and familiar to users of SPSS 项目地址: http…...