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

Linux驱动开发进阶(九)- SPI子系统BSP驱动

文章目录

  • 1、前言
  • 2、SPI总线注册
  • 3、SPI设备注册
  • 4、SPI驱动注册
  • 5、SPI BSP驱动

1、前言

  1. 学习参考书籍以及本文涉及的示例程序:李山文的《Linux驱动开发进阶》
  2. 本文属于个人学习后的总结,不太具备教学功能。

2、SPI总线注册

驱动源码文件:drivers/spi/spi.c,负责注册spi总线和spi设备以及spi控制器设备:

上面的spi_init是整个总线注册的入口,但是没有spi_exit出口函数。这一点很多驱动的总线也是这样,只有注册,没有注销,这是因为总线会一直存在,只有当系统关闭时,总线才不需要了。相关的spi总线结构体如下:

除了注册bus,还注册了class:

相关class结构体为:

注册类后,会在/sys/class目录下生成spi_master文件夹,而dev_groups会在/sys/class/spi_master目录下生成相关的属性文件分组目录。

3、SPI设备注册

在注册SPI控制器驱动时,此时会扫描设备树中的spi控制器节点,找到匹配的,则会注册成设备,并且会将控制器设备树节点下的子节点全部注册为spi设备。如下代码所示,spi注册控制器时,在最后会调用of_register_spi_devices函数来解析设备树,然后将其注册为spi设备。

int spi_register_controller(struct spi_controller *ctlr)
{.../* Register devices from the device tree and ACPI */of_register_spi_devices(ctlr);	// 将spi子设备注册进spi总线acpi_register_spi_devices(ctlr);return status;...
}

4、SPI驱动注册

SPI驱动分为控制器驱动和设备驱动,其中控制器驱动又称为BSP驱动程序,该驱动程序一般由芯片原厂的驱动工程师来编写,而设备驱动程序指对芯片的外部设备来编写驱动。

先看SPI控制器驱动的注册,该驱动注册由platform总线注册,这是因为在SPI控制器驱动注册前,SPI的设备驱动无法注册,原因就是SPI子设备驱动的注册依赖于SPI控制器驱动,只有当SPI控制器驱动注册完毕后,module_spi_driver宏才能使用(实际上是spi_register_driver和spi_unregister_driver函数)。所以SPI控制器驱动注册只能依靠platform平台总线注册,因为platform平台总线驱动编译到内核中的,一定会注册, 即module_platform_driver宏一定是可以使用的。

5、SPI BSP驱动

spi bsp驱动其实就是编写spi的控制器驱动。linux中spi控制器被抽象为spi_master或者spi_controller,在bsp中,我们常用spi_master,结构体如下:

