<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > 內存! 內存! 內存! 嵌入式裸機編程最重要的事

內存! 內存! 內存! 嵌入式裸機編程最重要的事

作者: 時(shí)間:2018-05-22 來(lái)源:網(wǎng)絡(luò ) 收藏

  在中,作為一名初級的CODER。經(jīng)常要與CPU、內存等打交道。CPU作為系統的動(dòng)力源,其重要程度不言而喻。但是,在中,對內存的管理也不容忽視。如果稍微不注意,輕則,可能造成內存泄漏,重則造成內存訪(fǎng)問(wèn)異常。導致系統死機。

本文引用地址:http://dyxdggzs.com/article/201805/380284.htm

  產(chǎn)品,對穩定性要求及其嚴格。動(dòng)不動(dòng)就死機,那可就麻煩大了。以下,是我本人對系統的內存管理的一些簡(jiǎn)介。

  1. 萬(wàn)萬(wàn)不可使用系統自帶的malloc和free。

  malloc和free在PC編程中是很好用的一種內存分配手段。但是,其在嵌入式中,就未必好用了。由于嵌入式裸機編程中,無(wú)MMU,即內存管理單元。無(wú)法實(shí)現對內存進(jìn)行動(dòng)態(tài)映射(不明白什么叫動(dòng)態(tài)映射的同學(xué),可以參考網(wǎng)上的資料)。也就是說(shuō),實(shí)際上,malloc和free并不能實(shí)現動(dòng)態(tài)的內存的管理。這需要在啟動(dòng)階段專(zhuān)門(mén)給其分配一段空閑的內存區域作為malloc的內存區。如STM32中的啟動(dòng)文件startup_stm32f10x_md.s中見(jiàn)以下信息:

  [plain] view plain copy

  Heap_Size EQU 0x00000800 AREA HEAP, NOINIT, READWRITE, ALIGN=3

  __heap_base

  Heap_Mem SPACE Heap_Size

  __heap_limit

  其中,Heap_Size即定義一個(gè)宏定義。數值為0x00000800。Heap_Mem則為申請一塊連續的內存,大小為 Heap_Size。簡(jiǎn)化為C語(yǔ)言版本如下:

  #define Heap_Size 0x00000800

  unsigned char Heap_Mem[Heap_Size] = {0};

  在這里申請的這塊內存,在接下來(lái)的代碼中,被注冊進(jìn)系統中給malloc和free函數所使用:

  __user_initial_stackheap

  LDR R0, = Heap_Mem ; 返回系統中堆內存起始地址

  LDR R1, =(Stack_Mem + Stack_Size)

  LDR R2, = (Heap_Mem + Heap_Size); 返回系統中堆內存的結束地址

  LDR R3, = Stack_Mem

  BX LR

  就如上面分析的那樣,其實(shí),在裸機編程的時(shí)候,對堆內存的管理。并非是智能化的,并非你想申請多少就多少。而是使用一塊固定的內存用作堆內存的分配。這在設計的時(shí)候,往往不是最佳的方案。這塊內存,如果被多次按照不同的大小進(jìn)行申請,就會(huì )造成內存碎片。最終導致無(wú)法申請到足夠的內存。導致系統運行出錯。這在原本內存就已經(jīng)很少的嵌入式系統中,更是不能接受的。所以,建議是把那個(gè)Heap_Size設置成 0 吧。放棄其使用吧。

  而更為致命的是,有些malloc,free函數,由于工程人員的偷懶。實(shí)現甚至可能如下:

  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. 更好的替代方案:內存池。

  可能有些同學(xué),覺(jué)得:內存池,這是什么東西?

  內存池,簡(jiǎn)潔地來(lái)說(shuō),就是預先分配一塊固定大小的內存。以后,要申請固定大小的內存的時(shí)候,即可從該內存池中申請。用完了,自然要放回去。注意,內存池,每次申請都只能申請固定大小的內存。這樣子做,有很多好處:

  (1)每次動(dòng)態(tài)內存申請的大小都是固定的,可以有效防止內存碎片化。(至于為什么,可以想想,每次申請的都是固定的大小,回收也是固定的大小)

  (2)效率高,不需要復雜的內存分配算法來(lái)實(shí)現。申請,釋放的時(shí)間復雜度,可以做到O(1)。

  (3)實(shí)現簡(jiǎn)單,易用。

  (4)內存的申請,釋放都在可控的范圍之內。不會(huì )出現以后運行著(zhù),運行著(zhù),就再也申請不到內存的情況。

  內存池,并非什么很厲害的技術(shù)。實(shí)現起來(lái),其實(shí)可以做到很簡(jiǎn)單。只需要一個(gè)鏈表即可。在初始化的時(shí)候,把全局變量申請來(lái)的內存,一個(gè)個(gè)放入該鏈表中。在申請的時(shí)候,只需要取出頭部并返回即可。在釋放的時(shí)候,只需要把該內存插入鏈表。以下是一種簡(jiǎn)單的例子(使用移植來(lái)的linux內核鏈表,對該鏈表的移植,以后有時(shí)間再去分析):

  #define MEM_BUFFER_LEN 5 //內存塊的數量

  #define MEM_BUFFER_SIZE 256 //每塊內存的大小

  //內存池的描述,使用聯(lián)合體,體現窮人的智慧。就如,我一同學(xué)說(shuō)的:一個(gè)字節,恨不得掰成8個(gè)字節來(lái)用。

  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);

  //分配內存

  void *mem_pop()

  {

  union mem *ret = NULL;

  psr_t psr;

  psr = ENTER_CRITICAL();

  if(!list_empty(&mem_pool)) { //有可用的內存池

  ret = list_first_entry(&mem_pool, union mem, list);

  //printf("mem_pool = 0x%p ret = 0x%pn", &mem_pool, &ret->list);

  list_del(&ret->list);

  }

  EXIT_CRITICAL(psr);

  return ret;//->buffer;

  }

  //回收內存

  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%pn", &tmp->list);

  EXIT_CRITICAL(psr);

  }

  //初始化內存池

  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%pn", &(gmem[i].list));

  }

  EXIT_CRITICAL(psr);

  }



關(guān)鍵詞: 嵌入式 裸機編程

評論


相關(guān)推薦

技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>