正点原子嵌入式linux驱动开发——Linux ADC驱动

在之前的笔记中,学习了如何给ICM20608编写IIO驱动,ICM20608本质就是ADC,因此纯粹的ADC驱动也是IIO驱动框架的。本章就学习一下如何使用STM32MP1内部的ADC,并且在学习巩固一下IIO驱动

ADC简介

ADC

ADC,Analog to Digital Converter的缩写,中文名称模数转换器。它可以将外部的模拟信号转化成数字信号。对于GPIO口来说高于某个电压值,它读出来的只有高电平,低于就是低电平。假如想知道具体的电压数值就要借助于ADC的帮助,它可以将一个范围内的电压精确的读取出来

ADC有几个比较重要的参数:

  • 测量范围:测量范围对于ADC来说就好比尺子的量程,ADC测量范围决定了外接的设备其信号输出电压范围,不能超过ADC的测量范围。如果所使用的外部传感器输出的电压信号范围和所使用的ADC测量范围不符合,那么就需要自行设计相关电压转换电路。
  • 分辨率:就是尺子上的能量出来的最小测量刻度,假如ADC的测量范围为0-5V,分辨率设置为12位,那么能测出来的最小电压就是 5V除以2的12次方,也就是5/4096=0.00122V。很明显,分辨率越高,采集到的信号越精确,所以分辨率是衡量 ADC 的一个重要指标。
  • 精度:是影响结果准确度的因素之一,比如在厘米尺上能测量出大概多少毫米的尺度但是毫米后一点点我们却不能准确的量出。经过计算ADC在12位分辨率下的最小测量值是0.00122V,但是ADC的精度最高只能到11位也就是0.00244V。也就是ADC测量出0.00244V的结果是要比0.00122V要可靠,也更准确。
  • 采样时间:当ADC在某时刻采集外部电压信号的时候,此时外部的信号应该保持不变,但实际上外部的信号是不停变化的。所以在ADC内部有一个保持电路,保持某一时刻的外部信号,这样ADC就可以稳定采集了,保持这个信号的时间就是采样时间
  • 采样率:也就是在一秒的时间内采集多少次。很明显,采样率越高越好,当采样率不够的时候可能会丢失部分信息,所以ADC采样率是衡量ADC性能的另一个重要指标。

总之,只要是需要模拟信号转为数字信号的场合,那么肯定要用到ADC。很多数字传感器内部会集成ADC,传感器内部使用ADC来处理原始的模拟信号,最终给用户输出数字信号。

STM32MP157 ADC简介

STM32MP157有两个ADC:ADC1和ADC2,ADC1和ADC2紧密耦合,可在双重模式下运行(ADC1为主器件)。每个ADC由一个16位逐次逼近模数转换器组成,每个ADC有20个通道,每个通道支持单次、连续、扫描或不连续采样模式。转换结果存储在一个左对齐或右对
齐的32位数据寄存器中。ADC主要特性如下:

  1. 多达2个ADC,可在双重模式下运行。
  2. 可以配置为16、14、12、10或8位分辨率。
  3. 自校准。
  4. 可独立配置各通道采样时间。
  5. ……

ADC驱动源码简析

设备树下ADC节点

STM32MP157有2个ADC,因此对应2个ADC控制器,所以在设备树里就有2个ADC控制器节点。这2个ADC的设备树节点内容都是一样的,除了reg属性不同(毕竟不同的控制器,其地址范围不同)。本章实验使用PA5这个引脚来完成ADC实验,而PA5就是ADC1_INP19通道引脚,所以这里就以ADC1为例进行讲解,stm32mp151.dtsi文件中的adc节点信息如下:

