0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
电子发烧友
开通电子发烧友VIP会员 尊享10大特权
海量资料免费下载
精品直播免费看
优质内容免费畅学
课程9折专享价
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

嵌入式系統(tǒng)裸機編程的內(nèi)存管理簡介

FPGA之家 ? 來源:嵌入式大雜燴 ? 作者:嵌入式大雜燴 ? 2020-12-28 09:44 ? 次閱讀

嵌入式裸機編程中,作為一名初級的CODER。經(jīng)常要與CPU、內(nèi)存等打交道。CPU作為系統(tǒng)的動力源,其重要程度不言而喻。 但是,在裸機編程中,對內(nèi)存的管理也不容忽視。如果稍微不注意,輕則,可能造成內(nèi)存泄漏,重則造成內(nèi)存訪問異常。導致系統(tǒng)死機。 嵌入式產(chǎn)品,對穩(wěn)定性要求及其嚴格。動不動就死機,那可就麻煩大了。以下,是我本人對嵌入式系統(tǒng)裸機編程的內(nèi)存管理的一些簡介。

1、盡量不使用庫自帶的malloc和free。

malloc和free在PC編程中是很好用的一種內(nèi)存分配手段。但是,其在嵌入式中,就未必好用了。由于嵌入式裸機編程中,無MMU,即內(nèi)存管理單元。無法實現(xiàn)對內(nèi)存進行動態(tài)映射(不明白什么叫動態(tài)映射的同學,可以參考網(wǎng)上的資料)。也就是說,實際上,malloc和free并不能實現(xiàn)動態(tài)的內(nèi)存的管理。這需要在啟動階段專門給其分配一段空閑的內(nèi)存區(qū)域作為malloc的內(nèi)存區(qū)。如STM32中的啟動文件startup_stm32f10x_md.s中可見以下信息:
其中,Heap_Size即定義一個宏定義。數(shù)值為 0x00000800。Heap_Mem則為申請一塊連續(xù)的內(nèi)存,大小為 Heap_Size。簡化為C語言版本如下:
#define Heap_Size 0x00000800unsigned char Heap_Mem[Heap_Size] = {0};
在這里申請的這塊內(nèi)存,在接下來的代碼中,被注冊進系統(tǒng)中給malloc和free函數(shù)所使用:
__user_initial_stackheapLDR     R0, =  Heap_Mem  ;  返回系統(tǒng)中堆內(nèi)存起始地址LDR     R1, =(Stack_Mem + Stack_Size)LDR     R2, = (Heap_Mem +  Heap_Size); 返回系統(tǒng)中堆內(nèi)存的結(jié)束地址LDR     R3, = Stack_MemBX      LR
就如上面分析的那樣,其實,在裸機編程的時候,對堆內(nèi)存的管理。并非是智能化的,并非你想申請多少就多少。而是使用一塊固定的內(nèi)存用作堆內(nèi)存的分配。這在設計的時候,往往不是最佳的方案。這塊內(nèi)存,如果被多次按照不同的大小進行申請,就會造成內(nèi)存碎片。最終導致無法申請到足夠的內(nèi)存。導致系統(tǒng)運行出錯。這在原本內(nèi)存就已經(jīng)很少的嵌入式系統(tǒng)中,更是不能接受的。所以,建議是把那個Heap_Size設置成 0 吧。放棄其使用吧。 而更為致命的是,有些malloc,free函數(shù),由于工程人員的偷懶。實現(xiàn)甚至可能如下:
unsigned char mem_buffer[512];unsigned char *mem_offset = & mem_buffer;void *malloc(int size){unsigned char *tmp = mem_offset;    mem_offset += size;return (void *)tmp;}void free(void *mem){ mem_offset = mem;}

2、不用malloc、free的原因

