rtthread stm32h743的使用(十二)spi設備fal驅動的使用

我們要在rtthread studio 開發環境中建立stm32h743xih6芯片的工程。我們使用一塊stm32h743及fpga的核心板完成相關實驗,核心板如圖:
在這里插入圖片描述
在這里插入圖片描述
fal驅動的使用是建立在sfud驅動之上的,所以我們在上一節使用的工程基礎上繼續實驗。
在這里插入圖片描述1.在上一節工程的基礎上,我們添加fal驅動:
在這里插入圖片描述
在這里插入圖片描述
2.打開工程目錄,將samples文件夾下的fal cfg.h復制到inc文件夾下,如圖:
在這里插入圖片描述
3.在drive文件夾下添加drv_flash_h7.c文件如圖:
在這里插入圖片描述
代碼如下:

#include <rtconfig.h>
#include <rtdef.h>
#include <board.h>#ifdef BSP_USING_ON_CHIP_FLASH
#include "drv_config.h"
#include "drv_flash.h"#if defined(RT_USING_FAL)
#include "fal.h"
#endif//#define DRV_DEBUG
#define LOG_TAG                "drv.flash"
#include <drv_log.h>/*** Read data from flash.* @note This operation's units is word.** @param addr flash address* @param buf buffer to store read data* @param size read bytes size** @retval The length of bytes that have been read*/
int stm32_flash_read(rt_uint32_t addr, rt_uint8_t *buf, size_t size)
{size_t i;if ((addr + size - 1) > FLASH_END){LOG_E("read outrange flash size! addr is (0x%p)", (void *)(addr + size));return -RT_ERROR;}for (i = 0; i < size; i++, buf++, addr++){*buf = *(rt_uint8_t *) addr;}return size;
}/*** Write data to flash.* @note This operation's units is word.* @note This operation must after erase. @see flash_erase.** @param addr flash address* @param buf the write data buffer* @param size write bytes size** @return The length of bytes that have been written*/
int stm32_flash_write(rt_uint32_t addr, const rt_uint8_t *buf, size_t size)
{rt_err_t result      = RT_EOK;rt_uint32_t end_addr = addr + size - 1, write_addr;rt_uint32_t write_granularity = FLASH_NB_32BITWORD_IN_FLASHWORD * 4;rt_uint32_t write_size = write_granularity;rt_uint8_t write_buffer[32] = {0};if ((end_addr) > FLASH_END){LOG_E("write outrange flash size! addr is (0x%p)", (void *)(addr + size));return -RT_EINVAL;}if(addr % 32 != 0){LOG_E("write addr must be 32-byte alignment");return -RT_EINVAL;}if (size < 1){return -RT_EINVAL;}HAL_FLASH_Unlock();write_addr = (uint32_t)buf;__HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_EOP | FLASH_FLAG_OPERR | FLASH_FLAG_WRPERR);while (addr < end_addr){if(end_addr - addr + 1 < write_granularity){write_size = end_addr - addr + 1;for(size_t i = 0; i < write_size; i++){write_buffer[i] = *((uint8_t *)(write_addr + i));}write_addr = (uint32_t)((rt_uint32_t *)write_buffer);}if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_FLASHWORD, addr, write_addr) == HAL_OK){for(rt_uint8_t i = 0; i < write_size; i++){if (*(rt_uint8_t *)(addr + i) != *(rt_uint8_t *)(write_addr + i)){result = -RT_ERROR;goto __exit;}}addr += write_granularity;write_addr  += write_granularity;}else{result = -RT_ERROR;goto __exit;}}__exit:HAL_FLASH_Lock();if (result != RT_EOK){return result;}return size;
}/*** Erase data on flash.* @note This operation is irreversible.* @note This operation's units is different which on many chips.** @param addr flash address* @param size erase bytes size** @return result*/
int stm32_flash_erase(rt_uint32_t addr, size_t size)
{rt_err_t result = RT_EOK;rt_uint32_t SECTORError = 0;if ((addr + size - 1) > FLASH_END){LOG_E("ERROR: erase outrange flash size! addr is (0x%p)\n", (void *)(addr + size));return -RT_EINVAL;}rt_uint32_t addr_bank1 = 0;rt_uint32_t size_bank1 = 0;
#ifdef FLASH_BANK_2rt_uint32_t addr_bank2 = 0;rt_uint32_t size_bank2 = 0;
#endifif((addr + size) < FLASH_BANK2_BASE){addr_bank1 = addr;size_bank1 = size;
#ifdef FLASH_BANK_2size_bank2 = 0;
#endif}else if(addr >= FLASH_BANK2_BASE){size_bank1 = 0;
#ifdef FLASH_BANK_2addr_bank2 = addr;size_bank2 = size;
#endif}else{addr_bank1 = addr;size_bank1 = FLASH_BANK2_BASE - addr_bank1;
#ifdef FLASH_BANK_2addr_bank2 = FLASH_BANK2_BASE;size_bank2 = addr + size - FLASH_BANK2_BASE;
#endif}/*Variable used for Erase procedure*/FLASH_EraseInitTypeDef EraseInitStruct;/* Unlock the Flash to enable the flash control register access */HAL_FLASH_Unlock();EraseInitStruct.TypeErase     = FLASH_TYPEERASE_SECTORS;EraseInitStruct.VoltageRange  = FLASH_VOLTAGE_RANGE_3;SCB_DisableDCache();if(size_bank1){EraseInitStruct.Sector    = (addr_bank1 - FLASH_BANK1_BASE) / FLASH_SECTOR_SIZE;EraseInitStruct.NbSectors = (addr_bank1 + size_bank1 -1 - FLASH_BANK1_BASE) / FLASH_SECTOR_SIZE - EraseInitStruct.Sector + 1;EraseInitStruct.Banks = FLASH_BANK_1;if (HAL_FLASHEx_Erase(&EraseInitStruct, &SECTORError) != HAL_OK){result = -RT_ERROR;goto __exit;}}#ifdef FLASH_BANK_2if(size_bank2){EraseInitStruct.Sector    = (addr_bank2 - FLASH_BANK2_BASE) / FLASH_SECTOR_SIZE;EraseInitStruct.NbSectors = (addr_bank2 + size_bank2 -1 - FLASH_BANK2_BASE) / FLASH_SECTOR_SIZE - EraseInitStruct.Sector + 1;EraseInitStruct.Banks = FLASH_BANK_2;if (HAL_FLASHEx_Erase(&EraseInitStruct, &SECTORError) != HAL_OK){result = -RT_ERROR;goto __exit;}}
#endif__exit:SCB_EnableDCache();HAL_FLASH_Lock();if (result != RT_EOK){return result;}LOG_D("erase done: addr (0x%p), size %d", (void *)addr, size);return size;
}#if defined(RT_USING_FAL)
static int fal_flash_read_128k(long offset, rt_uint8_t *buf, size_t size);
static int fal_flash_write_128k(long offset, const rt_uint8_t *buf, size_t size);
static int fal_flash_erase_128k(long offset, size_t size);
const struct fal_flash_dev stm32_onchip_flash_128k = { "onchip_flash_128k", STM32_FLASH_START_ADRESS, FLASH_SIZE_GRANULARITY_128K, (128 * 1024), {NULL, fal_flash_read_128k, fal_flash_write_128k, fal_flash_erase_128k} };static int fal_flash_read_128k(long offset, rt_uint8_t *buf, size_t size)
{return stm32_flash_read(stm32_onchip_flash_128k.addr + offset, buf, size);
}
static int fal_flash_write_128k(long offset, const rt_uint8_t *buf, size_t size)
{return stm32_flash_write(stm32_onchip_flash_128k.addr + offset, buf, size);
}static int fal_flash_erase_128k(long offset, size_t size)
{return stm32_flash_erase(stm32_onchip_flash_128k.addr + offset, size);
}#endif
#endif /* BSP_USING_ON_CHIP_FLASH */

