【Nordic博文分享系列】理解Zephyr编译与配置系统(上)

Nordic半导体 2025-02-07 17:01




转载自:cnblogs.com |作者:Jayant Tang

Zephyr Project是Linux基金会推出的一个Apache2.0开源项目,版权非常友好,适合用于商业项目开发。包含RTOS、编译系统、各类第三方库。NCS中的例程基本都跑在Zephyr RTOS上。

对于之前只接触过IDE+外设驱动库这种开发方式的开发者来说,Zephyr的配置和编译系统可能比较令人费解,但是一旦你能掌握,就会发现它的方便之处。

因篇幅较长,文章将分为上下两篇,本文重点介绍了NCS中的配置和编译工具,包含一些其他开发环境中常见的CMake,Kconfig,DeviceTree等的简单介绍。




通过CMake管理源码

本节只简要介绍NCS中常见的CMake使用方法,篇幅有限不可能完整的介绍CMake。希望完整学习CMake的话可以参考CMake官方文档.

1

CMake基本写法

通过zephyr/samples/hello_world例程的CMakeLists.txt,我们可以看到:

# SPDX-License-Identifier: Apache-2.0
# 指定CMake版本cmake_minimum_required(VERSION 3.20.0)
# 从系统环境变量${ZEPHYR_BASE}找到NCS中的Zephyr安装目录# 并把整个Zephyr系统当作包来导入find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
# 设定项目名称project(hello_world)
# 把main.c添加为app目标的源码target_sources(app PRIVATE src/main.c)

这里的编译目标是app,最终会编译为libapp.a,也就是把用户自己的应用层代码编译成库的形式。最后再链接进Zephyr系统。

这里的PRIVATE控制的是编译的行为:

  • PRIVATE:main.c修改后,只会重新编译app目标

  • PUBLIC:main.c修改后,app目标要重新编译,且所有与APP目标链接的其他目标也要重新编译


2

条件添加源码

条件添加也很好理解,就是某个CMake变量值为true时,才把源码添加到目标中去。例如:

# Include UART ASYNC API adaptertarget_sources_ifdef(CONFIG_BT_NUS_UART_ASYNC_ADAPTER app PRIVATE  src/uart_async_adapter.c)

这里就是CONFIG_BT_NUS_UART_ASYNC_ADAPTER为y时,才添加src/uart_async_adapter.c到源码中。


3

把整个目录添加源码

有时目录层级很多,我们没必要在一个CMakeLists.txt里把所有源码都添加完。

|-CMakeLists.txt|-aaa|  |-CMakeLists.txt|  `-main.c`-bbb   |-CMakeLists.txt   `-hello.c

这时,就可以在项目根目录的CMakeLists.txt中写:

add_subdirectory(aaa)add_subdirectory(bbb)

然后在两个子目录的CMakeLists.txt中添加对应的源码。

当然,目录也是可以条件添加的,最典型的就是在${NCS}/zephyr/driver/CMakeLists.txt中:

add_subdirectory_ifdef(CONFIG_ADC adc)

也就是说,只有启用了CONFIG_ADC=y,Zephyr才会去编译${NCS}/zephyr/driver/adc/目录下的驱动。

此外,如果再去看${NCS}/zephyr/driver/adc/CMakeLists.txt:

...zephyr_library_sources_ifdef(CONFIG_ADC_MCUX_LPADC  adc_mcux_lpadc.c)zephyr_library_sources_ifdef(CONFIG_ADC_SAM_AFEC  adc_sam_afec.c)zephyr_library_sources_ifdef(CONFIG_ADC_NRFX_ADC  adc_nrfx_adc.c)zephyr_library_sources_ifdef(CONFIG_ADC_NRFX_SAADC  adc_nrfx_saadc.c)...

就可以看到,这里又根据不同的MCU平台,来添加对应的adc驱动代码。



4

添加include目录

也就是存放头文件的目录,如:

# 添加CMakeLists.txt所在目录下的inc/目录到app目标target_include_directories(app PRIVATE inc)
# 也是可以条件添加的zephyr_include_directories_ifdef(CONFIG_MEMFAULT configuration/memfault)


5

设置变量

和宏定义类似,把A定义成B。主要是用来定义一些编译系统会用到的东西,例如:

# 指定自己项目的device tree overlay文件set(DTC_OVERLAY_FILE app.oerlay)

除了上述直接把变量定义写在CMakeLists.txt内,还可以在命令行编译时,通过-D选项传入的参数:

west build -b nrf52840dk/nrf52840 -d build --sysbuild -- -D DTC_OVERLAY_FILE app.overlay

注意,CMake参数的传递在--之后,再用多个-D分别传入。


上述通过编译命令添加的CMake变量,也可以在nRF Connect for VS Code的界面中编译时输入:

在CMakeLists.txt中用set()函数,或者在命令行编译时用-D参数,都可以设置你自定义的变量。但是更多时候,还是用来设置Zephyr编译系统的一些选项,这里给出一个表格,方便查找:

Zephyr系统自带选项

CMake中直接修改Kconfig配置项

直接在CMake中指定某个Kconfig选项的值。

命令行参数:

-D=

CONF_FILE

设置当前工程的Kconfig基本配置文件。通常是prj.conf。

命令行参数:

# 设置默认的配置文件-DCONF_FILE=.conf
# 设置特定Build Type下的配置文件-DCONF_FILE=prj_.conf`