一般單片機的內(nèi)存都比較小,而且沒有MMU,malloc 與free的使用容易造成內(nèi)存碎片。而且可能因為空間不足而分配失敗,從而導致系統(tǒng)崩潰,因此應該慎用,或者自己實現(xiàn)內(nèi)存管理。如:《一個簡單而強大的單片機內(nèi)存管理器》 在函數(shù)中使用malloc,如果是大的內(nèi)存分配,而且malloc與free的次數(shù)也不是特別頻繁,使用malloc與free是比較合適的,但是如果內(nèi)存分配比較小,而且次數(shù)特別頻繁,那么使用malloc與free就有些不太合適了。 因為過多的malloc與free容易造成內(nèi)存碎片,致使可使用的堆內(nèi)存變小。尤其是在對單片機等沒有MMU的芯片編程時,慎用malloc與free。如果需要對內(nèi)存的頻繁操作,可以自己實現(xiàn)一個內(nèi)存管理。 使用動態(tài)內(nèi)存分配,應分不同的應用場合。 對于在操作系統(tǒng)上運行的程序,實際的物理內(nèi)存分配與釋放使用操作系統(tǒng)來實現(xiàn)的,即使程序調(diào)用了 malloc和free物理內(nèi)存并不會馬上變化。物理內(nèi)存的變化,直到系統(tǒng)的內(nèi)存管理操作時才發(fā)生。 對于裸機跑在MCU上的程序,分配與釋放內(nèi)存都會造成實際物理內(nèi)存的變化。因為此時物理內(nèi)存的分配是由自己實現(xiàn)的,而內(nèi)存管理我們自己并沒有去做。這樣,盲目的使用malloc與free恰恰并不好,反而會造成內(nèi)存的不恰當使用。甚至于內(nèi)存溢出。 所以,動態(tài)內(nèi)存的使用前提是有一套好的內(nèi)存管理方法,這樣動態(tài)內(nèi)存的使用才會合理使用內(nèi)存。如果沒有合適的內(nèi)存管理代碼,還是用靜態(tài)內(nèi)存好一些。

3、 更好的替代方案:內(nèi)存池。