4.在在board.h文件打開BSP_USING_ON_CHIP_FLASH如圖:
在這里插入圖片描述
5.對fal_cfg.h做更改,使其和h7芯片對應,如圖:
在這里插入圖片描述
6.這時候編譯已經沒有錯誤了,我們下載測試下:
在這里插入圖片描述
可以看到,[I/SFUD] Probe SPI flash norflash0 by SPI device spi10 success.RT-Thread Flash Abstraction Layer initialize success.說明sfud和fal都init成功,并且打印了分區表。
7.測試幾個fal指令,都沒有問題:
在這里插入圖片描述
8.我們測試下性能,性能將會測試 Flash的擦除、寫入及讀取速度,同時將會測試寫入及讀取數據的準確性,保證整個 Flash 或整個分區的 寫入與讀取 數據的一致性。
指令為fal bench 4096 yes
在這里插入圖片描述
其余指令可以參考:

msh />fal
Usage:
fal probe [dev_name|part_name]   - probe flash device or partition by given name
fal read addr size               - read 'size' bytes starting at 'addr'
fal write addr data1 ... dataN   - write some bytes 'data' starting at 'addr'
fal erase addr size              - erase 'size' bytes starting at 'addr'
fal bench <blk_size>             - benchmark test with per block size

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/37082.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/37082.shtml
英文地址,請注明出處:http://en.pswp.cn/web/37082.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