示例代码 57.2.1.1 adc 节点内容
1  adc: adc@48003000 {
2      compatible = "st,stm32mp1-adc-core";
3      reg = <0x48003000 0x400>;
4      interrupts = <GIC_SPI 18 IRQ_TYPE_LEVEL_HIGH>,
5                  <GIC_SPI 90 IRQ_TYPE_LEVEL_HIGH>;
6      clocks = <&rcc ADC12>, <&rcc ADC12_K>;
7      clock-names = "bus", "adc";
8      interrupt-controller;
9      st,syscfg = <&syscfg>;
10     #interrupt-cells = <1>;
11     #address-cells = <1>;
12     #size-cells = <0>;
13     status = "disabled";
14
15     adc1: adc@0 {
16         compatible = "st,stm32mp1-adc";
17         #io-channel-cells = <1>;
18         reg = <0x0>;
19         interrupt-parent = <&adc>;
20         interrupts = <0>;
21         dmas = <&dmamux1 9 0x400 0x80000001>;
22         dma-names = "rx";
23         status = "disabled";
24     };
25
26     adc2: adc@100 {
27         compatible = "st,stm32mp1-adc";
28         #io-channel-cells = <1>;
29         reg = <0x100>;
30         interrupt-parent = <&adc>;
31         interrupts = <1>;
32         dmas = <&dmamux1 10 0x400 0x80000001>;
33         dma-names = "rx";
34         status = "disabled";
35     };
36 };

第2行,compatible属性值为“st,stm32mp1-adc-core”,所以在整个Linux源码里面搜索这个字符串即可找到STM32MP157的ADC驱动核心文件,这个文件就是drivers/iio/adc/stm32-adc-core.c

第16、27行,compatible属性值“st,stm32mp1-adc”,搜索这个字符串,可以找到ADC驱动文件,这个文件就是drivers/iio/adc/stm32-adc.c

关于STM32MP157的ADC节点更为详细的信息请参考对应的绑定文档:Documentation/devicetree/bindings/iio/adc/st,stm32-adc.txt。接下来简单分析一下绑定文档,后面需要根据绑定文档修改设备树,使能ADC对应的通道。

ADC首先需要一个根节点,adc根节点属性如下:

1、必要属性

  • compatible:兼容性属性,必须的,可以设置为“st,stm32mp1-adc-core”。
  • reg:ADC控制器寄存器信息。
  • interrupts:中断属性,ADC1和ADC2各对应一个中断信息。
  • clocks:时钟属性。
  • clock-names:时钟名字,可选“adc”或“bus”。
  • interrupt-controller:中断控制器。
  • vdda-supply:此属性对应vdda输入模拟电压句柄。
  • vref-supply:此属性对应vref参考电压句柄。
  • interrupt-cells:设置为1。
  • address-cells:设置为1。
  • size-cells:设置为0。

2、可选属性

  • :pinctrl引脚配置信息。
  • booster-supply:嵌入式booster调节器句柄。
  • vdd-supply:vdd输入电压句柄。
  • st,syscfg:系统配置控制器句柄。
  • st,max-clk-rate-hz:最大时钟。

STM32MP157有两个ADC,每个ADC对应一个子节点,ADC子节点相关属性如下:

1、必要属性

  • compatible:兼容性属性,必须的,可以设置为“st,stm32mp1-adc”。
  • reg:不同ADC控制器寄存器地址偏移信息。
  • interrupts:中断线信息,adc@0为0,adc@100为1。
  • st,adc-channels:ADC通道信息,可以设置0-19,分别对应20个通道。
  • st,adc-diff-channels:ADC差分通道信息,如果使用差分ADC功能的话。
  • io-channel-cells:设置为1。

2、可选属性

  • dmas:DMA 通道句柄。
  • dma-names:dma 名字,必须设置成“rx”。
  • assigned-resolution-bits:ADC分辨率,可以设置为8、10、12,、14或16。
  • st,min-sample-time-nsecs:最小采样时间,单位ns。

ADC驱动源码分析

STM32MP157 ADC驱动文件有两个:stm32-adc-core.c和stm32-adc.cstm32-adc-core.c是ADC核心层,主要用于ADC电源等初始化,需要重点关注的是 stm32-adc.c这个文件。stm32-adc.c主体框架是platform,配合IIO驱动框架实现ADC驱动

stm32_adc结构体

