Zephyr OS Nordic芯片的Flash 操作

news2025/5/17 17:38:23

目录

概述

1.  软硬件环境

1.1 软件开发环境

1.2 硬件环境 

2 Flash操作库函数

2.1 nRF52832的Flash 

2.2 Nordic 特有的 Flash 操作

2.2.1 nrfx_nvmc_bytes_write 函数

 2.2.2 nrfx_nvmc_page_erase函数

 2.2.3 nrfx_nvmc_write_done_check 函数

3 操作Flash的接口函数

3.1 接口实现

3.2 函数接口

4 验证

4.1 读取Flash的参数

4.2 大容量数据读写操作

5 使用Zephyr的Flash接口实现驱动

5.1 代码实现

5.2 测试代码实现

5.3 验证功能


概述

本文主要介绍基于 Zephyr RTOS 操作 Nordic 芯片的 Flash 存储器,其主要涉及以下几个方面:Nordic 内部Flash的资源,操作Flash的接口,验证读写数据功能等内容。

1.  软硬件环境

1.1 软件开发环境

nordic提供了基于zephyr平台sdk, 其提供了大量的demo可供开发者参考和使用,同时nordi还提供一个集成的软件库工具,方便开发者安装相应的SDK和编译工具链。集成环境同时包含了其他的一些软件,非常便于进行项目开发。

软件工具功能版本信息
nRF Connect SDK nordic提供基于zephyr的代码库v2.9.0 
nRF Connect SDK Toolchain代码编译工具v2.9.1
VS-CODE集成开发环境v1.99.3 
nRF Connect for Desktopnordic集成工具链v5.1.0
nRF Connect手机App

手机App下载地址:

https://nav.nordicsemi.com/search?query=nRF%20Connect

1.2 硬件环境 

本案例是在nRF52832开发板(nRF52-DK)上实现的,该开发板nRF52832的主要特点如下:

1)板载j-link调试接口

2)引出所有 IO接口,用户可根据实际应用,外载其他设备

3)支持4个LED

4)支持4路Key接口

5)板载UART调试接口,方便打印调试信息

2 Flash操作库函数

2.1 nRF52832的Flash 

在 Zephyr 的设备树中配置 Flash如下,本文以nRF52832的设备树为例:

 具体设备树代码如下:

&flash0 {

	partitions {
		compatible = "fixed-partitions";
		#address-cells = <1>;
		#size-cells = <1>;

		boot_partition: partition@0 {
			label = "mcuboot";
			reg = <0x00000000 0xc000>;
		};
		slot0_partition: partition@c000 {
			label = "image-0";
			reg = <0x0000C000 0x37000>;
		};
		slot1_partition: partition@43000 {
			label = "image-1";
			reg = <0x00043000 0x37000>;
		};
		storage_partition: partition@7a000 {
			label = "storage";
			reg = <0x0007a000 0x00006000>;
		};
	};

2.2 Nordic 特有的 Flash 操作

2.2.1 nrfx_nvmc_bytes_write 函数

nrfx_nvmc_bytes_write 是 Nordic 提供的用于向 Flash 写入数据的底层函数,属于 nrfx 驱动库的一部分。这个函数提供了比 Zephyr 通用 Flash API 更底层的访问方式。

函数原型

nrfx_err_t nrfx_nvmc_bytes_write(uint32_t addr, const void *p_src, uint32_t num_bytes);

参数说明

参数类型描述
addruint32_tFlash 中要写入的目标地址
p_srcconst void *包含要写入数据的源缓冲区指针
num_bytesuint32_t要写入的字节数

 返回值

返回 nrfx_err_t 类型,可能的值为:

  • NRFX_SUCCESS - 写入成功完成

  • NRFX_ERROR_INVALID_ADDR - 提供的地址无效

  • NRFX_ERROR_INVALID_LENGTH - 请求的长度无效

  重要注意事项

  1. 地址对齐

    • 虽然函数名为 bytes_write,但实际上写入操作是以32位字为单位进行的

    • 地址必须是4字节对齐的(addr % 4 == 0)

    • 长度也必须是4的倍数

  2. Flash状态

    • 目标区域必须已经被擦除(全为0xFF)

    • 只能将1改为0(不能将0改为1)

  3. 中断影响

    • Flash写入期间CPU会被暂停

    • 建议在写入关键代码段时禁用中断

 2.2.2 nrfx_nvmc_page_erase函数

nrfx_nvmc_page_erase 是 Nordic nRF 系列芯片提供的用于擦除 Flash 页面的底层函数,属于 nrfx 驱动库的一部分。这个函数执行的是对整个 Flash 页的擦除操作。

函数原型

void nrfx_nvmc_page_erase(uint32_t address);

参数说明

参数类型描述
addressuint32_t要擦除的 Flash 页中的任意地址

 注意事项:

  1. 页面大小

    • 不同 nRF 芯片的 Flash 页面大小不同

    • nRF51 系列:1024 字节 (1KB)

    • nRF52 系列:4096 字节 (4KB)

    • 可以使用 NRF_FICR->CODEPAGESIZE 获取实际的页面大小

  2. 地址对齐

    • 地址参数不需要严格对齐到页面起始地址

    • 函数会自动对齐到包含该地址的页面起始地址

  3. 擦除效果

    • 擦除后,整个页面的所有位将被设置为 1 (0xFF)

    • 擦除是写入操作的必要前提

 2.2.3 nrfx_nvmc_write_done_check 函数

nrfx_nvmc_write_done_check 是 Nordic nRF 系列芯片提供的用于检查 Flash 写入/擦除操作是否完成的辅助函数,属于 nrfx 驱动库的一部分。

函数原型

bool nrfx_nvmc_write_done_check(void);

功能说明

  1. 主要用途

    • 检查 NVMC (Non-Volatile Memory Controller) 是否已完成前一次 Flash 编程或擦除操作

    • 提供非阻塞式的操作完成状态检查机制

  2. 返回值

    • true:表示所有挂起的 Flash 操作已完成

    • false:表示 Flash 操作仍在进行中

  3. 底层原理

    • 通过检查 NVMC 的 READY 寄存器位来确定操作状态

    • 对应寄存器位:NRF_NVMC->READY

3 操作Flash的接口函数

3.1 接口实现

在Zephyr OS框架下使用Flash的相关接口,需要做如下配置:

1)在.conf文件中使能Flash的操作接口

CONFIG_NRFX_NVMC=y
CONFIG_FLASH=y

CONFIG_FLASH_PAGE_LAYOUT=y
CONFIG_FLASH_MAP=y

2)引用相关的头文件

#include <zephyr/kernel.h>
#include <zephyr/drivers/flash.h>
#include <nrfx_nvmc.h>

3.2 函数接口

 1)写操作

u32 user_flash_write(u32 addr, const u8 *buf, u32 size)
{
    nrfx_nvmc_bytes_write(addr, buf, size);

    while(nrfx_nvmc_write_done_check() == false)
        ;

    return size;
}

2)读操作

u32 user_flash_read(u32 addr, u8 *buf, u32 size)
{
    u32 i;
    const uint8_t *p = (const uint8_t *)addr;

    for (i = 0; i < size; i ++) {
        *buf ++ = *p ++;
    }

    return size;
}

3)擦除操作

s8 user_flash_erase(u32 addr)
{
    nrfx_err_t err_code = nrfx_nvmc_page_erase(addr);

    if (err_code != NRFX_SUCCESS)
    {
        LOG_ERR("Erase flash page(0x%08x) err_code(0x%08x)", addr, err_code);
    }

    while(nrfx_nvmc_write_done_check() == false)
    {
        // wait for erase to complete
    }

    return 0;
}

4 验证

4.1 读取Flash的参数

实现函数接口如下:

void user_drv_flash_msg( void )
{
    flash_msg.paga_size =  nrfx_nvmc_flash_page_size_get();
    flash_msg.total_bytes = nrfx_nvmc_flash_size_get();
    flash_msg.total_page = nrfx_nvmc_flash_page_count_get();

    printf("Flash page size:        %d bytes \r\n", flash_msg.paga_size );
	printf("Total flash size:       %d bytes \r\n", flash_msg.total_bytes );
	printf("Total flash page count: %d \r\n ", flash_msg.total_page);
}

