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

Linux设备树的驱动开发

概述

本文介绍了platform框架下的设备驱动开发流程和方法,主要包括设备树、驱动程序和应用程序的开发。以随机数驱动为例,实现了应用程序调用库函数,通过系统调用陷入内核,最后执行硬件驱动,获取真随机数的过程。

添加设备树节点

soc节点下添加名为trng子节点,内容如下:

trng: trng@0x53030000 {compatible = "acme,trng";reg = <0x00 0x53030000 0x00 0x1000>;interrupts = <0x34 IRQ_TYPE_LEVEL_HIGH>;interrupt-parent = <&plic>;
};

编译设备树dts,生成相应的dtb文件:

make dtbs

使用新的dtb启动Linux内核。Linux启动成功之后查看是否有trng这个节点:

ls /proc/device-tree/soc
# trng@0x53030000

进入trng目录,查看属性相关的文件:

/proc/device-tree/soc/trng@0x53030000# ls
compatible        interrupts        phandle
interrupt-parent  name              reg

编写设备驱动

Makefile

新建trng/driver目录,并创建Makefile,内如如下:

# 内核架构
ARCH := riscv
# 交叉工具链
CROSS_COMPILE := /path/to/riscv32-linux-
# 内核目录
KERNELDIR := /path/to/linux/linux-6.1
# 当前目录
PWD := $(shell pwd)
# 目标文件
obj-m := trng.o# 目标
build: kernel_modules# 编译模块
kernel_modules:$(MAKE) ARCH=$(ARCH) CROSS_COMPILE=$(CROSS_COMPILE) -C $(KERNELDIR) M=$(PWD) modules# 清理
clean:$(MAKE) ARCH=$(ARCH) CROSS_COMPILE=$(CROSS_COMPILE) -C $(KERNELDIR) M=$(PWD) clean

驱动