SHIELD

很多开发板都是支持Arduino接口的,因此很多器件厂商/分销商会制作Ardiono接口的扩展板:

Zephyr中,也会有这些扩展板的配置(包含device tree和Kconfig)。如果要在工程中启用扩展板,则需要设置CMake变量:

set(SHIELD nrf21540_ek)

或者在编译目标的配置中添加CMake参数:

编译时,会自动合并原始板子和扩展板的Kconfig和Devicetree。

更多CMake配置项,请参考Providing CMake Options


6

总结

项目通过CMake管理源码和include目录。项目本身会把应用代码编译成build/app/libapp.a,最后和Zephyr系统一起链接成可执行文件。

Zephyr系统本身的内核、库、驱动等源码也都是用CMake来管理的。





通过Kconfig管理配置

一个编译系统中,肯定有很多配置项的需求,如:

  • 布尔类型:开关某些功能,决定一些库和内核功能代码是否参与编译

  • 枚举类型:配置某些预设好的功能,比如日志打印级别(ERR/WRN/INF/OFF)等

  • 数值类型:设置具体参数,如线程栈大小、蓝牙MTU Size大小等

以上功能当然可以通过宏定义来实现。但是宏的作用比较有限,且所有的宏都是平等的,无法结构化地管理。

Kconfig就是用来结构化地管理整个项目以及SDK中所有的配置项的。

在Zephyr系统中,RTOS内核、各个功能模块都会有自己的配置项;并且,开发者自己的项目也会有很多配置项。这些配置项之间可能还有依赖关系。

Kconfig就是把一个模块的所有配置项组成一个菜单。所有模块的菜单,通过层级关系拼接在一起,形成一个大菜单。菜单有默认配置项,开发者可以随意修改配置项。只需把自己和默认配置项有差异的部分写到一个配置文件(*.conf)中,就可以方便地进行配置项的管理了。

在管理配置项时,Kconfig相比于宏定义有许多优势:
  • Kconfig不止适用于源码。编译系统(CMake)也可以用到其中的配置来决定源码是否参与编译。

  • Kconfig是结构化的,可以规定配置项之间的依赖关系;支持提前枚举好允许的配置范围。

  • Kconfig菜单方便互相引用。一个功能库在提供源码和API之外,还会提供一个Kconfig菜单,方便开发者使用。

  • 配置项可以保存到配置文件中。多个配置文件可以合并、覆盖。


1

Kconfig交互式菜单

我们知道,Kconfig实际上是定义了一个菜单,在哪里能看到这个菜单呢?

我们可以在VS Code中点击nRF Kconfig GUI:


也可以把鼠标悬浮在这个按钮上,点右边的三个点,然后用Guiconfig(弹窗)或Menuconfig(命令行)的方式进行配置。



这里就只介绍nRF Kconfig GUI:


2

