1 // Copyright 2018 The SwiftShader Authors. All Rights Reserved.
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 "VkDescriptorPool.hpp"
16
17 #include "VkDescriptorSet.hpp"
18 #include "VkDescriptorSetLayout.hpp"
19
20 #include <algorithm>
21 #include <memory>
22
23 namespace {
24
asMemory(VkDescriptorSet descriptorSet)25 inline uint8_t *asMemory(VkDescriptorSet descriptorSet)
26 {
27 return reinterpret_cast<uint8_t *>(vk::Cast(descriptorSet));
28 }
29
30 } // anonymous namespace
31
32 namespace vk {
33
DescriptorPool(const VkDescriptorPoolCreateInfo * pCreateInfo,void * mem)34 DescriptorPool::DescriptorPool(const VkDescriptorPoolCreateInfo *pCreateInfo, void *mem)
35 : pool(static_cast<uint8_t *>(mem))
36 , poolSize(ComputeRequiredAllocationSize(pCreateInfo))
37 {
38 }
39
destroy(const VkAllocationCallbacks * pAllocator)40 void DescriptorPool::destroy(const VkAllocationCallbacks *pAllocator)
41 {
42 vk::deallocate(pool, pAllocator);
43 }
44
ComputeRequiredAllocationSize(const VkDescriptorPoolCreateInfo * pCreateInfo)45 size_t DescriptorPool::ComputeRequiredAllocationSize(const VkDescriptorPoolCreateInfo *pCreateInfo)
46 {
47 size_t size = pCreateInfo->maxSets * sw::align(sizeof(DescriptorSetHeader), 16);
48
49 for(uint32_t i = 0; i < pCreateInfo->poolSizeCount; i++)
50 {
51 size += pCreateInfo->pPoolSizes[i].descriptorCount *
52 sw::align(DescriptorSetLayout::GetDescriptorSize(pCreateInfo->pPoolSizes[i].type), 16);
53 }
54
55 return size;
56 }
57
allocateSets(uint32_t descriptorSetCount,const VkDescriptorSetLayout * pSetLayouts,VkDescriptorSet * pDescriptorSets)58 VkResult DescriptorPool::allocateSets(uint32_t descriptorSetCount, const VkDescriptorSetLayout *pSetLayouts, VkDescriptorSet *pDescriptorSets)
59 {
60 // FIXME (b/119409619): use an allocator here so we can control all memory allocations
61 std::unique_ptr<size_t[]> layoutSizes(new size_t[descriptorSetCount]);
62 for(uint32_t i = 0; i < descriptorSetCount; i++)
63 {
64 pDescriptorSets[i] = VK_NULL_HANDLE;
65 layoutSizes[i] = vk::Cast(pSetLayouts[i])->getDescriptorSetAllocationSize();
66 }
67
68 VkResult result = allocateSets(&(layoutSizes[0]), descriptorSetCount, pDescriptorSets);
69 if(result == VK_SUCCESS)
70 {
71 for(uint32_t i = 0; i < descriptorSetCount; i++)
72 {
73 vk::Cast(pSetLayouts[i])->initialize(vk::Cast(pDescriptorSets[i]));
74 }
75 }
76 return result;
77 }
78
findAvailableMemory(size_t size)79 uint8_t *DescriptorPool::findAvailableMemory(size_t size)
80 {
81 if(nodes.empty())
82 {
83 return pool;
84 }
85
86 // First, look for space at the end of the pool
87 const auto itLast = nodes.rbegin();
88 ptrdiff_t itemStart = itLast->set - pool;
89 ptrdiff_t nextItemStart = itemStart + itLast->size;
90 size_t freeSpace = poolSize - nextItemStart;
91 if(freeSpace >= size)
92 {
93 return pool + nextItemStart;
94 }
95
96 // Second, look for space at the beginning of the pool
97 const auto itBegin = nodes.begin();
98 freeSpace = itBegin->set - pool;
99 if(freeSpace >= size)
100 {
101 return pool;
102 }
103
104 // Finally, look between existing pool items
105 const auto itEnd = nodes.end();
106 auto nextIt = itBegin;
107 ++nextIt;
108 for(auto it = itBegin; nextIt != itEnd; ++it, ++nextIt)
109 {
110 uint8_t *freeSpaceStart = it->set + it->size;
111 freeSpace = nextIt->set - freeSpaceStart;
112 if(freeSpace >= size)
113 {
114 return freeSpaceStart;
115 }
116 }
117
118 return nullptr;
119 }
120
allocateSets(size_t * sizes,uint32_t numAllocs,VkDescriptorSet * pDescriptorSets)121 VkResult DescriptorPool::allocateSets(size_t *sizes, uint32_t numAllocs, VkDescriptorSet *pDescriptorSets)
122 {
123 size_t totalSize = 0;
124 for(uint32_t i = 0; i < numAllocs; i++)
125 {
126 totalSize += sizes[i];
127 }
128
129 if(totalSize > poolSize)
130 {
131 return VK_ERROR_OUT_OF_POOL_MEMORY;
132 }
133
134 // Attempt to allocate single chunk of memory
135 {
136 uint8_t *memory = findAvailableMemory(totalSize);
137 if(memory)
138 {
139 for(uint32_t i = 0; i < numAllocs; i++)
140 {
141 pDescriptorSets[i] = *(new(memory) DescriptorSet());
142 nodes.insert(Node(memory, sizes[i]));
143 memory += sizes[i];
144 }
145
146 return VK_SUCCESS;
147 }
148 }
149
150 // Attempt to allocate each descriptor set separately
151 for(uint32_t i = 0; i < numAllocs; i++)
152 {
153 uint8_t *memory = findAvailableMemory(sizes[i]);
154 if(memory)
155 {
156 pDescriptorSets[i] = *(new(memory) DescriptorSet());
157 }
158 else
159 {
160 // vkAllocateDescriptorSets can be used to create multiple descriptor sets. If the
161 // creation of any of those descriptor sets fails, then the implementation must
162 // destroy all successfully created descriptor set objects from this command, set
163 // all entries of the pDescriptorSets array to VK_NULL_HANDLE and return the error.
164 for(uint32_t j = 0; j < i; j++)
165 {
166 freeSet(pDescriptorSets[j]);
167 pDescriptorSets[j] = VK_NULL_HANDLE;
168 }
169 return (computeTotalFreeSize() > totalSize) ? VK_ERROR_FRAGMENTED_POOL : VK_ERROR_OUT_OF_POOL_MEMORY;
170 }
171 nodes.insert(Node(memory, sizes[i]));
172 }
173
174 return VK_SUCCESS;
175 }
176
freeSets(uint32_t descriptorSetCount,const VkDescriptorSet * pDescriptorSets)177 void DescriptorPool::freeSets(uint32_t descriptorSetCount, const VkDescriptorSet *pDescriptorSets)
178 {
179 for(uint32_t i = 0; i < descriptorSetCount; i++)
180 {
181 freeSet(pDescriptorSets[i]);
182 }
183 }
184
freeSet(const VkDescriptorSet descriptorSet)185 void DescriptorPool::freeSet(const VkDescriptorSet descriptorSet)
186 {
187 const auto itEnd = nodes.end();
188 auto it = std::find(nodes.begin(), itEnd, asMemory(descriptorSet));
189 if(it != itEnd)
190 {
191 nodes.erase(it);
192 }
193 }
194
reset()195 VkResult DescriptorPool::reset()
196 {
197 nodes.clear();
198
199 return VK_SUCCESS;
200 }
201
computeTotalFreeSize() const202 size_t DescriptorPool::computeTotalFreeSize() const
203 {
204 size_t totalFreeSize = 0;
205
206 // Compute space at the end of the pool
207 const auto itLast = nodes.rbegin();
208 totalFreeSize += poolSize - ((itLast->set - pool) + itLast->size);
209
210 // Compute space at the beginning of the pool
211 const auto itBegin = nodes.begin();
212 totalFreeSize += itBegin->set - pool;
213
214 // Finally, look between existing pool items
215 const auto itEnd = nodes.end();
216 auto nextIt = itBegin;
217 ++nextIt;
218 for(auto it = itBegin; nextIt != itEnd; ++it, ++nextIt)
219 {
220 totalFreeSize += (nextIt->set - it->set) - it->size;
221 }
222
223 return totalFreeSize;
224 }
225
226 } // namespace vk