日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

LwIP 之五 详解动态内存管理 内存堆(mem.c/h)

發布時間:2024/10/14 编程问答 86 豆豆
生活随笔 收集整理的這篇文章主要介紹了 LwIP 之五 详解动态内存管理 内存堆(mem.c/h) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

寫在前面

??目前網上有很多介紹LwIP內存的文章,但是絕大多數都不夠詳細,甚至很多介紹都是錯誤的!無論是代碼的說明還是給出的圖例,都欠佳!下面就從源代碼,到圖例詳細進行說明。
??目前,網絡上多數文章所使用的LwIP版本為1.4.1。最新版本為2.0.3。從1.4.1到2.0.3(貌似從2.0.0開始),LwIP的源碼有了一定的變化,甚至于源碼的文件結構也不一樣,內部的一些實現源文件也被更新和替換了。

簡介

??對于嵌入式開發來說,內存管理及使用是至關重要的,內存的使用多少、內存泄漏等時刻需要注意!合理的內存管理策略將從根本上決定內存分配和回收效率,最終決定系統的整體性能。LwIP為了能夠靈活的使用內存,為使用者提供兩種簡單卻又高效的動態內存管理機制:***動態內存堆管理(heap)、動態內存池管理(pool)***。這兩中內存管理策略的實現分別對應著源碼文件mem.c/h和memp.c/h。
??其中,***動態內存池管理(heap)***又可以分為兩種: C 運行時庫自帶的內存分配策略、LwIP自己實現的內存堆分配策略。這兩者的選擇需要通過宏值MEM_LIBC_MALLOC來選擇,且二者只能選擇其一。
??其次,LwIP在自己的內存堆和內存池具體實現上也比較靈活。內存池可有由內存堆實現,反之,內存堆也可以有內存池實現。通過宏值MEM_USE_POOLS和MEMP_MEM_MALLOC來選擇,且二者只能選擇其一。

內存堆和內存池

??動態內存堆分配策略原理就是在一個事先定義好大小的內存塊中進行管理,其內存分配的策略是采用最快合適( First Fit)方式,只要找到一個比所請求的內存大的空閑塊,就從中切割出合適的塊,并把剩余的部分返回到動態內存堆中。內存的釋放時,重新將申請到的內存返回堆中。
??其優點就是內存浪費小,比較簡單,適合用于小內存的管理,其缺點就是如果頻繁的動態分配和釋放,可能會造成嚴重的內存碎片,如果在碎片情況嚴重的話,可能會導致內存分配不成功。

這其中也有個問題,就是內存合并問題。因為內存堆的管理通常為鏈表的形式進行管理。可選擇將小的鏈表節點(較小的內存)進行合并。

??內存池的特點是預先開辟多組固定大小的內存塊組織成鏈表,實現簡單,分配和回收速度快,不會產生內存碎片,但是大小固定,并且需要預估算準確。

內存對齊

??一般來說,每一種處理器都會有自己的內存對齊要求,這樣做的目的很大程度上是為了處理器讀取內存數據的效率,且與對應硬件上的設計也有很大的關系。LwIP中,對于內存的操作函數都用到了內存對齊。

??在LwIP中,用戶需要處理一個重要的部分sys_arch。具體可以參見另一片博文。整個arch框架中,就包含了內存對其這一塊的配置:

/** Allocates a memory buffer of specified size that is of sufficient size to align* its start address using LWIP_MEM_ALIGN.* You can declare your own version here e.g. to enforce alignment without adding* trailing padding bytes (see LWIP_MEM_ALIGN_BUFFER) or your own section placement* requirements.\n* e.g. if you use gcc and need 32 bit alignment:\n* \#define LWIP_DECLARE_MEMORY_ALIGNED(variable_name, size) u8_t variable_name[size] \_\_attribute\_\_((aligned(4)))\n* or more portable:\n* \#define LWIP_DECLARE_MEMORY_ALIGNED(variable_name, size) u32_t variable_name[(size + sizeof(u32_t) - 1) / sizeof(u32_t)]*//* 定義全局數組作為內存堆的內存,LwIP就是實現的如何管理這塊內存的。這塊內存時經過對其操作的 */ #ifndef LWIP_DECLARE_MEMORY_ALIGNED #define LWIP_DECLARE_MEMORY_ALIGNED(variable_name, size) u8_t variable_name[LWIP_MEM_ALIGN_BUFFER(size)] #endif/** Calculate memory size for an aligned buffer - returns the next highest* multiple of MEM_ALIGNMENT (e.g. LWIP_MEM_ALIGN_SIZE(3) and* LWIP_MEM_ALIGN_SIZE(4) will both yield 4 for MEM_ALIGNMENT == 4).*//* 數據占用空間大小對齊計算 */ #ifndef LWIP_MEM_ALIGN_SIZE #define LWIP_MEM_ALIGN_SIZE(size) (((size) + MEM_ALIGNMENT - 1U) & ~(MEM_ALIGNMENT-1U)) #endif/** Calculate safe memory size for an aligned buffer when using an unaligned* type as storage. This includes a safety-margin on (MEM_ALIGNMENT - 1) at the* start (e.g. if buffer is u8_t[] and actual data will be u32_t*)*/ #ifndef LWIP_MEM_ALIGN_BUFFER #define LWIP_MEM_ALIGN_BUFFER(size) (((size) + MEM_ALIGNMENT - 1U)) #endif/** Align a memory pointer to the alignment defined by MEM_ALIGNMENT* so that ADDR % MEM_ALIGNMENT == 0*//* 數據起始地址對齊 */ #ifndef LWIP_MEM_ALIGN #define LWIP_MEM_ALIGN(addr) ((void *)(((mem_ptr_t)(addr) + MEM_ALIGNMENT - 1) & ~(mem_ptr_t)(MEM_ALIGNMENT-1))) #endif

為什么要對齊

??這個其實和各種處理器的硬件設計息息相關,具體可以參見該博文《為什么要內存對齊》。

對齊的本質

??首先,這里所說的對其是指 2k2^k2k 字節的對其(其中k取0,1,2,3…正整數)。如果你糾結于什么3字節的對齊等。這里不適用! 在計算機中,所有的數據都是二進制!所謂對齊,就是將一串二進制的最后N位抹成 0。具體幾位呢?這就是根據自己要對齊的字節數來定了。1字節對齊時N == 0;2 字節對齊時N == 1,4 字節對齊時N == 2;以此類推。對齊字節數是根據硬件平臺來的,一般不會隨便來。
??再一點就是,對齊的抹 0,是需要向上取整的。為什么要向上取整呢?如果網下取整,那么,實際返回的大小就比用戶實際需要的要小了。這可就麻煩了,甚至由于不滿足自己需要的大小,申請的這塊內存都沒法使用!

關于任意自己對齊,可以參考一下文章:
??1. 實現任意字節對齊的內存分配和釋放
??2. 任意字節對齊的內存分配和釋放

LWIP_MEM_ALIGN_SIZE(size)

??這個宏的作用就是將指定的大小處理成對其后大小,對其的大小由用戶提供宏值MEM_ALIGNMENT來決定。其中size為想要分配的大小。下面來看看這個宏:

  • ~(MEM_ALIGNMENT-1U): 這一步就是按照對應的對齊字節數據,將二進制的最后最后幾位置為 0。例如,MEM_ALIGNMENT為4 ,則該步就將后2位置為了 0。自己可以轉為二進制計算一下!
  • ((size) + MEM_ALIGNMENT - 1U): 這里其實就是為了在處理時,能夠向上取整。

??下面是針對其中不同值得計算結果(MEM_ALIGNMENT表示要對其的字節,size 為想要的大小,ALIG表示對齊后實際的大小)