ST自己将ADC外设抽象成了结构体,stm32_adc就相当于自定义的设备结构体。stm32_adc结构体贯穿于整个驱动文件,结构体内容如下:

stm32_adc结构体

stm32_adc_probe函数

接下来看一下stm32_adc_probe函数,内容如下(有省略):

示例代码 57.2.2.2 stm32_adc_probe 函数
1   static int stm32_adc_probe(struct platform_device *pdev)
2   {
3       struct iio_dev *indio_dev;
4       struct device *dev = &pdev->dev;
5       irqreturn_t (*handler)(int irq, void *p) = NULL;
6       struct stm32_adc *adc;
7       int ret;
8 
9       if (!pdev->dev.of_node)
10           return -ENODEV;
11 
12      indio_dev = devm_iio_device_alloc(&pdev->dev, sizeof(*adc));
13      if (!indio_dev)
14          return -ENOMEM;
15 
16      adc = iio_priv(indio_dev);
17      adc->common = dev_get_drvdata(pdev->dev.parent);
18      spin_lock_init(&adc->lock);
19      init_completion(&adc->completion);
20      adc->cfg = (const struct stm32_adc_cfg *)
21          of_match_device(dev->driver->of_match_table, dev)->data;
22 
23      indio_dev->name = dev_name(&pdev->dev);
24      indio_dev->dev.parent = &pdev->dev;
25      indio_dev->dev.of_node = pdev->dev.of_node;
26      indio_dev->info = &stm32_adc_iio_info;
27      indio_dev->modes = INDIO_DIRECT_MODE | INDIO_HARDWARE_TRIGGERED;
28 
29      platform_set_drvdata(pdev, adc);
30 
31      ret = of_property_read_u32(pdev->dev.of_node, "reg",
&adc->offset);
32      if (ret != 0) {
33          dev_err(&pdev->dev, "missing reg property\n");
34          return -EINVAL;
35      }
36 
37      adc->irq = platform_get_irq(pdev, 0);
38      if (adc->irq < 0)
39          return adc->irq;
40 
41      ret = devm_request_threaded_irq(&pdev->dev, adc->irq,
stm32_adc_isr,
42                                      stm32_adc_threaded_isr,
43                                      0, pdev->name, adc);
44      if (ret) {
45          dev_err(&pdev->dev, "failed to request IRQ\n");
46          return ret;
47      }
48 
49      adc->clk = devm_clk_get(&pdev->dev, NULL);
50      if (IS_ERR(adc->clk)) {
51          ret = PTR_ERR(adc->clk);
52          if (ret == -ENOENT && !adc->cfg->clk_required) {
53              adc->clk = NULL;
54          } else {
55              dev_err(&pdev->dev, "Can't get clock\n");
56              return ret;
57          }
58      }
59 
60      ret = stm32_adc_of_get_resolution(indio_dev);
61      if (ret < 0)
62          return ret;
63 
64      ret = stm32_adc_chan_of_init(indio_dev);
65      if (ret < 0)
66          return ret;
67 
68      ret = stm32_adc_dma_request(indio_dev);
69      if (ret < 0)
70          return ret;
71 
72      if (!adc->dma_chan)
73          handler = &stm32_adc_trigger_handler;
74 
75      ret = iio_triggered_buffer_setup(indio_dev,
76                     &iio_pollfunc_store_time, handler,
77                     &stm32_adc_buffer_setup_ops);
78      if (ret) {
79          dev_err(&pdev->dev, "buffer setup failed\n");
80          goto err_dma_disable;
81      }
82 
83      /* Get stm32-adc-core PM online */
84      pm_runtime_get_noresume(dev);
85      pm_runtime_set_active(dev);
86      pm_runtime_set_autosuspend_delay(dev,
STM32_ADC_HW_STOP_DELAY_MS);
87      pm_runtime_use_autosuspend(dev);
88      pm_runtime_enable(dev);
89 
90      ret = stm32_adc_hw_start(dev);
91      if (ret)
92          goto err_buffer_cleanup;
93 
94      ret = iio_device_register(indio_dev);
95      if (ret) {
96          dev_err(&pdev->dev, "iio dev register failed\n");
97          goto err_hw_stop;
98      }
......
123     return ret;
124 }

