LwIP之動態內存堆

下面這段宏定義了內存堆

#ifndef LWIP_RAM_HEAP_POINTER
LWIP_DECLARE_MEMORY_ALIGNED(ram_heap, MEM_SIZE_ALIGNED + (2U * SIZEOF_STRUCT_MEM));
#define LWIP_RAM_HEAP_POINTER ram_heap    //內存堆指針
#endif

將宏LWIP_DECLARE_MEMORY_ALIGNED展開,發現內存堆就是一個數組。

#ifndef LWIP_DECLARE_MEMORY_ALIGNED
#define LWIP_DECLARE_MEMORY_ALIGNED(variable_name, size) u8_t variable_name[LWIP_MEM_ALIGN_BUFFER(size)]
#endif

動態堆最後被組織成內存塊分配給程序,或者從程序釋放到內存堆。

所謂的管理就是管理這些內存塊,下面是對內存塊的定義。而這些內存塊最終會被掛接到一條鏈表。

/* 內存塊結構體 */
struct mem {
  /* 指向後一個內存塊 */
  mem_size_t next;
  /* 指向前一個內存塊 */
  mem_size_t prev;
  /* 內存塊是否被使用 1:被使用 0:未使用 */
  u8_t used;
 
#if MEM_OVERFLOW_CHECK
  mem_size_t user_size;
#endif
};

 

初始化堆內存,將對內部組織成內存塊鏈表。

/* 初始化堆內存 */
void mem_init(void)
{
  struct mem *mem;

  LWIP_ASSERT("Sanity check alignment", (SIZEOF_STRUCT_MEM & (MEM_ALIGNMENT - 1)) == 0);

  /* 內存堆對齊 */
  ram = (u8_t *)LWIP_MEM_ALIGN(LWIP_RAM_HEAP_POINTER);

  /* 在內存堆起始處組織一個內存塊 */
  mem = (struct mem *)(void *)ram;
  /* 下一個內存塊指向內存堆結尾 */
  mem->next = MEM_SIZE_ALIGNED;
  /* 上一個內存塊爲空 */
  mem->prev = 0;
  /* 該內存塊未使用 */
  mem->used = 0;
  
  /* 在內存堆結尾處組織一個內存塊 */
  ram_end = ptr_to_mem(MEM_SIZE_ALIGNED);
  /* 該內存塊已使用 */
  ram_end->used = 1;
  /* 上一個內存塊指向自身 */
  ram_end->next = MEM_SIZE_ALIGNED;
  /* 下一個內存塊指向自身 */
  ram_end->prev = MEM_SIZE_ALIGNED;
  
  MEM_SANITY();

  /* 內存堆起始處初始化爲地址最低的空閒塊 */
  lfree = (struct mem *)(void *)ram;

  MEM_STATS_AVAIL(avail, MEM_SIZE_ALIGNED);

  /* 創建一個互斥鎖 */
  if (sys_mutex_new(&mem_mutex) != ERR_OK) {
    LWIP_ASSERT("failed to create mem_mutex", 0);
  }
}

 

內存分配,從鏈表中找到一個合適的內存塊,分配給程序。將剩下的內存組織成新的內存塊,並掛接到鏈表。