MEM_ALIGNMENT = 1 MEM_ALIGNMENT = 2 MEM_ALIGNMENT = 4 MEM_ALIGNMENT = 8 size = 1 ALIG = 1 size = 1 ALIG = 2 size = 1 ALIG = 4 size = 1 ALIG = 8 size = 2 ALIG = 2 size = 2 ALIG = 2 size = 2 ALIG = 4 size = 2 ALIG = 8 size = 3 ALIG = 3 size = 3 ALIG = 4 size = 3 ALIG = 4 size = 3 ALIG = 8 size = 4 ALIG = 4 size = 4 ALIG = 4 size = 4 ALIG = 4 size = 4 ALIG = 8 size = 5 ALIG = 5 size = 5 ALIG = 6 size = 5 ALIG = 8 size = 5 ALIG = 8 size = 6 ALIG = 6 size = 6 ALIG = 6 size = 6 ALIG = 8 size = 6 ALIG = 8 size = 7 ALIG = 7 size = 7 ALIG = 8 size = 7 ALIG = 8 size = 7 ALIG = 8 size = 8 ALIG = 8 size = 8 ALIG = 8 size = 8 ALIG = 8 size = 8 ALIG = 8 size = 9 ALIG = 9 size = 9 ALIG = 10 size = 9 ALIG = 12 size = 9 ALIG = 16 size = 10 ALIG = 10 size = 10 ALIG = 10 size = 10 ALIG = 12 size = 10 ALIG = 16 size = 11 ALIG = 11 size = 11 ALIG = 12 size = 11 ALIG = 12 size = 11 ALIG = 16 size = 12 ALIG = 12 size = 12 ALIG = 12 size = 12 ALIG = 12 size = 12 ALIG = 16 size = 13 ALIG = 13 size = 13 ALIG = 14 size = 13 ALIG = 16 size = 13 ALIG = 16 size = 14 ALIG = 14 size = 14 ALIG = 14 size = 14 ALIG = 16 size = 14 ALIG = 16 size = 15 ALIG = 15 size = 15 ALIG = 16 size = 15 ALIG = 16 size = 15 ALIG = 16 size = 16 ALIG = 16 size = 16 ALIG = 16 size = 16 ALIG = 16 size = 16 ALIG = 16 size = 17 ALIG = 17 size = 17 ALIG = 18 size = 17 ALIG = 20 size = 17 ALIG = 24 size = 18 ALIG = 18 size = 18 ALIG = 18 size = 18 ALIG = 20 size = 18 ALIG = 24 size = 19 ALIG = 19 size = 19 ALIG = 20 size = 19 ALIG = 20 size = 19 ALIG = 24 size = 20 ALIG = 20 size = 20 ALIG = 20 size = 20 ALIG = 20 size = 20 ALIG = 24

LWIP_MEM_ALIGN(addr)

??這個宏用來處理數據起始地址對齊。其處理方式與上面的數據大小的處理沒有任何區別。關于數據類型在arch.h的開頭部分有定義,這里有個文件stdint.h需要注意一下!在某寫平臺,可能沒有該文件,需要用戶自己來添加。

#if !LWIP_NO_STDINT_H #include <stdint.h> typedef uint8_t u8_t; typedef int8_t s8_t; typedef uint16_t u16_t; typedef int16_t s16_t; typedef uint32_t u32_t; typedef int32_t s32_t; typedef uintptr_t mem_ptr_t; /* 這個通常為一個 unsigned int 類型 */ #endif

LwIP中宏配置及使用