第12行,调用devm_iio_device_alloc函数申请iio_dev,这里也连stm32_adc内存一起申请了。

第16行,调用iio_priv函数从iio_dev里面的到stm32_adc首地址。

第23-27行,初始化iio_dev,重点是第26行的stm32_adc_iio_info,因为用户空间读取ADC数据最终就是由stm32_adc_iio_info来完成的

第37行,调用platform_get_irq获取中断号。

第41行,调用devm_request_threaded_irq函数申请中断,这里使用的是中断线程化。

第60行,调用stm32_adc_of_get_resolution函数获取ADC的分辨率。

第64行,调用stm32_adc_chan_of_init函数初始化ADC通道。

第68行,调用stm32_adc_dma_request函数初始化DMA。

第75行,调用iio_triggered_buffer_setup函数设置IIO触发缓冲区。

第90行,调用stm32_adc_hw_start函数开启ADC。

第94行,调用iio_device_register函数向内核注册iio_dev。

可以看出stm32_adc_probe函数核心就是初始化ADC,然后建立ADC的IIO驱动框架

stm32_adc_iio_info结构体

stm32_adc_iio_info结构体内容如下所示:

stm32_adc_iio_info结构体

重点来看一下第2行的stm32_adc_read_raw函数,因为此函数才是最终向用户空间发送ADC原始数据的,函数内容如下:

stm32_adc_read_raw函数

第9-18行,读取ADC原始数据值,第18行type值为IIO_VOLTAGE,也就是读取电压值。第14行调用stm32_adc_single_conv函数来完成ADC单次读取。stm32_adc_single_conv函数会设置采样率、配置通道、使用硬件触发、开启转换,最后等待转换完成中断发生。

第20-28行,返回ADC对应的分辨率。

第30-36行,返回差分ADC的偏移值。

stm32_adc_read_raw函数内容还是比较简单的,因为只是读取ADC原始值,不像ICM20608那么复杂。关于ADC驱动源码就讲解到这里,接下来学习如何使能ADC,然后编写应用程序读取ADC采集到的值

硬件原理图分析

STM32MP157开发板ADC硬件原理图如下图所示:

ADC原理图

上图中JP2是一个3P的排针,1脚连接到STM32MP157的DAC引脚上(PA4),2脚连
接到ADC引脚上(PA5),3脚连接到VR1这个可调电位器上。本章实验使用STM32MP157的ADC来采集VR1可调电位器的电压,因此要用跳线帽将JP1的2,3脚连接起来,如下图所示:

开发板ADC设置

ADC驱动编写

修改设备树

ADC驱动ST已经编写好了,只需要修改设备树即可。首先在stm32mp15-pinctrl.dtsi文件中添加ADC使用的PA5引脚配置信息:

示例代码 57.4.1.1 PA5 引脚配置信息
1 adc1_in19_pins_a: adc1-in19 {
2     pins {
3         pinmux = <STM32_PINMUX('A', 5, ANALOG)>;
4     };
5 };

接下来在stm32mp157d-atk.dts文件中向根节点添加vdd子节点信息,内容如下:

示例代码 57.4.1.2 vdd 子节点
1 vdd: regulator-vdd {
2     compatible = "regulator-fixed";
3     regulator-name = "vdd";
4     regulator-min-microvolt = <3300000>;
5     regulator-max-microvolt = <3300000>;
6     regulator-always-on;
7     regulator-boot-on;
8 };

最后在stm32mp157d-atk.dts文件中向adc节点追加一些内容,内容如下:

示例代码 57.4.1.3 adc 节点
1  &adc {
2      pinctrl-names = "default";
3      pinctrl-0 = <&adc1_in19_pins_a>;
4      vdd-supply = <&vdd>;
5      vdda-supply = <&vdd>;
6      vref-supply = <&vdd>;
7      status = "okay";
8 
9      adc1: adc@0 {
10         st,adc-channels = <19>;
11         st,min-sample-time-nsecs = <10000>;
12         assigned-resolution-bits = <16>;
13         status = "okay";
14     };
15 };

