【Nordic博文分享系列】nRFConnectSDKBasic

Nordic半导体 2025-01-15 17:00




转载自:博客园 |作者:zhengyi_hot

用户在使用 nRF connect SDK(NCS) 的时候经常会操作的外设有GPIO,I2C,SPI,UART。

我们就以 NCS 2.7.0 中的例程代码 nrf\samples\bluetooth\peripheral_lbs 为基础,来演示上述外设的简单使用。

使用的硬件是开发板 nRF52840 DK





具体操作

1

准备工作

  • 首先我们在原本的工程目录的 boards 文件夹里,添加文件 nrf52840dk_nrf52840.overlay。通过这个文件我们可以修改 devicetree。 编译完成后,我们可以查看 build\zephyr\zephyr.dts,以确认devicetree 的更改是否生效。

  • 我们还可以通过修改 prj.conf 来修改 Kconfig。编译完成后,我们可以查看 build\zephyr\.config 以确认 Kconfig 的更改是否生效。




2

GPIO 控制

  • 首先我们演示如何删除原有的按键和LED的 node 。按照下面的代码,来修改 devicetree,就可以删除 button3 和 led3。

/ {    aliases {        /delete-property/ sw3;        /delete-property/ led3;    };};  /delete-node/ &button3;/delete-node/ &led3;
  • 接着我们来更改控制 led2 的管脚。这里我们用 P0.04 控制 led2。

&led2 {    gpios = <&gpio0 4 GPIO_ACTIVE_LOW>;};
  • 最后我们添加一个用户 GPIO 。这里添加了一个名为 user_gpios 的 node。然后又定义了 user_io0,它是 user_gpios 的 subnode。

/ {    user_gpios {        compatible = "gpio-leds";        user_io0: user_io0 {            gpios = <&gpio0 16 GPIO_ACTIVE_LOW>;            label = "user gpio 0";        };    };};

我们不仅在 devicetree 里添加这个 GPIO ,还要在 main.c 里添加代码使用这个GPIO。下面这句代码中,我们声明了结构体变量  user_gpio0,并用宏 GPIO_DT_SPEC_GET 根据 devicetree 里的定义初始化它。

const struct gpio_dt_spec user_gpio_0 = GPIO_DT_SPEC_GET(DT_NODELABEL(user_io0),gpios);

下面这段代码中  gpio_is_ready_dt 是用来检查 GPIO 的状态是否是就绪。用函数 gpio_pin_configure_dt 把 user_gpio_0 配置成输出。gpio_pin_toggle_dt 用来翻转 GPIO。

if (!gpio_is_ready_dt(&user_gpio_0)) {    printk("%s: device not ready.\n", user_gpio_0.port->name);    return 0;}gpio_pin_configure_dt(&user_gpio_0, GPIO_OUTPUT_ACTIVE); for (index = 0; index < 100; index++) {    gpio_pin_toggle_dt(&user_gpio_0);    k_sleep(K_MSEC(100));}

从下面的代码可以看出翻转 GPIO 这个操作有两种 API 可以调用。二者的主要区别是 gpio_pin_toggle_dt 不需要指明引脚 。

/** * @brief Toggle pin level from a @p gpio_dt_spec. * * This is equivalent to: * *     gpio_pin_toggle(spec->port, spec->pin); * * @param spec GPIO specification from devicetree * @return a value from gpio_pin_toggle() */static inline int gpio_pin_toggle_dt(const struct gpio_dt_spec *spec){    return gpio_pin_toggle(spec->port, spec->pin);}



3

I2C 设备控制

Nordic 的芯片中 I2C 接口是由外设 TWI 来实现的,I2C master 由 TWIM 实现, I2C master 由 TWIS 实现。这里将演示如何用一个 TWIM 来连接两个 I2C slave 设备。

  • 首先我们还是先修改 devicetree我们使用 i2c1 这个 node。 一方面按照应用的要求修改这个 node 的 propertise,另一方面在这个 node 里创建两个 sub-node。