/* 申請內存 */
void *mem_malloc(mem_size_t size_in)
{
  mem_size_t ptr, ptr2, size;
  struct mem *mem, *mem2;
#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
  u8_t local_mem_free_count = 0;
#endif
  LWIP_MEM_ALLOC_DECL_PROTECT();

  if (size_in == 0) {
    return NULL;
  }

  /* 字節數對齊 */
  size = (mem_size_t)LWIP_MEM_ALIGN_SIZE(size_in);
  /* 內存最小12字節 */
  if (size < MIN_SIZE_ALIGNED) {
    size = MIN_SIZE_ALIGNED;
  }
#if MEM_OVERFLOW_CHECK
  size += MEM_SANITY_REGION_BEFORE_ALIGNED + MEM_SANITY_REGION_AFTER_ALIGNED;
#endif
  if ((size > MEM_SIZE_ALIGNED) || (size < size_in)) {
    return NULL;
  }

  /* 獲取互斥鎖 */
  sys_mutex_lock(&mem_mutex);
  
  LWIP_MEM_ALLOC_PROTECT();
  
#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
  do {
    local_mem_free_count = 0;
#endif

    /* 遍歷所有內存塊 */
    for (ptr = mem_to_ptr(lfree); ptr < MEM_SIZE_ALIGNED - size; ptr = ptr_to_mem(ptr)->next) 
    {
      /* 內存塊地址 */
      mem = ptr_to_mem(ptr);

#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
      mem_free_count = 0;
      LWIP_MEM_ALLOC_UNPROTECT();
 
      LWIP_MEM_ALLOC_PROTECT();
      if (mem_free_count != 0) 
      {
        local_mem_free_count = 1;
        break;
      }
#endif

		/* 內存塊未使用,且滿足分配大小(請求大小+內存塊結構體大小) */
       if ((!mem->used) && (mem->next - (ptr + SIZEOF_STRUCT_MEM)) >= size) 
       {
        /* 內存塊過大(剩餘內存還夠分配最小內存塊) */
        if (mem->next - (ptr + SIZEOF_STRUCT_MEM) >= (size + SIZEOF_STRUCT_MEM + MIN_SIZE_ALIGNED)) 
        {
          /* 剩下的內存地址 */
          ptr2 = (mem_size_t)(ptr + SIZEOF_STRUCT_MEM + size);
          LWIP_ASSERT("invalid next ptr",ptr2 != MEM_SIZE_ALIGNED);
          /* 將剩下的內存地址轉換爲內存塊結構體指針 */
          mem2 = ptr_to_mem(ptr2);
          /* 新的內存塊未使用 */
          mem2->used = 0;
          /* 將新的內存塊插入鏈表 */
          mem2->next = mem->next;
          mem2->prev = ptr;
          mem->next = ptr2;

          /* 該內存塊設置爲已使用 */
          mem->used = 1;

          /* 如果不是最後一個內存塊,則下一個內存塊指向新內存塊 */
          if (mem2->next != MEM_SIZE_ALIGNED) {
            ptr_to_mem(mem2->next)->prev = ptr2;
          }

          MEM_STATS_INC_USED(used, (size + SIZEOF_STRUCT_MEM));
        }
        /* 如果剩餘內存不夠重新組織一個新內存塊 */
        else
        {
          /* 該內存塊已使用 */
          mem->used = 1;
          MEM_STATS_INC_USED(used, mem->next - mem_to_ptr(mem));
        }

#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
mem_malloc_adjust_lfree:
#endif

		/* 如果該內存塊爲地址最低的空閒塊,則要更新地址最低空閒塊 */
        if (mem == lfree) 
        {
          struct mem *cur = lfree;

          /* 遍歷向後的所有內存塊,找出地址最低的空閒塊 */
          while (cur->used && cur != ram_end) 
          {
#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
            mem_free_count = 0;
            LWIP_MEM_ALLOC_UNPROTECT();
            LWIP_MEM_ALLOC_PROTECT();
            if (mem_free_count != 0) {
              goto mem_malloc_adjust_lfree;
            }
#endif
            cur = ptr_to_mem(cur->next);
          }

          /* 更新地址最低的空閒塊 */
          lfree = cur;

          LWIP_ASSERT("mem_malloc: !lfree->used", ((lfree == ram_end) || (!lfree->used)));
        }
        
        LWIP_MEM_ALLOC_UNPROTECT();

		/* 釋放互斥鎖 */
        sys_mutex_unlock(&mem_mutex);
        
        LWIP_ASSERT("mem_malloc: allocated memory not above ram_end.",
                    (mem_ptr_t)mem + SIZEOF_STRUCT_MEM + size <= (mem_ptr_t)ram_end);
        LWIP_ASSERT("mem_malloc: allocated memory properly aligned.",
                    ((mem_ptr_t)mem + SIZEOF_STRUCT_MEM) % MEM_ALIGNMENT == 0);
        LWIP_ASSERT("mem_malloc: sanity check alignment",
                    (((mem_ptr_t)mem) & (MEM_ALIGNMENT - 1)) == 0);

#if MEM_OVERFLOW_CHECK
        mem_overflow_init_element(mem, size_in);
#endif

        MEM_SANITY();

		/* 返回內存地址 */
        return (u8_t *)mem + SIZEOF_STRUCT_MEM + MEM_SANITY_OFFSET;
      }
    }
#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
  } 
  while (local_mem_free_count != 0);
#endif
  MEM_STATS_INC(err);
  
  LWIP_MEM_ALLOC_UNPROTECT();
  
  sys_mutex_unlock(&mem_mutex);
  
  LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SERIOUS, ("mem_malloc: could not allocate %"S16_F" bytes\n", (s16_t)size));

  return NULL;
}

 