??LwIP中,內存的選擇是通過以下這幾個宏值來決定的,根據用戶對宏值的定義值來判斷使用那種內存管理策略,具體如下:

  • MEM_LIBC_MALLOC: 該宏值定義是否使用C 運行時庫自帶的內存分配策略。該值默認情況下為0,表示不使用C 運行時庫自帶的內存分配策略。即默認使用LwIP提供的內存堆分配策略。
    ??如果要使用C運行時庫自帶的分配策略,則需要把該值定義為 1。此時,宏值MEM_USE_POOLS必然不能為 1。
  • MEMP_MEM_MALLOC: 該宏值定義是否使用lwip內存堆分配策略實現內存池分配(即:要從內存池中獲取內存時,實際是從內存堆中分配)。默認情況下為 0,表示不從內存堆中分配,內存池為獨立一塊內存實現。與MEM_USE_POOLS只能選擇其一
  • MEM_USE_POOLS:該宏值定時是否使用lwip內存池分配策略實現內存堆的分配(即:要從內存堆中獲取內存時,實際是從內存池中分配)。默認情況下為 0,表示不使用。與MEMP_MEM_MALLOC只能選擇其一
    ??要使用內存池的方式,則需要將該宏值定義為 1,且MEMP_MEM_MALLOC必須為 0,除此之外還需要做一下處理:
  • MEMP_USE_CUSTOM_POOLS == 1
  • 新建文件lwippools.h,并且在該文件中定義如下內存池(想多定義幾個時,必須在宏LWIP_MALLOC_MEMPOOL_START和LWIP_MALLOC_MEMPOOL_END之間添加):
  • LWIP_MALLOC_MEMPOOL(20, 256) LWIP_MALLOC_MEMPOOL(10, 512) LWIP_MALLOC_MEMPOOL(5, 1512) LWIP_MALLOC_MEMPOOL_END

    ??根據宏值的不同,mem.c/h會部分有效,下面簡單看看mem.c文件的結構:

    #if MEM_LIBC_MALLOC /* 這里表示使用C庫時的部分 */ #elif MEM_USE_POOLS /* 這里表示使用內存池方式實現的內存堆函數 */ #else /* 使用內存堆實現分配函數 */ #endif

    而對于memp.c/h來說,就沒這么復雜了。因為該文件就是獨立實現內存池的管理策略的,無論宏值怎么配置,里面的函數就是那些。唯一有影響的宏值是MEMP_MEM_MALLOC后面會說明。
    ??總結來說,無論宏值怎么配置,LwIP都有兩種內存管理策略:內存堆內存池

    • 內存堆: 可以來自C庫,也可以使用LwIP自己實現的。
    • 內存池: 可以單獨實現,也可以從內存堆中分配實現。

    ??上面說了如何進行配置,那么在LwIP內部是如何做到兼容以上的靈活配置的呢?其實很簡單,LwIP內部全部使用統一的內存分配函數,只是在不同模式下,對相應的函數進行了重新定義,具體函數如下:

    • void mem_init(void):內存堆初始化函數,主要設置內存堆的起始地址,以及初始化空閑列表,lwip初始化時調用,內部接口。配置不同時,其實現也不同(可能為空)
    • void *mem_trim(void *mem, mem_size_t size): 減小mem_malloc所分配的內存。mem為申請到的內存的地址,size為新的大小。

    與標準C函數realloc不同,該函數只能縮減分配內存

    • void *mem_malloc(mem_size_t size):申請分配內存,size為需要申請的內存字節數,返回值為最新分配的內存塊的數據地址。
    • void *mem_calloc(mem_size_t count, mem_size_t size):是對mem_malloc()函數的簡單包裝,兩個入口參數,count為元素的總個數,size為每個元素大小,兩個參數的乘積就是實際要分配的內存空間的大小。與mem_malloc()不同的是它會把動態分配的內存清零
    • void mem_free(void *mem):內存釋放函數,mem前面申請到的內存時分配得地址。

    這樣,無論用戶選擇了什么配置方式,對于LwIP內部來說,函數全都是一個樣的!下面,詳細說明每種方式下,以上函數是怎么實現的。

    C庫分配方式

    ??上面已經說了要使用該方式如何進行配置,下面結合源碼看看,如果用戶選擇了該方式,LwIP內部是如何處理的。該部分對應的源碼文件為mem.c。
    ??首先,如果用戶選擇了該方式,那么內存處理函數void mem_init(void)和void* mem_trim(void *mem, mem_size_t size)將沒有實際的實現內容,既然選擇了C庫策略,此時也必然沒法實現。
    ??接下來我們將看到如下代碼:

    /* in case C library malloc() needs extra protection,* allow these defines to be overridden.*/ #ifndef mem_clib_free #define mem_clib_free free #endif #ifndef mem_clib_malloc #define mem_clib_malloc malloc #endif #ifndef mem_clib_calloc #define mem_clib_calloc calloc #endif#if LWIP_STATS && MEM_STATS #define MEM_LIBC_STATSHELPER_SIZE LWIP_MEM_ALIGN_SIZE(sizeof(mem_size_t)) #else #define MEM_LIBC_STATSHELPER_SIZE 0 #endif

    這里將C庫的標準內存處理函數進行了重定義,沒啥太大作用,就是為了后面使用的方便。下面以void * mem_malloc(mem_size_t size)為例看看以上宏值的用法(其他函數類似,注意:有幾個函數在文件的最末尾統一處理了):

    void * mem_malloc(mem_size_t size) {/* 這里就是調用的malloc有木有! */void* ret = mem_clib_malloc(size + MEM_LIBC_STATSHELPER_SIZE);if (ret == NULL) { /* 分配失敗 */MEM_STATS_INC(err);} else { /* 需要重點注意的就是,內存的對其問題,下面就是處理對其的。然后直接返回申請到的內存地址。 */LWIP_ASSERT("malloc() must return aligned memory", LWIP_MEM_ALIGN(ret) == ret); #if LWIP_STATS && MEM_STATS*(mem_size_t*)ret = size;ret = (u8_t*)ret + MEM_LIBC_STATSHELPER_SIZE;MEM_STATS_INC_USED(used, size); #endif}return ret; }

    ??最后,使用C庫內存處理策略時,LwIP的處理就是這么簡單,也沒啥需要特殊注意的地方!如果說非得注意點啥,就是一旦選擇了該種模式,內存池的配置問題。

    LwIP內存堆

    ??上面已經說了要使用該方式如何進行配置,下面結合源碼看看,如果用戶選擇了該方式,LwIP內部是如何處理的。該部分對應的源碼文件為mem.c。

    用內存池實現

    ??首先,LwIP會判斷用戶是否定義了宏值MEM_USE_POOLS,如果定義了該宏值,表示內存堆的內存來自于內存池,需要調用內存池的相關函數來處理內存。
    ??既然是調用內存堆中的各函數,那么這邊的實現也相對簡單。這與上面的調用C庫的各函數沒啥太大區別,至少在處理邏輯上是一致的,當然,由于內存池的實現上的原因,該部分看著還是挺復雜的!下面仍舊以函數void *mem_malloc(mem_size_t size)來看看LwIP是如何處理的。

    void * mem_malloc(mem_size_t size) {void *ret;struct memp_malloc_helper *element = NULL;memp_t poolnr;mem_size_t required_size = size + LWIP_MEM_ALIGN_SIZE(sizeof(struct memp_malloc_helper));for (poolnr = MEMP_POOL_FIRST; poolnr <= MEMP_POOL_LAST; poolnr = (memp_t)(poolnr + 1)) {/* is this pool big enough to hold an element of the required sizeplus a struct memp_malloc_helper that saves the pool this element came from? */if (required_size <= memp_pools[poolnr]->size) {element = (struct memp_malloc_helper*)memp_malloc(poolnr);if (element == NULL) {/* No need to DEBUGF or ASSERT: This error is already taken care of in memp.c */ #if MEM_USE_POOLS_TRY_BIGGER_POOL/** Try a bigger pool if this one is empty! */if (poolnr < MEMP_POOL_LAST) {continue;} #endif /* MEM_USE_POOLS_TRY_BIGGER_POOL */MEM_STATS_INC(err);return NULL;}break;}}if (poolnr > MEMP_POOL_LAST) {LWIP_ASSERT("mem_malloc(): no pool is that big!", 0);MEM_STATS_INC(err);return NULL;}/* save the pool number this element came from */element->poolnr = poolnr;/* and return a pointer to the memory directly after the struct memp_malloc_helper */ret = (u8_t*)element + LWIP_MEM_ALIGN_SIZE(sizeof(struct memp_malloc_helper));#if MEMP_OVERFLOW_CHECK || (LWIP_STATS && MEM_STATS)/* truncating to u16_t is safe because struct memp_desc::size is u16_t */element->size = (u16_t)size;MEM_STATS_INC_USED(used, element->size); #endif /* MEMP_OVERFLOW_CHECK || (LWIP_STATS && MEM_STATS) */ #if MEMP_OVERFLOW_CHECK/* initialize unused memory (diff between requested size and selected pool's size) */memset((u8_t*)ret + size, 0xcd, memp_pools[poolnr]->size - size); #endif /* MEMP_OVERFLOW_CHECK */return ret; }

    ??該部分許多地方需要看看后面的內存池。再次不做過多深入說明。

    獨立實現內存堆

    ??如果沒有定義宏值MEM_USE_POOLS,那么LwIP提供了一套獨立實現的內存堆處理接口。
    ??內存堆的本質是對一個事先定義好的內存塊進行合理有效的組織和管理。主要用于任意大小的內存分配,實現較復雜,分配需要查找,回收需要合并,容易產生內存碎片,需要合理估算內存堆的總大小。LwIP使用類似于鏈表的結構來組織管理堆內存。節點的結構如下(在該部分的源碼實現中,第一部分的代碼便是以下結構體):

    struct mem {/** index (-> ram[next]) of the next struct */mem_size_t next; /* 下一個內存塊的索引,不是指針,而是個索引號 *//** index (-> ram[prev]) of the previous struct */mem_size_t prev; /* 前一個內存塊的索引,不是指針,而是個索引號 *//** 1: this area is used; 0: this area is unused */u8_t used; /* 此內存快是否被用。1使用、0 未使用 */ };

    ??LwIP的內存堆實現中,分配的內存塊有個最小大小的限制,要求請求的分配大小不能小于 MIN_SIZE,默認 MIN_SIZE為 12 字節。所以在該部分實現中,我看到的第二部分便是如下結構(第一部分是鏈表結構體):

    /** All allocated blocks will be MIN_SIZE bytes big, at least!* MIN_SIZE can be overridden to suit your needs. Smaller values save space,* larger values could prevent too small blocks to fragment the RAM too much. */ #ifndef MIN_SIZE #define MIN_SIZE 12 #endif /* MIN_SIZE */ /* some alignment macros: we define them here for better source code layout */ #define MIN_SIZE_ALIGNED LWIP_MEM_ALIGN_SIZE(MIN_SIZE) /* 最小大小做對齊處理,后面均用對齊后的該宏值 */ #define SIZEOF_STRUCT_MEM LWIP_MEM_ALIGN_SIZE(sizeof(struct mem)) /* 內存塊頭大小做對齊處理,后面均用對齊后的該宏值 */ #define MEM_SIZE_ALIGNED LWIP_MEM_ALIGN_SIZE(MEM_SIZE) /* 用戶定義的堆大小做對齊處理,后面均用對齊后的該宏值 */

    這就有一個問題,為什么默認值是12呢?因為這正好是2個sizeof(struct mem)的長度。為什么是2個sizeof(struct mem)。后面再說內存堆的定義及初始化時,我們會詳細說明。
    ??接下來就是使用宏值LWIP_DECLARE_MEMORY_ALIGNED定義內存堆所使用的內存空間的了,這個宏值在上面一節由介紹,具體如下。從下面的定義中,我們可以看到***內存堆空間就是定義的一個名為ram_heap,大小為MEM_SIZE_ALIGNED + (2U*SIZEOF_STRUCT_MEM)的數組***。內存堆管理的具體實現,就是實現如何管理使用這個數組!

    /** If you want to relocate the heap to external memory, simply define* LWIP_RAM_HEAP_POINTER as a void-pointer to that location.* If so, make sure the memory at that location is big enough (see below on* how that space is calculated). */ #ifndef LWIP_RAM_HEAP_POINTER /** the heap. we need one struct mem at the end and some room for alignment */ /* 下面用全局變量的形式(名字為ram_heap),定義堆內存空間。并且由于對齊問題,不能直接使用數組名。因此,后面再用時全部處理為指針 */ LWIP_DECLARE_MEMORY_ALIGNED(ram_heap, MEM_SIZE_ALIGNED + (2U*SIZEOF_STRUCT_MEM)); #define LWIP_RAM_HEAP_POINTER ram_heap #endif /* LWIP_RAM_HEAP_POINTER */

    注意:這里的大小多加了兩個SIZEOF_STRUCT_MEM

    ??在定義完內存空間后,接下來就是一些為了管理內存堆,而定義的結構性的全局變量的定義。再后面是一些內存保護相關的宏,這里不具體說明。下面看看,為了管理內存堆,LwIP都定義了那些變量。

    /** pointer to the heap (ram_heap): for alignment, ram is now a pointer instead of an array */ static u8_t *ram; /* 指向對齊后的內存堆的地址。由于對齊問題,不能直接使用數組名上面的數組名。*/ /** the last entry, always unused! */ static struct mem *ram_end; /* 指向對齊后的內存堆的最后一個內存塊。*/ /** pointer to the lowest free block, this is used for faster search */ static struct mem *lfree; /* 指向已被釋放的索引號最小的內存塊(內存堆最前面的已被釋放的)。*/

    下面,就結合各接口函數來說說LwIP對內存堆的管理是如何實現的。先說第一個函數void mem_init(void)。先好看看源碼,具體說明見其中的注釋部分:

    void mem_init(void) {struct mem *mem;LWIP_ASSERT("Sanity check alignment",(SIZEOF_STRUCT_MEM & (MEM_ALIGNMENT-1)) == 0);/* align the heap 對內存堆的地址(全局變量的名)進行對齊。*/ram = (u8_t *)LWIP_MEM_ALIGN(LWIP_RAM_HEAP_POINTER);/* initialize the start of the heap 建立第一個內存塊,內存塊由內存塊頭+空間組成。 */mem = (struct mem *)(void *)ram;mem->next = MEM_SIZE_ALIGNED; /* 下一個內存塊不存在,因此指向內存堆的結束 */mem->prev = 0; /* 前一個內存塊就是它自己,因為這是第一個內存塊 */mem->used = 0; /* 第一個內存塊沒有被使用 *//* initialize the end of the heap 建立最后一個內存塊 */ram_end = (struct mem *)(void *)&ram[MEM_SIZE_ALIGNED];ram_end->used = 1; /* 最后一個內存塊被使用。因為其后面沒有可用空間,必須標記為已被使用 */ram_end->next = MEM_SIZE_ALIGNED;/* 下一個不存在,因此指向內存堆的結束 */ram_end->prev = MEM_SIZE_ALIGNED;/* 前一個不存在,因此指向內存堆的結束 *//* initialize the lowest-free pointer to the start of the heap */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);} }

    ??經過初始化,LwIP將內存堆劃分為了如下格式:

    上圖中已經非常詳細的給出了從定義到初始化后,詳細的內存堆的結構!在整個內存堆的開頭和結尾,各有一個內存塊頭。開頭的內存塊頭用來定義第一個內存塊(也就是整個內存堆只有一個內存塊),最后一個內存塊頭用來標識內存塊的結尾。 看圖的時候,結合源碼和上一節的內存對齊部分。因為在上圖中,到處都是對齊!
    ??接下來,在看看函數void *mem_malloc(mem_size_t size)。仍舊是先看源代碼,具體語句在源碼中都有注釋,對于源碼中,保護部分暫不作說明!

    void * mem_malloc(mem_size_t size) {mem_size_t ptr, ptr2;struct mem *mem, *mem2; #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXTu8_t local_mem_free_count = 0; #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT */LWIP_MEM_ALLOC_DECL_PROTECT();/* 第一步:參數檢查 */if (size == 0) {return NULL;}/* Expand the size of the allocated memory region so that we canadjust for alignment. */size = LWIP_MEM_ALIGN_SIZE(size);if (size < MIN_SIZE_ALIGNED) { /* 這里處理 最小內存塊的限制問題 不能小于前面定義的宏值 *//* every data block must be at least MIN_SIZE_ALIGNED long */size = MIN_SIZE_ALIGNED;}if (size > MEM_SIZE_ALIGNED) {return NULL;}/* 第二步:從內存堆中劃分需要的內存塊 *//* protect the heap from concurrent access */sys_mutex_lock(&mem_mutex);LWIP_MEM_ALLOC_PROTECT(); #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT/* run as long as a mem_free disturbed mem_malloc or mem_trim */do {local_mem_free_count = 0; #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT *//* Scan through the heap searching for a free block that is big enough,* beginning with the lowest free block.*/for (ptr = (mem_size_t)((u8_t *)lfree - ram); ptr < MEM_SIZE_ALIGNED - size;ptr = ((struct mem *)(void *)&ram[ptr])->next) {/* 遍歷內存堆,注意遍歷時的開始地址:永遠從最前面開始找。因為lfree指向已釋放的最靠前的內存塊 */mem = (struct mem *)(void *)&ram[ptr]; /* 內存塊的頭 */ #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXTmem_free_count = 0;LWIP_MEM_ALLOC_UNPROTECT();/* allow mem_free or mem_trim to run */LWIP_MEM_ALLOC_PROTECT();if (mem_free_count != 0) {/* If mem_free or mem_trim have run, we have to restart since theycould have altered our current struct mem. */local_mem_free_count = 1;break;} #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT */if ((!mem->used) &&(mem->next - (ptr + SIZEOF_STRUCT_MEM)) >= size) { /* 空間大小必須排除內存塊頭大小 *//* mem is not used and at least perfect fit is possible:* mem->next - (ptr + SIZEOF_STRUCT_MEM) gives us the 'user data size' of mem */if (mem->next - (ptr + SIZEOF_STRUCT_MEM) >= (size + SIZEOF_STRUCT_MEM + MIN_SIZE_ALIGNED)) { /* 這個地方需要判斷 最后一個內存塊頭和最小大小 *//* (in addition to the above, we test if another struct mem (SIZEOF_STRUCT_MEM) containing* at least MIN_SIZE_ALIGNED of data also fits in the 'user data space' of 'mem')* -> split large block, create empty remainder,* remainder must be large enough to contain MIN_SIZE_ALIGNED data: if* mem->next - (ptr + (2*SIZEOF_STRUCT_MEM)) == size,* struct mem would fit in but no data between mem2 and mem2->next* @todo we could leave out MIN_SIZE_ALIGNED. We would create an empty* region that couldn't hold data, but when mem->next gets freed,* the 2 regions would be combined, resulting in more free memory*//* 上面注釋一大堆,主要就是說,剩余內存可能連一個內存塊的頭都放不下了,這個時候就沒法新建空內存塊。其索引也就不能移動 */ptr2 = ptr + SIZEOF_STRUCT_MEM + size; /* 指向申請后的位置,即:建立下一個未使用的內存塊的頭部。即:插入一個新空內存塊 *//* create mem2 struct */mem2 = (struct mem *)(void *)&ram[ptr2];mem2->used = 0;mem2->next = mem->next; /* */mem2->prev = ptr; /* 空閑內存塊的前一個指向上面分配的內存塊 *//* and insert it between mem and mem->next 前一個內存塊指向上面建立的空閑內存塊 */mem->next = ptr2;mem->used = 1; /* 將當前分配的內存塊標記為 已使用 */if (mem2->next != MEM_SIZE_ALIGNED) {/* 中間節點 建立后,原來的下一個節點的前一個要指向 新建的空節點 */((struct mem *)(void *)&ram[mem2->next])->prev = ptr2;}MEM_STATS_INC_USED(used, (size + SIZEOF_STRUCT_MEM));} else {/* 進入了這一步,則表示內存塊太小了,即:生產的碎片 *//* (a mem2 struct does no fit into the user data space of mem and mem->next will always* be used at this point: if not we have 2 unused structs in a row, plug_holes should have* take care of this).* -> near fit or exact fit: do not split, no mem2 creation* also can't move mem->next directly behind mem, since mem->next* will always be used at this point!*/mem->used = 1;MEM_STATS_INC_USED(used, mem->next - (mem_size_t)((u8_t *)mem - ram));} #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT mem_malloc_adjust_lfree: #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT */if (mem == lfree) { /* 這里處理:當分配出去的內存正好是lfree時,因為該內存塊已經被分配出去了,必須修改lfree的指向下一個最其前面的已釋放的內存塊*/struct mem *cur = lfree;/* Find next free block after mem and update lowest free pointer */while (cur->used && cur != ram_end) { /* 只要內存塊已使用且沒到結尾,則繼續往后找 */ #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXTmem_free_count = 0;LWIP_MEM_ALLOC_UNPROTECT();/* prevent high interrupt latency... */LWIP_MEM_ALLOC_PROTECT();if (mem_free_count != 0) {/* If mem_free or mem_trim have run, we have to restart since theycould have altered our current struct mem or lfree. */goto mem_malloc_adjust_lfree;} #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT */cur = (struct mem *)(void *)&ram[cur->next]; /* 下一個內存塊 */}lfree = cur; /* 指向找到的 第一個已釋放的內存塊。如果上面沒有找到,則lfree = lfree不變 */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);return (u8_t *)mem + SIZEOF_STRUCT_MEM; /* 這里返回 內存塊的空間的地址,排除內存塊的頭 */}} #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT/* if we got interrupted by a mem_free, try again */} while (local_mem_free_count != 0); #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT */LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SERIOUS, ("mem_malloc: could not allocate %"S16_F" bytes\n", (s16_t)size));MEM_STATS_INC(err);LWIP_MEM_ALLOC_UNPROTECT();sys_mutex_unlock(&mem_mutex);return NULL; }

    ??分配內存的過程,就是從初始化后的內存塊中,劃分新內存塊的過程。新內存塊劃分完成后,將其標記為已使用,并且新建的空內存塊。在劃分過程中,需要處理已經被釋放的內存塊的問題。劃分后如下圖所示:

    ??首先需要重點注意的就是遍歷查找新內存塊時的起始位置:ptr = (mem_size_t)((u8_t *)lfree - ram)。前面說過,變量lfree永遠指向內存堆中最靠前的那個已經釋放的內存塊。這也就是意味值,每次總是從整個內存堆的開始的第一個空閑內存塊開始找合適的內存塊的,即便后面可能有更大的空間!
    ??這其中有個需要注意的地方:if (mem->next - (ptr + SIZEOF_STRUCT_MEM) >= (size + SIZEOF_STRUCT_MEM + MIN_SIZE_ALIGNED))。這個地方具體是干嘛的呢?看下圖(圖例不考慮對齊問題):

    ??接下來一個注意的地方就是if (mem2->next != MEM_SIZE_ALIGNED) { 省略...。這個地方又是干什么用的呢?具體看下圖(圖例不考慮對齊問題)

    ??其實上面兩個問題,就是針對中間已經釋放過的內存塊由重新劃分時的處理。一旦從釋放過的內存塊中劃分出新的內存后,則可能會導致剩余的內存塊有可能充足,也可能不充足,這就是針對以上兩個情況!
    ??最后再看看內存的釋放函數,和上面相同,仍舊從源代碼說起,具體見注釋。

    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;}LWIP_ASSERT("mem_free: sanity check alignment", (((mem_ptr_t)rmem) & (MEM_ALIGNMENT-1)) == 0);LWIP_ASSERT("mem_free: 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) { /* 檢查是否在內存堆范圍內 */SYS_ARCH_DECL_PROTECT(lev);LWIP_DEBUGF(MEM_DEBUG | LWIP_DBG_LEVEL_SEVERE, ("mem_free: illegal memory\n"));/* protect mem stats from concurrent access */SYS_ARCH_PROTECT(lev);MEM_STATS_INC(illegal);SYS_ARCH_UNPROTECT(lev);return;}/* 第二步:查找指定的內存塊,標記為未使用 *//* protect the heap from concurrent access */LWIP_MEM_FREE_PROTECT();/* Get the corresponding struct mem ... *//* cast through void* to get rid of alignment warnings */mem = (struct mem *)(void *)((u8_t *)rmem - SIZEOF_STRUCT_MEM); /* 通過mem_malloc的到的地址是不含 struct mem 的 *//* ... which has to be in a used state ... */LWIP_ASSERT("mem_free: mem->used", mem->used);/* ... and is now unused. */mem->used = 0;/* 第三步:需要移動全局的釋放指針,因為lfree始終指向內存堆中最小索引的那個已經釋放的內存塊 */if (mem < lfree) {/* the newly freed struct is now the lowest */lfree = mem;}MEM_STATS_DEC_USED(used, mem->next - (mem_size_t)(((u8_t *)mem - ram)));/* finally, see if prev or next are free also */plug_holes(mem); #if LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXTmem_free_count = 1; #endif /* LWIP_ALLOW_MEM_FREE_FROM_OTHER_CONTEXT */LWIP_MEM_FREE_UNPROTECT(); }

    釋放過程非常簡單,就是將指定的內存塊標記為未使用,其他不做任何處理。唯一處理的一個地方就就是全局變量lfree,必須檢查是否為最前面的內存塊,釋放后如下圖:

    內存塊釋放后,其原先建立的連接關系是不會改變的。
    ??這部分中還有一個函數:static void plug_holes(struct mem *mem),用來對相鄰且未用的內存塊進行合并。該函數暫時未使用,這里就不過多說明了!

    LwIP內存池

    見后文 LwIP 之 詳解動態內存管理 內存池(memp.c/h)

    總結

    以上是生活随笔為你收集整理的LwIP 之五 详解动态内存管理 内存堆(mem.c/h)的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    久久久久欧美精品 | 天天操,夜夜操 | 亚洲视频免费在线观看 | 免费看黄色91 | 国产99中文字幕 | 国产999精品久久久久久 | 精品在线视频观看 | 国产区网址 | 亚洲一区美女视频在线观看免费 | 青春草视频 | 亚洲免费高清视频 | 婷婷精品在线视频 | a视频在线 | 成年人视频在线免费播放 | 成人91视频 | 玖玖国产精品视频 | 在线观看视频在线观看 | 久久草在线视频国产 | 久久国产福利 | 九九热精品视频在线观看 | 国产盗摄精品一区二区 | a天堂中文在线 | 免费av的网站 | 免费亚洲黄色 | 91超级碰| 久久不卡视频 | 日韩高清不卡一区二区三区 | 久久综合五月天婷婷伊人 | 丝袜av一区| 国产精品亚 | 日韩伦理片一区二区三区 | 亚洲作爱| 中文字幕在线视频一区二区 | 亚洲精品免费在线 | 亚洲高清视频在线观看 | www.久久久| 在线看91| 亚洲色图激情文学 | 日本一区二区三区免费观看 | 国产一在线精品一区在线观看 | 欧美精品久久久久久久久久久 | 综合网伊人 | 最新精品国产 | 337p西西人体大胆瓣开下部 | 亚洲网站在线看 | 玖玖视频国产 | 在线观看成年人 | 五月婷婷丁香综合 | 黄色av三级在线 | 亚洲一级电影 | 美女黄视频免费 | 久久久久久久久影视 | 国产亚洲一区二区在线观看 | 中日韩欧美精彩视频 | 国产中文字幕网 | 国产精品久久久久久999 | 日韩欧美在线一区 | 五月婷香蕉久色在线看 | 97色在线观看免费视频 | 在线免费高清视频 | 亚洲成人av片 | 在线成人性视频 | 久久精品79国产精品 | 免费看黄的视频 | 国产伦精品一区二区三区高清 | 日日天天av | 永久免费精品视频网站 | 中文字幕4| 综合av在线 | 国产午夜三级 | 天天天操天天天干 | 麻豆视频在线观看免费 | 91丨九色丨国产丨porny精品 | 玖玖视频精品 | 亚洲精品字幕 | 亚洲欧美日韩一区二区三区在线观看 | 成人av免费在线 | 草久在线观看视频 | 日韩午夜视频在线观看 | 亚洲理论在线 | 国产精品成人一区 | 久久国产精品第一页 | 色婷婷在线视频 | 日韩av一区二区在线播放 | 狂野欧美激情性xxxx欧美 | 中文字幕在线观看不卡 | 亚洲精品在线免费播放 | 黄色的网站免费看 | 久久成人欧美 | 精品在线视频播放 | 国产高清日韩 | 黄色大片日本免费大片 | 日韩理论在线视频 | 91精彩视频 | 久久久精品综合 | 精品国产一区二区三区日日嗨 | 最新av免费在线观看 | 美女免费视频一区二区 | 国产破处在线播放 | av免费在线看网站 | 久久国产午夜精品理论片最新版本 | 日韩中文字幕a | 91久久丝袜国产露脸动漫 | а天堂中文最新一区二区三区 | 免费色视频 | 亚洲国内精品在线 | 96精品高清视频在线观看软件特色 | 亚洲欧美激情精品一区二区 | wwwww.国产| 亚洲黄色免费网站 | 91精品视频导航 | 狠狠干免费 | 国产69精品久久久久99尤 | 欧美黑吊大战白妞欧美 | 国产精品色婷婷视频 | 99re8这里有精品热视频免费 | 国产精品中文字幕av | 免费观看一级特黄欧美大片 | 国产精品久久久久久久午夜片 | 在线观看国产一区二区 | 日日夜夜精品免费视频 | 91麻豆产精品久久久久久 | 亚洲精选视频免费看 | 天天天操天天天干 | 中文字幕av一区二区三区四区 | 天堂av在线网址 | 国产一区二区三区黄 | 中文字幕一区二区三区四区在线视频 | 九色视频网址 | 久久综合中文字幕 | 国产黄色精品在线 | 免费a现在观看 | 日韩网站中文字幕 | 成人h动漫在线看 | 国产一区二区久久久久 | 中文字幕色播 | 五月婷婷操 | 国产日韩精品一区二区在线观看播放 | 在线免费国产视频 | 天天摸天天舔 | 国产精品v a免费视频 | 91看片在线看片 | 欧亚久久 | 日本一区二区三区视频在线播放 | 国产精品久久久久三级 | 亚州欧美视频 | 欧美一级片在线免费观看 | 免费手机黄色网址 | 成年人视频免费在线播放 | 黄色a一级片 | 国产黄网站在线观看 | a在线免费观看视频 | 国产真实在线 | 99久久精品免费看国产 | 欧美日韩中文国产一区发布 | 91自拍视频在线观看 | 婷婷综合网 | 日本特黄一级片 | 成人在线视频网 | av线上免费观看 | 精品国产成人av | 美女视频免费一区二区 | 天天天射 | 成人黄色一级视频 | 大荫蒂欧美视频另类xxxx | 国产精品大片免费观看 | 91手机在线看片 | 天天鲁一鲁摸一摸爽一爽 | 探花视频免费观看 | 国产精品99蜜臀久久不卡二区 | 成人avav| 操久在线 | 日韩欧美在线视频一区二区 | 97国产 | 久久国产免费看 | 狠狠伊人| 一二区av | av观看在线观看 | 国产96av| 日韩免费电影一区二区三区 | 国内少妇自拍视频一区 | 国内精品美女在线观看 | 成人h动漫在线看 | 91精品国产三级a在线观看 | 五月天综合色 | 欧亚久久 | 欧美日韩免费网站 | 一级免费看 | 亚洲永久精品国产 | 日日干影院 | 久久久免费精品视频 | 久久久久久久亚洲精品 | 欧美一二区在线 | 精品久久综合 | 日本黄色a级大片 | 久久久精品网站 | 人人插人人 | 成人综合日日夜夜 | 国产91aaa | 91视频 - x99av| 肉色欧美久久久久久久免费看 | 久久99精品一区二区三区三区 | 狠狠躁日日躁狂躁夜夜躁av | 成人黄色在线看 | 波多野结衣精品在线 | 91激情视频在线观看 | 夜色资源站wwwcom | 奇米777777| 国产高清久久久久 | 国产高h视频 | 精品中文字幕在线观看 | 国产成人av网 | 黄色成人在线网站 | 天堂av网址| 亚洲国产午夜视频 | 亚一亚二国产专区 | 日韩av综合网站 | 黄色在线观看免费 | av中文字幕在线观看网站 | 天堂av免费看 | 亚洲专区视频在线观看 | 996久久国产精品线观看 | 91精品色 | 精品一区二区在线看 | www.色午夜,com| 亚洲91在线| 91麻豆视频网站 | 久久深夜福利免费观看 | 日韩有码网站 | 国产精品黄色 | 国产精品区二区三区日本 | 中文字幕免费一区 | 免费黄a | 免费看国产一级片 | 日韩二区三区在线观看 | 丁香高清视频在线看看 | 国产一区精品在线 | 国产小视频在线观看免费 | 国产一级视频 | 午夜电影久久久 | 中文字幕在线日本 | 91看片在线播放 | 欧美日韩中文另类 | 中文字幕观看视频 | 久久久久久久久久伊人 | 国产精品永久免费观看 | 国内成人av | 一区二区三区四区五区在线 | 99爱视频在线观看 | 免费的国产精品 | 精品国产一区二区三区久久久久久 | 99久久精品国产网站 | 人人躁| 九九免费精品视频在线观看 | 国产在线最新 | 久久综合九色欧美综合狠狠 | 国产精品无| 91经典在线 | 日韩电影一区二区三区在线观看 | 国产在线小视频 | 一级做a爱片性色毛片www | www.天天射.com| 久久久久久久久久亚洲精品 | 欧美精品乱码久久久久久 | 日韩三级在线观看 | 99色资源| 国产一区二区精品在线 | 国产一级免费视频 | 免费一级特黄录像 | 黄在线免费看 | 伊人色综合久久天天 | 爱爱av在线| 欧美色图30p| 91av视屏| 国产精品毛片一区二区在线 | 国产成人一区二区精品非洲 | 在线国产日本 | 91精品一区二区三区蜜臀 | 日p在线观看 | 久久婷婷五月综合色丁香 | 国内精品中文字幕 | 永久免费毛片 | 超碰97在线资源 | 国产手机在线观看 | 2020天天干夜夜爽 | 久久精品国亚洲 | 操夜夜操 | av中文在线播放 | 国产精品综合久久久久 | 国产日韩一区在线 | 视频三区在线 | 九九热免费视频在线观看 | 国产99免费视频 | 精品一二 | 国产精品综合av一区二区国产馆 | 亚洲精品乱码久久久久久久久久 | 天天综合网入口 | 黄色片免费在线 | 国产亚洲精品久久久久5区 成人h电影在线观看 | 日韩xxxx视频| 天天插天天 | 中文一区二区三区在线观看 | 99资源网| 国产精品美女视频网站 | 黄色av电影一级片 | 天天干天天在线 | 伊人一级 | 国产打女人屁股调教97 | 国产精品免费小视频 | 亚洲欧美日韩国产一区二区 | 国产福利a| 97夜夜澡人人双人人人喊 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 国产精品久久久久久久久久久久 | 国产伦精品一区二区三区四区视频 | 99在线国产 | www.色国产 | 成人午夜影视 | 99精品在线视频观看 | 中文亚洲欧美日韩 | 天天草天天操 | 久久国产午夜精品理论片最新版本 | 精精国产xxxx视频在线播放 | 国产精品美女久久久久aⅴ 干干夜夜 | 色综合久久综合中文综合网 | 久久精品99国产精品酒店日本 | 国产精品自产拍在线观看网站 | 97超碰色偷偷 | 黄色成人av网址 | 伊人春色电影网 | 免费观看黄 | 在线观看一级片 | 香蕉久久久久 | 天天操天天摸天天干 | 国产视频一区二区三区在线 | 91在线观看视频网站 | 日韩av区 | 日韩在线观看三区 | 成人av亚洲 | 中文字幕在线不卡国产视频 | 日韩高清免费无专码区 | 综合激情| 日韩av不卡在线 | 91免费视频黄 | 亚洲精品色视频 | 精品国产aⅴ麻豆 | 韩国av一区二区三区 | japanesexxxhd奶水| 久久免费精彩视频 | 天天操狠狠操夜夜操 | 人人爽人人爽 | 色香com. | 国产精品第一页在线观看 | 中文字幕日韩免费视频 | 亚洲污视频 | 四虎影视成人精品国库在线观看 | 日韩欧美一区二区三区在线观看 | 香蕉视频在线播放 | 亚洲精品美女久久17c | 亚洲欧美日韩中文在线 | 婷婷丁香色综合狠狠色 | 午夜.dj高清免费观看视频 | 日本精品中文字幕在线观看 | 2021国产视频| 久久久免费视频播放 | 激情久久五月天 | 久久久久久久久久久影院 | 成人黄色大片在线免费观看 | 日本黄色黄网站 | 日韩四虎| 久久精品成人 | 国产精品久久久久9999吃药 | 美女亚洲精品 | 69国产盗摄一区二区三区五区 | 五月天激情视频 | 免费成人av | 黄色大片日本 | 丁香花在线视频观看免费 | 精品99免费 | 在线激情网| 久久男人中文字幕资源站 | 国产一区播放 | 国产精品不卡一区 | 久久视频精品在线观看 | 国产精品24小时在线观看 | 婷婷色社区 | 免费在线观看av电影 | 人人爱人人爽 | 91久久国产自产拍夜夜嗨 | 国产污视频在线观看 | 成人网在线免费视频 | 最新免费中文字幕 | 久久免费一级片 | 91视频在线免费看 | 免费在线观看国产精品 | 亚洲视频久久久久 | 国产精品免费视频久久久 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 免费看一级黄色 | 日韩在线在线 | 99爱视频在线观看 | 亚洲综合日韩在线 | 夜夜躁狠狠燥 | 狠狠干在线 | 国产精品成人品 | 国产精品精品国产色婷婷 | 久久综合免费 | 13日本xxxxxⅹxxx20| 国产中文字幕在线播放 | 69久久夜色精品国产69 | 午夜视频在线网站 | 三级动图| 婷婷5月色 | 美女在线观看网站 | 欧美精品一二三 | 国产99久久九九精品 | 免费a级毛片在线看 | 国产第页 | 中文字幕一区二区三区视频 | 精品国内自产拍在线观看视频 | 久久免费看av | 一本—道久久a久久精品蜜桃 | 五月天开心 | 国产精品福利午夜在线观看 | 国产丝袜网站 | 精品国产乱码久久久久久1区二区 | 九九免费在线观看 | 婷婷亚洲综合五月天小说 | 国产精品久久久久久久久久99 | 在线免费观看不卡av | av中文字幕网 | 欧美精品在线观看一区 | 中文字幕在 | 黄色com | 午夜成人免费影院 | 国产精品永久久久久久久www | 日日摸日日爽 | 国内亚洲精品 | 色综合久久88色综合天天6 | 国产精品18久久久久白浆 | 色网站免费在线看 | 亚洲国产中文字幕 | 久久国产精品网站 | 中文字幕a∨在线乱码免费看 | 99精品免费久久久久久久久 | 青青网视频 | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 欧美激情视频一二三区 | www.天天射.com | 欧美精品久久久久久久久久白贞 | 999国内精品永久免费视频 | 欧美日韩色婷婷 | 精品一二三四视频 | 国产精品mv在线观看 | 日本免费久久高清视频 | 久久久久久国产精品免费 | 欧美日韩高清在线一区 | 中文字幕一区二区三 | 国内精品视频久久 | 久久经典国产视频 | 欧美精品在线观看免费 | 国产精品久一 | 久久综合九色综合97_ 久久久 | 中文字幕日韩高清 | 在线中文字幕视频 | 久久国产精品99久久久久久丝袜 | 国产精品3区| 亚洲精品久久久久www | 欧美性极品xxxx娇小 | 人人爽久久久噜噜噜电影 | 精品视频久久久久久 | 美女搞黄国产视频网站 | 精品国产乱码久久 | 日韩欧三级 | 国产一区自拍视频 | 日日夜夜婷婷 | 久久午夜鲁丝片 | 国产69精品久久久久99 | 国产清纯在线 | 国产精品一区二区 91 | 午夜视频亚洲 | 97视频精品 | 欧美日韩不卡在线 | 欧美日高清视频 | 国产日韩欧美中文 | 国产美女精品视频 | 国产一区在线视频观看 | 久草香蕉在线视频 | 波多野结衣视频一区 | 中文字幕黄色网 | 国产在线传媒 | 成人影片在线免费观看 | 国产99久久久国产精品免费看 | 国产在线观看a | 十八岁免进欧美 | 亚洲韩国一区二区三区 | 探花视频在线观看免费版 | 欧美综合色在线图区 | 3d黄动漫免费看 | 久久久国产日韩 | 免费在线视频一区二区 | 国产高清一区二区 | 亚洲国产经典视频 | 国产二区免费视频 | 日韩中文字幕在线观看 | 丁香六月天婷婷 | 亚洲一区美女视频在线观看免费 | 日韩在线观看免费 | 99在线热播精品免费 | 欧美在线视频精品 | 亚洲国产影院 | 午夜国产福利视频 | 亚洲欧洲日韩 | 亚洲综合视频在线 | 欧美国产亚洲精品久久久8v | 成人av电影网址 | 国产精品乱码一区二区视频 | 人人揉人人揉人人揉人人揉97 | 国产视频午夜 | 日韩午夜高清 | 成人影视免费 | 黄色软件在线观看 | 日韩午夜精品 | 日韩久久在线 | 日韩精品免费一区二区在线观看 | www.久久久| 国产精品专区在线观看 | 黄色小网站在线 | 日韩视频一区二区三区在线播放免费观看 | 久久99精品久久久久久久久久久久 | 欧美日韩在线视频免费 | 99精品影视 | 久久99热久久99精品 | 人人爽人人爽人人 | 二区视频在线 | 黄色小网站免费看 | 美女福利视频网 | 欧美色一色 | www.亚洲视频.com | 亚洲综合色激情五月 | www.香蕉视频在线观看 | 久久艹影院| 激情小说久久 | 韩国视频一区二区三区 | 久草在线免费看视频 | 日三级在线| 午夜美女av| 国产91aaa| 天天操夜夜曰 | 国内精自线一二区永久 | 国产精品第一视频 | 久久综合免费视频影院 | 性日韩欧美在线视频 | 91视频免费视频 | 午夜av免费看 | 久久伊人热 | 在线观看亚洲视频 | 九九视频免费观看视频精品 | 婷婷成人在线 | 国产91精品久久久久久 | 美女黄频| 日韩精品免费在线视频 | 国产亚洲精品久久久久久网站 | 欧美亚洲国产日韩 | 天天干com| 91看片在线| 免费在线成人 | 九九热在线精品 | 久久av观看 | 欧美精彩视频在线观看 | 久久综合9988久久爱 | 成人免费一区二区三区在线观看 | 欧洲视频一区 | 精品久久久久国产 | 久久免费观看少妇a级毛片 久久久久成人免费 | 在线观看网站你懂的 | 超碰在线99 | 久久国产露脸精品国产 | 97精品国产91久久久久久 | 亚洲精品日韩av | 久久综合久久综合这里只有精品 | 国产精品成人一区二区三区 | 麻豆av一区二区三区在线观看 | 黄色一区二区在线观看 | 国产69精品久久99不卡的观看体验 | 国产成人久久精品77777 | 国产资源站 | 色狠狠婷婷 | 国产精品久久久久久久久久免费看 | 在线免费观看一区二区三区 | 久久精品久久精品久久精品 | 狠狠操夜夜 | 久久在线影院 | 日韩伦理片一区二区三区 | 91手机在线看片 | 色婷婷在线视频 | 在线观看免费 | 九九热免费在线视频 | 国产96在线| 久久精品这里都是精品 | 久久久夜色 | 在线免费观看av网站 | 免费在线播放黄色 | 久久新视频 | 欧美午夜理伦三级在线观看 | 蜜桃av观看 | 不卡在线一区 | 欧美精品久久久久久久久老牛影院 | 久久精品视频在线 | 国产精品高潮久久av | 婷婷免费在线视频 | 久久国产片 | 日本精品久久久久中文字幕 | 二区视频在线观看 | 久草在线观看视频免费 | 成人欧美一区二区三区在线观看 | 91麻豆精品国产午夜天堂 | 在线日韩精品视频 | 日韩久久视频 | 不卡的av在线播放 | 天天操人 | 最近中文字幕高清字幕免费mv | 免费看网站在线 | 日韩理论片在线观看 | 欧美在线久久 | 国产亚洲午夜高清国产拍精品 | 99在线免费视频观看 | 在线播放日韩av | 久久久久久久久久电影 | 日韩在线观看a | 欧洲精品一区二区 | 国产精品九九久久久久久久 | 国产视频在线一区二区 | 国产99久久九九精品免费 | 91网站免费观看 | 国产精品国产三级在线专区 | 国内精品久久久久久久久久久 | 午夜精品久久 | 福利一区二区 | 国产视频亚洲 | 国产精品igao视频网网址 | 日韩网站在线播放 | 天天操天天添 | 在线观看精品视频 | 欧美日韩在线免费观看 | 黄网站a| 久久精品中文字幕少妇 | www.超碰| 久久视频在线观看 | 人人看人人爱 | 久久综合成人网 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 91精品在线麻豆 | 黄色国产区 | 久久久午夜精品福利内容 | 波多野结依在线观看 | 天天做天天干 | 黄色a视频 | 中文字幕一区二区三区在线播放 | 欧洲亚洲女同hd | 久久99中文字幕 | av噜噜噜在线播放 | 国产一区视频导航 | 91精品一区二区三区久久久久久 | 亚洲精品国产欧美在线观看 | 手机看国产毛片 | 人人爽人人爽人人片 | 中文资源在线官网 | 中文字幕免费高 | 国产999精品久久久影片官网 | 麻豆网站免费观看 | 波多野结衣在线中文字幕 | 亚洲免费av在线 | 麻豆视频在线观看免费 | 日韩美一区二区三区 | 久久久久久草 | 国产第一页精品 | av电影在线免费 | 久视频在线播放 | 中文字幕一区二区三区四区久久 | 久久久久www | 97天天综合网 | 国产精品久久麻豆 | 久久精品99北条麻妃 | 国产精品国产亚洲精品看不卡 | 黄色网中文字幕 | 久精品视频 | 欧美怡红院视频 | 欧美久久电影 | 亚洲五月激情 | 视频一区二区三区视频 | 欧美日韩视频一区二区三区 | 激情五月五月婷婷 | 日韩字幕| 国产精品人成电影在线观看 | 九九九九精品 | 久久视频网 | 国产a免费| 日韩精品中文字幕久久臀 | 日韩sese | 久久精品视频网址 | 99在线精品视频观看 | 国产最新视频在线 | 国产专区第一页 | 色视频在线免费 | 99r精品视频在线观看 | 在线免费av网 | 又湿又紧又大又爽a视频国产 | 亚洲伦理精品 | 国产裸体bbb视频 | 久久精品女人毛片国产 | 久久成人18免费网站 | 天天添夜夜操 | 黄色美女免费网站 | 亚洲精品国产麻豆 | 国产视频精选在线 | 欧美整片sss | 香蕉视频免费看 | 日本精品久久久久久 | 久久精品99精品国产香蕉 | 日本久久综合视频 | 99免费看片| 国产视频一二区 | 久草在线资源观看 | 亚洲精品国产自产拍在线观看 | 在线免费日韩 | 天天草网站 | 欧美先锋影音 | 国产精品 视频 | 四虎影视8848dvd| 91丨精品丨蝌蚪丨白丝jk | 久草在线视频精品 | 成人福利在线 | 黄色大全免费观看 | 国产aaa免费视频 | 在线精品视频在线观看高清 | 91成人免费观看视频 | 亚洲黄色在线免费观看 | 久久久电影网站 | 色视频在线免费观看 | 香蕉在线观看视频 | 国产精品美乳一区二区免费 | 色噜噜在线观看 | 日韩,精品电影 | 久久视频在线观看中文字幕 | 中文字幕色综合网 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 免费在线观看视频a | 西西www4444大胆视频 | 国产精品久久久久久久妇 | 色瓜 | 国产美女永久免费 | 午夜久久久久久久久久久 | 伊人av综合| 人人精品| 综合精品久久久 | 色综合久久久久久久久五月 | 亚洲黄色av一区 | 91久久黄色 | 久久不卡av | 一区二区三区国产精品 | 欧美黄污视频 | 国产黄a三级| 国产精品久久久久av福利动漫 | 国产精品久久久久久久久久久久午夜片 | 亚洲免费在线看 | 二区三区中文字幕 | 欧美日韩在线精品一区二区 | 91精品久 | 中文字幕 成人 | 精品91在线 | 国色天香在线 | 免费一级片视频 | 91久久国产露脸精品国产闺蜜 | 日韩午夜一级片 | 国产精品成人在线 | www.狠狠干 | 丁香久久久| 亚洲精品国精品久久99热 | 婷婷五天天在线视频 | 国产精品99久久久久的智能播放 | 久久99久久99精品免观看粉嫩 | 92中文资源在线 | 成人av在线资源 | 国内精品久久久久 | 久久国产精品免费一区 | 色天天综合网 | 久草青青在线观看 | 久久在线一区 | 99精品视频中文字幕 | av资源中文字幕 | 亚洲色图激情文学 | 精品在线视频一区二区三区 | 在线视频麻豆 | 日韩在线小视频 | 国产原厂视频在线观看 | 人人爽人人做 | 欧美精品在线一区二区 | 国产精品视频在线观看 | 91麻豆精品国产91久久久久久 | 一区二区视频在线看 | 久草视频手机在线 | 99国产成+人+综合+亚洲 欧美 | 亚洲精品国产精品99久久 | 99久久精品免费一区 | 男女视频久久久 | 日韩欧美极品 | 91亚洲精品久久久 | 欧美精品在线观看免费 | 日日夜夜狠狠干 | 五月婷婷在线综合 | 久久精品三级 | 中文字幕日韩在线播放 | 国产专区在线播放 | 婷婷久久一区二区三区 | 国产+日韩欧美 | 人人玩人人添人人澡97 | 国产成人精品一区在线 | 久久久国产精品视频 | 精品国产成人在线 | 国产精品黄色影片导航在线观看 | 国产99久久久国产精品免费二区 | 国产系列精品av | 在线成人免费av | 在线观看av中文字幕 | 五月婷婷网站 | 韩日电影在线 | 中文字幕丝袜一区二区 | 久久亚洲免费 | 日韩av成人在线 | 我要色综合天天 | 国产精品久久久久久久午夜片 | 免费在线黄色av | 日日夜夜av| 在线91av| 天天综合色天天综合 | 久久国产精品99久久久久久老狼 | 91精品国产自产在线观看永久 | zzijzzij亚洲日本少妇熟睡 | 天天干天天操天天入 | 日b视频在线观看网址 | 久久免费美女视频 | 开心婷婷色 | 国产在线色站 | 日韩在线观看精品 | 91精品网站| 国产色视频123区 | 中文字幕亚洲高清 | 国产精品久久久久久久久久三级 | 国产一级二级在线播放 | 黄色一集片| 97国产大学生情侣酒店的特点 | 俺要去色综合狠狠 | 亚洲精品自拍视频在线观看 | 国产成人福利 | 国产亚洲欧美精品久久久久久 | 天天摸日日摸人人看 | 日韩视频一区二区三区 | 国产日韩欧美综合在线 | 久久亚洲热 | 成人av高清在线观看 | 国产黄色免费电影 | 久久综合一本 | 久久精品二区 | 亚洲h在线播放在线观看h | 波多野结衣精品在线 | 欧美性极品xxxx做受 | 欧美先锋影音 | 久久久免费少妇 | 久精品视频免费观看2 | 久久99国产精品免费 | 久久国产综合视频 | 午夜精品久久久久久久99 | 久久视频网| 91精品视频在线免费观看 | 婷婷深爱网 | 国产成人三级 | 香蕉在线影院 | 91九色porny在线 | 新版资源中文在线观看 | 日韩毛片在线一区二区毛片 | 亚洲精品中文在线资源 | 天天艹 | 免费在线观看亚洲视频 | 欧美成人免费在线 | 视频成人永久免费视频 | 精品国产乱码久久久久久1区二区 | 人人藻人人澡人人爽 | 日日夜夜中文字幕 | 91人人网| 日韩精品极品视频 | www.av免费| 日韩理论视频 | 91精品免费在线观看 | 狠狠综合网 | 1024手机看片国产 | 久久艹在线观看 | 热热热热热色 | 成年人黄色av | 国产一级淫片免费看 | 欧美黑人性爽 | 亚洲国产精品va在线看黑人动漫 | 96视频免费在线观看 | 亚洲自拍自偷 | 亚洲视频在线观看免费 | 精品播放| 日韩在线视频网站 | 视频二区 | 五月综合激情网 | 国产一区二区三区高清播放 | 香蕉色综合 | 在线观看中文字幕第一页 | 99久久精品午夜一区二区小说 | 99精品热视频 | 久草网首页 | 亚洲一区二区黄色 | 99久久国产免费,99久久国产免费大片 | 久久久亚洲精品 | 天天色天天搞 | 一区二区三区久久精品 | 国模视频一区二区三区 | 狠狠干免费 | 久久99视频免费观看 | 久久久久久久久久久久亚洲 | 亚洲三级精品 | 97国产视频| 婷婷亚洲综合五月天小说 | 国产精品激情偷乱一区二区∴ | 日日干精品 | 91视频 - x99av | 欧美日韩国产xxx | 天堂va在线高清一区 | 天天做天天爱天天爽综合网 | 久久中文网 | 亚洲精品啊啊啊 | 欧美日韩一区二区三区不卡 | 久草在线最新 | 亚洲少妇xxxx | 五月天婷婷免费视频 | 久久天天操 | 深夜免费网站 | 在线欧美最极品的av | 国产精品美女久久久久久网站 | 一区二区中文字幕在线播放 | 国产精品免费观看在线 | 欧美日韩国产免费视频 | 精品国产一区二区三区男人吃奶 | 国产精品99久久免费黑人 | 91资源在线观看 | 在线看av的网址 | 99热网站| 国产视频精品久久 | 在线观看mv的中文字幕网站 | 日本在线观看一区 | 亚洲综合在线五月 | 不卡av电影在线观看 | 在线观看黄网站 | 久久影院午夜论 | 日韩极品视频在线观看 | 久久久久福利视频 | 91最新国产 | 亚洲成人免费在线观看 | 久久精品中文字幕一区二区三区 | 成人免费网视频 | 亚洲精品国偷自产在线91正片 | 国产精品视屏 | 色综合狠狠干 | 亚洲视频六区 | 国产精选在线观看 | 成人免费一区二区三区在线观看 | 国内揄拍国内精品 | 亚洲成人资源 | 97视频入口免费观看 | 成年人在线看片 | www.亚洲精品在线 | 在线视频国产区 | 91视频成人免费 | 国产精品免费久久 | 久久伊人八月婷婷综合激情 | 日日干日日 | 中日韩欧美精彩视频 | 欧美精品三级在线观看 | 日本黄色免费播放 | 国产午夜精品一区二区三区嫩草 | 免费观看全黄做爰大片国产 | h网站免费在线观看 | 日韩在线视频在线观看 | 97在线视频免费看 | 日韩av中文字幕在线免费观看 | 色爱区综合激月婷婷 | 草久在线观看视频 | 国产小视频你懂的在线 | av性在线| 天天操天天操天天操天天操天天操天天操 | 粉嫩av一区二区三区四区在线观看 | 国产美女网 | 视频二区在线 | 久久久久激情 |