-
Notifications
You must be signed in to change notification settings - Fork 0
/
Tiny Buddy Book.html
69 lines (50 loc) · 32 KB
/
Tiny Buddy Book.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
<!DOCTYPE html><html><head><title>Tiny Buddy Book</title><meta charset='utf-8'><link href='https://dn-maxiang.qbox.me/res-min/themes/marxico.css' rel='stylesheet'><style>
.note-content {font-family: 'Helvetica Neue', Arial, 'Hiragino Sans GB', STHeiti, 'Microsoft YaHei', 'WenQuanYi Micro Hei', SimSun, Song, sans-serif;}
</style></head><body><div id='preview-contents' class='note-content'>
<div id="wmd-preview" class="preview-content"></div>
<div id="wmd-preview-section-48433" class="wmd-preview-section preview-content">
</div><div id="wmd-preview-section-48434" class="wmd-preview-section preview-content">
<h1 id="tiny-buddy-book">Tiny Buddy Book</h1>
</div><div id="wmd-preview-section-48435" class="wmd-preview-section preview-content">
<h3 id="a-simple-implementation-of-buddy-memory-allocation">A simple implementation of Buddy memory allocation</h3>
<p></p>
</div><div id="wmd-preview-section-48436" class="wmd-preview-section preview-content">
<h3 id="the-physical-memory-abstracion"><strong>The physical memory abstracion</strong></h3>
<p>We divided whole memory space into many chunks with the same size, and define each chunk as 4KB in conventional ,which is the size of a page in x86-32 platform. Then in uCore we defien data type <code>struct Page</code>and use an array of <code>pages</code>to represent entire physical memory, our physical memory manager(<strong>pmm</strong>) is built based on this array.Another thing we should notice is that the usable physical memory is not the continuous address space in ideal, for example ,some memory is reserved for kernel,or perhaps memory chipset on motherboard lacks some chips, which lead to “memory hole” . we flag these unallocatable pages so that they will be never touched by any processes. The pages that are not flaged make up the free area.The responsibility of <strong>pmm</strong> is allocate allocatable pages from free area to process and reclaim freed memory to free area.</p>
</div><div id="wmd-preview-section-48437" class="wmd-preview-section preview-content">
<h3 id="buddy-system-as-a-pmm-for-ucore"><strong>Buddy system as a pmm for uCore</strong></h3>
<p>Based on the page structure we have described above, we use a powerful tool to manage physical memory. <br>
The traditional pmm just like <strong>first-fit</strong>,<strong>best-fit</strong>, etc. neraly all use doublely linked list to manage free pages,this cause the allocation or free algorithm is <span class="" rel="1f08ccc9cd7309ba1e756c3d9345ad9f"><span class="MathJax_SVG" id="MathJax-Element-12-Frame" role="textbox" aria-readonly="true" style="font-size: 100%; display: inline-block;"><span><img type="image/png" width="37" height="18" longdesc="__SVG__undefined" src="" style="margin-top:0;margin-bottom:0;"></span></span></span> in complexity at worst,if the page number is very large ,this seems too slow. <br>
Buddy memory allocation describe a series exquisite algorithms that can finish allocation and free work at worst <span class="" rel="4840badd5478c5e2fa10e45907a181df"><span class="MathJax_SVG" id="MathJax-Element-13-Frame" role="textbox" aria-readonly="true" style="font-size: 100%; display: inline-block;"><span><img type="image/png" width="62" height="18" longdesc="__SVG__undefined" src="" style="margin-top:0;margin-bottom:0;"></span></span></span> <br>
Now let’us treat the memory as a complete binary tree(CBT). Sounds a little stranger,right ? First, walk into the bottom of this CBT,we can see many tree node stand in a line,define every node as a page and are continuous in address. So these page make up entire memory space, and then go up stairs, every higher level nodes can be treated as a root of subtree, this subtree owns some pages in his bottom, and this is a subset of memory space,the higher level it is, the more pages it owns, the root of CBT owns whole memory space.Thus we treat each node as a memory block and each block is exactly power of 2 pages in size.The reason why we use CBT to represent memory map is that we can use binary search in allocation algorithm, it’ll bring <span class="" rel="4840badd5478c5e2fa10e45907a181df"><span class="MathJax_SVG" id="MathJax-Element-14-Frame" role="textbox" aria-readonly="true" style="font-size: 100%; display: inline-block;"><span><img type="image/png" width="62" height="18" longdesc="__SVG__undefined" src="" style="margin-top:0;margin-bottom:0;"></span></span></span> in complexity. </p>
<ul><li><p><strong>Buddy Data Structure</strong> <br>
It is time to define the datastrcture of this CBT,before to do this , we need to make clear how to measure size of memory. Block size are be measured using term <code>order</code>, a block with order <code>o</code> means this block have <span class="" rel="c4ac6abb3be2ee9441487b37b7baf1c7"><span class="MathJax_SVG" id="MathJax-Element-15-Frame" role="textbox" aria-readonly="true" style="font-size: 100%; display: inline-block;"><span><img type="image/png" width="35" height="18" longdesc="__SVG__undefined" src="" style="margin-top:0;margin-bottom:0;"></span></span></span> continuous pages, the start address of this block is start address of first block. Every node we called it a buddy, a buddy is just a unsigned short which separate into three field as follows.</p>
<table>
<thead>
<tr>
<th align="left">order</th>
<th align="right">left</th>
<th align="center">right</th>
</tr>
</thead>
</table>
<ul>
<li>order : The order of this block.</li>
<li>left : The allocable memory of left tree, measure in order</li>
<li>right : The allocable memory of right tree, measure in order</li></ul>
<p>Each field is 5 bit width, the most significant bit is not used at present. We can extract this info through bit-wise operation, and are defined as macros in <code>tiny_buddy.h</code>. <br>
The CBT is an array of buddys, which is defined as <code>Buddy *mem</code> in uCore .We need a continuous memory reserved for Buddy system.</p></li>
<li><strong>The allocation algorithm</strong> <br>
The algorithm is just like binary search. Given the number of pages need to allocated , calculate the minimum order that can satisfy requirement.Then start from root,if the left allocable satisfy request then go left,else if the right allocable satisfy request then go right, if neither left nor right could satisfy request, then the allocation failed and return <code>NULL</code>,otherwise it will finally stop at a buddy. and this buddy’s order must equal allocation order, it is what we want. Attention! Don’t forget flag this buddy as unallocatable and update its ancestor’s allocable status.<code>struct Page* buddy_alloc_pages(size_t n)</code>do this work in uCore, return value is the head page of a memory block, and the page’s property flag should be set, the property should be assigned to n, which means the next following continous pages are allocated.</li>
<li><strong>The free algorithm</strong> <br>
Given an allocated block,first we should translate physical address to buddy index,then modify buddy’s field. Next flag this buddy as allocatable and update ancestors’ allocatable status. Because of using of CBT,free block merge has finished implicitly.<code>void buddy_free_pages(struct Page* page)</code> do this work in uCore, the argument page is the header page of memory block need to be freed, the page’s property flag should be clean and property should be assigned to zero. </li>
<li><p><strong>The initialization algorithm</strong> <br>
Initialization is divided into 2 steps.</p>
<ul>
<li>Flag all pages as unallocable </li>
<li>Flag usable page as allocable one by one <br>
Bootloader will provide usable continous pages to pmm, once we flag one page as allocable, update his ancestor’s allocable status immediately. Each page will cost <span class="" rel="4840badd5478c5e2fa10e45907a181df"><span class="MathJax_SVG" id="MathJax-Element-16-Frame" role="textbox" aria-readonly="true" style="font-size: 100%; display: inline-block;"><span><img type="image/png" width="62" height="18" longdesc="__SVG__undefined" src="" style="margin-top:0;margin-bottom:0;"></span></span></span> to do update, The complexity of this algorithm is finally <span class="" rel="743c00b1fa7d59c887892656948b54b8"><span class="MathJax_SVG" id="MathJax-Element-17-Frame" role="textbox" aria-readonly="true" style="font-size: 100%; display: inline-block;"><span><img type="image/png" width="71" height="18" longdesc="__SVG__undefined" src="" style="margin-top:0;margin-bottom:0;"></span></span></span></li></ul>
<p><code>static void buddy_init_memmap(struct Page* page, size_t n)</code> do this work in uCore. If <code>mem_init</code> equals false, which means <code>Buddy* mem</code> has not been initialized, let <code>mem</code> point to a continous memory block with size of CBT in the front of detected memory, so these memory are be reserved for buddy system. then call <code>buddy_fill(0)</code> to flag all pages as unallocable. </p></li>
<li><strong>The update algorithm</strong> <br>
We have mentiond update many times. Now let’s see this important algorithm who guarantee a buddy’s allocable info is always right.Each buddy has its left tree and right tree, we can set buddy’s left/right field as allocable(left tree/right tree). what is allocable size of a buddy? If this block is totally free, which means no pages is allocated in this block, the allocable size is just the size of this buddy. else the allocable size is maximum of allocable(left tree ) and allocable(right tree) . When an allocation or a free happend in a buddy, the allocable size of his ancestor might be changed, update ancestor’s left and right field iteratively until reach root. <code>static void update(int i)</code>do this work in uCore.</li>
</ul></div><div id="wmd-preview-section-footnotes" class="preview-content"></div></div></body></html>