1. i2c 的时钟频率通过 clock-frequency 来定义。

2. i2c 的引脚通过 pinctrl-0 和 pinctrl-1 定义。我们将在后面分析 i2c1_default 和 i2c1_sleep 的定义。

3. 这两个 sub-node 一个是 user_i2c_sensor,另一个是 user_i2c_eeprom。这两个 sub-node 通过 propertise reg 来定义各自的 I2C 地址。  

&i2c1 {    status = "ok";    clock-frequency = ;    pinctrl-0 = < &i2c1_default >;    pinctrl-1 = < &i2c1_sleep >;    pinctrl-names = "default", "sleep";    user_i2c_sensor: user_i2c_sensor@0 {        compatible = "i2c-user-define";        reg = <0xA>;    };    user_i2c_eeprom: user_i2c_eeprom@0 {        compatible = "i2c-user-define";        reg = <0x5>;    };      };     

4. i2c1_default 和 i2c1_sleep的定义如下。TWIM_SDA 信号使用的是引脚 P0.04,TWIM_SCL 信号使用的是引脚 P0.03。

&pinctrl {    i2c1_default: i2c1_default {        group1 {            psels = 0, 4)>,                0, 3)>;        };    };     i2c1_sleep: i2c1_sleep {        group1 {            psels = 0, 4)>,                0, 3)>;            low-power-enable;        };    }; };
  • 修改 prj.conf 添加 CONFIG_I2C=y

  • 修改完 devicetree 我们在来添加操作 i2c 的代码。分别定义 i2c1_sensor 和 i2c1_eeprom,它们对应刚才 i2c1 的两个子节点。

const struct i2c_dt_spec i2c1_sensor = I2C_DT_SPEC_GET(DT_NODELABEL(user_i2c_sensor));const struct i2c_dt_spec i2c1_eeprom= I2C_DT_SPEC_GET(DT_NODELABEL(user_i2c_eeprom));

i2c 设备在读写操作前无需调用 API 来配置 ,直接调用下面的写函数。

err = i2c_write_dt(&i2c1_sensor, buf, 1); err = i2c_write_dt(&i2c1_eeprom, buf, 1);

通过逻辑分析仪我们可以看到如下的总线数据,操作的目标地址分别是我们在 devicetree 里设置的数值 0x05 和 0x0A 。



4

SPI 设备控制

Nordic 的芯片中 SPI 接口的 master 端通过 SPIM 实现, slave 端通过 SPIS 实现。这里将演示如何用一个 SPIM 来连接两个 SPI slave 设备。

  • 首先修改 devicetree。

1. 这里我们使用 spi2, 并且关闭 spi1。在 nordic 的nRF52 系列芯片中,相同数字编号的 TWIM, TWIS, SPIM, SPIS 是共用一组硬件模块的。在上面 i2c 中我们已经使用 i2c1, 所以这里我们就不能同时使用 spi1了。

2. cs-gpios 定义了 P0.26 和 P0.27 两 个CS 信号。 SPI 用不同的片选信号,区分不同的 slave 设备。

3. devicetree node spi2 下定义了两个 sub-node 分别是 user_spi_adc 和 user_spi_flash。 sub-node 里定义了三个 propertise。propertise compatible 的取值来自于我们在工程里新添加的文件 dts\bindings\spi-user-define.yaml。 propertise reg 的取值和前面的 propertise cs-gpios 呼应,reg = <0> 的 sub-node 使用 cs-gpios 里面定义的第一个 CS 引脚。reg = <1> 的 sub-node 使用 cs-gpios 里面定义的第二个 CS 引脚。propertise spi-max-frequency 定义 SPI 的时钟频率。两个不同的 SPI 设备可以使用不同的时钟频率驱动。                       