釋放內存,將內存重新組織成內存塊插入鏈表。相鄰內存堆都空閒,需要合併。

/* 釋放內存 */
void mem_free(void *rmem)
{
  struct mem *mem;

  LWIP_MEM_FREE_DECL_PROTECT();

  /* 參數合法性驗證,釋放的指針不能爲空 */
  if (rmem == NULL) 
  {
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_TRACE | LWIP_DBG_LEVEL_SERIOUS, ("mem_free(p == NULL) was called.\n"));
    return;
  }

  /* 參數合法性驗證,釋放的指針必須要字節對齊 */
  if ((((mem_ptr_t)rmem) & (MEM_ALIGNMENT - 1)) != 0) 
  {
    LWIP_MEM_ILLEGAL_FREE("mem_free: sanity check alignment");
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SEVERE, ("mem_free: sanity check alignment\n"));
    MEM_STATS_INC_LOCKED(illegal);
    return;
  }

  /* 內存指針向前偏移一個內存塊結構體大小 */
  mem = (struct mem *)(void *)((u8_t *)rmem - (SIZEOF_STRUCT_MEM + MEM_SANITY_OFFSET));

  /* 參數合法性驗證,釋放的指針必須在內存堆中 */
  if ((u8_t *)mem < ram || (u8_t *)rmem + MIN_SIZE_ALIGNED > (u8_t *)ram_end)
  {
    LWIP_MEM_ILLEGAL_FREE("mem_free: illegal memory");
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SEVERE, ("mem_free: illegal memory\n"));
    MEM_STATS_INC_LOCKED(illegal);
    return;
  }
#if MEM_OVERFLOW_CHECK
  mem_overflow_check_element(mem);
#endif

  LWIP_MEM_FREE_PROTECT();

  /* 參數合法性驗證,內存塊必須已經分配 */
  if (!mem->used) 
  {
    LWIP_MEM_ILLEGAL_FREE("mem_free: illegal memory: double free");
    LWIP_MEM_FREE_UNPROTECT();
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SEVERE, ("mem_free: illegal memory: double free?\n"));
    MEM_STATS_INC_LOCKED(illegal);
    return;
  }

  /* 參數合法性驗證,檢查內存塊是否鏈接在鏈表中 */
  if (!mem_link_valid(mem)) 
  {
    LWIP_MEM_ILLEGAL_FREE("mem_free: illegal memory: non-linked: double free");
    LWIP_MEM_FREE_UNPROTECT();
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SEVERE, ("mem_free: illegal memory: non-linked: double free?\n"));
    MEM_STATS_INC_LOCKED(illegal);
    return;
  }

  /* 將內存塊設置爲未使用 */
  mem->used = 0;

  /* 內存塊地址小於最低地址空閒塊 */
  if (mem < lfree) 
  {
    /* 更新最低地址空閒塊 */
    lfree = mem;
  }

  MEM_STATS_DEC_USED(used, mem->next - (mem_size_t)(((u8_t *)mem - ram)));

  /* 合併空閒塊 */
  plug_holes(mem);
  
  MEM_SANITY();
#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
  mem_free_count = 1;