SpringCloud Alibaba Seata2.0基礎入門與安裝

官網地址&#xff1a;https://seata.apache.org/zh-cn/ GitHub下載地址&#xff1a;https://github.com/apache/incubator-seata/releases 本文這里下載的是seata2.0.0版本。 【1】概述 ① Seata是什么 Simple Extensible Autonomous Transaction Architecture&#xff0c…

C++ 設計模式之訪問者模式

C 設計模式之訪問者模式 簡介 1、訪問者模式 &#xff08;Visitor&#xff09;是一種行為型設計模式&#xff0c;它表示一個作用于某對象結構中的各元素的操作。它使你可以在不改變各元素的類的前提下定義作用于這些元素的新操作。 使用該模式可以在不修改已有程序結構的前提…

vue3 全局引入 onMounted, reactive, ref 的插件全局引入

webpack 的引入 npm install -D unplugin-auto-import const AutoImport require(unplugin-auto-import/webpack).default;configureWebpack: {devtool: source-map,module: {rules: [{test: /\.mjs$/,include: /node_modules/,type: javascript/auto}],}, plugins: [Aut…

Java對象創建過程

在日常開發中&#xff0c;我們常常需要創建對象&#xff0c;那么通過new關鍵字創建對象的執行中涉及到哪些流程呢&#xff1f;本文主要圍繞這個問題來展開。 類的加載 創建對象時我們常常使用new關鍵字。如下 ObjectA o new ObjectA();對虛擬機來講首先需要判斷ObjectA類的…

Java代碼質量管理與持續集成

Java代碼質量管理與持續集成 大家好&#xff0c;我是免費搭建查券返利機器人省錢賺傭金就用微賺淘客系統3.0的小編&#xff0c;也是冬天不穿秋褲&#xff0c;天冷也要風度的程序猿&#xff01; 引言 在當今軟件開發的環境中&#xff0c;高質量的代碼和持續集成是保證軟件項目…

# Sharding-JDBC從入門到精通(4)- Sharding-JDBC 入門程序幾種配置方式

Sharding-JDBC從入門到精通&#xff08;4&#xff09;- Sharding-JDBC 入門程序幾種配置方式 一、Sharding-JDBC 入門程序&#xff08;水平分表&#xff09;-使用 application.yml 配置文件的 方式 1、打開 idea 創建 artifactId 名為 dbsharding 的 maven 父工程。 --> i…

python sklearn機械學習模型-回歸

&#x1f308;所屬專欄&#xff1a;【機械學習】?作者主頁&#xff1a; Mr.Zwq??個人簡介&#xff1a;一個正在努力學技術的Python領域創作者&#xff0c;擅長爬蟲&#xff0c;逆向&#xff0c;全棧方向&#xff0c;專注基礎和實戰分享&#xff0c;歡迎咨詢&#xff01; 您…

redis實戰-添加商戶緩存

為什么要使用緩存 言簡意賅&#xff1a;速度快&#xff0c;好用緩存數據存儲于代碼中&#xff0c;而代碼運行在內存中&#xff0c;內存的讀寫性能遠高于磁盤&#xff0c;緩存可以大大降低用戶訪問并發量帶來的服務器讀寫壓力實際開發中&#xff0c;企業的數據量&#xff0c;少…

找不到mfc100.dll文件怎么辦?推薦這7個解決方法快速解決mfc100.dll丟失問題

使用電腦中&#xff0c;會遇到各種各樣的問題&#xff0c;比如找不到mfc100.dll&#xff0c;或mfc100.dll丟失導致軟件程序無法繼續運行&#xff0c;就是日常中比較常見的問題之一&#xff0c;今天我教大家遇到這個mfc100.dll丟失問題時候&#xff0c;要怎么解決&#xff0c;以…

【List集合排序】

List集合排序Demo import com.google.common.collect.Lists; import lombok.AllArgsConstructor; import lombok.NoArgsConstructor;import java.util.*;/*** list order demo*/ public class ListOrderDemo {public static void main(String[] args) {List<String> lis…

以太網幀格式是如何識別有效負載類型的

注&#xff1a;機翻&#xff0c;未校對。 識別以太網幀有效負載 Identifying Ethernet Frame Payloads Ethernet frames contain payload data encapsulated within header and trailer fields used to deliver packets over Layer 2 networks. This article provides an ov…

套接字通信類的封裝

在掌握了基于TCP的套接字通信流程之后&#xff0c;為了方便使用&#xff0c;提高編碼效率&#xff0c;可以對通信操作進行封裝&#xff0c;本著有淺入深的原則&#xff0c;先基于C語言進行面向過程的函數封裝&#xff0c;然后再基于C進行面向對象的類封裝。 1. 基于C語言的封裝…

Linux基礎篇——學習Linux基本工具安裝教程視頻鏈接

本篇文章就是記錄一下學習Linux需要用到的基本工具的視頻教程鏈接&#xff0c;方便以后查看 VMware15.5安裝 安裝視頻教程&#xff1a;VMware15.5安裝教程 centos7.6安裝&#xff08;這個視頻教程真的很nice&#xff09; 視頻教程&#xff1a;centos7.6 虛擬機克隆、快照、…

ansible 模塊擴展

uri模塊 在Ansible中&#xff0c;uri模塊是一個用于發送HTTP、HTTPS、FTP等請求的模塊&#xff0c;可以用于獲取網頁內容、下載文件、上傳文件等。本質上&#xff0c;它是一個HTTP客戶端模塊。 使用uri模塊&#xff0c;需要指定一些參數來定義HTTP請求。下面是一些uri模塊的常…

學習平臺推薦_菜鳥教程官網

網址&#xff1a; 菜鳥教程 - 學的不僅是技術&#xff0c;更是夢想&#xff01;菜鳥教程(www.runoob.com)提供了編程的基礎技術教程, 介紹了HTML、CSS、Javascript、Python&#xff0c;Java&#xff0c;Ruby&#xff0c;C&#xff0c;PHP , MySQL等各種編程語言的基礎知識。 同…

Nginx-2

一、高級配置 1.1網頁狀態頁 基于nginx 模塊 ngx_http_stub_status_module 實現&#xff0c;在編譯安裝nginx的時候需要添加編譯參數 --with-http_stub_status_module&#xff0c;否則配置完成之后監測會是提示語法錯誤注意: 狀態頁顯示的是整個服務器的狀態,而非虛擬主機的狀…

opencv實現surface_matching記錄

1 說明 使用的cv版本為4.7.0 , surface_matching功能是附加在contrib中的,并未直接包含在opencv 4.7.0中,因此編譯的時候需要考慮contrib。 VS版本為2022, CMake版本為3.30-rc4. 2 編譯opencv(含contrib) 參見: Win10 下編譯 OpenCV 4.7.0詳細全過程,包含xfeatures2…

Open3D (C++) 點云邊界提取

邊界提取 一、算法原理1、詳細流程2、主要函數3、參考文獻二、代碼實現三、結果展示四、注意事項本文由CSDN點云俠原創,原文鏈接。如果你不是在點云俠的博客中看到該文章,那么此處便是不要臉的爬蟲。 一、算法原理 1、詳細流程 該算法完全復刻自PCL。 2、主要函數 /// \…

算法入門(上)

什么是算法&#xff1f; 算法&#xff08;Algorithm&#xff09;是解決特定問題求解步驟的描述&#xff0c;在計算機中表現為指令的有限序列&#xff0c;并且每條指令表示一個或多個操作。 給定一個問題&#xff0c;能夠解決這個問題的算法是有很多種的。算式中的問題是千奇百怪…

C++為什么將 0.1f 更改為 0 性能會降低 10 倍

一、浮點數與整數的表示差異 在計算機內部&#xff0c;浮點數和整數的表示方式截然不同。浮點數遵循IEEE 754標準&#xff0c;通過符號位、指數位和尾數位來存儲和表示數值&#xff0c;而整數則是直接的二進制表示。這種表示上的差異導致了它們在內存占用、處理速度以及精度上…