• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2019 The Dawn Authors
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #include "dawn_native/BuddyAllocator.h"
16 
17 #include "common/Assert.h"
18 #include "common/Math.h"
19 
20 namespace dawn_native {
21 
BuddyAllocator(uint64_t maxSize)22     BuddyAllocator::BuddyAllocator(uint64_t maxSize) : mMaxBlockSize(maxSize) {
23         ASSERT(IsPowerOfTwo(maxSize));
24 
25         mFreeLists.resize(Log2(mMaxBlockSize) + 1);
26 
27         // Insert the level0 free block.
28         mRoot = new BuddyBlock(maxSize, /*offset*/ 0);
29         mFreeLists[0] = {mRoot};
30     }
31 
~BuddyAllocator()32     BuddyAllocator::~BuddyAllocator() {
33         if (mRoot) {
34             DeleteBlock(mRoot);
35         }
36     }
37 
ComputeTotalNumOfFreeBlocksForTesting() const38     uint64_t BuddyAllocator::ComputeTotalNumOfFreeBlocksForTesting() const {
39         return ComputeNumOfFreeBlocks(mRoot);
40     }
41 
ComputeNumOfFreeBlocks(BuddyBlock * block) const42     uint64_t BuddyAllocator::ComputeNumOfFreeBlocks(BuddyBlock* block) const {
43         if (block->mState == BlockState::Free) {
44             return 1;
45         } else if (block->mState == BlockState::Split) {
46             return ComputeNumOfFreeBlocks(block->split.pLeft) +
47                    ComputeNumOfFreeBlocks(block->split.pLeft->pBuddy);
48         }
49         return 0;
50     }
51 
ComputeLevelFromBlockSize(uint64_t blockSize) const52     uint32_t BuddyAllocator::ComputeLevelFromBlockSize(uint64_t blockSize) const {
53         // Every level in the buddy system can be indexed by order-n where n = log2(blockSize).
54         // However, mFreeList zero-indexed by level.
55         // For example, blockSize=4 is Level1 if MAX_BLOCK is 8.
56         return Log2(mMaxBlockSize) - Log2(blockSize);
57     }
58 
GetNextFreeAlignedBlock(size_t allocationBlockLevel,uint64_t alignment) const59     uint64_t BuddyAllocator::GetNextFreeAlignedBlock(size_t allocationBlockLevel,
60                                                      uint64_t alignment) const {
61         ASSERT(IsPowerOfTwo(alignment));
62         // The current level is the level that corresponds to the allocation size. The free list may
63         // not contain a block at that level until a larger one gets allocated (and splits).
64         // Continue to go up the tree until such a larger block exists.
65         //
66         // Even if the block exists at the level, it cannot be used if it's offset is unaligned.
67         // When the alignment is also a power-of-two, we simply use the next free block whose size
68         // is greater than or equal to the alignment value.
69         //
70         //  After one 8-byte allocation:
71         //
72         //  Level          --------------------------------
73         //      0       32 |               S              |
74         //                 --------------------------------
75         //      1       16 |       S       |       F2     |       S - split
76         //                 --------------------------------       F - free
77         //      2       8  |   Aa  |   F1  |              |       A - allocated
78         //                 --------------------------------
79         //
80         //  Allocate(size=8, alignment=8) will be satisfied by using F1.
81         //  Allocate(size=8, alignment=4) will be satified by using F1.
82         //  Allocate(size=8, alignment=16) will be satisified by using F2.
83         //
84         for (size_t ii = 0; ii <= allocationBlockLevel; ++ii) {
85             size_t currLevel = allocationBlockLevel - ii;
86             BuddyBlock* freeBlock = mFreeLists[currLevel].head;
87             if (freeBlock && (freeBlock->mOffset % alignment == 0)) {
88                 return currLevel;
89             }
90         }
91         return kInvalidOffset;  // No free block exists at any level.
92     }
93 
94     // Inserts existing free block into the free-list.
95     // Called by allocate upon splitting to insert a child block into a free-list.
96     // Note: Always insert into the head of the free-list. As when a larger free block at a lower
97     // level was split, there were no smaller free blocks at a higher level to allocate.
InsertFreeBlock(BuddyBlock * block,size_t level)98     void BuddyAllocator::InsertFreeBlock(BuddyBlock* block, size_t level) {
99         ASSERT(block->mState == BlockState::Free);
100 
101         // Inserted block is now the front (no prev).
102         block->free.pPrev = nullptr;
103 
104         // Old head is now the inserted block's next.
105         block->free.pNext = mFreeLists[level].head;
106 
107         // Block already in HEAD position (ex. right child was inserted first).
108         if (mFreeLists[level].head != nullptr) {
109             // Old head's previous is the inserted block.
110             mFreeLists[level].head->free.pPrev = block;
111         }
112 
113         mFreeLists[level].head = block;
114     }
115 
RemoveFreeBlock(BuddyBlock * block,size_t level)116     void BuddyAllocator::RemoveFreeBlock(BuddyBlock* block, size_t level) {
117         ASSERT(block->mState == BlockState::Free);
118 
119         if (mFreeLists[level].head == block) {
120             // Block is in HEAD position.
121             mFreeLists[level].head = mFreeLists[level].head->free.pNext;
122         } else {
123             // Block is after HEAD position.
124             BuddyBlock* pPrev = block->free.pPrev;
125             BuddyBlock* pNext = block->free.pNext;
126 
127             ASSERT(pPrev != nullptr);
128             ASSERT(pPrev->mState == BlockState::Free);
129 
130             pPrev->free.pNext = pNext;
131 
132             if (pNext != nullptr) {
133                 ASSERT(pNext->mState == BlockState::Free);
134                 pNext->free.pPrev = pPrev;
135             }
136         }
137     }
138 
Allocate(uint64_t allocationSize,uint64_t alignment)139     uint64_t BuddyAllocator::Allocate(uint64_t allocationSize, uint64_t alignment) {
140         if (allocationSize == 0 || allocationSize > mMaxBlockSize) {
141             return kInvalidOffset;
142         }
143 
144         // Compute the level
145         const uint32_t allocationSizeToLevel = ComputeLevelFromBlockSize(allocationSize);
146 
147         ASSERT(allocationSizeToLevel < mFreeLists.size());
148 
149         uint64_t currBlockLevel = GetNextFreeAlignedBlock(allocationSizeToLevel, alignment);
150 
151         // Error when no free blocks exist (allocator is full)
152         if (currBlockLevel == kInvalidOffset) {
153             return kInvalidOffset;
154         }
155 
156         // Split free blocks level-by-level.
157         // Terminate when the current block level is equal to the computed level of the requested
158         // allocation.
159         BuddyBlock* currBlock = mFreeLists[currBlockLevel].head;
160 
161         for (; currBlockLevel < allocationSizeToLevel; currBlockLevel++) {
162             ASSERT(currBlock->mState == BlockState::Free);
163 
164             // Remove curr block (about to be split).
165             RemoveFreeBlock(currBlock, currBlockLevel);
166 
167             // Create two free child blocks (the buddies).
168             const uint64_t nextLevelSize = currBlock->mSize / 2;
169             BuddyBlock* leftChildBlock = new BuddyBlock(nextLevelSize, currBlock->mOffset);
170             BuddyBlock* rightChildBlock =
171                 new BuddyBlock(nextLevelSize, currBlock->mOffset + nextLevelSize);
172 
173             // Remember the parent to merge these back upon de-allocation.
174             rightChildBlock->pParent = currBlock;
175             leftChildBlock->pParent = currBlock;
176 
177             // Make them buddies.
178             leftChildBlock->pBuddy = rightChildBlock;
179             rightChildBlock->pBuddy = leftChildBlock;
180 
181             // Insert the children back into the free list into the next level.
182             // The free list does not require a specific order. However, an order is specified as
183             // it's ideal to allocate lower addresses first by having the leftmost child in HEAD.
184             InsertFreeBlock(rightChildBlock, currBlockLevel + 1);
185             InsertFreeBlock(leftChildBlock, currBlockLevel + 1);
186 
187             // Curr block is now split.
188             currBlock->mState = BlockState::Split;
189             currBlock->split.pLeft = leftChildBlock;
190 
191             // Decend down into the next level.
192             currBlock = leftChildBlock;
193         }
194 
195         // Remove curr block from free-list (now allocated).
196         RemoveFreeBlock(currBlock, currBlockLevel);
197         currBlock->mState = BlockState::Allocated;
198 
199         return currBlock->mOffset;
200     }
201 
Deallocate(uint64_t offset)202     void BuddyAllocator::Deallocate(uint64_t offset) {
203         BuddyBlock* curr = mRoot;
204 
205         // TODO(crbug.com/dawn/827): Optimize de-allocation.
206         // Passing allocationSize directly will avoid the following level-by-level search;
207         // however, it requires the size information to be stored outside the allocator.
208 
209         // Search for the free block node that corresponds to the block offset.
210         size_t currBlockLevel = 0;
211         while (curr->mState == BlockState::Split) {
212             if (offset < curr->split.pLeft->pBuddy->mOffset) {
213                 curr = curr->split.pLeft;
214             } else {
215                 curr = curr->split.pLeft->pBuddy;
216             }
217 
218             currBlockLevel++;
219         }
220 
221         ASSERT(curr->mState == BlockState::Allocated);
222 
223         // Ensure the block is at the correct level
224         ASSERT(currBlockLevel == ComputeLevelFromBlockSize(curr->mSize));
225 
226         // Mark curr free so we can merge.
227         curr->mState = BlockState::Free;
228 
229         // Merge the buddies (LevelN-to-Level0).
230         while (currBlockLevel > 0 && curr->pBuddy->mState == BlockState::Free) {
231             // Remove the buddy.
232             RemoveFreeBlock(curr->pBuddy, currBlockLevel);
233 
234             BuddyBlock* parent = curr->pParent;
235 
236             // The buddies were inserted in a specific order but
237             // could be deleted in any order.
238             DeleteBlock(curr->pBuddy);
239             DeleteBlock(curr);
240 
241             // Parent is now free.
242             parent->mState = BlockState::Free;
243 
244             // Ascend up to the next level (parent block).
245             curr = parent;
246             currBlockLevel--;
247         }
248 
249         InsertFreeBlock(curr, currBlockLevel);
250     }
251 
252     // Helper which deletes a block in the tree recursively (post-order).
DeleteBlock(BuddyBlock * block)253     void BuddyAllocator::DeleteBlock(BuddyBlock* block) {
254         ASSERT(block != nullptr);
255 
256         if (block->mState == BlockState::Split) {
257             // Delete the pair in same order we inserted.
258             DeleteBlock(block->split.pLeft->pBuddy);
259             DeleteBlock(block->split.pLeft);
260         }
261         delete block;
262     }
263 
264 }  // namespace dawn_native
265