當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > 程序喵大人
[導(dǎo)讀]Understandingglibcmalloc日志:[2019-10-10]經(jīng)評(píng)論@kwdecsdn提醒,新增對(duì)「UnsortedBin中的chunks何時(shí)移至small/largechunk中」的補(bǔ)充解釋。[2019-02-06]勘誤與代碼著色優(yōu)化;[2018-05-22]內(nèi)...

Understanding glibc malloc

日志

  1. [2019-10-10] 經(jīng)評(píng)論 @kwdecsdn 提醒,新增對(duì)「Unsorted Bin 中的 chunks 何時(shí)移至 small/large chunk 中」的補(bǔ)充解釋。

  2. [2019-02-06] 勘誤與代碼著色優(yōu)化;

  3. [2018-05-22] 內(nèi)容優(yōu)化與排版優(yōu)化;

  4. [2017-03-17] 優(yōu)化排版.

譯者言:

  1. [2018-05-22] 在寫(xiě)完這篇博客之后,我抽空將 glibc malloc 的源碼閱讀了一遍,并參與編撰了一篇有關(guān)分配器的綜述文獻(xiàn)1,最后我動(dòng)手實(shí)現(xiàn)了自己的分配器。當(dāng)然,這都是 17 年暑期之前的工作了。一年后的今天,我打開(kāi)這篇藏在記憶角落里的文章,看著它驚人的點(diǎn)擊量,我覺(jué)得我有必要認(rèn)真地校準(zhǔn)一下本文,從而盡量為大家提供一篇內(nèi)容正確、閱讀舒適的博文,這樣才對(duì)得起大家的厚望。在修訂過(guò)程中,為了避免令人尷尬的翻譯腔,我會(huì)盡量意譯與技術(shù)無(wú)關(guān)的文本,希望大家喜歡!

  2. [2016-07-21] 本篇文章主要完成了「Understanding glibc malloc」的翻譯工作。限于本人翻譯水平與專業(yè)技術(shù)水平(純粹為了了解內(nèi)存分配而翻),本文章必定會(huì)有很多不足之處,請(qǐng)大家見(jiàn)諒,也歡迎大家的指正!


文章目錄

  • Understanding glibc malloc

    • 5.1. Fast Bin

    • 5.2. Unsorted Bin

    • 5.3. Small Bin

    • 5.4. Large Bin

    • 5.5. Top Chunk

    • 5.6. Last Remainder Chunk

    • 4.1. Allocated chunk

    • 4.2. Free chunk

    • 3.1. Arena 的數(shù)量

    • 3.2. Multiple Arena

    • 3.3. Multiple Heaps

    • 2.1. 案例代碼

    • 2.2. 案例輸出

    • 2.2.1. 在主線程 malloc 之前

    • 2.2.2. 在主線程 malloc 之后

    • 2.2.3. 在主線程 free 之后

    • 2.2.4. 在 thread1 malloc 之前

    • 2.2.5. 在 thread1 malloc 之后

    • 2.2.6. 在 thread1 free 之后

    • 前言

    • 1. 申請(qǐng)堆的系統(tǒng)調(diào)用

    • 2. 多線程支持

    • 3. Arena

    • 4. Chunk

    • 5. Bins


前言

堆內(nèi)存(Heap Memory)是一個(gè)很有意思的領(lǐng)域。你可能和我一樣,也困惑于下述問(wèn)題很久了:

  • 如何從內(nèi)核申請(qǐng)堆內(nèi)存?

  • 誰(shuí)管理它??jī)?nèi)核、庫(kù)函數(shù),還是應(yīng)用本身?

  • 內(nèi)存管理效率怎么這么高?!

  • 堆內(nèi)存的管理效率可以進(jìn)一步提高嗎?