验证结果如下: 

4.2 大容量数据读写操作

 实现一个写大容量的数据函数:

#define NOR_FLASH_PAGE_SIZE      4096

u8 wrflash_buff[NOR_FLASH_PAGE_SIZE];
void nor_flash_Write( u32 WriteAddr, u8 *pBuffer, u16 NumByteToWrite)   
{ 
    u32 secpos;
    u16 secoff;
    u16 secremain;
    u16 i;

    secpos = WriteAddr/NOR_FLASH_PAGE_SIZE;
    secoff = WriteAddr%NOR_FLASH_PAGE_SIZE;
    secremain = NOR_FLASH_PAGE_SIZE-secoff;

    if(NumByteToWrite <= secremain)
        secremain = NumByteToWrite;
    
    while(1) 
    {
            user_flash_read(secpos*NOR_FLASH_PAGE_SIZE, wrflash_buff, NOR_FLASH_PAGE_SIZE); 
            for( i=0; i<secremain; i++)
            {
                if(wrflash_buff[secoff+i]!=0XFF)
                     break;  
            }

            if(i<secremain)
            {
                  user_flash_erase(secpos*NOR_FLASH_PAGE_SIZE);
                  for(i=0;i<secremain;i++)
                  {
                          wrflash_buff[i+secoff]=pBuffer[i];
                  }
                  user_flash_write(secpos*NOR_FLASH_PAGE_SIZE,wrflash_buff, NOR_FLASH_PAGE_SIZE);
            }
            else
            {
                  user_flash_write( WriteAddr, pBuffer, secremain);
            }    
            
            if( NumByteToWrite == secremain)
                  break;
            else
            {
                  secpos++;
                  secoff=0;

                  pBuffer += secremain; 
                  WriteAddr += secremain;
                  NumByteToWrite -= secremain;
                  if( NumByteToWrite>NOR_FLASH_PAGE_SIZE )
                       secremain = NOR_FLASH_PAGE_SIZE;
                  else
                      secremain = NumByteToWrite;	
            }
    }
}

验证函数:

u8 write_buff[1024];
u8 read_fuff[1024];
void test_flash()
{
    user_drv_flash_msg();

    for( int i = 0; i < 1024; i++ )
    {
        write_buff[i] = 0x5a;
    }

    u32 address = flash_msg.total_bytes - (flash_msg.total_page-3)*4096 - 12;
    nor_flash_Write(address,write_buff,  1024);

    // read buff 
    user_flash_read(address, read_fuff,  1024);

    for( int i = 0; i < 1024; i++ )
    {
        if( read_fuff[i] != write_buff[i])
        {
             printk(" test_flash: failed \n");   
             return;
        }
    }
    printk(" test_flash: pass \n");   

}

验证结果如下:

5 使用Zephyr的Flash接口实现驱动

5.1 代码实现

 使用的主要接口:

#include <drivers/flash.h>

// 获取 Flash 设备
const struct device *flash_dev = DEVICE_DT_GET(DT_CHOSEN(zephyr_flash_controller));

// 读取数据
int flash_read(const struct device *dev, off_t offset, void *data, size_t len);

// 写入数据
int flash_write(const struct device *dev, off_t offset, const void *data, size_t len);

// 擦除扇区
int flash_erase(const struct device *dev, off_t offset, size_t size);

5.2 测试代码实现

u8 write_buff[1024];
u8 read_fuff[1024];

const struct device *flash_dev = DEVICE_DT_GET(DT_CHOSEN(zephyr_flash_controller));

void test_flash( void )
{
    u32 address;

    user_drv_flash_msg();

    address = flash_msg.total_bytes - (flash_msg.total_page-3)*4096;
    for( int i = 0; i < 1024; i++ )
    {
        write_buff[i] = 0x5a;
    }

    // 擦除一个页面 (通常4KB)
    flash_erase(flash_dev, address, 4096);
    
    // 写入数据
    flash_write(flash_dev, address, write_buff, sizeof(write_buff));
    
    // 读取验证
    flash_read(flash_dev, address, read_fuff, sizeof(read_fuff));

    for( int i = 0; i < 1024; i++ )
    {
        if( read_fuff[i] != write_buff[i])
        {
             printk(" test_flash: failed \n");   
             return;
        }
    }
    printk(" test_flash: pass \n");   
}