trng/driver下新建trng.c文件,内容如下:

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/interrupt.h>
#include <linux/fs.h>
#include <linux/of.h>
#include <linux/cdev.h>
#include <linux/io.h>
#include <linux/iopoll.h>
#include <linux/platform_device.h>/* TRNG寄存器 */
#define TRNG_CTRL           (0x0)#define TRNG_CTRL_CMD_MASK      (0x07)#define TRNG_CTRL_CMD_RNG       (0x01)#define TRNG_CTRL_CMD_SEED      (0x02)
#define TRNG_STAT           (0x4)#define TRNG_STAT_SEEDED        BIT(9)
#define TRNG_MODE           (0x8)#define TRNG_MODE_R256          BIT(3)
#define TRNG_ISTAT          (0x14)#define TRNG_ISTAT_RAND_RDY     BIT(0)#define TRNG_ISTAT_SEED_DONE    BIT(1)
#define TRNG_RAND0          (0x20)#define TRNG_TIMEOUT        (50000)#define DRIVER_NAME         "trng"struct trng_dev {dev_t devid;            /* 设备号 */struct cdev cdev;       /* cdev */struct class *class;    /* 类 */struct device *dev;     /* 设备 */int major;              /* 主设备号 */int minor;              /* 次设备号 */int irq;                /* 中断号 */void __iomem *base;     /* 基地址 */
};static int trng_init(void *base)
{int ret;unsigned int value;/* 模式 */value = readl(base + TRNG_MODE);value |= TRNG_MODE_R256;writel(value, base + TRNG_MODE);    /* 播种 */value = readl(base + TRNG_CTRL);value &= ~TRNG_CTRL_CMD_MASK;value |= TRNG_CTRL_CMD_SEED;writel(value, base + TRNG_CTRL);/* 等待播种完成 */ret = readl_relaxed_poll_timeout_atomic(base + TRNG_ISTAT,value, (value & TRNG_ISTAT_SEED_DONE),10, TRNG_TIMEOUT);if (ret == 0) {value |= TRNG_ISTAT_SEED_DONE;writel(value, base + TRNG_ISTAT);}return ret;
}static int trng_generate_random(void *base, unsigned char *buf)
{int ret;unsigned int value;/* 启动生成随机数 */value = readl(base + TRNG_CTRL);value &= ~TRNG_CTRL_CMD_MASK;value |= TRNG_CTRL_CMD_RNG;writel(value, base + TRNG_CTRL);/* 等待随机数准备好 */ret = readl_relaxed_poll_timeout_atomic(base + TRNG_ISTAT,value, (value & TRNG_ISTAT_RAND_RDY),10, TRNG_TIMEOUT);    if (ret) {return ret;}/* 清除准备好标志 */value = readl(base + TRNG_ISTAT);value &= ~TRNG_ISTAT_RAND_RDY;writel(value, base + TRNG_ISTAT);/* 读取随机数 */for (int i = 0; i < 8; i++) {*(unsigned int*)buf = readl(base + TRNG_RAND0 + i*4);buf += 4;}return 0;
}static irqreturn_t trng_irq_handler(int irq, void *dev_id)
{struct trng_dev *trng;trng = (struct trng_dev*)dev_id;dev_dbg(trng->dev, "TRNG interrupt received\n");return IRQ_HANDLED;
}static int trng_open(struct inode *inode, struct file *filp) 
{int ret;struct trng_dev *trng;trng = container_of(inode->i_cdev, struct trng_dev, cdev);filp->private_data = trng;dev_dbg(trng->dev, "Open trng\n");ret = trng_init(trng->base);if (ret) {dev_err(trng->dev, "Failed to init trng, ret=%d\n", ret);return ret;}return 0;
}static int trng_release(struct inode *inode, struct file *filp) 
{struct trng_dev *trng;trng = filp->private_data;dev_dbg(trng->dev, "Release trng\n");return 0;
}static ssize_t trng_read(struct file *filp, char __user *buffer, size_t len, loff_t *offset)
{  int ret;unsigned char random[32];size_t copyed_len, len_to_copy;struct trng_dev *trng;trng = filp->private_data;dev_info(trng->dev, "Read trng\n");copyed_len = 0;while (len) {ret = trng_generate_random(trng->base, random);if (ret) {dev_err(trng->dev, "Failed to generate random, ret=%d\n", ret);return ret;}// print_hex_dump(KERN_INFO, "random: ", DUMP_PREFIX_NONE, 16, 1, random, sizeof(random), false);len_to_copy = (len < sizeof(random)) ? len : sizeof(random);ret = copy_to_user(buffer, random, len_to_copy);if (ret) {dev_err(trng->dev, "Failed to copy to user\n");return -EFAULT;}copyed_len += len_to_copy;buffer += len_to_copy;len -= len_to_copy;}return copyed_len;  
}  static ssize_t trng_write(struct file *filp, const char __user *buffer, size_t len, loff_t *offset) 
{struct trng_dev *trng;trng = filp->private_data;dev_dbg(trng->dev, "Write trng\n");return len;
}static long trng_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
{struct trng_dev *trng;trng = filp->private_data;dev_dbg(trng->dev, "Ioctl trng\n");switch (cmd) {default:dev_err(trng->dev, "Unknown ioctl = 0x%x\n", cmd);break;}return -ENOTTY;
}static int trng_mmap(struct file *filp, struct vm_area_struct *vma) 
{struct trng_dev *trng;trng = filp->private_data;dev_dbg(trng->dev, "Mmap trng\n");return 0;
}static const struct file_operations trng_fops = {.owner = THIS_MODULE,.open = trng_open,.release = trng_release,.read = trng_read,.write = trng_write,.unlocked_ioctl = trng_ioctl,.mmap = trng_mmap,
};static int trng_probe(struct platform_device *pdev) 
{struct device *dev = &pdev->dev;struct trng_dev *trng;int ret;/* 分配内存 */trng = devm_kzalloc(dev, sizeof(*trng), GFP_KERNEL);if (!trng) {dev_err(dev, "Failed to allocate memory\n");return -ENOMEM;}/* 将设备的资源映射到内存空间 */trng->base = devm_platform_ioremap_resource(pdev, 0);if (IS_ERR(trng->base)) {dev_err(dev, "Failed to map device registers\n");return PTR_ERR(trng->base);}/* 获取设备的中断号 */trng->irq = platform_get_irq(pdev, 0);if (trng->irq <= 0) {dev_err(dev, "Failed to get irq %d\n", trng->irq);return trng->irq;        }/* 请求中断 */ret = devm_request_irq(dev, trng->irq, trng_irq_handler, 0,DRIVER_NAME, trng);if (ret) {dev_err(dev, "Failed to request IRQ\n");return ret;}/* 申请设备号 */ret = alloc_chrdev_region(&trng->devid, 0, 1, DRIVER_NAME);  if (ret < 0) {  dev_err(dev, "Failed to allocate device number\n");  return ret;  }trng->major = MAJOR(trng->devid); trng->minor = MINOR(trng->devid); /* 初始化cdev */trng->cdev.owner = THIS_MODULE;cdev_init(&trng->cdev, &trng_fops);/* 添加一个cdev */ret = cdev_add(&trng->cdev, trng->devid, 1);if (ret < 0) {dev_err(dev, "Failed to add cdev\n");unregister_chrdev_region(trng->devid, 1);return ret;} /* 创建类 */trng->class = class_create(THIS_MODULE, DRIVER_NAME);if (IS_ERR(trng->class)) {cdev_del(&trng->cdev);unregister_chrdev_region(trng->devid, 1);dev_err(dev, "Failed to create class\n");return PTR_ERR(trng->class);}/* 创建设备 */trng->dev = device_create(trng->class, NULL, trng->devid, NULL, DRIVER_NAME);if (IS_ERR(trng->dev)) {cdev_del(&trng->cdev);unregister_chrdev_region(trng->devid, 1);class_destroy(trng->class);dev_err(dev, "Failed to create device\n");return PTR_ERR(trng->dev);}/* 保存设备私有结构体 */platform_set_drvdata(pdev, trng);dev_info(dev, "TRNG platform driver probed\n");return 0;
}static int trng_remove(struct platform_device *pdev) 
{struct trng_dev *trng;/* 获取设备私有结构体 */trng = platform_get_drvdata(pdev);/* 删除cdev */cdev_del(&trng->cdev);/* 释放设备号 */unregister_chrdev_region(trng->devid, 1);/* 删除设备 */device_destroy(trng->class, trng->devid);/* 删除类 */class_destroy(trng->class);/* 释放设备内存 */// devm_kfree(&pdev->dev, trng);    // devm_kzalloc为设备分配的内存,在设备移除时会自动释放dev_info(&pdev->dev, "TRNG platform driver removed\n");return 0;
}static const struct of_device_id trng_of_match[] = {{ .compatible = "acme,trng" },{ /* sentinel */ }
};
MODULE_DEVICE_TABLE(of, trng_of_match);static struct platform_driver trng_driver = {.driver = {.name = DRIVER_NAME,.of_match_table = trng_of_match,},.probe = trng_probe,.remove = trng_remove,
};module_platform_driver(trng_driver);MODULE_AUTHOR("Author");
MODULE_DESCRIPTION("Trng driver");
MODULE_LICENSE("GPL");

使用platform平台驱动设备模型编写trng的驱动程序。

  • 当设备树中的节点与驱动匹配成功会执行trng_probe函数,完成驱动的加载。

  • 当应用需要获取随机数时,读取这个trng设备,陷入内核调用函数trng_read,进而调用函数trng_generate_random完成从硬件获取随机数。

  • 如果需要释放设备,会调用trng_remove函数卸载设备驱动。

执行make编译驱动程序,编译成功生成trng.ko

Linux启动成功之后,可以挂载nfs,将trng.ko拷贝到trng目录:

mkdir trng
mount -t nfs -o nolock xx.xx.xx.xx:/nfs/trng /root/trng

执行如下命令,加载设备驱动:

insmod trng.ko
# [  177.821055] trng 53030000.trng: TRNG platform driver probed

如果设备驱动加载成功,可以在/dev下找到设备:

ls /dev/trng

另外可以查看trng的设备号:

cat /proc/devices
# 249 trng

如果需要卸载设备驱动,执行:

rmmod trng.ko
# [ 2947.495906] trng 53030000.trng: TRNG platform driver removed

应用App

Makefile

新建trng/app目录,并创建Makefile,内如如下:

# 交叉工具链
CROSS_COMPILE ?= /opt/andestech/nds32le-linux-glibc-v5d/bin/riscv32-linux-# 指定C编译器
CC := $(CROSS_COMPILE)gcc# 目标文件名
TARGET := trng# 源文件名
SRC := trng.c# 默认目标
all: $(TARGET)# 编译并链接
$(TARGET): $(SRC)$(CC) $(SRC) -o $(TARGET)# 清理
clean:rm -f $(TARGET)

应用

trng/app下新建trng.c文件,内容如下:

#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h> #define TRNG_DEVICE                       "/dev/trng"static void hexdump(const char *name, const unsigned char *buffer, unsigned int len)
{printf("****************%s****************\n", name);for (unsigned int i = 0; i < len; i++) {printf("%02x ", buffer[i]);if ((i + 1) % 16 == 0) {printf("\n");}}if (len % 16 ) {printf("\n");}
}int main(int argc, char *argv[])
{uint8_t *buf = NULL;size_t num;int ret, fd;if (argc < 2) {printf("Usage: trng <num>\n");return -1;}num = atoi(argv[1]);buf = malloc(num);if (buf == NULL) {printf("Failed to malloc\n");return -1;}/* 打开设备 */fd = open(TRNG_DEVICE, O_RDONLY);if (fd < 0) {printf("Failed to open trng device\n");goto exit;}/* 读取随机数 */ret = read(fd, buf, num);if (ret < 0) {printf("Failed to read random, ret=%d\n", ret);goto exit;}hexdump("random", buf, num);
exit:close(fd);free(buf);return ret;
}

执行make编译应用程序,编译成功生成trng

同理,将trng应用拷贝到trng目录,并执行:

./trng 16
# ****************random****************
# 6c 95 ea 3c a0 1f e8 c2 03 db 66 f6 19 4b 07 e3 
# c0 96 a3 93 20 a9 68 c5 9f 1f a1 55 c0 9c 24 c9 
# 5f 06 47 45 be 2c 21 b5 11 23 23 e6 36 94 3f d6 
# 9a 30 68 91 da c4 6d ff af 46 26 c9 ab f8 79 7c 

如果随机数获取成功,说明驱动和应用程序运行正常。

编译进内核

在开发前期阶段,一般将驱动编译成模块,方便调试。当驱动开发完成后,可以将其编译进内核。

驱动

linux-6.1/drivers下新建trng目录,并创建MakefileKconfig文件,内容分别如下:

# SPDX-License-Identifier: GPL-2.0
#
# Makefile for the TRNG device drivers.
#obj-$(CONFIG_TRNG) := trng.o
# SPDX-License-Identifier: GPL-2.0-only
#
# TRNG device configuration
#config TRNGtristate "TRNG support"helpThis driver provides support for TRNG in SoCs.To compile this driver as a module, choose M here: the modulewill be called acme-trng.If unsure, say Y.

linux/drivers/Makefile中添加:

obj-$(CONFIG_TRNG)	+= trng/

linux/drivers/Kconfig中添加:

source "drivers/trng/Kconfig"

trng.c驱动文件拷贝到trng目录,最终目录文件如下:

$ tree linux-6.1/drivers/trng/
├── Kconfig
├── Makefile
└── trng.c

内核配置

配置内核,输入命令:

make menuconfig

选择Device Drivers->TRNG support,选择将trng编译进内核,这里可以有三种选择:

  • *:将该功能编译进内核
  • 空:不编译该功能
  • M:将该功能编译成内核中的模块

运行

编译Linux并启动,在启动日志中,如果打印如下,说明TRNG驱动运行正常:

[    4.974450] trng 53030000.trng: TRNG platform driver probed

可以执行命令,获取随机数:

cat /dev/trng | hexdump -n 32
0000000 df01 f5bc de33 2509 8d16 7b5f 8868 8bea
0000010 40f3 00f2 97a4 324d 03c2 10c8 b943 3d6d
0000020

问题解决

  1. 加载KO报异常trng: loading out-of-tree module taints kernel.

    原因在于没有将此驱动模块加入到Kconfig导致。

  2. 使用函数devm_kzalloc为设备分配的内存,在设备移除时会自动释放,可以不进行显示释放devm_kfree

  3. container_of用于从结构体的某个成员的地址反推出整个结构体的地址,尤其注意第一个参数必须为成员的地址,如果结构体成员为指针变量,需要取该指针变量的地址。

相关文章:

Linux设备树的驱动开发

概述 本文介绍了platform框架下的设备驱动开发流程和方法&#xff0c;主要包括设备树、驱动程序和应用程序的开发。以随机数驱动为例&#xff0c;实现了应用程序调用库函数&#xff0c;通过系统调用陷入内核&#xff0c;最后执行硬件驱动&#xff0c;获取真随机数的过程。 添…...

连锁?下沉?AI?2025年餐饮新活力!

如果要用几个词来形容 2024 年的餐饮业&#xff0c;这些词大概率会是「卷、难、惨」&#xff0c;用著名商业顾问刘润的话来说就是「卷到极致」。虽然餐饮人在社交平台上叫苦连天&#xff0c;但当我们查看餐饮大盘数据时发现&#xff0c;大盘在涨&#xff0c;与个体餐饮人的实感…...

Javascript中如何实现函数缓存?函数缓存有哪些应用场景?

今天要聊的一个很经典的问题——如何在JavaScript中实现函数缓存&#xff0c;以及它有哪些应用场景。 我们先来明确一下&#xff0c;函数缓存是什么。简单来说&#xff0c;函数缓存是将函数的运算结果存储起来&#xff0c;以便下次用到相同的输入时&#xff0c;可以直接返回结…...

子页面访问父页面

子页面访问父页面的方式主要依赖于页面之间的关系&#xff0c;特别是它们是否处于同一域、是否是嵌套在 <iframe> 中、或者通过弹出窗口打开。下面是几种常见的子页面访问父页面的方法&#xff1a; 1. 通过 window.parent 访问父页面&#xff08;适用于嵌套的 iframe&am…...

芯片级IO (Pad) Ring IP Checklist

SoC top顶层数字后端实现都会涉及到IO Ring &#xff08;PAD Ring&#xff09;的设计。这里面包括VDD IO,VDDIO IO, Signal IO, Corner IO&#xff0c;Filler IO&#xff0c;IO power cut cell等等。 数字后端零基础入门系列 | Innovus零基础LAB学习Day2 数字IC后端实现TOP F…...

计算机毕业设计论文指导

计算机毕业设计论文指导 计算机毕业设计辅导一站式&#xff01;太香了&#x1f4aa; [赞R][赞R][赞R]嗨喽&#xff01;计算机专业的宝子们&#xff01; 计算机毕设辅导专业靠谱的他来了&#xff01;&#xff01; 是不是还在为选题程序不会做而感到苦难&#xff1f; 论文没思路赶…...

Electron-Vue 开发下 dev/prod/webpack server各种路径设置汇总

背景 在实际开发中&#xff0c;我发现团队对于这几个路径的设置上是纯靠猜的&#xff0c;通过一点点地尝试来找到可行的路径&#xff0c;这是不应该的&#xff0c;我们应该很清晰地了解这几个概念&#xff0c;以下通过截图和代码进行细节讲解。 npm run dev 下的路径如何处理&…...

Vue.js前端框架教程9:Vue插槽slot用法

文章目录 插槽&#xff08;Slots&#xff09;无名插槽&#xff08;默认插槽&#xff09;具名插槽reference 插槽使用 v-slot 的缩写语法 插槽&#xff08;Slots&#xff09; 在 Vue 中&#xff0c;插槽&#xff08;Slots&#xff09;是一种组件内容分发的机制&#xff0c;允许…...

初学stm32 --- NVIC中断

目录 STM32 NVIC 中断优先级管理 NVIC_Type: ISER[8]&#xff1a; ICER[8]&#xff1a; ISPR[8]&#xff1a; ICPR[8]&#xff1a; IABR[8]&#xff1a; IP[240]&#xff1a; STM32 的中断分组&#xff1a; 中断优先级分组函数 NVIC_PriorityGroupConfig 中断初始化函…...

Jest 入门指南:从零开始编写 JavaScript 单元测试

前言 在前端开发中&#xff0c;单元测试已经成为确保代码质量和稳定性的关键步骤。Jest 作为由 Facebook 开发和维护的功能强大的 JavaScript 测试框架&#xff0c;以其易于配置、丰富的功能和开箱即用的特性&#xff0c;成为众多开发者的首选工具。本文旨在引导你从零开始&am…...

【Java Web】Axios实现前后端数据异步交互

目录 一、Promise概述 二、Promise基本用法 三、async和await关键字 四、Axios介绍 4.1 Axios基本用法 4.2 Axios简化用法之get和post方法 五、Axios拦截器 六、跨域问题处理 一、Promise概述 axios是代替原生的ajax实现前后端数据交互的一套新解决方案&#xff0c;而…...

React 第十七节 useMemo用法详解

概述 useMemo 是React 中的一个HOOK&#xff0c;用于根据依赖在每次渲染时候缓存计算结果&#xff1b; 大白话就是&#xff0c;只有依赖项发生变化时候&#xff0c;才会重新渲染为新计算的值&#xff0c;否则就还是取原来的值&#xff0c;有点类似 vue 中的 computed 计算属性…...

鸿蒙项目云捐助第十五讲云数据库的初步使用

鸿蒙项目云捐助第十五讲云数据库的初步使用 在华为云技术使用中&#xff0c;前面使用了云函数&#xff0c;接下来看一下华为云技术中的另外一个技术云数据库的使用。 一、云数据库的创建 这里使用华为云数据库也需要登录到AppGallery Connect平台中&#xff0c;点击进入到之…...

如何构建一个可信的联邦RAG系统。

今天给大家分享一篇论文。 题目是&#xff1a;C-RAG&#xff1a;如何构建一个可信的联邦检索RAG系统。 论文链接:https://arxiv.org/abs/2412.13163 论文概述 尽管大型语言模型 (LLM) 在各种应用中展现出令人印象深刻的能力&#xff0c;但它们仍然存在可信度问题&#xff…...

【深度学习之三】FPN与PAN网络详解

FPN与PAN&#xff1a;深度学习中的特征金字塔网络与路径聚合网络 在深度学习的领域里&#xff0c;特征金字塔网络&#xff08;Feature Pyramid Networks&#xff0c;简称FPN&#xff09; 和 路径聚合网络&#xff08;Path Aggregation Network&#xff0c;简称PAN&#xff09;…...

Qt学习笔记第71到80讲

第71讲 事件过滤器的方式实现滚轮按键放大 事件体系&#xff08;事件派发 -> 事件过滤->事件分发->事件处理&#xff09;中&#xff0c;程序员主要操作的是事件分发与事件处理。我们之前已经通过继承QTextEdit来重写事件实现Ctrl加滚轮的检测&#xff0c;还有一种处理…...

以管理员身份运行

同时按下Ctrl Shift Esc键打开任务管理器&#xff0c;在任务管理器的左上角&#xff0c;点击“文件”菜单&#xff0c;在下拉菜单中选择“新建任务” 在弹出的对话框中&#xff0c;输入您想要运行的程序的名称。如果您不确定程序的确切名称&#xff0c;可以点击“浏览”来找到…...

用 Python 实现井字棋游戏

一、引言 井字棋&#xff08;Tic-Tac-Toe&#xff09;是一款经典的两人棋类游戏。在这个游戏中&#xff0c;玩家轮流在 3x3 的棋盘上放置自己的标记&#xff0c;通常是 “X” 和 “O”&#xff0c;第一个在棋盘上连成一线&#xff08;横、竖或斜&#xff09;的玩家即为获胜者。…...

06 实现自定义AXI DMA驱动

为什么要实现自定义AXI DMA驱动 ZYNQ 的 AXI DMA 在 Direct Register DMA &#xff08;即 Simple DMA&#xff09;模式下可以通过 AXIS 总线的 tlast 提前结束传输&#xff0c;同时还可以在 BUFFLEN 寄存器中读取到实际传输的字节数&#xff0c;但是通过 Linux 的 DMA 驱动框架…...

SpringBoot集成ENC对配置文件进行加密

在线MD5生成工具 配置文件加密&#xff0c;集成ENC 引入POM依赖 <!-- ENC配置文件加密 --><dependency><groupId>com.github.ulisesbocchio</groupId><artifactId>jasypt-spring-boot-starter</artifactId><version>2.1.2</ver…...

变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析

一、变量声明设计&#xff1a;let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性&#xff0c;这种设计体现了语言的核心哲学。以下是深度解析&#xff1a; 1.1 设计理念剖析 安全优先原则&#xff1a;默认不可变强制开发者明确声明意图 let x 5; …...

Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)