struct spi_controller {struct device	dev;struct list_head list;s16			bus_num;u16			num_chipselect;u16			dma_alignment;u32			mode_bits;u32			buswidth_override_bits;u32			bits_per_word_mask;
#define SPI_BPW_MASK(bits) BIT((bits) - 1)
#define SPI_BPW_RANGE_MASK(min, max) GENMASK((max) - 1, (min) - 1)u32			min_speed_hz;u32			max_speed_hz;u16			flags;
#define SPI_CONTROLLER_HALF_DUPLEX	BIT(0)	/* can't do full duplex */
#define SPI_CONTROLLER_NO_RX		BIT(1)	/* can't do buffer read */
#define SPI_CONTROLLER_NO_TX		BIT(2)	/* can't do buffer write */
#define SPI_CONTROLLER_MUST_RX		BIT(3)	/* requires rx */
#define SPI_CONTROLLER_MUST_TX		BIT(4)	/* requires tx */
#define SPI_MASTER_GPIO_SS		BIT(5)	/* GPIO CS must select slave */bool			slave;size_t (*max_transfer_size)(struct spi_device *spi);size_t (*max_message_size)(struct spi_device *spi);struct mutex		io_mutex;spinlock_t		bus_lock_spinlock;struct mutex		bus_lock_mutex;bool			bus_lock_flag;int			(*setup)(struct spi_device *spi);int (*set_cs_timing)(struct spi_device *spi, struct spi_delay *setup,struct spi_delay *hold, struct spi_delay *inactive);int			(*transfer)(struct spi_device *spi,struct spi_message *mesg);void			(*cleanup)(struct spi_device *spi);bool			(*can_dma)(struct spi_controller *ctlr,struct spi_device *spi,struct spi_transfer *xfer);bool				queued;struct kthread_worker		*kworker;struct kthread_work		pump_messages;spinlock_t			queue_lock;struct list_head		queue;struct spi_message		*cur_msg;bool				idling;bool				busy;bool				running;bool				rt;bool				auto_runtime_pm;bool                            cur_msg_prepared;bool				cur_msg_mapped;bool				last_cs_enable;bool				last_cs_mode_high;bool                            fallback;struct completion               xfer_completion;size_t				max_dma_len;int (*prepare_transfer_hardware)(struct spi_controller *ctlr);int (*transfer_one_message)(struct spi_controller *ctlr,struct spi_message *mesg);int (*unprepare_transfer_hardware)(struct spi_controller *ctlr);int (*prepare_message)(struct spi_controller *ctlr,struct spi_message *message);int (*unprepare_message)(struct spi_controller *ctlr,struct spi_message *message);int (*slave_abort)(struct spi_controller *ctlr);void (*set_cs)(struct spi_device *spi, bool enable);int (*transfer_one)(struct spi_controller *ctlr, struct spi_device *spi,struct spi_transfer *transfer);void (*handle_err)(struct spi_controller *ctlr,struct spi_message *message);const struct spi_controller_mem_ops *mem_ops;struct spi_delay	cs_setup;struct spi_delay	cs_hold;struct spi_delay	cs_inactive;int			*cs_gpios;struct gpio_desc	**cs_gpiods;bool			use_gpio_descriptors;
#ifdef __GENKSYMS__u8			unused_native_cs;u8			max_native_cs;
#elses8			unused_native_cs;s8			max_native_cs;
#endifstruct spi_statistics	statistics;struct dma_chan		*dma_tx;struct dma_chan		*dma_rx;void			*dummy_rx;void			*dummy_tx;int (*fw_translate_cs)(struct spi_controller *ctlr, unsigned cs);bool			ptp_sts_supported;unsigned long		irq_flags;ANDROID_KABI_RESERVE(1);ANDROID_KABI_RESERVE(2);
};

以下列出几个比较重要字段的解释,后面会重点看看transfer_one_message字段,该字段实现真实数据的传输:

linux提供了一个函数来分配一个spi控制器:

static inline struct spi_controller *spi_alloc_master(struct device *host,unsigned int size)
{return __spi_alloc_controller(host, size, false);
}

然后使用如下函数注册:

int devm_spi_register_controller(struct device *dev,struct spi_controller *ctlr)

对于bsp而言,在使用前必须开启spi时钟。如下所示,设备树中spi时钟属性,用来指定时钟源:

spi bsp示例程序可以参考:李山文的《Linux驱动开发进阶》spi子系统示例程序。