5.3 验证功能

烧写代码,运行后结果如下:

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

uv python 卸载

又是查了半天 官网wiki没有 网上一堆傻子胡说 uv提示也不对 AI还在这尼玛胡编乱造 开始 我原来装了这几个环境 uv python list 现在python3.7.7不需要了&#xff0c;卸载&#xff0c;直接 uv python uninstall 3.7.7 去找你自己要卸载的版本号&#xff0c;不需要整个包名复制…

浮点数截断法:四舍五入的精确模拟

理论解释&#xff1a; 1. 目标 假设 a 3.14159&#xff0c;我们想四舍五入到 小数点后两位&#xff08;即 3.14 或 3.15&#xff09;。 2. 步骤拆解 (1) a * 100 把 a 放大 100 倍&#xff0c;让小数点后两位变成整数部分&#xff1a; 3.14159 * 100 314.159 (2) 0.5 关…

技术文章:解决汇川MD500系列变频器干扰问题——GRJ9000S EMC滤波器的应用

1. 引言 汇川MD500系列变频器&#xff08;Variable Frequency Drive, VFD&#xff09;以其高性能、宽功率范围&#xff08;0.4kW-500kW&#xff09;和灵活的控制方式&#xff0c;广泛应用于工业自动化领域&#xff0c;如风机、水泵、传送带和压缩机等。然而&#xff0c;MD500系…

大模型数据分析破局之路20250512

大模型数据分析破局之路 本文面向 AI 初学者、数据分析从业者与企业技术负责人&#xff0c;围绕大模型如何为数据分析带来范式转变展开&#xff0c;从传统数据分析困境谈起&#xff0c;延伸到 LLM MCP 的协同突破&#xff0c;最终落脚在企业实践建议。 &#x1f30d; 开篇导语…

基于javaweb的SSM驾校管理系统设计与实现(源码+文档+部署讲解)

技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论文…

Java内存泄露生产环境排查过程,通透了

昨天线上环境崩了 java堆内存溢出。。。 报错&#xff1a;java.lang.OutOfMemoryError: Java heap space 下面我将我排查问题的思路和过程记录了下来 1. 场景 客户端跟Java服务端通过websocket连接建立长链接并发送语音数据&#xff08;text格式&#xff09;Java服务端跟听…

NHANES指标推荐:MDS

文章题目&#xff1a;The association between magnesium depletion score (MDS) and overactive bladder (OAB) among the U.S. population DOI&#xff1a;10.1186/s41043-025-00846-x 中文标题&#xff1a;美国人群镁耗竭评分 &#xff08;MDS&#xff09; 与膀胱过度活动症…

【HTML5学习笔记1】html标签(上)

web标准&#xff08;重点&#xff09; w3c 构成&#xff1a;结构、表现、行为&#xff0c;结构样式行为相分离 结构&#xff1a;网页元素整理分类 html 表现&#xff1a;外观css 行为&#xff1a;交互 javascript html标签 1.html语法规范 1&#xff09; 所有标签都在…

计算机视觉---目标检测(Object Detecting)概览

一、目标检测定义与核心任务 1. 定义 任务&#xff1a;在图像/视频中定位并分类所有感兴趣目标&#xff0c;输出边界框&#xff08;Bounding Box&#xff09;和类别标签。核心输出&#xff1a; 坐标&#xff1a;((x_1, y_1, x_2, y_2))&#xff08;左上角右下角&#xff09;或…

在vue3中使用Cesium的保姆教程

1. 软件下载与安装 1. node安装 Vue.js 的开发依赖于 Node.js 环境&#xff0c;因此我们首先需要安装 Node.js。Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境&#xff0c;它允许你在服务器端运行 JavaScript 代码&#xff0c;同时也为前端开发提供了强大的工具支…

IP地址、端口、TCP介绍、socket介绍、程序中socket管理