&spi1 {    status = "disabled";}; &spi2 {    status = "okay";    cs-gpios = <&gpio0 26 GPIO_ACTIVE_LOW>,               <&gpio0 27 GPIO_ACTIVE_LOW>;    pinctrl-0 = < &spi2_default >;    pinctrl-1 = < &spi2_sleep >;    pinctrl-names = "default", "sleep";     user_spi_adc: user_spi_adc@0 {        compatible = "spi-user-define";        reg = <0>;        spi-max-frequency = 8)>;    };    user_spi_flash: user_spi_flash@0 {        compatible = "spi-user-define";        reg = <1>;        spi-max-frequency = 8)>;    };       };      

4. 来看一下我们新添加的 dts\bindings\spi-user-define.yaml 里面的内容。如下图 spi-user-define.yaml 里面包含了 spi-device.yaml 文件,这个文件的位置在目录 zephyr\dts\bindings\spi 。

compatible: "spi-user-define" include: [spi-device.yaml]

spi-device.yaml 文件里面定义了 spi 节点需要的一些 propertise。   比如我们在 sub-node 里定义的 propertise spi-max-frequency。 

# Copyright (c) 2018, I-SENSE group of ICCS# SPDX-License-Identifier: Apache-2.0 # Common fields for SPI devices include: [base.yaml, power.yaml] on-bus: spi properties:  reg:    required: true  spi-max-frequency:    type: int    required: true    description: Maximum clock frequency of device's SPI interface in Hz  duplex:    type: int    default: 0    description: |      Duplex mode, full or half. By default it's always full duplex thus 0      as this is, by far, the most common mode.      Use the macros not the actual enum value, here is the concordance      list (see dt-bindings/spi/spi.h)        0    SPI_FULL_DUPLEX        2048 SPI_HALF_DUPLEX    enum:      - 0      - 2048  frame-format:    type: int    default: 0    description: |      Motorola or TI frame format. By default it's always Motorola's,      thus 0 as this is, by far, the most common format.      Use the macros not the actual enum value, here is the concordance      list (see dt-bindings/spi/spi.h)        0     SPI_FRAME_FORMAT_MOTOROLA        32768 SPI_FRAME_FORMAT_TI    enum:      - 0      - 32768  spi-cpol:

5. SPI 引脚定义如下 CLK P0.28, MISO P0.29, MOSI P0.30。

spi2_default: spi2_default {    group1 {        psels = 0, 28)>,                0, 29)>,                0, 30)>;    };};spi2_sleep: spi2_sleep {    group1 {        psels = 0, 28)>,                0, 29)>,                0, 30)>;        low-power-enable;    };}; 
  • 修改 prj.conf 添加 CONFIG_SPI=y   CONFIG_SPI_ASYNC=y。

  • 在 main.c 里添加 SPI 的应用代码下面这段代码定义了两个结构体变量,并通过宏 SPI_DT_SPEC_GET 用 devicetree 里的参数初始化了这两个结构体变量。

#define SPI_OP     SPI_OP_MODE_MASTER | SPI_MODE_CPOL | SPI_MODE_CPHA \                   | SPI_WORD_SET(8) | SPI_LINES_SINGLEstatic struct spi_dt_spec spim2_adc = SPI_DT_SPEC_GET(DT_NODELABEL(user_spi_adc), SPI_OP, 0);static struct spi_dt_spec spim2_flash = SPI_DT_SPEC_GET(DT_NODELABEL(user_spi_flash), SPI_OP, 0);

spi 驱动支持多 buffer 所以要定义 buffer 个数,和每个 buffer 的长度。同样 spi 在读写之前无需调用配置函数,直接调用读写函数就行。

struct spi_buf_set tx_bufs;struct spi_buf spi_tx_buf; tx_bufs.buffers = &spi_tx_buf;tx_bufs.count = 1;spi_tx_buf.buf = buf;spi_tx_buf.len = 2; err = spi_write_dt(&spim2_adc, &tx_bufs);err = spi_write_dt(&spim2_flash, &tx_bufs);