下面贴出dummy-spimaster.c

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/device.h>
#include <linux/platform_device.h>
#include <linux/of.h>
#include <linux/spi/spi.h>
#include <linux/sysfs.h>
#include <linux/slab.h>static int dummy_spimaster_setup(struct spi_device *spi)
{int ret = 0;if(spi->chip_select >= spi->master->num_chipselect){printk(KERN_INFO " invalid chip_select\n");return -1;}return ret;
}static void dummy_spimaster_set_cs(struct spi_device *spi, bool enable)
{if (enable) {//set IO(spi->chip_select) high levelprintk(KERN_INFO "spi cs[%d]: high\n", spi->chip_select);}else {//set IO(spi->chip_select) low levelprintk(KERN_INFO "spi cs[%d]: low\n", spi->chip_select);}
}static int dummy_spimaster_transfer_one(struct spi_master *master, struct spi_device *spi, struct spi_transfer *tfr)
{int bytes;unsigned char *rx_buff = tfr->rx_buf;const unsigned char *tx_buff = tfr->tx_buf;bytes = tfr->len;while(bytes) {if(tx_buff) {printk(KERN_INFO "TX: %c \n", *tx_buff);tx_buff ++ ;}if(rx_buff) {*rx_buff = 'a';//读取时每次获取'a'rx_buff ++ ;}bytes -- ;}return 0;
}static int dummy_spimaster_probe(struct platform_device *pdev)
{int ret;struct spi_master *master;master = spi_alloc_master(&pdev->dev, 0);if (!master) {printk(KERN_ERR "unable to alloc SPI master\n");return -EINVAL;}master->dev.of_node = pdev->dev.of_node;master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_CS_HIGH | SPI_LSB_FIRST;master->bus_num = -1;master->auto_runtime_pm = false;//不支持PM,如果需要支持,则需要实现platform_driver中.driver.pmmaster->num_chipselect = 4;master->transfer_one = dummy_spimaster_transfer_one;master->setup = dummy_spimaster_setup;master->max_speed_hz = 100 * 1000 * 1000; //100MHzmaster->min_speed_hz = 3 * 1000;    //3kHzmaster->set_cs = dummy_spimaster_set_cs;ret = devm_spi_register_master(&pdev->dev, master);if (ret) {dev_err(&pdev->dev, "spi register master failed!\n");spi_master_put(master);return ret;}platform_set_drvdata(pdev, master);return 0;
} static int dummy_spimaster_remove(struct platform_device *pdev)
{struct spi_master	*master = platform_get_drvdata(pdev);spi_unregister_master(master);printk("%s:%d\n", __FUNCTION__, __LINE__);return 0;
}static const struct of_device_id dummy_spimaster_match[] = {{ .compatible = "dummy-spi-master", },{}
};
MODULE_DEVICE_TABLE(of, dummy_spimaster_match);static struct platform_driver dummy_spimaster_driver = {.driver = {.name = "dummy_spimaster_driver",.owner = THIS_MODULE,.of_match_table = dummy_spimaster_match,},.probe = dummy_spimaster_probe,.remove = dummy_spimaster_remove,//.driver.pm = 
};module_platform_driver(dummy_spimaster_driver);MODULE_LICENSE("GPL");
MODULE_AUTHOR("1477153217@qq.com");
MODULE_VERSION("0.1");
MODULE_DESCRIPTION("dummy spi controller driver test");

static int dummy_spimaster_transfer_one(struct spi_master *master, struct spi_device *spi, struct spi_transfer *tfr)
{int bytes;unsigned char *rx_buff = tfr->rx_buf;const unsigned char *tx_buff = tfr->tx_buf;bytes = tfr->len;while(bytes) {if(tx_buff) {printk(KERN_INFO "TX: %c \n", *tx_buff);tx_buff ++ ;}if(rx_buff) {*rx_buff = 'a';//读取时每次获取'a'rx_buff ++ ;}bytes -- ;}return 0;
}

接下来再看spi设备驱动程序:

相关文章:

Linux驱动开发进阶(九)- SPI子系统BSP驱动

文章目录 1、前言2、SPI总线注册3、SPI设备注册4、SPI驱动注册5、SPI BSP驱动 1、前言 学习参考书籍以及本文涉及的示例程序&#xff1a;李山文的《Linux驱动开发进阶》本文属于个人学习后的总结&#xff0c;不太具备教学功能。 2、SPI总线注册 驱动源码文件&#xff1a;dri…...

wabpack学习记录

wabpack学习记录 前言 项目写了不少 对webpack了解甚少 只记住一些 必要的概念以及指令 所以像深究一下具体是什么 可以做什么 如何做等 package.json 文件详解 name: 项目的名称。 version: 项目的版本号。 description: 项目的描述。 author: 项目的作者或维护者信息。 l…...

计算机视觉——基于 Yolov8 目标检测与 OpenCV 光流实现目标追踪

1. 概述 目标检测&#xff08;Object Detection&#xff09;和目标追踪&#xff08;Object Tracking&#xff09;是计算机视觉中的两个关键技术&#xff0c;它们在多种实际应用场景中发挥着重要作用。 目标检测指的是在静态图像或视频帧中识别出特定类别的目标对象&#xff0…...

React 更新 state 中的数组

更新 state 中的数组 数组是另外一种可以存储在 state 中的 JavaScript 对象&#xff0c;它虽然是可变的&#xff0c;但是却应该被视为不可变。同对象一样&#xff0c;当你想要更新存储于 state 中的数组时&#xff0c;你需要创建一个新的数组&#xff08;或者创建一份已有数组…...

[250415] OpenAI 推出 GPT-4.1 系列,支持 1M token

目录 OpenAI 推出 GPT-4.1 系列 OpenAI 推出 GPT-4.1 系列 OpenAI 宣布&#xff0c;新一代 GPT-4.1 模型系列正式发布&#xff0c;包括 GPT-4.1, GPT-4.1 mini 和 GPT-4.1 nano 三款模型&#xff0c;该系列模型在各项性能指标上全面超越 GPT-4o 和 GPT-4o mini&#xff0c;尤其…...

分布式锁+秒杀异步优化

文章目录 问题思路setnx实现锁误删问题和解决方案Redis Lua脚本问题引出解决方案 setnx实现的问题Redission快速入门redission可重入锁原理 秒杀优化(异步优化)异步秒杀思路秒杀资格判断Redis消息队列 问题 比如我们两个机器都部署了我们项目&#xff0c;这里nginx使用轮询的方…...

数据服务化 VS 数据中台:战略演进中的价值重构

在企业数据战略的演进历程中&#xff0c;数据中台曾被视为解决数据孤岛的 “万能钥匙”&#xff0c;而数据服务化的兴起则标志着企业从 “数据资源囤积” 向 “数据价值释放” 的深刻转型。两者的核心差异不仅在于技术架构&#xff0c;更在于对数据资产的定位与使用理念的根本分…...

PL/SQL登录慢,程序连接Oracle 提示无法连接或无监听

PL/SQL登录慢&#xff0c;程序连接Oracle 提示无法连接或无监听 错误提示&#xff1a;ORA-12541: TNS: 无监听程序 的解决办法&#xff0c; 现象&#xff1a;PL/SQL登录慢&#xff0c;程序连接Oracle 提示无法连接或无监听 监听已经正常开起&#xff0c;但还是PL/SQL登录慢或…...

【JAVAFX】自定义FXML 文件存放的位置以及使用

情况 1&#xff1a;FXML 文件与调用类在同一个包中&#xff08;推荐&#xff09; 假设类 MainApp 的包是 com.example&#xff0c;且 FXML 文件放在 resources/com/example 下&#xff1a; 项目根目录 ├── src │ └── sample │ └── Main.java ├── src/s…...

DDoS(分布式拒绝服务)攻击

DDoS(分布式拒绝服务)攻击 这是一份全面系统的 DDoS&#xff08;分布式拒绝服务攻击&#xff09;知识总结&#xff0c;适合用于学习、报告、讲稿或者面试准备。内容涵盖定义、原理、危害、利用、工具、防护策略等。 一、什么是DDoS DDoS&#xff08;Distributed Denial of Se…...

scikit-learn初探

KFold k交叉验证&#xff0c;k-1个作为训练集&#xff0c;剩下的作为测试集 split split(X, yNone, groupsNone)X&#xff1a; (n_samples, n_features)的矩阵&#xff0c;行数为n_samples&#xff0c;列数为n_features y&#xff1a;(n_samples,)为列向量&#xff0c;表示监…...

深入解析 sklearn 中的多种特征编码方式:功能、适用场景与选择建议

标题&#xff1a;深入解析 sklearn 中的多种特征编码方式&#xff1a;功能、适用场景与选择建议 摘要&#xff1a; 在机器学习中&#xff0c;特征编码是数据预处理的重要环节&#xff0c;直接影响模型的性能和效果。本文详细介绍了 sklearn 及其生态中&#xff08;含第三方库…...

windows10 wsl2 安装ubuntu和docker

见 弃用Docker Desktop&#xff1a;在WSL2中玩转Docker之Docker Engine 部署与WSL入门-阿里云开发者社区 如果启动docker时报下面这个错&#xff0c; 那是因为systemctl没有启用 sudo systemctl start docker System has not been booted with systemd as init system (PID 1)…...

一文读懂WPF系列之依赖属性与附加属性

依赖属性与附加属性 依赖属性对比C#属性WPF依赖属性&#xff08;Dependency Properties&#xff09;优先级计算与值决策​​回调与验证机制​​WPF 自带的依赖属性自定义依赖属性 附加属性本质与定义​​与依赖属性的区别​​附加属性的典型应用场景自定义附加属性注意事项 属性…...

1×1卷积与GoogleNet

11卷积 卷积核的尺寸等于1的卷积核 11卷积有什么用 1. 通道混合与特征转换 背景&#xff1a;在卷积神经网络中&#xff0c;输入数据通常有多个通道&#xff08;例如RGB图像有3个通道&#xff0c;经过卷积层后通道数可能会增加&#xff09;。不同通道的特征图可能包含了不同的…...

Handsontable 表格组件的使用

文章目录 1. 安装 Handsontable2. 创建一个基本表格3. 主要配置3.1、 data 数据3.2、 columns 指定列配置 4. Handsontable 高级功能4.1、 添加排序4.2、 过滤数据4.3、 选中行高亮4.4、 只读单元格4.5、 校验数据 5. Handsontable 与 Vue结合6. 总结 Handsontable 是一个强大的…...

消息中间件面试题

前言 本章内容来自B站黑马程序员java大厂面试题与小林coding 如有侵权立即删除 博主学习笔记&#xff0c;如果有不对的地方&#xff0c;海涵。 如果这篇文章对你有帮助&#xff0c;可以点点关注&#xff0c;点点赞&#xff0c;谢谢你&#xff01; 1.通用篇 1.1 什么是消息…...

数据结构与算法--1.判断数组中元素是否有重复

在C语言中&#xff0c;我们可以使用类似的方法来实现判断数组中是否有重复值的功能。由于C语言没有内置的哈希集合&#xff08;如Python的set或C的unordered_set&#xff09;&#xff0c;我们需要自己实现一个简单的哈希表或使用其他方法。 方法一&#xff1a;暴力法&#xff…...

硬件工程师面试常见问题(1)

第一问&#xff1a;单片机上电后没有运转&#xff0c;首先要检查什么&#xff1f; &#xff08;1&#xff09;单片机供电是否正常& 电路焊接检查 用万用表测量对应引脚的供电电压&#xff0c;检查对不对。 &#xff08;2&#xff09;单片机复位是否释放 用万用表测量复位引…...

测试100问:web测试和APP测试的区别

哈喽&#xff0c;大家好&#xff0c;我是十二&#xff0c;那今天要为大家分享的是高频面试题&#xff1a;web测试和 App测试的区别。 从功能测试方面来讲&#xff0c;web测试和 App测试在测试的流程以及测试用例的设计上是没有区别的&#xff0c;那主要的区别包含以下三个方面&…...

Leetcode 3518. Smallest Palindromic Rearrangement II

Leetcode 3518. Smallest Palindromic Rearrangement II 1. 解题思路2. 代码实现 题目链接&#xff1a;Leetcode 3518. Smallest Palindromic Rearrangement II 1. 解题思路 这一题是题目Leetcode 3517. Smallest Palindromic Rearrangement I的升级版本&#xff0c;其主要的…...

Golang|订单相关

文章目录 秒杀写库策略确保缓存的订单数据不丢失 秒杀写库策略 在我们的抽奖函数中&#xff0c;抽中奖品、减库存成功返回给前端后就应该生成订单写入数据库 但是这里有问题&#xff0c;我们的抽奖函数是支持高并发的&#xff0c;并发量大的情况下mysql无法支持这么大并发量的写…...

Python+Playwright:编写自动化测试的避坑策略

PythonPlaywright&#xff1a;编写自动化测试的避坑策略 前言一、告别 time.sleep()&#xff0c;拥抱 Playwright 的智能等待二、选择健壮、面向用户的选择器&#xff0c;优先使用 data-testid三、严格管理环境与依赖&#xff0c;确保一致性四、分离测试数据与逻辑&#xff0c;…...

P12130 [蓝桥杯 2025 省 B] 移动距离

P12130 [蓝桥杯 2025 省 B] 移动距离 - 洛谷 题目描述 小明初始在二维平面的原点&#xff0c;他想前往坐标 (233, 666)。在移动过程中&#xff0c;他只能采用以下两种移动方式&#xff0c;并且这两种移动方式可以交替、不限次数地使用&#xff1a; 水平向右移动&#xff0c;…...

关于 人工智能(AI)发展简史 的详细梳理,按时间阶段划分,涵盖关键里程碑、技术突破、重要人物及挑战

以下是关于 人工智能&#xff08;AI&#xff09;发展简史 的详细梳理&#xff0c;按时间阶段划分&#xff0c;涵盖关键里程碑、技术突破、重要人物及挑战&#xff1a; 字数&#xff1a;约2500字 逻辑结构&#xff1a;时间线清晰&#xff0c;分阶段描述技术突破、关键事件与挑战…...

Formality:Bug记录

相关阅读 Formalityhttps://blog.csdn.net/weixin_45791458/category_12841971.html?spm1001.2014.3001.5482 本文记录博主在使用Synopsys的形式验证工具Formality中遇到的几个Bug。 Bug复现 情况一 // 例1 module dff (input clk, input d_in, output d_out …...

react-07React提交表单数据调用同一方法(高阶函数,函数柯里化概念)

1.高阶函数与函数柯里化概念 高阶函数&#xff1a;符合其中之一&#xff0c;那该函数为高阶函数 1.A函数&#xff0c;接收的参数是一个函数&#xff0c;那么A就是高阶函数2.A函数&#xff0c;调用的返回值是一个函数&#xff0c;那么A就是高阶函数 常见的高阶函数&#xff1a…...

js ES6箭头函数的作用

前置知识 1、箭头函数语法简洁&#xff0c;相较于传统的函数表达式&#xff0c;箭头函数的语法更为简洁&#xff0c;尤其适用于简单的函数。 2、解决this取向问题&#xff0c;在传统函数中&#xff0c;this 的值取决于函数的调用方式&#xff0c;这可能会导致一些难以理解和调…...

淘宝商品数据高并发采集方案:API 接口限流机制与分布式调用实战

一、引言 在电商领域&#xff0c;对淘宝商品数据进行采集是一项常见且重要的任务。随着业务规模的扩大&#xff0c;高并发采集需求日益凸显。然而&#xff0c;淘宝 API 接口存在限流机制&#xff0c;为了高效且合规地完成数据采集&#xff0c;需要采用分布式调用的策略。本文将…...

python爬虫 线程,进程,协程

0x00 线程 线程是一个执行单位&#xff0c;是在一个进程里面的&#xff0c;是共享进程里面的提供的内存等资源&#xff0c;使用多个线程时和使用多个进程相比&#xff0c;多个线程使用的内存等资源较少。进程像一座“房子”&#xff08;独立资源&#xff09;&#xff0c;线程是…...

《忘尘谷》音阶与调性解析

一、音高与音名的对应关系 根据搜索结果及音乐理论&#xff0c;结合《忘尘谷》的曲谱信息&#xff0c;其音阶与调性分析如下&#xff1a; 调性判定 原曲调性为 D调&#xff08;原曲标注为D调&#xff09;&#xff0c;但曲谱编配时采用 C调指法&#xff0c;通过变调夹夹2品&…...

实验一 HDFS的Shell操作

一、实验目的 熟悉HDFS Shell的语法格式&#xff0c;完成对HDFS上文件和目录的一系列操作 二、实验要求 2.1 练习dfs常用的子命令 ① -ls ② -du ③ -mv ④ -cp ⑤ -rm ⑥ -put ⑦ -cat ⑧ -help ⑨ -mkdir ⑩ -get 2.2通过Shell脚本定时采集数据到HDFS 三、实…...

安装SQLServer管理工具

1.回到安装SQLServer的页面&#xff0c;选择安装SQLServer的管理工具2.跳转官网下载 3.安装SSMS 4.安装中5.安装成功以后打开软件&#xff0c;输入信息连接数据库 也可以在本机通过证书链接选择Windows身份验证&#xff0c;就不需要输入账号密码&#xff0c;但只能在安装sql…...

从JSON到SQL:基于业务场景的SQL生成器实战

引言 在数据驱动的业务场景中&#xff0c;将业务需求快速转化为SQL查询是常见需求。本文将通过一个轻量级的sql_json_to_sql函数&#xff0c;展示如何将JSON格式的查询描述转换为标准SQL语句&#xff0c;并结合实际业务场景验证其功能。 核心代码解析 1. 代码实现 def sql_j…...

记录鸿蒙应用上架应用未配置图标的前景图和后景图标准要求尺寸1024px*1024px和标准要求尺寸1024px*1024px

审核报错【①应用未配置图标的前景图和后景图,标准要求尺寸1024px*1024px且需下载HUAWEI DevEco Studio 5.0.5.315或以上版本进行图标再处理、②应用在展开状态下存在页面左边距过大的问题, 应用在展开状态下存在页面右边距过大的问题, 当前页面左边距: 504 px, 当前页面右边距…...

蓝桥杯嵌入式十六届赛前复习总结与准备

一.软件使用 赛点是没有网络的&#xff0c;要自己下载原件与数据包&#xff0c;这里给大家一个演示 在updater Settings这里设置文件存放位置&#xff0c;为了方便查找和提交文件&#xff0c;建议在桌面建立一个文件夹来存放。 把赛点的芯片包复制到创建的文件夹然后解压缩 之…...

了解一下Unity的RenderQueue

在Unity中&#xff0c;场景里的每个物体都需要通过渲染管线绘制到屏幕上。渲染管线处理光照、材质、纹理等信息&#xff0c;最终决定物体的显示效果。但当场景中有多个物体时&#xff0c;它们的绘制顺序会直接影响画面结果&#xff0c;尤其是在涉及透明物体或特效时。这时&…...

使用CS Roofline Toolkit测量带宽

使用CS Roofline Toolkit测量带宽 工程下载&#xff1a;使用CS Roofline Toolkit测量带宽-案例工程文件&#xff0c;也可以按照下面的说明使用git clone下载 目录 使用CS Roofline Toolkit测量带宽0、Roofline模型理解1、CS Roofline Toolkit下载1.1、设置代理1.2、git clone下…...

第三篇:深入 Framer Motion Variants:掌握组件动画编排的艺术

&#x1f3af; 前言 在动态交互主导的现代前端开发中&#xff0c;优雅的动画效果已成为提升用户体验的重要元素。Framer Motion 的 Variants&#xff08;动画变体&#xff09;功能&#xff0c;通过状态化管理和动画编排能力&#xff0c;让复杂动效的实现变得前所未有的高效。本…...

狂神SQL学习笔记四:基本的命令行操作

注&#xff1a;所有语句用 ; 结尾&#xff0c;– 单行注释&#xff0c;/* 多行注释 */ 连接数据库 mysql -uroot -p123456 --连接数据库查看所有的数据库 切换数据库 查看数据库中所有的表 显示数据库中所有表的信息 创建一个数据库 退出连接...

架构思维:缓存层场景实战_读缓存(下)

文章目录 Pre业务场景缓存存储数据的时机与常见问题解决方案1. 缓存读取与存储逻辑2. 高并发下的缓存问题及解决方案3. 缓存预热&#xff08;减少冷启动问题&#xff09; 缓存更新策略&#xff08;双写问题&#xff09;1. 先更新缓存&#xff0c;再更新数据库&#xff08;不推荐…...

软件架构设计:MVC、MVP、MVVM、RIA 四大风格优劣剖析

MVC、MVP、MVVM 和 RIA 都是软件架构中常见的设计风格&#xff0c;以下是对它们的详细介绍&#xff1a; 一、MVC 架构风格&#xff08;Model - View - Controller&#xff09; 1.简介&#xff1a;MVC 架构风格将软件应用程序分为三个核心部分&#xff0c;通过这种划分来分离不…...

java基础课程-springmvc课程

一. 回顾MVC&#xff1a; tomcat是servlet容器&#xff0c; servlet实现方式&#xff1a; xml中配置&#xff1a; 二. 回顾servlet: 三. SpringMvc学习 3.1 springMvc搭建和讲解&#xff1a; jar包引入spring-webmvc即可。 核心&#xff1a;DispatcherServlet Spring的web…...

NLP高频面试题(四十二)——RAG系统评估:方法、指标与实践指南

1. 引言:RAG系统概述与评估挑战 检索增强生成(Retrieval-Augmented Generation,简称 RAG)是近年来自然语言处理领域的一个重要进展。RAG系统在大型语言模型生成文本的过程中引入了外部检索模块,从外部知识库获取相关信息,以缓解纯生成模型可能出现的幻觉和知识盲点。通过…...

Flutter学习 滚动组件(1):ListView基本使用

目录 一、ListView构造方法1.1 常规方法1.2 ListView.builder1.3 ListView.separated 二、自定义ListView样式和布局&#xff1a;三、ListView性能优化&#xff1a;总结&#xff1a; 一、ListView构造方法 主要以下几种方法&#xff1a; 常规方法&#xff0c;直接使用默认的构…...

处理 Flutter 没有反应

现象 有以下几种 VS Code 中 Initializing the Flutter SDK. This may take a few minutes. 会一直维持在这个右下角提示窗, 但是无后续动作 Flutter CMD flutter_console.bat 执行 --version 或者 doctor [-v] 没有任何输出, 命令卡住 解决办法 参考官方说明 管理员身份…...

java面向对象06:封装

封装 该露的露&#xff0c;该藏的藏 我们程序设计要追求“高内聚&#xff0c;低耦合”。高内聚就是类的内部数据操作细节自己完成&#xff0c;不允许外部干涉&#xff1b;低耦合&#xff1a;仅暴露少量的方法给外部使用。 封装&#xff08;数据的隐藏&#xff09; 通常&#x…...

测试定时发布

测试定时发布 测试定时发布 测试定时发布 测试定时发布 欢迎使用Markdown编辑器 你好&#xff01; 这是你第一次使用 Markdown编辑器 所展示的欢迎页。如果你想学习如何使用Markdown编辑器, 可以仔细阅读这篇文章&#xff0c;了解一下Markdown的基本语法知识。 新的改变 我…...

2025年4月15日 百度一面 面经

目录 1. 代理相关 从静态代理到动态代理 2. cglib可以代理被final修饰的类吗,为什么 3. JVM 体系结构 4. 垃圾回收算法 5. 什么是注解 如何使用 底层原理 6. synchronized和reentrantlock 7. 讲一下你项目中 redis的分布式锁 与java自带的锁有啥区别 8. post 请求和 ge…...

抖音卸载了ip属地还会更新吗?深度解析

近期&#xff0c;关于抖音IP属地显示功能的讨论热度持续攀升&#xff0c;许多用户提出疑问&#xff1a;如果卸载抖音APP&#xff0c;账号的IP属地还会继续更新吗&#xff1f;这一问题背后&#xff0c;既涉及平台算法的运作逻辑&#xff0c;也关乎用户对隐私保护的深层需求。本文…...