修改并保存配置项

如果我们只是单纯点击界面右上角的"Apply",那么这些配置是保存在.config中的。这是编译过程中生成的一个临时文件,是把各种配置项来源整合到一起,得到的最终配置文件。

如果我们进行pristine build,那么.config文件就会重新生成,我们之前的修改就消失了。

要想永久保存,应该点击“save to file”。然后保存到配置文件(如prj.conf)中。

当你熟练后,就不需要再去这个菜单中找选项了,直接修改配置文件(如prj.conf)即可。


3

构建时配置项的合并

配置项有许多来源。在构建可执行文件时,会在configure阶段,compile之前,对所有来源的配置项按顺序进行合并,合并后的文件就是前面说的临时配置文件.config,路径为:

//zephyr/.config



注:在NCS v2.7.0之前,未采用Sysbuild。不使用Sysbuild时,合并后的配置文件位于build/zephyr/.config


那么,配置项总共有哪些来源呢?

  1. Kconfig菜单中的默认值

  2. 选择板子后,板子自带的一些config。可以在zephyr/boards或者nrf/boards中查看。

  3. CMake变量CONF_FILE指定的配置文件内的配置项,这也是最常用的。默认情况下是以下两个文件:
    项目的prj.conf,它可以覆盖默认值;
    项目的boards/.conf,当编译目标中选择的板子和这里的board_name一致时,可以覆盖默认值。此配置和前一项会合并。

  4. CMake变量EXTRA_CONF_FILE指定的额外配置文件,也就是在VS Code中创建新的build target时,可以选择的"Extra Kconfig fragments"


4

了解Kconfig菜单基本写法

可以先从一个简单的例子${NCS}/nrf/samples/bluetooth/peripheral_uart来参考:

# 引用Zephyr的Kconfig菜单source "Kconfig.zephyr"
# 自定义本项目的菜单menu "Nordic UART BLE GATT service sample"   ... 此处省略...endmenu

菜单中的选项,可以配置它的类型、说明,和默认值:

# 此选项用来设置Nordic UART Service线程的栈大小# 并且具有默认值config BT_NUS_THREAD_STACK_SIZE  int "Thread stack size"  default 1024  help    Stack size used in each of the two threads

菜单中的选项可以连锁使能:

# 当本选项被设置成y时,通过select,同时把CONFIG_BT_SMP的值设置成yconfig BT_NUS_SECURITY_ENABLED  bool "Enable security"  default y  select BT_SMP  help    "Enable BLE security for the UART service"

此外,一个选项也可以指定一个依赖项。如果本选项被启用,但依赖项未被启用,则编译前的配置过程就会报错:

# 配置是否在系统启动时,自动初始化USB ACM设备 (用于输出日志)# 此配置依赖于CONFIG_USB_CDC_ACM=y,也就是说,起码要把USB_CDC_ACM的代码编译进来config USB_DEVICE_INITIALIZE_AT_BOOT  bool "Initialize USB device support at boot"  depends on USB_CDC_ACM  help    Use CDC ACM UART as backend for console, shell, or logging.

当然,Kconfig也不是说要写的非常大,把整个项目的配置都写进去。你也可以每个子文件夹下单独写Kconfig,然后在项目的Kconfig中进行包含:

# 通过绝对路径进行包含source "xxx.Kconfig"
# 通过相对路径进行包含rsource "src/xxx.Kconfig"

某些简单例程,例如zephyr/samples/hello_world,没有什么配置项,所以是可以没有自己的Kconfig的。这种情况下,相当于直接用了Zephyr的Kconfig菜单,也就是相当于:

source "Kconfig.zephyr"



5

显性与隐性配置项

在Kconfig中定义菜单选项时,我们会发现,大多数选项,在变量类型后面会有一个说明字符串(prompt)。

如bool后面的"Support floating point operations":

config FPU   bool "Support floating point operations"   depends on HAS_FPU

这意味着,这个配置项会出现在Kconfig交互式菜单中,我们可以在交互式菜单中修改它的值:

[ ] Support floating point operations

也可以用prj.conf之类的配置文件来直接改它的值:

CONFIG_FPU=y

