在嵌入式裸機編程中,作為一名初級的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中可見以下信息:

在這里申請的這塊內(nèi)存,在接下來的代碼中,被注冊進系統(tǒng)中給malloc和free函數(shù)所使用:unsigned char Heap_Mem[Heap_Size] = {0};
就如上面分析的那樣,其實,在裸機編程的時候,對堆內(nèi)存的管理。并非是智能化的,并非你想申請多少就多少。而是使用一塊固定的內(nèi)存用作堆內(nèi)存的分配。這在設(shè)計的時候,往往不是最佳的方案。這塊內(nèi)存,如果被多次按照不同的大小進行申請,就會造成內(nèi)存碎片。最終導致無法申請到足夠的內(nèi)存。導致系統(tǒng)運行出錯。這在原本內(nèi)存就已經(jīng)很少的嵌入式系統(tǒng)中,更是不能接受的。所以,建議是把那個Heap_Size設(shè)置成 0 吧。放棄其使用吧。 而更為致命的是,有些malloc,free函數(shù),由于工程人員的偷懶。實現(xiàn)甚至可能如下:__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
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)崩潰,因此應(yī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)存分配,應(yīng)分不同的應(yīng)用場合。 對于在操作系統(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)存池,簡潔地來說,就是預(yù)先分配一塊固定大小的內(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)存池,并非什么很厲害的技術(shù)。實現(xiàn)起來,其實可以做到很簡單。只需要一個鏈表即可。在初始化的時候,把全局變量申請來的內(nèi)存,一個個放入該鏈表中。在申請的時候,只需要取出頭部并返回即可。在釋放的時候,只需要把該內(nèi)存插入鏈表。以下是一種簡單的例子(使用移植來的linux內(nèi)核鏈表,對該鏈表的移植,以后有時間再去分析):
責任編輯:xj//內(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; ilist_add(&(gmem[i].list), &mem_pool);//printf("add mem 0x%p ", &(gmem[i].list));}EXIT_CRITICAL(psr);}
原文標題:嵌入式裸機編程中使用malloc、free會怎樣?
文章出處:【微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。
舉報投訴
-
嵌入式
+關(guān)注
關(guān)注
5209文章
20655瀏覽量
336958 -
編程
+關(guān)注
關(guān)注
90文章
3723瀏覽量
97434 -
Free
+關(guān)注
關(guān)注
0文章
16瀏覽量
11389 -
內(nèi)存管理
+關(guān)注
關(guān)注
0文章
171瀏覽量
14921
原文標題:嵌入式裸機編程中使用malloc、free會怎樣?
文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
熱點推薦
如何成為一名高薪嵌入式工程師?
精通指針、內(nèi)存管理、結(jié)構(gòu)體、位操作等,理解底層硬件交互原理。
C++(可選):面向?qū)ο?b class='flag-5'>編程在嵌入式Linux或復雜系統(tǒng)中應(yīng)用廣泛,如驅(qū)動開
發(fā)表于 04-09 09:03
嵌入式驅(qū)動開發(fā),需要掌握哪些技能?
有經(jīng)驗的老司機都知道,做嵌入式驅(qū)動開發(fā)需要掌握一系列的技能,包含編程、硬件、操作系統(tǒng)、調(diào)試等多個方面。
1、硬件知識熟悉原理圖:能看懂原理圖以及各種元器件的含義,以及PCB布局等。
了解
發(fā)表于 01-20 16:46
什么是嵌入式應(yīng)用開發(fā)?
包括ARM7、ARM9、ARM11、PowerPC、MIPS等。這些處理器和編程語言的選擇取決于具體的硬件條件和開發(fā)需求?。
綜上所述,嵌入式應(yīng)用開發(fā)是一個綜合性的技術(shù)領(lǐng)域,涉及硬件和軟件的緊密結(jié)合,廣泛應(yīng)用于各種設(shè)備和系統(tǒng)中
發(fā)表于 01-12 16:13
C語言嵌入式系統(tǒng)編程注意事項-內(nèi)存操作
C語言嵌入式系統(tǒng)編程注意事項之內(nèi)存操作
在嵌入式系統(tǒng)的編程中,常常要求在特定的
發(fā)表于 01-04 07:31
什么是嵌入式操作系統(tǒng)?
、嵌入式 OS 的核心作用(嵌入式開發(fā)中最實用的 3 個功能)
1. 多任務(wù)管理:
裸機開發(fā)的痛點:所有功能寫在一個main函數(shù)的循環(huán)里,代碼臃腫、邏輯混亂,一個功能出問題會影響整
發(fā)表于 12-09 10:33
嵌入式應(yīng)掌握的幾種能力
1. 編程語言
基本掌握嵌入式必備的編程語言。C語言為主,C++為輔。在資源有限的情況下,大多數(shù)嵌入式產(chǎn)品還是使用C語言為主的,特別是底層驅(qū)動開發(fā)。上層應(yīng)用開發(fā)大多用C,在一些資源比
發(fā)表于 12-08 06:05
CW32嵌入式軟件開發(fā)的必備知識
合適的數(shù)據(jù)結(jié)構(gòu)和算法。
3 、計算機體系結(jié)構(gòu)
了解處理器架構(gòu),如ARM、x86等,以及指令集和內(nèi)存管理。
熟悉嵌入式系統(tǒng)的硬件組成,如微控制器、FPGA、DSP等。
可以很熟練地根據(jù)C
發(fā)表于 11-28 07:48
嵌入式開發(fā)的關(guān)鍵點介紹
嵌入式開發(fā)在現(xiàn)代科技中扮演著非常重要的角色。隨著物聯(lián)網(wǎng)的發(fā)展,嵌入式系統(tǒng)的需求也越來越大。嵌入式開發(fā)不僅需要開發(fā)人員具備深入的硬件知識和編程
發(fā)表于 11-13 08:12
嵌入式實時操作系統(tǒng)的特點
操作系統(tǒng)具備高效的中斷處理機制,能夠快速響應(yīng)和處理系統(tǒng)的中斷事件。
資源管理:實時嵌入式操作系統(tǒng)提供有效的資源
發(fā)表于 11-13 06:30
嵌入式需要掌握哪些核心技能?
: 1)C語言與底層編程 核心地位:C語言是嵌入式開發(fā)的基石,需精通指針操作、內(nèi)存管理、位運算,直接操控硬件資源。 延伸技能:C++用于復雜項目架構(gòu)設(shè)計,匯編語言優(yōu)化底層性能(如啟動代
發(fā)表于 10-21 16:25
嵌入式達到什么水平才能就業(yè)?
、LoRa、NB-IoT,能實現(xiàn)嵌入式設(shè)備與云端平臺的數(shù)據(jù)交互了解 RTOS 實時操作系統(tǒng):如 FreeRTOS、RT-Thread,能進行任務(wù)創(chuàng)建、信號量管理、內(nèi)存分配
發(fā)表于 09-15 10:20
入行嵌入式應(yīng)該怎么準備?
知識:
一、C/C++編程C/C++是嵌入式系統(tǒng)開發(fā)中最常用的編程語言。熟練掌握C/C++語言將使你能夠理解和編寫底層驅(qū)動程序、操作系統(tǒng)以
發(fā)表于 08-06 10:34
Linux嵌入式和單片機嵌入式的區(qū)別?
:
單片機嵌入式 :開發(fā)環(huán)境相對簡單,通常使用C語言或匯編語言進行編程,開發(fā)工具包括Keil、IAR等。
Linux嵌入式 :開發(fā)環(huán)境較為復雜,除了需要掌握C語言,還需要了解Linux操作
發(fā)表于 06-20 09:46
嵌入式開發(fā)入門指南:從零開始學習嵌入式
基礎(chǔ) 3. 學習路徑推薦第一階段:熟悉開發(fā)環(huán)境(如Keil、IAR、STM32)第二階段:掌握裸機編程與驅(qū)動開發(fā)第三階段:學習RTOS(實時操作系統(tǒng))如FreeRTOS第四階段:深入理解Linux
發(fā)表于 05-15 09:29
嵌入式系統(tǒng)裸機編程的內(nèi)存管理簡介
評論