在嵌入式裸機(jī)編程中,作為一名初級的CODER。經(jīng)常要與CPU、內(nèi)存等打交道。CPU作為系統(tǒng)的動力源,其重要程度不言而喻。 但是,在裸機(jī)編程中,對內(nèi)存的管理也不容忽視。如果稍微不注意,輕則,可能造成內(nèi)存泄漏,重則造成內(nèi)存訪問異常。導(dǎo)致系統(tǒng)死機(jī)。 嵌入式產(chǎn)品,對穩(wěn)定性要求及其嚴(yán)格。動不動就死機(jī),那可就麻煩大了。以下,是我本人對嵌入式系統(tǒng)裸機(jī)編程的內(nèi)存管理的一些簡介。
1、盡量不使用庫自帶的malloc和free。
malloc和free在PC編程中是很好用的一種內(nèi)存分配手段。但是,其在嵌入式中,就未必好用了。由于嵌入式裸機(jī)編程中,無MMU,即內(nèi)存管理單元。無法實(shí)現(xiàn)對內(nèi)存進(jìn)行動態(tài)映射(不明白什么叫動態(tài)映射的同學(xué),可以參考網(wǎng)上的資料)。也就是說,實(shí)際上,malloc和free并不能實(shí)現(xiàn)動態(tài)的內(nèi)存的管理。這需要在啟動階段專門給其分配一段空閑的內(nèi)存區(qū)域作為malloc的內(nèi)存區(qū)。如STM32中的啟動文件startup_stm32f10x_md.s中可見以下信息:
![](https://file.elecfans.com/web1/M00/D7/D1/pIYBAF_pOTiAGn4EAAAlRLndINo348.jpg)
在這里申請的這塊內(nèi)存,在接下來的代碼中,被注冊進(jìn)系統(tǒng)中給malloc和free函數(shù)所使用:unsigned char Heap_Mem[Heap_Size] = {0};
就如上面分析的那樣,其實(shí),在裸機(jī)編程的時(shí)候,對堆內(nèi)存的管理。并非是智能化的,并非你想申請多少就多少。而是使用一塊固定的內(nèi)存用作堆內(nèi)存的分配。這在設(shè)計(jì)的時(shí)候,往往不是最佳的方案。這塊內(nèi)存,如果被多次按照不同的大小進(jìn)行申請,就會造成內(nèi)存碎片。最終導(dǎo)致無法申請到足夠的內(nèi)存。導(dǎo)致系統(tǒng)運(yùn)行出錯(cuò)。這在原本內(nèi)存就已經(jīng)很少的嵌入式系統(tǒng)中,更是不能接受的。所以,建議是把那個(gè)Heap_Size設(shè)置成 0 吧。放棄其使用吧。 而更為致命的是,有些malloc,free函數(shù),由于工程人員的偷懶。實(shí)現(xiàn)甚至可能如下:__user_initial_stackheap
LDR 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_Mem
BX 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的原因
一般單片機(jī)的內(nèi)存都比較小,而且沒有MMU,malloc 與free的使用容易造成內(nèi)存碎片。而且可能因?yàn)榭臻g不足而分配失敗,從而導(dǎo)致系統(tǒng)崩潰,因此應(yīng)該慎用,或者自己實(shí)現(xiàn)內(nèi)存管理。如:《一個(gè)簡單而強(qiáng)大的單片機(jī)內(nèi)存管理器》 在函數(shù)中使用malloc,如果是大的內(nèi)存分配,而且malloc與free的次數(shù)也不是特別頻繁,使用malloc與free是比較合適的,但是如果內(nèi)存分配比較小,而且次數(shù)特別頻繁,那么使用malloc與free就有些不太合適了。 因?yàn)檫^多的malloc與free容易造成內(nèi)存碎片,致使可使用的堆內(nèi)存變小。尤其是在對單片機(jī)等沒有MMU的芯片編程時(shí),慎用malloc與free。如果需要對內(nèi)存的頻繁操作,可以自己實(shí)現(xiàn)一個(gè)內(nèi)存管理。 使用動態(tài)內(nèi)存分配,應(yīng)分不同的應(yīng)用場合。 對于在操作系統(tǒng)上運(yùn)行的程序,實(shí)際的物理內(nèi)存分配與釋放使用操作系統(tǒng)來實(shí)現(xiàn)的,即使程序調(diào)用了 malloc和free物理內(nèi)存并不會馬上變化。物理內(nèi)存的變化,直到系統(tǒng)的內(nèi)存管理操作時(shí)才發(fā)生。 對于裸機(jī)跑在MCU上的程序,分配與釋放內(nèi)存都會造成實(shí)際物理內(nèi)存的變化。因?yàn)榇藭r(shí)物理內(nèi)存的分配是由自己實(shí)現(xiàn)的,而內(nèi)存管理我們自己并沒有去做。這樣,盲目的使用malloc與free恰恰并不好,反而會造成內(nèi)存的不恰當(dāng)使用。甚至于內(nèi)存溢出。 所以,動態(tài)內(nèi)存的使用前提是有一套好的內(nèi)存管理方法,這樣動態(tài)內(nèi)存的使用才會合理使用內(nèi)存。如果沒有合適的內(nèi)存管理代碼,還是用靜態(tài)內(nèi)存好一些。
3、 更好的替代方案:內(nèi)存池。
可能有些同學(xué),覺得:內(nèi)存池,這是什么東西? 內(nèi)存池,簡潔地來說,就是預(yù)先分配一塊固定大小的內(nèi)存。以后,要申請固定大小的內(nèi)存的時(shí)候,即可從該內(nèi)存池中申請。用完了,自然要放回去。注意,內(nèi)存池,每次申請都只能申請固定大小的內(nèi)存。這樣子做,有很多好處: (1)每次動態(tài)內(nèi)存申請的大小都是固定的,可以有效防止內(nèi)存碎片化。(至于為什么,可以想想,每次申請的都是固定的大小,回收也是固定的大小) (2)效率高,不需要復(fù)雜的內(nèi)存分配算法來實(shí)現(xiàn)。申請,釋放的時(shí)間復(fù)雜度,可以做到O(1)。 (3)實(shí)現(xiàn)簡單,易用。 (4)內(nèi)存的申請,釋放都在可控的范圍之內(nèi)。不會出現(xiàn)以后運(yùn)行著,運(yùn)行著,就再也申請不到內(nèi)存的情況。 內(nèi)存池,并非什么很厲害的技術(shù)。實(shí)現(xiàn)起來,其實(shí)可以做到很簡單。只需要一個(gè)鏈表即可。在初始化的時(shí)候,把全局變量申請來的內(nèi)存,一個(gè)個(gè)放入該鏈表中。在申請的時(shí)候,只需要取出頭部并返回即可。在釋放的時(shí)候,只需要把該內(nèi)存插入鏈表。以下是一種簡單的例子(使用移植來的linux內(nèi)核鏈表,對該鏈表的移植,以后有時(shí)間再去分析):
責(zé)任編輯:xj
//內(nèi)存池的描述,使用聯(lián)合體,體現(xiàn)窮人的智慧。就如,我一同學(xué)說的:一個(gè)字節(jié),恨不得掰成8個(gè)字節(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);
}
原文標(biāo)題:嵌入式裸機(jī)編程中使用malloc、free會怎樣?
文章出處:【微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。
舉報(bào)投訴
-
嵌入式
+關(guān)注
關(guān)注
5087文章
19148瀏覽量
306179 -
編程
+關(guān)注
關(guān)注
88文章
3628瀏覽量
93816 -
Free
+關(guān)注
關(guān)注
0文章
16瀏覽量
11095 -
內(nèi)存管理
+關(guān)注
關(guān)注
0文章
168瀏覽量
14160
原文標(biāo)題:嵌入式裸機(jī)編程中使用malloc、free會怎樣?
文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
新手怎么學(xué)嵌入式?
嵌入式系統(tǒng)的發(fā)展,嵌入式操作系統(tǒng)也變得越來越重要。學(xué)習(xí)嵌入式操作系統(tǒng)可以幫助你更好地
發(fā)表于 12-12 10:51
嵌入式系統(tǒng)開發(fā)與硬件的關(guān)系 嵌入式系統(tǒng)開發(fā)常見問題解決
系統(tǒng)開發(fā)與硬件關(guān)系的幾個(gè)關(guān)鍵點(diǎn): 硬件依賴性 :嵌入式系統(tǒng)的軟件必須能夠在特定的硬件上運(yùn)行,這包括處理器、內(nèi)存、輸入/輸出接口等。軟件必須能夠充分利用硬件的特性,同時(shí)繞過其限制。 資源
嵌入式系統(tǒng)與物聯(lián)網(wǎng)的結(jié)合
。這兩者的結(jié)合,為智能設(shè)備和智能系統(tǒng)的發(fā)展提供了強(qiáng)大的動力。 一、嵌入式系統(tǒng)與物聯(lián)網(wǎng)的基本概念 嵌入式系統(tǒng)
嵌入式主板是什么意思?嵌入式主板全面解析
嵌入式主板,通常被稱為嵌入式系統(tǒng)的核心組件,是一種用于控制和數(shù)據(jù)處理的計(jì)算機(jī)硬件,其設(shè)計(jì)旨在嵌入特定設(shè)備中執(zhí)行專門任務(wù)。嵌入式主板如同是設(shè)備
嵌入式常用數(shù)據(jù)結(jié)構(gòu)有哪些
在嵌入式編程中,數(shù)據(jù)結(jié)構(gòu)的選擇和使用對于程序的性能、內(nèi)存管理以及開發(fā)效率都具有重要影響。嵌入式系統(tǒng)
freertos和裸機(jī)有什么區(qū)別
FreeRTOS 和裸機(jī)編程是兩種不同的嵌入式系統(tǒng)開發(fā)方法,它們在設(shè)計(jì)理念、資源使用、功能實(shí)現(xiàn)等方面有著顯著的差異。 1. 基本概念 1.1 FreeRTOS FreeRTOS 是一個(gè)
嵌入式系統(tǒng)怎么學(xué)?
一系列課程和技術(shù),包括但不限于以下內(nèi)容:
1、基礎(chǔ)知識:學(xué)習(xí)計(jì)算機(jī)組成原理、數(shù)字電路、模擬電路等基礎(chǔ)知識,建立對計(jì)算機(jī)硬件的認(rèn)知與理解。
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
嵌入式系統(tǒng)的外設(shè)器件選擇
作者:DigiKey Editor 嵌入式系統(tǒng)除了最重要的為處理器選擇之外,配合的相關(guān)外設(shè)器件也是嵌入式系統(tǒng)的重要組成部分,包括內(nèi)存、時(shí)鐘(
![<b class='flag-5'>嵌入式</b><b class='flag-5'>系統(tǒng)</b>的外設(shè)器件選擇](https://file1.elecfans.com/web2/M00/D4/03/wKgaomYkbU2AfzuwAAASqtl4vhI345.jpg)
嵌入式可編程片上系統(tǒng)是什么
嵌入式可編程片上系統(tǒng)(Embedded Programmable System-on-Chip,或簡稱EPSoC)是一種特殊的嵌入式系統(tǒng),它
嵌入式fpga是什么意思
嵌入式FPGA是指將FPGA技術(shù)集成到嵌入式系統(tǒng)中的一種解決方案。嵌入式系統(tǒng)是一種為特定應(yīng)用而設(shè)計(jì)的計(jì)算機(jī)
fpga是嵌入式嗎
FPGA(現(xiàn)場可編程門陣列)不是嵌入式系統(tǒng),但FPGA在嵌入式系統(tǒng)中有著重要的應(yīng)用。
嵌入式系統(tǒng)發(fā)展前景?
設(shè)備、健康監(jiān)測等領(lǐng)域有著廣泛的應(yīng)用前景。隨著人們對健康的重視程度不斷提高,嵌入式系統(tǒng)將更為深入地應(yīng)用于醫(yī)療設(shè)備和健康護(hù)理中,實(shí)現(xiàn)個(gè)人健康管理的智能化和實(shí)時(shí)化。 汽車電子和自動駕駛是嵌入式
發(fā)表于 02-22 14:09
嵌入式軟件開發(fā)應(yīng)該掌握哪些知識?
和 Thumb 模式的區(qū)別,以及 ARM Cortex 系列處理器的特性。 嵌入式 ARM 開發(fā):學(xué)習(xí)如何在嵌入式系統(tǒng)中使用 ARM 處理器進(jìn)行開發(fā),包括交叉編譯工具鏈的配置、裸機(jī)
發(fā)表于 02-19 11:23
嵌入式學(xué)習(xí)步驟
語言編寫。但是,有些嵌入式系統(tǒng)也使用其他編程語言,例如Python或Java。 (3).了解硬件:深入了解您要控制的硬件設(shè)備的功能和特性。您需要了解嵌入式
發(fā)表于 02-02 15:24
評論