下面是SPI的波形。可以看到和不同的 spi slave 设备通讯的时候, spi master 会拉低不同的 CS 引脚。



5

UART 控制

Nordic 的芯片中 UART 接口叫做 UARTE。这里的 E 是指 EasyDMA , UART 可以使用 DMA 来连续收发。

  • 修改 Devicetree。这里使用 uart1。propertise current-speed 设置 uart 的波特率。

&uart1 {    status = "okay";    current-speed = <115200>;    pinctrl-0 = < &uart1_default >;    pinctrl-1 = < &uart1_sleep >;    pinctrl-names = "default", "sleep";};

TXD pin 为 P1.02, RXD pin 为 P1.01。     

uart1_default: uart1_default {    group1 {        psels = 1, 1)>;        bias-pull-up;    };    group2 {        psels = 1, 2)>;    };}; uart1_sleep: uart1_sleep {    group1 {        psels = 1, 1)>,            1, 2)>;        low-power-enable;    };};
  • 修改 prj.conf 在里面添加 CONFIG_UART_ASYNC_API=y    CONFIG_UART_ASYNC_RX_HELPER=y。

  • 修改 main.c 添加 uart 收发代码。 uart_callback_set 设置 callback 函数 uart_cb。因为这里采用的是异步收发的模式,所以设置callback 函数是必备的。uart_rx_enable 使能接收。uart_tx 发送数据。

err = uart_callback_set(uart1, uart_cb, NULL);//printk("uart_callback_set return %d\n", err); err = uart_rx_enable(uart1, uart_rx_buf, MAX_UART_BUF_LEN, UART_RX_TIMEOUT_MS);//printk("uart_rx_enable return %d\n", err); err = uart_tx(uart1, uart_tx_buf, 6, SYS_FOREVER_MS);//printk("uart_tx return %d\n", err);

callback 函数 uart_cb 可能由多种事件触发。比如当接收到数据后会触发回调,并在参数 EVT 传递 UART_RX_RDY 和接收到的数据和长度。