最近,我終于有時(shí)間去深入了解這些問(wèn)題。下面就讓我來(lái)談?wù)勎业恼{(diào)研成果。

開(kāi)源社區(qū)公開(kāi)了很多現(xiàn)成的內(nèi)存分配器(Memory Allocators,以下簡(jiǎn)稱為分配器):

  • dlmalloc – 第一個(gè)被廣泛使用的通用動(dòng)態(tài)內(nèi)存分配器;

  • ptmalloc2 – glibc 內(nèi)置分配器的原型;

  • jemalloc – FreeBSD & Firefox 所用分配器;

  • tcmalloc – Google 貢獻(xiàn)的分配器;

  • libumem – Solaris 所用分配器;

每一種分配器都宣稱自己快(fast)、可拓展(scalable)、效率高(memory efficient)!但是并非所有的分配器都適用于我們的應(yīng)用。內(nèi)存吞吐量大(memory hungry)的應(yīng)用程序,其性能很大程度上取決于分配器的性能。

在這篇文章中,我只談「glibc malloc」分配器。為了方便大家理解「glibc malloc」,我會(huì)聯(lián)系最新的源代碼。

歷史:ptmalloc2 基于 dlmalloc 開(kāi)發(fā),其引入了多線程支持,于 2006 年發(fā)布。發(fā)布之后,ptmalloc2 整合進(jìn)了 glibc 源碼,此后其所有修改都直接提交到了 glibc malloc 里。因此,ptmalloc2 的源碼和 glibc malloc 的源碼有很多不一致的地方。(譯者注:1996 年出現(xiàn)的 dlmalloc 只有一個(gè)主分配區(qū),該分配區(qū)為所有線程所爭(zhēng)用,1997 年發(fā)布的 ptmalloc 在 dlmalloc 的基礎(chǔ)上引入了非主分配區(qū)的概念。)

1. 申請(qǐng)堆的系統(tǒng)調(diào)用

我在之前的文章中提到過(guò),malloc?內(nèi)部通過(guò)?brk?或?mmap?系統(tǒng)調(diào)用向內(nèi)核申請(qǐng)堆區(qū)。

譯者注:在內(nèi)存管理領(lǐng)域,我們一般用「堆」指代用于分配動(dòng)態(tài)內(nèi)存的虛擬地址空間,而用「棧」指代用于分配靜態(tài)內(nèi)存的虛擬地址空間。具體到虛擬內(nèi)存布局(Memory Layout),堆維護(hù)在通過(guò)?brk?系統(tǒng)調(diào)用申請(qǐng)的「Heap」及通過(guò)?mmap?系統(tǒng)調(diào)用申請(qǐng)的「Memory Mapping Segment」中;而棧維護(hù)在通過(guò)匯編棧指令動(dòng)態(tài)調(diào)整的「Stack」中。在 Glibc 里,「Heap」用于分配較小的內(nèi)存及主線程使用的內(nèi)存。

下圖為 Linux 內(nèi)核 v2.6.7 之后,32 位模式下的虛擬內(nèi)存布局方式。

2. 多線程支持

Linux 的早期版本采用 dlmalloc 作為它的默認(rèn)分配器,但是因?yàn)?ptmalloc2 提供了多線程支持,所以 后來(lái) Linux 就轉(zhuǎn)而采用 ptmalloc2 了。多線程支持可以提升分配器的性能,進(jìn)而間接提升應(yīng)用的性能。

在 dlmalloc 中,當(dāng)兩個(gè)線程同時(shí)?malloc?時(shí),只有一個(gè)線程能夠訪問(wèn)臨界區(qū)(critical section)——這是因?yàn)樗芯€程共享用以緩存已釋放內(nèi)存的「空閑列表數(shù)據(jù)結(jié)構(gòu)」(freelist data structure),所以使用 dlmalloc 的多線程應(yīng)用會(huì)在?malloc?上耗費(fèi)過(guò)多時(shí)間,從而導(dǎo)致整個(gè)應(yīng)用性能的下降。

在 ptmalloc2 中,當(dāng)兩個(gè)線程同時(shí)調(diào)用?malloc?時(shí),內(nèi)存均會(huì)得以立即分配——每個(gè)線程都維護(hù)著單獨(dú)的堆,各個(gè)堆被獨(dú)立的空閑列表數(shù)據(jù)結(jié)構(gòu)管理,因此各個(gè)線程可以并發(fā)地從空閑列表數(shù)據(jù)結(jié)構(gòu)中申請(qǐng)內(nèi)存。這種為每個(gè)線程維護(hù)獨(dú)立堆與空閑列表數(shù)據(jù)結(jié)構(gòu)的行為就「per thread arena」。

2.1. 案例代碼