但是,也有一些隐性配置项,它们的变量类型后面不带说明字符串,我们无法直接修改它的值:

config CPU_HAS_FPU   bool   help     This symbol is y if the CPU has a hardware floating point unit.

一个CPU到底带不带FPU,肯定不由开发者的配置决定,因此不能直接修改是很合理的。

这种配置,通常是通过连锁使能select的方式,被其他配置项使能的,例如zephyr/soc/arm/nordic_nrf/nrf52/Kconfig.soc:

# 隐性配置项config SOC_NRF52840  bool  select CPU_CORTEX_M_HAS_DWT  select CPU_HAS_FPU     ...   # 显性配置项 config SOC_NRF52840_QIAA  bool "NRF52840_QIAA"  select SOC_NRF52840

而这个SOC_NRF52840_QIAA,是我们选择板子时,52840DK的板子自带的默认配置,来自于

zephyr/samples/application_development/out_of_tree_board/boards/arm/nrf52840dk_nrf52840/nrf52840dk_nrf52840_defconfig:

CONFIG_SOC_NRF52840_QIAA=y



6

总结

Zephyr的配置系统是Kconfig定义的菜单。可以用prj.conf之类的文件来修改配置项的值。

Kconfig中的配置项,可以影响CMake中的条件,选择是否添加哪些源码,从而剪裁内核。

Kconfig中的配置项,最终会生成到build//zephyr/include/generated/autoconf.h中,成为源代码中也可以用到的宏。

不要去尝试修改隐性的Kconfig配置项。





DeviceTree和Zephyr驱动模型

device tree比较复杂,具体的语法、使用方法可以参考我的另一篇文章:《详解Zephyr设备树(DeviceTree)与驱动模型》。https://www.cnblogs.com/jayant97/articles/17209392.html

本文中尽量简洁地说明device tree的用途。


1

设备树文件

device tree的文件是Device Tree Source (DTS)。这里用最简洁的语言描述一下dts文件的产生:

  • 芯片级的dts文件,定义了芯片上的各种外设资源及其地址;

  • 板级的dts文件,可以包含芯片级的dts文件。除了芯片之外,也会包含板子上的资源,如按键、LED、i2c等总线上挂的外设等等;

  • 在工程中选板子时,实际上就是选择了板级的dts文件。在工程中,如果想修改默认的dts,是通过*.overlay文件进行覆盖;例如开发板默认的dts(SDK中的文件)默认没有打开串口1,那么就可以在Overlay文件(你的工程中的文件)中打开串口1;

  • build target时,所有这些dts会在编译目录下合并成zephyr.dts。这就是最终的dts。

合并dts的位置

NCS v2.7.0引入了sysbuild,zephyr.dts的路径为build//zephyr/zephyr.dts;

在NCS v2.6.x之前,zephyr.dts的路径为:build/zephyr/zephyr.dts;

overlay文件

如果说*.conf文件是你当前工程的软件配置,那么*.overlay文件就是你的当前工程的硬件配置。

  • app.overlay是整个项目的overlay,如果CMake不设置DTC_OVERLAY_FILE,则默认使用app.overlay

  • boards/.overlay是板子对应的overlay


外设的使能与关闭,引脚的分配等与硬件相关的内容,都在dts overlay文件中编写。修改时,注意不要修改SDK里的dts,因为这会影响其他的工程。只在自己的工程内用Overlay修改就好。

// 例如,在overlay中使能串口1. uart1是label,可以直接引用&uart1 {  compatible = "nordic,nrf-uarte";  status = "okay";}
// 另一种写法,不用label,而用绝对路径/{  soc{    uart@40028000{      compatible = "nordic,nrf-uarte";      status = "okay";    }  }}


2

Zephyr驱动程序

在main()函数运行起来之前,zephyr设备驱动的初始化程序就已经先运行了。设备的驱动程序根据device tree中的配置,自动把外设进行相应的初始化,配置寄存器。然后driver还会提供一个struct device结构体,方便应用层操作这个外设。

程序的application层起来之后,开发者就可以用driver初始化好的device结构体,用标准的Zephyr API进行操作。