文章目录 1.什么是Redis&#xff1f;2.为什么要使用redis作为mysql的缓存&#xff1f;3.什么是缓存雪崩、缓存穿透、缓存击穿&#xff1f;3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...

大数据零基础学习day1之环境准备和大数据初步理解

学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 &#xff08;1&#xff09;设置网关 打开VMware虚拟机&#xff0c;点击编辑…...

五年级数学知识边界总结思考-下册

目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解&#xff1a;由来、作用与意义**一、知识点核心内容****二、知识点的由来&#xff1a;从生活实践到数学抽象****三、知识的作用&#xff1a;解决实际问题的工具****四、学习的意义&#xff1a;培养核心素养…...

【ROS】Nav2源码之nav2_behavior_tree-行为树节点列表

1、行为树节点分类 在 Nav2(Navigation2)的行为树框架中,行为树节点插件按照功能分为 Action(动作节点)、Condition(条件节点)、Control(控制节点) 和 Decorator(装饰节点) 四类。 1.1 动作节点 Action 执行具体的机器人操作或任务,直接与硬件、传感器或外部系统…...

如何将联系人从 iPhone 转移到 Android

从 iPhone 换到 Android 手机时&#xff0c;你可能需要保留重要的数据&#xff0c;例如通讯录。好在&#xff0c;将通讯录从 iPhone 转移到 Android 手机非常简单&#xff0c;你可以从本文中学习 6 种可靠的方法&#xff0c;确保随时保持连接&#xff0c;不错过任何信息。 第 1…...

数据库分批入库

今天在工作中&#xff0c;遇到一个问题&#xff0c;就是分批查询的时候&#xff0c;由于批次过大导致出现了一些问题&#xff0c;一下是问题描述和解决方案&#xff1a; 示例&#xff1a; // 假设已有数据列表 dataList 和 PreparedStatement pstmt int batchSize 1000; // …...

如何理解 IP 数据报中的 TTL?

目录 前言理解 前言 面试灵魂一问&#xff1a;说说对 IP 数据报中 TTL 的理解&#xff1f;我们都知道&#xff0c;IP 数据报由首部和数据两部分组成&#xff0c;首部又分为两部分&#xff1a;固定部分和可变部分&#xff0c;共占 20 字节&#xff0c;而即将讨论的 TTL 就位于首…...

Rapidio门铃消息FIFO溢出机制

关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系&#xff0c;以下是深入解析&#xff1a; 门铃FIFO溢出的本质 在RapidIO系统中&#xff0c;门铃消息FIFO是硬件控制器内部的缓冲区&#xff0c;用于临时存储接收到的门铃消息&#xff08;Doorbell Message&#xff09;。…...

服务器--宝塔命令

一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行&#xff01; sudo su - 1. CentOS 系统&#xff1a; yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...