可能有些同學,覺得:內(nèi)存池,這是什么東西? 內(nèi)存池,簡潔地來說,就是預先分配一塊固定大小的內(nèi)存。以后,要申請固定大小的內(nèi)存的時候,即可從該內(nèi)存池中申請。用完了,自然要放回去。注意,內(nèi)存池,每次申請都只能申請固定大小的內(nèi)存。這樣子做,有很多好處: (1)每次動態(tài)內(nèi)存申請的大小都是固定的,可以有效防止內(nèi)存碎片化。(至于為什么,可以想想,每次申請的都是固定的大小,回收也是固定的大?。?/span> (2)效率高,不需要復雜的內(nèi)存分配算法來實現(xiàn)。申請,釋放的時間復雜度,可以做到O(1)。 (3)實現(xiàn)簡單,易用。 (4)內(nèi)存的申請,釋放都在可控的范圍之內(nèi)。不會出現(xiàn)以后運行著,運行著,就再也申請不到內(nèi)存的情況。 內(nèi)存池,并非什么很厲害的技術。實現(xiàn)起來,其實可以做到很簡單。只需要一個鏈表即可。在初始化的時候,把全局變量申請來的內(nèi)存,一個個放入該鏈表中。在申請的時候,只需要取出頭部并返回即可。在釋放的時候,只需要把該內(nèi)存插入鏈表。以下是一種簡單的例子(使用移植來的linux內(nèi)核鏈表,對該鏈表的移植,以后有時間再去分析):
#define MEM_BUFFER_LEN  5    //內(nèi)存塊的數(shù)量#define MEM_BUFFER_SIZE 256 //每塊內(nèi)存的大小
//內(nèi)存池的描述,使用聯(lián)合體,體現(xiàn)窮人的智慧。就如,我一同學說的:一個字節(jié),恨不得掰成8個字節(jié)來用。typedef union mem {struct list_head list;unsigned char buffer[MEM_BUFFER_SIZE];}mem_t;
static union mem gmem[MEM_BUFFER_LEN];
LIST_HEAD(mem_pool);
//分配內(nèi)存void *mem_pop(){    union mem *ret = NULL;    psr_t psr;
    psr = ENTER_CRITICAL();    if(!list_empty(&mem_pool)) { //有可用的內(nèi)存池         ret = list_first_entry(&mem_pool, union mem, list);        //printf("mem_pool = 0x%p  ret = 0x%p
", &mem_pool, &ret->list);        list_del(&ret->list); } EXIT_CRITICAL(psr); return ret;//->buffer;}

//回收內(nèi)存void mem_push(void *mem){    union mem *tmp = NULL;     psr_t psr;
    tmp = (void *)mem;//container_of(mem, struct mem, buffer);    psr = ENTER_CRITICAL();    list_add(&tmp->list, &mem_pool);    //printf("free = 0x%p
", &tmp->list);
    EXIT_CRITICAL(psr);}
//初始化內(nèi)存池void mem_pool_init(){    int i;    psr_t psr;    psr = ENTER_CRITICAL();    for(i=0; i        list_add(&(gmem[i].list), &mem_pool);        //printf("add mem 0x%p
", &(gmem[i].list)); } EXIT_CRITICAL(psr);}
責任編輯:xj
原文標題:嵌入式裸機編程中使用malloc、free會怎樣?

文章出處:【微信公眾號:FPGA之家】歡迎添加關注!文章轉(zhuǎn)載請注明出處。


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 嵌入式
    +關注

    關注

    5143

    文章

    19569

    瀏覽量

    315652
  • 編程
    +關注

    關注

    88

    文章

    3686

    瀏覽量

    94951
  • Free
    +關注

    關注

    0

    文章

    16

    瀏覽量

    11236
  • 內(nèi)存管理

    關注

    0

    文章

    168

    瀏覽量

    14504

原文標題:嵌入式裸機編程中使用malloc、free會怎樣?

文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關注!文章轉(zhuǎn)載請注明出處。

收藏 1人收藏

    評論

    相關推薦
    熱點推薦

    嵌入式開發(fā)入門指南:從零開始學習嵌入式

    基礎 3. 學習路徑推薦第一階段:熟悉開發(fā)環(huán)境(如Keil、IAR、STM32)第二階段:掌握裸機編程與驅(qū)動開發(fā)第三階段:學習RTOS(實時操作系統(tǒng))如FreeRTOS第四階段:深入理解Linux
    發(fā)表于 05-15 09:29

    嵌入式編程設計模式

    嵌入式編程設計模式,介紹如何使用設計模式為嵌入式系統(tǒng)創(chuàng)建高效且優(yōu)化的C語言設計。 純分享貼,有需要可以直接下載附件獲取完整資料! (如果內(nèi)容有幫助可以關注、點贊、評論支持一下哦~
    發(fā)表于 04-15 14:47

    CPLD 在嵌入式系統(tǒng)中的應用

    在現(xiàn)代電子設計領域,復雜可編程邏輯器件(CPLD)因其靈活性、成本效益和快速開發(fā)周期而在嵌入式系統(tǒng)中扮演著重要角色。 1. CPLD簡介 CPLD是一種集成電路,其內(nèi)部包含可
    的頭像 發(fā)表于 01-23 09:50 ?698次閱讀

    如何提高嵌入式代碼質(zhì)量?

    。 3. 嵌入式操作系統(tǒng)的使用:選擇和配置合適的嵌入式操作系統(tǒng)(如FreeRTOS、uC/OS等),能夠提供任務調(diào)度、內(nèi)存
    發(fā)表于 01-15 10:48

    嵌入式主板的概述與發(fā)展

    未來發(fā)展趨勢進行深入探討。嵌入式主板的基本概念嵌入式主板是指為嵌入式系統(tǒng)設計的電路板,它通常集成了微處理器、內(nèi)存、輸入輸出接口等關鍵組件。與
    的頭像 發(fā)表于 01-13 16:30 ?715次閱讀
    <b class='flag-5'>嵌入式</b>主板的概述與發(fā)展

    新手怎么學嵌入式?

    嵌入式系統(tǒng)的發(fā)展,嵌入式操作系統(tǒng)也變得越來越重要。學習嵌入式操作系統(tǒng)可以幫助你更好地
    發(fā)表于 12-12 10:51

    嵌入式系統(tǒng)開發(fā)與硬件的關系 嵌入式系統(tǒng)開發(fā)常見問題解決

    系統(tǒng)開發(fā)與硬件關系的幾個關鍵點: 硬件依賴性 :嵌入式系統(tǒng)的軟件必須能夠在特定的硬件上運行,這包括處理器、內(nèi)存、輸入/輸出接口等。軟件必須能夠充分利用硬件的特性,同時繞過其限制。 資源
    的頭像 發(fā)表于 12-09 09:38 ?878次閱讀

    【「嵌入式系統(tǒng)設計與實現(xiàn)」閱讀體驗】全書概覽與內(nèi)容簡介

    嵌入式系統(tǒng)設計與實現(xiàn)》,瞬間就被吸引了。非常感謝電子發(fā)燒友平臺提供這次寶貴機會,讓我有機會接觸到這么多優(yōu)秀的設計案例。 今天剛收到書籍,接下來讓我們一起先大概瀏覽一下這本書的內(nèi)容吧! 內(nèi)容簡介 本書
    發(fā)表于 12-01 17:05

    嵌入式系統(tǒng)與物聯(lián)網(wǎng)的結(jié)合

    。這兩者的結(jié)合,為智能設備和智能系統(tǒng)的發(fā)展提供了強大的動力。 一、嵌入式系統(tǒng)與物聯(lián)網(wǎng)的基本概念 嵌入式系統(tǒng)
    的頭像 發(fā)表于 11-06 10:23 ?1014次閱讀

    嵌入式主板是什么意思?嵌入式主板全面解析

    嵌入式主板,通常被稱為嵌入式系統(tǒng)的核心組件,是一種用于控制和數(shù)據(jù)處理的計算機硬件,其設計旨在嵌入特定設備中執(zhí)行專門任務。嵌入式主板如同是設備
    的頭像 發(fā)表于 09-30 10:05 ?1537次閱讀

    嵌入式常用數(shù)據(jù)結(jié)構(gòu)有哪些

    嵌入式編程中,數(shù)據(jù)結(jié)構(gòu)的選擇和使用對于程序的性能、內(nèi)存管理以及開發(fā)效率都具有重要影響。嵌入式系統(tǒng)
    的頭像 發(fā)表于 09-02 15:25 ?973次閱讀

    freertos和裸機有什么區(qū)別

    FreeRTOS 和裸機編程是兩種不同的嵌入式系統(tǒng)開發(fā)方法,它們在設計理念、資源使用、功能實現(xiàn)等方面有著顯著的差異。 1. 基本概念 1.1 FreeRTOS FreeRTOS 是一個
    的頭像 發(fā)表于 09-02 14:13 ?2849次閱讀

    學習hypervisor嵌入式產(chǎn)品安全設計

    Hypervisor的設計與實現(xiàn),涵蓋嵌入式Hypervisor架構(gòu)與核心組件、中斷隔離技術、內(nèi)存隔離技術、循環(huán)表調(diào)度器、健康監(jiān)控、分區(qū)間通信技術、內(nèi)核資源管理模型、系統(tǒng)初始化過程、內(nèi)
    發(fā)表于 08-25 09:11

    嵌入式系統(tǒng)怎么學?

    一系列課程和技術,包括但不限于以下內(nèi)容: 1、基礎知識:學習計算機組成原理、數(shù)字電路、模擬電路等基礎知識,建立對計算機硬件的認知與理解。 2、編程語言:掌握至少一種嵌入式系統(tǒng)常用的編程
    發(fā)表于 07-02 10:10

    如何提升嵌入式編程能力?

    代碼以提高性能,包括減少內(nèi)存使用、提高處理速度等。 16. 調(diào)試技巧:掌握嵌入式系統(tǒng)的調(diào)試技巧,包括使用調(diào)試器、日志記錄和性能分析工具。 17. 參加競賽和挑戰(zhàn):參加編程競賽或黑客馬拉
    發(fā)表于 06-21 10:01

    電子發(fā)燒友

    中國電子工程師最喜歡的網(wǎng)站

    • 2931785位工程師會員交流學習
    • 獲取您個性化的科技前沿技術信息
    • 參加活動獲取豐厚的禮品