第3行,配置adc引脚。

第4-6行,设置电压属性。

第9-12行,adc1子节点,第10行st,adc-channels属性设置adc通道为19,第11行st,min-sample-time-nsecs属性设置最小采样时间为10000ns,第12行设置分辨率为16位。

使能ADC驱动

ST官方默认已经使能了ADC驱动,所以不需要修改,但是为了学习,看一下如何使能Linux内核自带的ADC驱动。打开Linux内核配置界面,配置路径如下:

-> Device Drivers
-> Industrial I/O support (IIO [=y])
-> Analog to digital converters
-> STMicroelectronics STM32 adc core (STM32_ADC_CORE [=y])
-> <*>STMicroelectronics STM32 adc //使能 STM32 ADC

如下图所示:

ADC配置项

编写测试APP

编译修改后的设备树,然后使用新的设备树启动系统。进入/sys/bus/iio/devices目录下,此目录下就有ADC对应的iio设备:iio:deviceX,本章例程如下图所示:

ADC iio设备

上图中的“iio:device0”就是ADC设备,因为此时并没有加载其他的IIO设备驱动,只有一个ADC。如果还加载了其他IIO设备驱动,那么就要依次进入iio设备目录,查看一下都对应的是什么设备。

进入“iio:device0”目录,内容如下图所示:

iio:device0目录文件

标准的IIO设备文件目录,只关心三个文件:

  • in_voltage19_raw:ADC1通道19原始值文件。
  • in_voltage_offset:ADC1偏移文件。
  • in_voltage_scale:ADC1比例文件(分辨率),单位为mV。实际电压值(mV)=in_voltage19_raw*in_voltage_scale。

开发板此时in_voltage19_raw和in_voltage_scale这两个文件内容如下:

当前电压

经过计算,上图中实际电压:20779*0.050354003≈1046.3mV,也就是1.0463V。

编写测试APP,其中需要:

新建char数组指针file_path存储iio框架对应文件路径,并enum对应的文件索引。然后欣姐设备结构体,里面存储raw、scale和act就可以了。

编写file_data_read,这个跟之前iio的很类似,就是fopen之后fscanf,到EOF处就fseek把指针调回文件头然后fclose。

之后编写adc_read,里面就是调用file_data_read读出来之后分别atoi和atof转成数字,最后把raw和scale乘起来/1000得到实际值存到adc_dev的结构体指针dev->act之中。

最后是main函数,argc就1个,在while中adc_read然后printf就好了。

运行测试

编译驱动程序和测试APP

由于不需要编写ADC驱动程序,因此也就不需要编译驱动程序。设备树前面已经编译过了,所以这里就只剩下编译测试APP。由于adcApp.c用到了浮点运算,因此编译的时候要使能硬件浮点,输入如下编译adcApp.c 这个测试程序:

arm-none-linux-gnueabihf-gcc -march=armv7-a -mfpu=neon -mfloat-abi=hard adcApp.c -o adcApp

运行测试

注意,在测试之前一定要先按照之前的连接示意图所示,将JP2跳线帽接到左边,也就是将ADC1_CH19通道连接到开发板上的可调电位器上!

输入如下命令,使用adcApp测试程序:

./adcApp

测试APP会不断的读取ADC值并输出到终端,可以通过调节开发板上的电位器来改变电压值,如下图所示:

测试结果

从上图可以看到ADC原始值以及对应的电压值,因为STM32MP157的ADC可采集电压范围为0-3.3V,因此扭动开发板上的电位器的时候,电压会在0-3.3V之间变化。

总结

这一章的学习放在了IIO的驱动后面,所以其实比之前要简单了很多,个人感觉可以先看这个,学一下IIO的驱动框架,然后再去看上一篇笔记,基本的内容都一样,反而上一篇笔记的IIO驱动来搞ICM20608难度大很多

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

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

相关文章

做一个springboot用户信息模块