#endif

  LWIP_MEM_FREE_UNPROTECT();
}
/* 合併空閒塊 */
static void plug_holes(struct mem *mem)
{
  struct mem *nmem;
  struct mem *pmem;

  LWIP_ASSERT("plug_holes: mem >= ram", (u8_t *)mem >= ram);
  LWIP_ASSERT("plug_holes: mem < ram_end", (u8_t *)mem < (u8_t *)ram_end);
  LWIP_ASSERT("plug_holes: mem->used == 0", mem->used == 0);

  LWIP_ASSERT("plug_holes: mem->next <= MEM_SIZE_ALIGNED", mem->next <= MEM_SIZE_ALIGNED);

  /* 下一個內存塊 */
  nmem = ptr_to_mem(mem->next);

  /* 下一個內存塊未被使用,並且不是最後一個內存塊 */
  if (mem != nmem && nmem->used == 0 && (u8_t *)nmem != (u8_t *)ram_end)
  {
    /* 下一個空閒塊是最小地址空閒塊 */
    if (lfree == nmem) 
    {
      /* 更新最小地址空閒塊 */
      lfree = mem;
    }

    /* 合併後一個內存塊 */
    mem->next = nmem->next;
    if (nmem->next != MEM_SIZE_ALIGNED)
    {
      ptr_to_mem(nmem->next)->prev = mem_to_ptr(mem);
    }
  }

  /* 上一個內存塊 */
  pmem = ptr_to_mem(mem->prev);

  /* 上一個內存塊未使用 */
  if (pmem != mem && pmem->used == 0) 
  {
    /* 該內存塊是最小地址空閒塊 */
    if (lfree == mem) 
    {
      /* 更新最小地址空閒塊 */
      lfree = pmem;
    }

    /* 合併上一個內存塊 */
    pmem->next = mem->next;
    if (mem->next != MEM_SIZE_ALIGNED) 
    {
      ptr_to_mem(mem->next)->prev = mem_to_ptr(pmem);
    }
  }
}

 

除了基本的分配和釋放,協議棧還mem_trim函數和mem_calloc函數

mem_trim函數,能夠收縮內存,將原有的內存縮小。縮小的部分重新組織成內存塊,掛接到鏈表。

/* 收縮內存 */
void *mem_trim(void *rmem, mem_size_t new_size)
{
  mem_size_t size, newsize;
  mem_size_t ptr, ptr2;
  struct mem *mem, *mem2;

  LWIP_MEM_FREE_DECL_PROTECT();

  /* 新的內存大小字節對齊 */
  newsize = (mem_size_t)LWIP_MEM_ALIGN_SIZE(new_size);
  /* 字節大小大於最小內存字節 */
  if (newsize < MIN_SIZE_ALIGNED) 
  {
    newsize = MIN_SIZE_ALIGNED;
  }

#if MEM_OVERFLOW_CHECK
  newsize += MEM_SANITY_REGION_BEFORE_ALIGNED + MEM_SANITY_REGION_AFTER_ALIGNED;
#endif

  /* 參數合法性驗證,新的內存大小不能大於內存堆 */
  if ((newsize > MEM_SIZE_ALIGNED) || (newsize < new_size)) {
    return NULL;
  }

  LWIP_ASSERT("mem_trim: legal memory", (u8_t *)rmem >= (u8_t *)ram && (u8_t *)rmem < (u8_t *)ram_end);

  /* 參數合法性驗證,內存必須在內存堆中 */
  if ((u8_t *)rmem < (u8_t *)ram || (u8_t *)rmem >= (u8_t *)ram_end)
  {
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SEVERE, ("mem_trim: illegal memory\n"));
    MEM_STATS_INC_LOCKED(illegal);
    return rmem;
  }
  
  /* 內存指針向前偏移內存塊結構體大小 */
  mem = (struct mem *)(void *)((u8_t *)rmem - (SIZEOF_STRUCT_MEM + MEM_SANITY_OFFSET));
#if MEM_OVERFLOW_CHECK
  mem_overflow_check_element(mem);