static void uart_cb(const struct device *dev, struct uart_event *evt, void *user_data){    ARG_UNUSED(dev);     //LOG_INF("uart_cb evt->type:%d", evt->type);    switch (evt->type) {    case UART_TX_DONE:        printk("UART_TX_DONE\n");        break;     case UART_RX_RDY:        printk("UART_RX_RDY\n");        printk("received %d bytes\n", evt->data.rx.len);        break;     case UART_RX_DISABLED:        printk("UART_RX_DISABLED\n");        break;     case UART_RX_BUF_REQUEST:        printk("UART_RX_BUF_REQUEST\n");        uart_rx_buf_rsp(uart1, uart_rx_buf2, MAX_UART_BUF_LEN);        break;     case UART_RX_BUF_RELEASED:        printk("UART_RX_BUF_RELEASED\n");        break;     case UART_TX_ABORTED:        printk("UART_TX_ABORTED\n");        break;     default:        break;    }}
  • 我们在 DK 上把 TXD 引脚和 RXD 引脚短接来测试 UART 的收发,可以看到如下的 log 信息。UART 收到了自己发送的6字节的数据。 



6

UART 应用代码的优化

上面的 uart 演示代码中,我们只实现了简单的收发。下面我们将进一步在此基础上优化 UART 的收发代码。这一部分的修改都在 main.c 里,主要涉及下面几个部分:

  1. Thread 线程

  2. Semaphore 信号量

  3. 线程间通讯 Message queue

  • 线程 下面的代码中通过 K_THREAD_DEFINE 定义了 一个独立的线程来处理 uart 相关的代码。线程处理函数 uart_thread_task 中:也是先用 uart_callback_set 设置了回调函数;再用 uart_rx_enable 使能了接收;然后是一个 for 循环,在里面不断的接收消息,根据消息中的指令发送数据,或者处理接收到的数据。     

#define UART_THREAD_STACK_SIZE    512#define UART_THREAD_PRIORITY      -1 void uart_thread_task(void){    int err;    struct uart_data_item_type uart_msgq;     k_sem_take(&uart_thread_start, K_FOREVER);     printk("uart_thread_task\n");     err = uart_callback_set(uart1, uart_cb, NULL);     err = uart_rx_enable(uart1, uart_rx_buf, MAX_UART_BUF_LEN, UART_RX_TIMEOUT_MS);         for (;;) {        k_msgq_get(&uart_data_msgq, &uart_msgq, K_FOREVER);        printk("received uart data item\n");                 switch(uart_msgq.cmd) {        case UART_CMD_TX:        memcpy(uart_tx_buf,&uart_msgq.data, sizeof(uint32_t));        err = uart_tx(uart1, uart_tx_buf, sizeof(uint32_t), SYS_FOREVER_MS);        break;         case UART_CMD_DATA_PROCESS:        break;         default:            break;             }    }} K_THREAD_DEFINE(uart_thread_id, UART_THREAD_STACK_SIZE, uart_thread_task, NULL, NULL,                NULL, UART_THREAD_PRIORITY, 0, 0);

上面的代码中用 K_THREAD_DEFINE 定义线程的时候,需要指定此线程的优先级 UART_THREAD_PRIORITY。

UART_THREAD_PRIORITY 的数据类型是 integer,可以是正数也可以是负数。优先级的数字越小,优先级越高,负数的优先级比正数高。

thread 的优先级取值为负数时,此 thread 为协同线程 cooperative thread 。当这种线程正在执行的时候,其它更高优先级的线程不能打断它,必须等它执行完再执行下一个线程。

当 thread 的优先级取值为正数,此 thread 为抢占线程 preemptible thread。当这种线程正在执行的时候,其它更高优先级的线程可以打断它,跳转到高优先级的任务。等高优先级的线程执行完才返回原 thread 继续执行。

回到例程代码,从应用的角度出发,我们希望 uart_thread_task 的执行优先级大于 main 函数。通过查询文件 build\zephyr\.config 我们得知 CONFIG_MAIN_THREAD_PRIORITY 的取值为 0,也就是说 main thread 当前的优先级为 0, 所以我们定义了 UART_THREAD_PRIORITY 为 -1。这样 uart thread 的优先级就高于 main thread, 而且 uart thread 的执行不会被其它更高优先级的 thread 打断。

需要注意的是这里的不能被打断只是对 thread 而言,中断是可以打断 cooperative thread 的。

  • 信号量 函数 uart_thread_task 的优先级比 main 函数高,所以会先于main 函数执行。如果之前的函数 uart_thread_task 里没有 k_sem_take(&uart_thread_start, K_FOREVER),就会出现如下图的现象。我们看到 uart thread 的 log 是先于 main thread 被打印出来的。 


从应用的角度,我们希望 uart_thread_task  在 main 函数启动完广播之后再执行。这就引入了一个不同线程之间的同步问题。

Zephyr RTOS 中可以通过 semaphore 解决不同 thread 间的同步问题。下面的代码中通过 K_SEM_DEFINE 定义了一个为 uart_thread_start 的 semaphore 。 

函数 uart_thread_task 执行到函数 k_sem_take 时,如果 uart_thread_start 没有被释放,当前 thread 会被挂起等待,直到 semaphore 被释放。   

static K_SEM_DEFINE(uart_thread_start, 0, 1); #define UART_THREAD_STACK_SIZE    512#define UART_THREAD_PRIORITY      -1 void uart_thread_task(void){    int err;    struct uart_data_item_type uart_msgq;     k_sem_take(&uart_thread_start, K_FOREVER);     printk("uart_thread_task\n");     err = uart_callback_set(uart1, uart_cb, NULL);     err = uart_rx_enable(uart1, uart_rx_buf, MAX_UART_BUF_LEN, UART_RX_TIMEOUT_MS);         for (;;) {

在 main 里通过 k_sem_give 释放 uart_thread_start。uart 线程会打断当前的 main thread 从 k_sem_take 继续执行。

err = spi_write_dt(&spim2_adc, &tx_bufs);err = spi_write_dt(&spim2_flash, &tx_bufs); k_sem_give(&uart_thread_start); struct uart_data_item_type main_msgq; main_msgq.cmd = UART_CMD_TX;main_msgq.data = 0; for (;;) {     while (k_msgq_put(&uart_data_msgq, &main_msgq, K_NO_WAIT) != 0) {        /* message queue is full: purge old data & try again */        k_msgq_purge(&uart_data_msgq);    }    main_msgq.data++;         dk_set_led(RUN_STATUS_LED, (++blink_status) % 2);    k_sleep(K_MSEC(RUN_LED_BLINK_INTERVAL));}
  • 线程间通讯 演示代码中 main thread 会把要发送的数据通过线程通讯发送到  uart thread, uart thread 调用驱动函数发送。zephyr 中提供了多种线程间通讯方式,具体如下图,这里使用的是 message queue。

下面的代码中 K_MSGQ_DEFINE 定义了一个名为 uart_data_msgq 的 message queue。uart_data_msgq 的缓冲区里最多可以容纳 8 个消息。

struct uart_data_item_type {    uint8_t cmd;    uint32_t data;};  K_MSGQ_DEFINE(uart_data_msgq, sizeof(struct uart_data_item_type), 8, 4);

下面这段代码来自于 main thread 的 main 函数。代码会定时循环把待发送的数据打包成一个 message,然后推送到 message queue 里面。

struct uart_data_item_type main_msgq; main_msgq.cmd = UART_CMD_TX;main_msgq.data = 0; for (;;) {     while (k_msgq_put(&uart_data_msgq, &main_msgq, K_NO_WAIT) != 0) {        /* message queue is full: purge old data & try again */        k_msgq_purge(&uart_data_msgq);    }    main_msgq.data++;        dk_set_led(RUN_STATUS_LED, (++blink_status) % 2);    k_sleep(K_MSEC(RUN_LED_BLINK_INTERVAL));}

下面的代码来自 uart thread 的 uart_thread_task 函数。 函数等待 message queue 里推送来的 message。得到 message 后,根据里面的 cmd 字段来处理发送或者接收数据。

void uart_thread_task(void){    int err;    struct uart_data_item_type uart_msgq;     k_sem_take(&uart_thread_start, K_FOREVER);     printk("uart_thread_task\n");     err = uart_callback_set(uart1, uart_cb, NULL);     err = uart_rx_enable(uart1, uart_rx_buf, MAX_UART_BUF_LEN, UART_RX_TIMEOUT_MS);         for (;;) {        k_msgq_get(&uart_data_msgq, &uart_msgq, K_FOREVER);        printk("received uart data item\n");                 switch(uart_msgq.cmd) {        case UART_CMD_TX:        memcpy(uart_tx_buf,&uart_msgq.data, sizeof(uint32_t));        err = uart_tx(uart1, uart_tx_buf, sizeof(uint32_t), SYS_FOREVER_MS);        break;         case UART_CMD_DATA_PROCESS:        break;         default:            break;             }    }}

下面是加入线程间通讯的代码后得到的 log,当我们把 TX 和 RX 引脚短接后可以看出 uart thread 不断的发送从 main thread 传输的数据。   






总结

本文从实际操作出发,介绍了用户最常用的一些外设如 GPIO, I2C, SPI, UART 的配置和使用方法。并介绍了一些简单 RTOS 组件的应用如 thread, semaphore, message queue。希望能帮助 Nordic 用户加快 nRF Connect SDK 的开发速度。          


【联系我们】

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