目录 用户信息部分 1、获取用户详细信息 前言 代码分析 代码实现 测试 2、更新用户信息 前言 代码实现 测试 3、更新用户头像 前言 代码实现 测试 4、更新用户密码 前言 代码实现 测试 用户信息部分 1、获取用户详细信息 前言 承接上一篇博客登录注册功能…

找工作去哪个网站比较好

吉鹿力招聘网是一个专注于互联网岗位求职招聘的网站&#xff0c;提供海量的互联网人才储备。它主要覆盖了互联网类招聘&#xff0c;包括技术、产品、设计、运营、市场、销售等。吉鹿力招聘网的特点是用户量大&#xff0c;需求旺盛。如果你希望找工作&#xff0c;吉鹿力招聘网是…

[HCTF 2018]admin 1(四种解法!)

题目环境&#xff1a; 有登录和注册两个按钮 先注册一个admin用户 注册admin用户 显示admin用户已经被注册了 好&#xff0c;这就简单了&#xff0c;admin用户存在&#xff0c;但是不清楚admin用户的密码 尝试以下弱口令 第一种解法&#xff1a;密码爆破-尝试弱口令 进去login登…

基于selenium的pyse自动化测试框架

介绍&#xff1a; pyse基于selenium&#xff08;webdriver&#xff09;进行了简单的二次封装&#xff0c;比selenium所提供的方法操作更简洁。 特点&#xff1a; 默认使用CSS定位&#xff0c;同时支持多种定位方法&#xff08;id\name\class\link_text\xpath\css&#xff09…

Python+unittest+requests接口自动化测试框架搭建 完整的框架搭建过程

首先配置好开发环境&#xff0c;下载安装Python并下载安装pycharm&#xff0c;在pycharm中创建项目功能目录。如果不会的可以百度Google一下&#xff0c;该内容网上的讲解还是比较多比较全的&#xff01; 大家可以先简单了解下该项目的目录结构介绍&#xff0c;后面会针对每个文…

第 19 章 网络编程

网络可以使不同物理位置上的计算机达到资源共享和通信的目的&#xff0c;在Java中也提供了专门的网络开发程序包--java.net&#xff0c;以方便开发者进行网络程序的开发&#xff0c;本章将讲解TCP与UDP程序开发 19.1 网络编程简介 将地理位置不同的、具有独立功能的多台计算机…

uview的u-calendar日历组件,当设置了 minDate配置项后,会导致第一次打开日历弹窗,不会精准的滚动到选中的日期(设置了默认日期都没用)

发现需要给month.vue文件里的getMonth方法加一个延时器&#xff0c;猜测是因为设置最小日期后&#xff0c;日历没渲染完毕的时候就已经开始获取节点信息了

C# Onnx Yolov8 Detect 印章 指纹捺印 检测

应用场景 检测文件中的印章和指纹捺印&#xff0c;用于判断文件是否合规&#xff08;是否盖章&#xff0c;是否按印&#xff09; 效果 项目 代码 using Microsoft.ML.OnnxRuntime; using Microsoft.ML.OnnxRuntime.Tensors; using OpenCvSharp; using System; using System.…

Makefile应用

Makefile实例 在c.c里面&#xff0c;包含一个头文件c.h&#xff0c;在c.h里面定义一个宏&#xff0c;把这个宏打印出来。 c.c&#xff1a; #include <stdio.h> #include <c.h>void func_c() {printf("This is C %d\n", C); }c.h #define C 1然后上传…

跨国企业如何选择安全靠谱的跨国传输文件软件?

随着全球化的不断发展&#xff0c;跨国企业之间的合作变得越来越频繁。而在这种合作中&#xff0c;如何安全、可靠地将文件传输给合作伙伴或客户&#xff0c;成为了跨国企业必须面对的问题。 然而&#xff0c;跨国文件传输并不是一件容易的事情&#xff0c;由于网络物理条件的…

Mysql-库的操作