#endif

  /* 內存指針轉偏移量 */
  ptr = mem_to_ptr(mem);

  /* 當前內存塊,內存大小 */
  size = (mem_size_t)((mem_size_t)(mem->next - ptr) - (SIZEOF_STRUCT_MEM + MEM_SANITY_OVERHEAD));
  LWIP_ASSERT("mem_trim can only shrink memory", newsize <= size);

  /* 參數合法性驗證,新內存大小不能大於原內存大小 */
  if (newsize > size) 
  {
    return NULL;
  }

  /* 新內存大小如果等於原內存大小,什麼也不用做 */
  if (newsize == size) 
  {
    return rmem;
  }

  LWIP_MEM_FREE_PROTECT();

  /* 下一個內存塊指針 */
  mem2 = ptr_to_mem(mem->next);
  /* 下一個內存塊未使用,需要合併 */
  if (mem2->used == 0) 
  {
    mem_size_t next;
    LWIP_ASSERT("invalid next ptr", mem->next != MEM_SIZE_ALIGNED);

    /* 下下個內存偏移量 */
    next = mem2->next;
    /* 截取之後,剩餘內存塊指針 */
    ptr2 = (mem_size_t)(ptr + SIZEOF_STRUCT_MEM + newsize);
	/* 下一個內存塊是地址最低空閒塊 */
    if (lfree == mem2) 
    {
      /* 更新最低空閒塊指針 */
      lfree = ptr_to_mem(ptr2);
    }
    /* 剩餘內存塊指針 */
    mem2 = ptr_to_mem(ptr2);
    /* 剩餘內存塊設置爲未使用 */
    mem2->used = 0;
    
    /* 將剩餘空閒塊和下一個內存塊合併插入鏈表 */
    mem2->next = next;
    mem2->prev = ptr;
    mem->next = ptr2;

    /* 不是最後一個內存塊,要讓下一個內存塊指向自己 */
    if (mem2->next != MEM_SIZE_ALIGNED) 
    {
      ptr_to_mem(mem2->next)->prev = ptr2;
    }
    MEM_STATS_DEC_USED(used, (size - newsize));
  }
  /* 下一個內存塊不空閒,並且剩餘的內存還夠組織新的內存塊 */
  else if (newsize + SIZEOF_STRUCT_MEM + MIN_SIZE_ALIGNED <= size) 
  {
    /* 剩餘的內存塊指針 */
    ptr2 = (mem_size_t)(ptr + SIZEOF_STRUCT_MEM + newsize);

    LWIP_ASSERT("invalid next ptr", mem->next != MEM_SIZE_ALIGNED);

    /* 內存塊轉換爲偏移量 */
    mem2 = ptr_to_mem(ptr2);

    /* 剩餘內存塊地址小於最小地址空閒塊 */
    if (mem2 < lfree) 
    {
      /* 更新最小地址空閒塊 */
      lfree = mem2;
    }
    /* 剩餘內存塊未使用 */
    mem2->used = 0;
    
    /* 將剩餘內存塊插入鏈表 */
    mem2->next = mem->next;
    mem2->prev = ptr;
    mem->next = ptr2;
    if (mem2->next != MEM_SIZE_ALIGNED)
    {
      ptr_to_mem(mem2->next)->prev = ptr2;
    }
    MEM_STATS_DEC_USED(used, (size - newsize));
  }
  
#if MEM_OVERFLOW_CHECK
  mem_overflow_init_element(mem, new_size);
#endif

  MEM_SANITY();
  
#if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT
  mem_free_count = 1;
#endif

  LWIP_MEM_FREE_UNPROTECT();
  
  return rmem;
}

mem_calloc函數,申請內存的同時,將內存清空。

/* 申請並清空內存 */
void *mem_calloc(mem_size_t count, mem_size_t size)
{
  void *p;
  size_t alloc_size = (size_t)count * (size_t)size;

  if ((size_t)(mem_size_t)alloc_size != alloc_size) 
  {
    LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SERIOUS, ("mem_calloc: could not allocate %"SZT_F" bytes\n", alloc_size));
    return NULL;
  }

  /* 申請內存 */
  p = mem_malloc((mem_size_t)alloc_size);
  /* 申請成功 */
  if (p) 
  {
    /* 清空內存 */
    memset(p, 0, alloc_size);
  }
  return p;
}

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章