有以下5个阶段可以用来初始化外设驱动:

Zephyr外设驱动的整个流程:


【编译阶段】

1. 开发者在Kconfig中,使能了某个外设驱动,如CONFIG_SERIAL=y

2. zephyr/driver/下的CMakeLists.txt,根据CONFIG_SERIAL=y,把zephyr/driver/serial/添加到工程中

3. zephyr/driver/serial/下有各个半导体厂商向Zephyr提交的串口驱动代码。此目录下的CMakeLists.txt根据你的当前Kconfig配置,来选择哪个驱动文件编译进来:

zephyr_library_sources_ifdef(CONFIG_UART_NRFX_UART uart_nrfx_uart.c)if (CONFIG_UART_NRFX_UARTE)  if (CONFIG_UART_NRFX_UARTE_LEGACY_SHIM)    zephyr_library_sources(uart_nrfx_uarte.c)  else()    zephyr_library_sources(uart_nrfx_uarte2.c)  endif()endif()

4. 驱动代码中,会通过宏来匹配zephyr.dts中的所有串口节点,也就是匹配哪些节点的compatible与当前驱动是一致的。然后,再匹配这些节点的status="okay",就说明这个外设被使能了,于时就定义一个device结构体实例。


device结构体的定义:

struct device { const char *name;           // 设备的名称 const void *config;         // 设备的初始配置 const void *api;            // 设备的api函数集合 struct device_state *state; // 设备的工作状态 void *data;                 // 设备的运行数据 /* ... */                   // 其他参数,例如电源管理};

【运行阶段】

1. 系统启动后,在设备驱动程序预设好的阶段(上图5个阶段之一),进行外设的初始化和配置。配置的值就来自于dts overlay中节点的配置。
如果是外挂芯片的驱动,则会在这个阶段完成外挂芯片的配置(如SPI总线的液晶屏、I2C总线的RTC时钟等)。
以上只是两个示例,具体的行为,要看根据驱动程序的代码。

2. 程序进入到应用层之后,所有需要的外设就已经被初始化好了。在应用层代码中,开发者只需先获得这个device结构体的指针,后续调用Zephyr标准外设API时,把这个指针作为参数传入即可。

// 例如,获取串口1的device结构体指针static const struct device *uart1_dev = DEVICE_DT_GET(DT_NODELABEL(uart1));
// 使用串口1发送数据uart_tx(uart1_dev, buf, len, timeout);


3

Zephyr驱动模型的优劣

优点:

  • 代码里调用的都是Zephyr标准API,与硬件细节无关。如果后需要更换MCU平台,几乎没有什么移植成本,只需要更换所选的board即可。

  • 通用性强,无论是普通的串口,还是USB串口,抑或是LPUART,它们的应用层代码均是Zephyr标准API,只需要更换底层驱动即可。

  • 开发者无需花精力在标准、通用的基本功能上,如串口、SPI、网络、按钮等。因为这些驱动都是厂商提供的,在性能、健壮性、功能性上往往都强于开发者自己用寄存器或外设驱动库开发的代码。


缺点:

  • 上手难度稍高,需要花精力去学习语法,并且要简单了解驱动代码