1.创建数据库 CREATE DATABASE [IF NOT EXISTS] name name后可以加 CHARACTER SET 或者是 charsetname COLLATE collation_name &#xff08;mysql数据库不区分大小写&#xff09; 说明&#xff1a; name表示想创建的库的名字大写的表示关键字 [] 是可选项 CHARACTER SET…

Linux调试器-gdb使用

程序的开发过程 debug和release ​ 程序的发布版本有debug和release版本&#xff0c;debug就是可被调试的&#xff08;debug环境下程序可以进行调试&#xff09;&#xff0c;release是发行版本&#xff0c;测试人员和用户使用的版本。Linux下我们编译代码时默认是release版本&…

苍穹外卖项目学习日记(13)

苍穹外卖项目学习日记(13) day10 Spring task cron表达式 导入spring task坐标&#xff08;spring-context已经存在&#xff09;&#xff0c;添加EnableScheduling注解 订单状态定时处理 新建task包&#xff0c;并且创建OrderTask定时类&#xff0c;添加处理超时订单和处…

网络运维Day12

文章目录 yum概述部署阿里镜像源yum基本使用 NFS网络文件系统NFS共享概述部署NFS服务端部署NFS客户端 Tomcat服务实验拓扑安装Tomcat启动服务客户端&#xff08;真机&#xff09;浏览访问页面测试 Tomcat虚拟主机NGINX服务web服务器对比NGINX简介实验拓扑虚拟机A源码编译安装NG…

ESP32-DHT11温湿度数据上传MQTT服务器

ESP32-DHT11温湿度数据上传MQTT服务器 简介ESP32DHT11 实验实验说明接线MQTT服务器建立连接添加订阅 ESP32驱动DHT11ESP32向MQTT服务器发送数据上传温湿度数据实验结果 简介 ESP32 点击图片购买 ESP32 系列模组集成 Wi-Fi、传统蓝牙和低功耗蓝牙功能&#xff0c;具有广泛的用途…

centos7系统离线安装tcpdump抓包软件、使用教程

tcpdump 是Linux系统下的一个强大的命令&#xff0c;可以将网络中传送的数据包完全截获下来提供分析。它支持针对网络层、协议、主机、网络或端口的过滤&#xff0c;并提供and、or、not等逻辑语句来帮助你去掉无用的信息。 本教程对tcpdump命令使用进行讲解说明&#xff0c;通…

对话金山云高级副总裁刘涛 | 做大模型公司的助力者

“ AIGC的应用化阶段将很快到来&#xff1b;AGI&#xff0c;已步入助手阶段。” 口述 | 刘涛 整理 | 梦婕&云舒 出品&#xff5c;极新 在10月的最后一天&#xff0c;极新有幸采访到了金山云高级副总裁刘涛。我们深入探讨了“云计算逐鹿 AIGC”的话题。作为中国公有云互…

Redhat Linux v8.2 实时内核环境配置及参数调优

BC-Linux V8.2 实时内核环境配置及参数调优 -------物理机 & 虚拟机 一、前言 本文档包含有关Redhat Linux for Real Time的基本安装和调试信息。许多行业和组织需要极高性能的计算&#xff0c;并且可能需要低且可预测的延迟&#xff0c;尤其是在金融和电信行业中。延迟&…

【MySQL】入门基础

文章目录 1 :peach:数据库基础:peach:1.1 :apple:什么是数据库:apple:1.2 :apple:主流数据库:apple:1.3 :apple:服务器/数据库/表关系:apple:1.4 :apple:MySQL架构:apple:1.5 :apple:SQL分类:apple:1.6 :apple:存储引擎:apple: 2 :peach:库的操作:peach:2.1 :apple:创建数据库:…

Xilinx MIPI4.3——bg<x>_pin<y>_nc

摘要&#xff1a;由于使用的需要&#xff0c;我要在一个bank上面使用4个MIPI D-PHY&#xff1b;如果pin的指定&#xff0c;跨了bytegroup就会出现bg<x>_pin<y>_nc信号&#xff0c;而且如果一个bit slice control被多个byte group 使用会发生报错&#xff1b;所以我…