/* Per thread arena example. */#include #include #include #include #include
void* threadFunc(void* arg) { printf("Before malloc in thread 1\n"); getchar(); char* addr = (char*) malloc(1000); printf("After malloc and before free in thread 1\n"); getchar(); free(addr); printf("After free in thread 1\n"); getchar();}
int main() { pthread_t t1; void* s; int ret; char* addr;
printf("Welcome to per thread arena example::%d\n",getpid()); printf("Before malloc in main thread\n"); getchar(); addr = (char*) malloc(1000); printf("After malloc and before free in main thread\n"); getchar(); free(addr); printf("After free in main thread\n"); getchar(); ret = pthread_create(
本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

為增進(jìn)大家對(duì)分配器的認(rèn)識(shí),本文將對(duì)遞進(jìn)式分配器予以介紹。

關(guān)鍵字: 遞進(jìn)式分配器 指數(shù) 分配器

為增進(jìn)大家對(duì)分配器的認(rèn)識(shí),本文將對(duì)VGA分配器以及VGA分配器的工作原理予以介紹。

關(guān)鍵字: VGA 指數(shù) 分配器

為增進(jìn)大家對(duì)功率分配器的認(rèn)識(shí),本文將對(duì)功率分配器的功能、功率分配器的技術(shù)指標(biāo)予以介紹。

關(guān)鍵字: 功率分配器 指數(shù) 分配器

↓推薦關(guān)注↓前言堆內(nèi)存(HeapMemory)是一個(gè)很有意思的領(lǐng)域。你可能和我一樣,也困惑于下述問(wèn)題很久了:如何從內(nèi)核申請(qǐng)堆內(nèi)存?誰(shuí)管理它??jī)?nèi)核、庫(kù)函數(shù),還是應(yīng)用本身??jī)?nèi)存管理效率怎么這么高?!堆內(nèi)存的管理效率可以進(jìn)一步...

關(guān)鍵字: 分配器 內(nèi)存分配 BSP THREAD

關(guān)注、星標(biāo)公眾號(hào),直達(dá)精彩內(nèi)容文章來(lái)源:技術(shù)讓夢(mèng)想更偉大整理:李肖遙前言由于malloc()的源碼十分的繁瑣,并且會(huì)調(diào)用OS所提供的API,所以我不在對(duì)malloc()的源碼進(jìn)行分析了,而只是會(huì)分析malloc()的動(dòng)作...

關(guān)鍵字: 內(nèi)存分配 GROUP TE COOKIE

Understandingglibcmalloc日志:[2019-10-10]經(jīng)評(píng)論@kwdecsdn提醒,新增對(duì)「UnsortedBin中的chunks何時(shí)移至small/largechunk中」的補(bǔ)充解釋。[2019...

關(guān)鍵字: 分配器 內(nèi)存分配

  網(wǎng)絡(luò)分配器就是把一路網(wǎng)絡(luò)信號(hào)分成幾路網(wǎng)絡(luò)信號(hào),且每路信號(hào)電平相等的設(shè)備。使用分配器會(huì)有信號(hào)強(qiáng)度的損耗。2分配為4dB,3分配為6dB,4分配為8dB。分配器分成過(guò)流(電)型,和不過(guò)流型。就是

關(guān)鍵字: 分配器

  網(wǎng)絡(luò)分配器是什么   網(wǎng)絡(luò)分配器,俗稱貓”,它能把計(jì)算機(jī)的數(shù)字信號(hào)翻譯成可沿普通電話線傳送的脈沖信號(hào),而這些脈沖信號(hào)又可被線路另一端的另一個(gè)調(diào)制解調(diào)器接收,并譯成計(jì)算機(jī)可

關(guān)鍵字: 交換器 分配器

新加坡–2019年9月11日–Molex旗下的Phillips-Medisize公司今天宣布與一家全球性的制藥公司達(dá)成協(xié)議,在該協(xié)議下,Phillips-Medisize收購(gòu)了有關(guān)一種創(chuàng)新性的小片藥分配器的專有專利權(quán)歸屬...

關(guān)鍵字: 分配器 制藥公司 小片藥

內(nèi)存分配方面:堆: 操作系統(tǒng)有一個(gè)記錄空閑內(nèi)存地址的鏈表,當(dāng)系統(tǒng)收到程序的申請(qǐng)時(shí),會(huì)遍歷該鏈表,尋找第一個(gè)空間大于所申請(qǐng)空間的堆結(jié)點(diǎn),然后將該結(jié)點(diǎn)從空閑結(jié)點(diǎn)鏈表中刪 除,并將該結(jié)點(diǎn)的空間分配給程序,另

關(guān)鍵字: c++ 內(nèi)存分配

程序喵大人

185 篇文章

關(guān)注

發(fā)布文章

編輯精選

技術(shù)子站

關(guān)閉