  • 功能不完全。Zephyr只提供最标准的用法,当用到串口、spi、i2c等协议时,就是最标准的协议。一旦有不符合标准的,或者Zephyr标准库未提供的功能,就无法在Zephyr驱动模型的框架下实现了。


例如,nordic的芯片有PPI的功能,可以让一个外设的event触发另一个外设的task。这个功能Zephyr是没有标准驱动的。

Nordic可以在提交给Zephyr的驱动代码中用PPI。例如,在串口驱动中,通过uart外设和timer外设,加上PPI,实现异步流控串口(Timer的作用是记录发送/接收了多少字节,然后用PPI控制GPIO CTS/RTS),Nordic提供的驱动代码,把他们整体封装成串口,也就是说,Zephyr标准驱动操作的串口,实际并不是单独对应uart这一个外设,而是UART+GPIOTE+TIMER+PPI的复合外设。

如果用户想自己用PPI实现一些自定义功能,只能直接调用nrfx api。


4

Nordic NRFX外设驱动库

如果你的需求比较特殊,想要绕过Zephyr驱动层,直接在底层驱动甚至寄存器和中断的级别来进行开发,NCS也是支持的。

请参考《在NCS中使用NRFX外设驱动库》。
https://www.cnblogs.com/jayant97/articles/17835258.html


5

总结

dts怎么写,本质上取决于驱动代码里怎么读取dts。dts的本质就是保存硬件细节相关的信息,使自己的应用代码与硬件细节解耦。

要更详细地了解Device Tree,请参考《详解Zephyr设备树(DeviceTree)与驱动模型》。

https://jayant-tang.github.io/jayant97.github.io/2023/03/4b274a50e575/


后半部分将于下期连载,敬请期待




❤️ 感谢伙伴们的热情支持,新的一年,Nordic会带着大家的期待推出更多优秀产品和技术解决方案,增加大家呼声很高的nRF54系列等产品产量,一起加油!

任何产品和技术相关问题,欢迎随时私信Nordic !


本期获奖名单如下

@iam铭哥、@Figo、@pretty、@kinggate、@文若、@W11、@Dan、@安静、@Lucifer、@Rain、@释怀、@Rabbit米唐、@一两、@如去如来、@小瞌睡虫、@🌙 、@nikkkkk 、@悠悠海


请以上朋友于2月14日前后台私信联系我们,及时领取红包卡券!




【联系我们】

中文官网:www.nordicsemi.cn

英文官网:www.nordicsemi.com

微信公众号:nordicsemi


【Nordic 开发者论坛】

https://devzone.nordicsemi.com


【销售接洽】

北京分公司: +86 010 8438 2767

上海分公司: +86 21 6330 0620

深圳分公司: +86 755 8322 0147

sales.cn@nordicsemi.no

