電子產(chǎn)業(yè)一站式賦能平臺(tái)

PCB聯(lián)盟網(wǎng)

搜索
查看: 59|回復(fù): 0
收起左側(cè)

為什么說(shuō)內(nèi)部Flash驅(qū)動(dòng)是個(gè)既冷門(mén)又不冷門(mén)的話題 | LPC Flash IAP

[復(fù)制鏈接]

302

主題

307

帖子

1896

積分

三級(jí)會(huì)員

Rank: 3Rank: 3

積分
1896
跳轉(zhuǎn)到指定樓層
樓主
發(fā)表于 2023-3-30 09:02:00 | 只看該作者 回帖獎(jiǎng)勵(lì) |倒序?yàn)g覽 |閱讀模式
大家好,我是痞子衡,是正經(jīng)搞技術(shù)的痞子。今天痞子衡給大家介紹的是恩智浦經(jīng)典LPC系列MCU內(nèi)部Flash IAP驅(qū)動(dòng)。  E0 Z  |2 T# j$ C3 }4 d8 W: K
LPC 系列 MCU 是恩智浦公司于 2003 年開(kāi)始推出的非常具有代表性的產(chǎn)品,距今已經(jīng)有近 20 年的生命。按時(shí)間線演進(jìn)來(lái)說(shuō),其主要分為三代:4 V+ N! A7 Q# e9 K( Q; [, U1 g, u
- 元老:基于 ARM7/9 內(nèi)核的 LPC2000/3000 系列
6 L: s! F) M* S2 W+ b, T8 f- |- 中堅(jiān):基于 Cortex-M0/0+/3/4 內(nèi)核的 LPC800/1100/1200/1300/1500/1700/1800/4000/4300/54000
$ L2 S; q: f5 @, N5 ~+ ?9 ~7 ]- 新銳:基于 Cortex-M33 內(nèi)核的 LPC5500 系列。- D: M  y8 X+ X; y3 }: E% ^6 Q& r$ N+ u
其中堅(jiān)產(chǎn)品即是痞子衡今天要重點(diǎn)聊的經(jīng)典 MCU,從其第一顆 LPC1800 到至今仍有新型號(hào)出來(lái)的 LPC800,仍然深受廣大開(kāi)發(fā)者喜愛(ài)。今天痞子衡想討論的是內(nèi)部 Flash 驅(qū)動(dòng)這個(gè)對(duì)嵌入式軟件開(kāi)發(fā)者來(lái)說(shuō)既冷門(mén)又不冷門(mén)的話題:
7 m) j' G5 b% [9 L. i& i/ S. {
  • Note:本文內(nèi)容主要以 LPC845 這個(gè)型號(hào)為例,未必完全適用其它經(jīng)典 LPC 型號(hào),具體需要查看相應(yīng)手冊(cè)。一、關(guān)于MCU內(nèi)部Flash的基本概念痞子衡先解釋下為什么內(nèi)部 Flash 驅(qū)動(dòng)這個(gè)話題既冷門(mén)又不冷門(mén)。說(shuō)它冷門(mén)是因?yàn)榇蟛糠智度胧杰浖_(kāi)發(fā)工程師寫(xiě)的應(yīng)用代碼里很少包含 Flash 操作功能(除非應(yīng)用需要 OTA 升級(jí)或者斷電保存參數(shù)),因此對(duì) Flash 模塊的關(guān)注度不如其它外設(shè)模塊。說(shuō)它不冷門(mén)則是在 IDE 中調(diào)試或者編程器做量產(chǎn)又離不開(kāi) Flash 操作,所以避不可免地關(guān)注 Flash 擦寫(xiě)算法、性能、壽命、效率等。
    ) }6 i/ k; ^. T  `話說(shuō)回來(lái),F(xiàn)lash 外設(shè)一般由兩部分組成:Flash 控制器 + Flash Memory 介質(zhì),其 Memory 介質(zhì)部分從原理上屬于并行 NOR Flash,MCU 上電 Flash 外設(shè)總是使能的,可以通過(guò) AHB 總線直接讀取其映射空間內(nèi)任意 Flash 地址處的數(shù)據(jù)/指令,所以其最主要的作用就是存儲(chǔ)可執(zhí)行代碼。
      [  F' j9 B0 w9 V8 [3 a: V如果應(yīng)用程序需要做 OTA 升級(jí),則需要借助 Flash 控制器完成擦除和寫(xiě)入操作。這里就有一些概念性的東西出現(xiàn)了,比如 Flash 擦除正常是按 Block/Sector 為單元(不排除有些支持按 Page 擦除),并且擦除操作是將 Block/Sector 里全部 bit 從 0 恢復(fù)為 1。而 Flash 寫(xiě)入則是按 PUnit 為最小單元的(可能是 1/2/4/8 bytes),一次性最多寫(xiě)入一個(gè) Page 的數(shù)據(jù)(這里指一次完整命令執(zhí)行等待過(guò)程)。擦除和寫(xiě)入操作都不是立刻就完成的,需要等待 Memory 介質(zhì)更新完成(讀 Flash 控制器相應(yīng)狀態(tài)位寄存器)。
    7 G% r5 O( Q" B, z0 zLPC845 內(nèi)部 Flash 一共 64KB,劃分為 64 個(gè) Sector,每個(gè) Sector 大小為 1KB。每個(gè) Sector 包含 16 個(gè) Page,每個(gè) Page 大小為 64Bytes。支持按 Sector/Page 擦除,IAP 僅支持按 Page 寫(xiě)入(但是控制器底層最小寫(xiě)入單元是 4bytes),不支持 RWW 特性。+ L' X( O* p( v8 C
        64KB          N/A            N/A           1KB          64Bytes       4Bytes7 b. k- b' P, f6 G
    Flash Memory > Flash Bank >= Flash Block > Flash Sector > Flash Page >= Flash PUnit >= Flash Byte
    * R, j# P3 Y$ q, [+ s3 {                   |              |             |              |             |
    2 Y3 [/ ]: i/ k                RWW單元        擦除單元        擦除單元      最大寫(xiě)入單元    最小寫(xiě)入單元) R' Q! I3 g* J% h) X7 c) I
    關(guān)于 Flash 擦寫(xiě)操作,還有一個(gè)重要概念叫 Read-While-Write(簡(jiǎn)稱(chēng) RWW),因?yàn)槟J(rèn)代碼是執(zhí)行在 Flash 里,如果我們這個(gè)時(shí)候還做 Flash 擦寫(xiě)操作,就會(huì)讓同一個(gè) Flash 處于又做擦寫(xiě)處理同時(shí)也要響應(yīng) AHB 總線來(lái)的讀指令請(qǐng)求,大部分 Flash 是無(wú)法支持這個(gè)特性的,因此常見(jiàn)的操作是將觸發(fā) Flash 擦寫(xiě)命令以及讀 Flash 狀態(tài)的代碼重定向到 RAM 里去執(zhí)行。而 LPC 上不一樣的 Flash IAP 驅(qū)動(dòng)設(shè)計(jì)正是為了解決這個(gè) RWW 限制的。
    8 s9 U% Q% W& ^! z$ i* \二、一般Flash驅(qū)動(dòng)設(shè)計(jì)在講 LPC Flash IAP 特色驅(qū)動(dòng)之前,我們先來(lái)看看一般 MCU 上 Flash 驅(qū)動(dòng)設(shè)計(jì),就以恩智浦 Kinetis MK60DN512Z 系列為例。它的 Flash 外設(shè)是 FTFL (詳見(jiàn)參考手冊(cè)里 Chapter 28 Flash Memory Module (FTFL) 章節(jié)),F(xiàn)lash 大小為 512KB,分為兩個(gè) 256KB Block (這里就相當(dāng)于Bank),支持 RWW 特性(以 Block 為單元)。每個(gè) Block 包含 128 個(gè) Sector,每個(gè) Sector 大小為 2KB。它其實(shí)沒(méi)有明確的 Page 概念(但是最大寫(xiě)入單元是專(zhuān)用 4KB FLEXRAM 的一半,可以理解為 Page 大小就是 2KB),支持的最小寫(xiě)入單元是 4bytes。8 ^: X' G$ ~# ~6 J' {, H, m
       512KB         256KB          256KB          2KB            2KB         4Bytes, _6 H  p" I2 M
    Flash Memory > Flash Bank >= Flash Block > Flash Sector >= Flash Page > Flash PUnit >= Flash Byte
    / @# k- X3 R# L1 H; ?3 i8 f# d                   |              |             |              |             |
    " N2 [' g0 V! s' H; X5 Q. _7 I                RWW單元        擦除單元        擦除單元      最大寫(xiě)入單元    最小寫(xiě)入單元
    " K$ k: j2 V3 I6 F* i, t, e在官方驅(qū)動(dòng) \SDK_2_2_0_TWR-K60D100M\devices\MK60D10\drivers\fsl_flash.c 里我們重點(diǎn)關(guān)注如下 5 個(gè)基本函數(shù),這些函數(shù)都是直接操作 FTFL 外設(shè)寄存器來(lái)完成相應(yīng) Flash 擦寫(xiě)功能的。其中 flash_command_sequence() 內(nèi)部函數(shù)設(shè)計(jì)是核心,每一個(gè) API 基本都會(huì)調(diào)用它,這里面有一個(gè)關(guān)于解決 RWW 限制的黑科技設(shè)計(jì),后面痞子衡會(huì)寫(xiě)文章專(zhuān)門(mén)介紹。' r. H0 x( [$ Q) v
    // 一般初始化函數(shù),主要是軟件層面初始化% }. P( [, n( K7 I; ?' m# N- c
    status_t FLASH_Init(flash_config_t *config);' k/ e) f; o2 K6 @
    // 為了解決 RWW 限制而特殊設(shè)計(jì)的命令觸發(fā)執(zhí)行函數(shù)
    : J  M. F/ }* o; ^+ @status_t FLASH_PrepareExecuteInRamFunctions(flash_config_t *config);- G2 z( q% o2 `2 S+ P$ V' r
    static status_t flash_command_sequence(flash_config_t *config)
    / u& V. M  E' X// 擦除函數(shù),長(zhǎng)度不限(需要按 Sector 對(duì)齊),key 參數(shù)是為了降低誤擦除風(fēng)險(xiǎn)2 C4 G% {, e9 q& A# h
    status_t FLASH_Erase(flash_config_t *config, uint32_t start, uint32_t lengthInBytes, uint32_t key);
    & `' Q  `7 ]/ Y// 寫(xiě)入函數(shù),長(zhǎng)度不限(僅最小寫(xiě)入單元對(duì)齊限制),函數(shù)內(nèi)部自動(dòng)結(jié)合 Page 和 PUnit 寫(xiě)入命令做處理
    % K% X% g& n+ E5 m1 ?" Lstatus_t FLASH_Program(flash_config_t *config, uint32_t start, uint32_t *src, uint32_t lengthInBytes);, \/ V6 _, j" e0 q/ B
    三、LPC Flash IAP驅(qū)動(dòng)設(shè)計(jì)原理終于來(lái)到本文核心 - LPC Flash IAP 驅(qū)動(dòng)了。按照我們一般經(jīng)驗(yàn),首先是翻看 LPC845 用戶(hù)手冊(cè)尋找 Flash 外設(shè),但是很遺憾,用戶(hù)手冊(cè)里并沒(méi)有 Flash 外設(shè)詳細(xì)介紹,取而代之的是 Chapter 5: LPC84x ISP and IAP 章節(jié)。因?yàn)?LPC 全系列都包含 BootROM(映射地址為 0x0F00_0000 - 0x0F00_3FFF),而 BootROM 代碼里包含了 Flash 擦寫(xiě)驅(qū)動(dòng),因此官方直接推薦用戶(hù)調(diào)用 ROM 里的 Flash 驅(qū)動(dòng) API 來(lái)完成操作,而不是按照傳統(tǒng)方式提供直接操作 Flash 外設(shè)寄存器的 SDK 源碼。
    % L- U+ D( h. m2 U4 tBootROM 提供的 API 不止 Flash IAP 一個(gè),可以在 Boot Process 章節(jié)里如下圖里找到全部 API。這里我們可以看到 Flash IAP 函數(shù)的統(tǒng)一入口地址是 0x0F001FF1,這在 SDK 里 LPC845_features.h 文件里有如下專(zhuān)門(mén)宏:
    0 j5 _% M/ a2 i" D/* @brief Pointer to ROM IAP entry functions */
    ) w( ~) f$ s) T% `#define FSL_FEATURE_SYSCON_IAP_ENTRY_LOCATION (0x0F001FF1)
  • 發(fā)表回復(fù)

    本版積分規(guī)則


    聯(lián)系客服 關(guān)注微信 下載APP 返回頂部 返回列表