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