 点击“阅读原文” 探索更多Nordic资讯

Nordic半导体 Nordic 半导体开发支持蓝牙智能、ANT+和2.4GHz应用的超低功耗短距无线通信技术,用于物联网 、可穿戴产品、智能家居、玩具等应用。Nordic 提供现成可用的设计框架、世界级文档资料和支持,以加快专业工程师和业余爱好者的开发速度。
评论 (0)
  • 体积大小:14*11*2.6CM,电气参数:输入100V-240V/10A,输出16V24A。PCB 正面如下图。PCB 背面如下图。根据实际功能可以将PCB分成几部分:EMI滤波,PFC电路,LLC电路。EMI滤波区域,两级共模电感,LN各用了保险丝加压敏电阻,继电器(HF32FV-G)用来切除NTC的,为了提高效率点,如下图。PFC电路区域,如下图。LLC电路区域,如下图。详细分析一下该电源用的主要IC还有功率器件。AC侧采用了两颗整流桥进行并联,器件增加电流应力,如下图。共模电感都有放电针
    liweicheng 2025-05-10 20:03 25浏览
  •         信创产业含义的“信息技术应用创新”一词,最早公开信息见于2019年3月26日,在江苏南京召开的信息技术应用创新研讨会。本次大会主办单位为江苏省工业和信息化厅和中国电子工业标准化技术协会安全可靠工作委员会。        2019年5月16日,美国将华为列入实体清单,在未获得美国商务部许可的情况下,美国企业将无法向华为供应产品。       2019年6
    天涯书生 2025-05-11 10:41 106浏览
  • 递交招股书近一年后,曹操出行 IPO 进程终于迎来关键节点。从 2024 年 4 月首次递表,到 2025 年 4 月顺利通过中国证监会境外发行上市备案,并迅速更新招股书。而通过上市备案也标志着其赴港IPO进程进入实质性推进阶段,曹操出行最快有望于2025年内完成港股上市,成为李书福商业版图中又一关键落子。行路至此,曹操出行面临的挑战依然不容忽视。当下的网约车赛道,早已不是当年群雄逐鹿的草莽时代,市场渐趋饱和,竞争近乎白热化。曹操出行此时冲刺上市,既是背水一战,也是谋篇布局。其招股书中披露的资金
    用户1742991715177 2025-05-10 21:18 44浏览
  • ‌磁光克尔效应(Magneto-Optic Kerr Effect, MOKE)‌ 是指当线偏振光入射到磁性材料表面并反射后,其偏振状态(偏振面旋转角度和椭偏率)因材料的磁化强度或方向发生改变的现象。具体表现为:1、‌偏振面旋转‌:反射光的偏振方向相对于入射光发生偏转(克尔旋转角 θK)。2、‌椭偏率变化‌:反射光由线偏振变为椭圆偏振(克尔椭偏率 εK)。这一效应直接关联材料的磁化状态,是表征磁性材料(如铁磁体、反铁磁体)磁学性质的重要非接触式光学探测手段,广泛用于
    锦正茂科技 2025-05-12 11:02 54浏览
  • 文/Leon编辑/cc孙聪颖‍在新能源汽车赛道的残酷洗牌中,威马、爱驰等数十个品牌黯然退场,极越、哪吒汽车也深陷经营困局,“跨界造车” 早已褪去曾经的光环,成为吞噬企业资金与精力的风险泥潭,尤其对上市公司而言,稍有不慎便会被拖入业绩泥沼。当行业共识已清晰显现 —— 新能源汽车市场这片红海正上演着惨烈的生存之战,石头科技创始人昌敬却逆势入局,掌舵极石汽车,其押注造车的抉择,正让本就面临挑战的石头科技主业雪上加霜。2025 年 4 月中旬,昌敬突然清空微博、抖音等社交媒体账号的举动,迅速引爆舆论场。
    华尔街科技眼 2025-05-09 20:53 24浏览
  • 在 AI 浪潮席卷下,厨电行业正经历着深刻变革。AWE 2025期间,万得厨对外首次发布了wan AiOS 1.0组织体超智能系统——通过AI技术能够帮助全球家庭实现从健康检测、膳食推荐,到食材即时配送,再到一步烹饪、营养总结的个性化健康膳食管理。这一创新之举并非偶然的个案,而是整个厨电行业大步迈向智能化、数字化转型浪潮的一个关键注脚,折射出全行业对 AI 赋能的热切渴求。前有标兵后有追兵,万得厨面临着高昂的研发成本与技术迭代压力,稍有懈怠便可能被后来者赶
    用户1742991715177 2025-05-11 22:44 48浏览
  • 1.概述MYD-YG2LX采用瑞萨RZ/G2L作为核心处理器,该处理器搭载双核Cortex-A55@1.2GHz+Cortex-M33@200MHz处理器,其内部集成高性能3D加速引擎Mail-G31 GPU(500MHz)和视频处理单元(支持H.264硬件编解码),16位的DDR4-1600 / DDR3L-1333内存控制器、千兆以太网控制器、USB、CAN、SD卡、MIPI-CSI等外设接口,在工业、医疗、电力等行业都得到广泛的应用。米尔基于瑞萨RZ/G2L开发板本文主要介绍基于MYD-Y
    米尔电子嵌入式 2025-05-09 17:38 23浏览
  • 蓝牙耳机是长这个样子,如下图。背部图,如下图。拆开L耳的一侧,有NFC和电池包(501230 3.7V 150mAh)如下图。电池包(501230 3.7V 150mAh)如下图。NFC正面,如下图。NFC背面,如下图。如何理解NFC的工作原理呢,搜集一下相关的资料,如下图。拆开R耳的一侧,PCB正面,如下图。PCB背面,如下图。有两组红黑的线,一组连接到了喇叭,另一组连接到了MIC头上,MIC头参数如下图。蓝牙模块(CSR 8635),有蛇形PCB走线做成天线,节约了天线成本,如下图。该IC介
    liweicheng 2025-05-10 00:45 24浏览
  • 行车记录仪是长这个样子的,如下图。从前面拆去玻璃挡板,可以清晰的看见里面的部件,5个按键电路板,液晶显示屏,摄像头,喇叭,电池包,还有一块主电路板。液晶显示屏正面,如下图。液晶显示屏背面,如下图。喇叭,如下图。5个按键的电路板,MENU,DOWN,POWER,UP,OK总共5个按键功能,导线连接到主电路板上,如下图。电池包,303040聚合物锂电池,3.7V,300mAH,如下图。如下图。摄像头,如下图。拿去摄像头外壳,如下图。分离广角聚集镜头和PCB板,如下图。广角聚焦镜头,具体结构如下图。P
    liweicheng 2025-05-09 22:50 25浏览
  • 【拆解】+CamFi卡菲单反无线传输器拆解 对于单反爱好者,想要通过远程控制自拍怎么办呢。一个远程连接,远程控制相机拍摄的工具再合适不过了。今天给大伙介绍的是CamFi卡菲单反无线传输器。 CamFi 是专为数码单反相机打造的无线传输控制器,自带的 WiFi 功能(无需手机流量),不但可通过手机、平板、电脑等设备远程连接操作单反相机进行拍摄,而且还可实时传输相机拍摄的照片到 iPad 和电视等大屏设备进行查看和分享。 CamFi 支持大部分佳能和尼康单反相机,内置可充电锂离子电池,无需相机供电。
    zhusx123 2025-05-11 14:14 69浏览
  • 【拆解】+自动喷香机拆解 家里之前买了从PDD买了一个小型自动喷香机放在厕所里。来增加家里的温馨感,这东西看着确实小巧,精致。可是这东西吧,耗电就是快,没过几天就没电了。今个就让我拆开看看什么在捣鬼。如下是产品的实物和宣传图: 由于螺丝孔太小和限位很深。对于我的螺丝刀套装没用。只能使用那种螺丝刀细头,同时又长的小螺丝刀进行拆解 拧下三颗螺丝钉,用一字螺丝刀撬开外壳,内部结构就呈现在眼前。 内部构造相当简单,部件没多少。就是锂电池供电,通过MCU实现按键控制,段码屏控制,LE
    zhusx123 2025-05-10 19:55 40浏览
  • 在印度与巴基斯坦的军事对峙情境下,歼10C的出色表现如同一颗投入平静湖面的巨石,激起层层涟漪,深刻印证了“质量大于数量”这一铁律。军事领域,技术优势就是决定胜负的关键钥匙。歼10C凭借先进的航电系统、强大的武器挂载能力以及卓越的机动性能,在战场上大放异彩。它能够精准捕捉目标,迅速发动攻击,以一敌多却毫不逊色。与之形成鲜明对比的是,单纯依靠数量堆砌的军事力量,在面对先进技术装备时,往往显得力不从心。这一现象绝非局限于军事范畴,在当今社会的各个领域,“质量大于数量”都已成为不可逆转的趋势。在科技行业
    curton 2025-05-11 19:09 161浏览
  •   定制软件开发公司推荐清单   在企业数字化转型加速的2025年,定制软件开发需求愈发多元复杂。不同行业、技术偏好与服务模式的企业,对开发公司的要求大相径庭。以下从技术赛道、服务模式及行业场景出发,为您提供适配的定制软件开发公司推荐及选择建议。   华盛恒辉科技有限公司:是一家专注于高端软件定制开发服务和高端建设的服务机构,致力于为企业提供全面、系统的开发制作方案。在部队政企开发、建设到运营推广领域拥有丰富经验,在教育,工业,医疗,APP,管理,商城,人工智能,部队软件、工业软件、数字化转
    华盛恒辉l58ll334744 2025-05-12 15:55 44浏览
  •   基于 2025 年行业权威性与时效性,以下梳理国内知名软件定制开发企业,涵盖综合型、垂直领域及特色技术服务商:   华盛恒辉科技有限公司:是一家专注于高端软件定制开发服务和高端建设的服务机构,致力于为企业提供全面、系统的开发制作方案。在部队政企开发、建设到运营推广领域拥有丰富经验,在教育,工业,医疗,APP,管理,商城,人工智能,部队软件、工业软件、数字化转型、新能源软件、光伏软件、汽车软件,ERP,系统二次开发,CRM等领域有很多成功案例。   五木恒润科技有限公司:是一家专业的部队信
    华盛恒辉l58ll334744 2025-05-12 16:13 31浏览
我要评论
0
5
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