1、IP地址&#xff1a;IP 地址就是 标识网络中设备的一个地址&#xff0c;好比现实生活中的家庭地址。IP 地址的作用是 标识网络中唯一的一台设备的&#xff0c;也就是说通过IP地址能够找到网络中某台设备。 2、端口&#xff1a;代表不同的进程,如下图&#xff1a; 3、socket:…

搭建运行若依微服务版本ruoyi-cloud最新教程

搭建运行若依微服务版本ruoyi-cloud 一、环境准备 JDK > 1.8MySQL > 5.7Maven > 3.0Node > 12Redis > 3 二、后端 2.1数据库准备 在navicat上创建数据库ry-seata、ry-config、ry-cloud运行SQL文件ry_20250425.sql、ry_config_20250224.sql、ry_seata_2021012…

RK3568-鸿蒙5.1与原生固件-扇区对比分析

编译生成的固件目录地址 ../openharmony/out/rk3568/packages/phone/images鸿蒙OS RK3568固件分析 通过查看提供的信息&#xff0c;分析RK3568开发板固件的各个组件及其用途&#xff1a; 主要固件组件 根据终端输出的文件列表&#xff0c;RK3568固件包含以下关键组件&#x…

常见激活函数——作用、意义、特点及实现

文章目录 激活函数的意义常见激活函数及其特点1. Sigmoid&#xff08;Logistic 函数、S型函数&#xff09;2. Tanh&#xff08;双曲正切函数&#xff09;3. ReLU&#xff08;Rectified Linear Unit修正线性单元&#xff09;4. Softmax5. Swish&#xff08;Google 提出&#xff…

基于微信小程序的在线聊天功能实现:WebSocket通信实战

基于微信小程序的在线聊天功能实现&#xff1a;WebSocket通信实战 摘要 本文将详细介绍如何使用微信小程序结合WebSocket协议开发一个实时在线聊天功能。通过完整的代码示例和分步解析&#xff0c;涵盖界面布局、WebSocket连接管理、消息交互逻辑及服务端实现&#xff0c;适合…

小波变换+注意力机制成为nature收割机

小波变换作为一种新兴的信号分析工具&#xff0c;能够高效地提取信号的局部特征&#xff0c;为复杂数据的处理提供了有力支持。然而&#xff0c;它在捕捉数据中最为关键的部分时仍存在局限性。为了弥补这一不足&#xff0c;我们引入了注意力机制&#xff0c;借助其能够强化关注…

【无标题】威灏光电哲讯科技MES项目启动会圆满举行

5月14日&#xff0c;威灏光电与哲讯科技MES项目启动会在威灏光电总部隆重举行。威灏光电董事长江轮、总经理刘明星、哲讯科技总经理崔新华、副总王子文及双方项目组成员共同出席&#xff0c;标志着两家企业在数字化领域的第二次深度合作正式启航。 强强联手&#xff0c;二度合作…

display:grid网格布局属性说明

网格父级 &#xff1a;display:grid&#xff08;块级网格&#xff09;/ inline-grid&#xff08;行内网格&#xff09; 注意&#xff1a;当设置网格布局&#xff0c;column、float、clear、vertical-align的属性是无效的。 HTML: <ul class"ls02 f18 mt50 sysmt30&…

排序算法之高效排序:快速排序,归并排序,堆排序详解

排序算法之高效排序&#xff1a;快速排序、归并排序、堆排序详解 前言一、快速排序&#xff08;Quick Sort&#xff09;1.1 算法原理1.2 代码实现&#xff08;Python&#xff09;1.3 性能分析 二、归并排序&#xff08;Merge Sort&#xff09;2.1 算法原理2.2 代码实现&#xf…

Java 并发编程归纳总结(可重入锁 | JMM | synchronized 实现原理)

1、锁的可重入 一个不可重入的锁&#xff0c;抢占该锁的方法递归调用自己&#xff0c;或者两个持有该锁的方法之间发生调用&#xff0c;都会发生死锁。以之前实现的显式独占锁为例&#xff0c;在递归调用时会发生死锁&#xff1a; public class MyLock implements Lock {/* 仅…