xref: /aosp_15_r20/external/swiftshader/src/Vulkan/VkBuffer.cpp (revision 03ce13f70fcc45d86ee91b7ee4cab1936a95046e)
1*03ce13f7SAndroid Build Coastguard Worker // Copyright 2018 The SwiftShader Authors. All Rights Reserved.
2*03ce13f7SAndroid Build Coastguard Worker //
3*03ce13f7SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
4*03ce13f7SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
5*03ce13f7SAndroid Build Coastguard Worker // You may obtain a copy of the License at
6*03ce13f7SAndroid Build Coastguard Worker //
7*03ce13f7SAndroid Build Coastguard Worker //    http://www.apache.org/licenses/LICENSE-2.0
8*03ce13f7SAndroid Build Coastguard Worker //
9*03ce13f7SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*03ce13f7SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
11*03ce13f7SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*03ce13f7SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
13*03ce13f7SAndroid Build Coastguard Worker // limitations under the License.
14*03ce13f7SAndroid Build Coastguard Worker 
15*03ce13f7SAndroid Build Coastguard Worker #include "VkBuffer.hpp"
16*03ce13f7SAndroid Build Coastguard Worker 
17*03ce13f7SAndroid Build Coastguard Worker #include "VkConfig.hpp"
18*03ce13f7SAndroid Build Coastguard Worker #include "VkDeviceMemory.hpp"
19*03ce13f7SAndroid Build Coastguard Worker 
20*03ce13f7SAndroid Build Coastguard Worker #include <algorithm>
21*03ce13f7SAndroid Build Coastguard Worker #include <cstring>
22*03ce13f7SAndroid Build Coastguard Worker #include <limits>
23*03ce13f7SAndroid Build Coastguard Worker 
24*03ce13f7SAndroid Build Coastguard Worker namespace vk {
25*03ce13f7SAndroid Build Coastguard Worker 
Buffer(const VkBufferCreateInfo * pCreateInfo,void * mem)26*03ce13f7SAndroid Build Coastguard Worker Buffer::Buffer(const VkBufferCreateInfo *pCreateInfo, void *mem)
27*03ce13f7SAndroid Build Coastguard Worker     : flags(pCreateInfo->flags)
28*03ce13f7SAndroid Build Coastguard Worker     , size(pCreateInfo->size)
29*03ce13f7SAndroid Build Coastguard Worker     , usage(pCreateInfo->usage)
30*03ce13f7SAndroid Build Coastguard Worker     , sharingMode(pCreateInfo->sharingMode)
31*03ce13f7SAndroid Build Coastguard Worker {
32*03ce13f7SAndroid Build Coastguard Worker 	if(pCreateInfo->sharingMode == VK_SHARING_MODE_CONCURRENT)
33*03ce13f7SAndroid Build Coastguard Worker 	{
34*03ce13f7SAndroid Build Coastguard Worker 		queueFamilyIndexCount = pCreateInfo->queueFamilyIndexCount;
35*03ce13f7SAndroid Build Coastguard Worker 		queueFamilyIndices = reinterpret_cast<uint32_t *>(mem);
36*03ce13f7SAndroid Build Coastguard Worker 		memcpy(queueFamilyIndices, pCreateInfo->pQueueFamilyIndices, sizeof(uint32_t) * queueFamilyIndexCount);
37*03ce13f7SAndroid Build Coastguard Worker 	}
38*03ce13f7SAndroid Build Coastguard Worker 
39*03ce13f7SAndroid Build Coastguard Worker 	const auto *nextInfo = reinterpret_cast<const VkBaseInStructure *>(pCreateInfo->pNext);
40*03ce13f7SAndroid Build Coastguard Worker 	for(; nextInfo != nullptr; nextInfo = nextInfo->pNext)
41*03ce13f7SAndroid Build Coastguard Worker 	{
42*03ce13f7SAndroid Build Coastguard Worker 		if(nextInfo->sType == VK_STRUCTURE_TYPE_EXTERNAL_MEMORY_BUFFER_CREATE_INFO)
43*03ce13f7SAndroid Build Coastguard Worker 		{
44*03ce13f7SAndroid Build Coastguard Worker 			const auto *externalInfo = reinterpret_cast<const VkExternalMemoryBufferCreateInfo *>(nextInfo);
45*03ce13f7SAndroid Build Coastguard Worker 			supportedExternalMemoryHandleTypes = externalInfo->handleTypes;
46*03ce13f7SAndroid Build Coastguard Worker 		}
47*03ce13f7SAndroid Build Coastguard Worker 		else if(nextInfo->sType == VK_STRUCTURE_TYPE_BUFFER_OPAQUE_CAPTURE_ADDRESS_CREATE_INFO)
48*03ce13f7SAndroid Build Coastguard Worker 		{
49*03ce13f7SAndroid Build Coastguard Worker 			const auto *opaqueCaptureAddressInfo = reinterpret_cast<const VkBufferOpaqueCaptureAddressCreateInfo *>(nextInfo);
50*03ce13f7SAndroid Build Coastguard Worker 			opaqueCaptureAddress = opaqueCaptureAddressInfo->opaqueCaptureAddress;
51*03ce13f7SAndroid Build Coastguard Worker 		}
52*03ce13f7SAndroid Build Coastguard Worker 	}
53*03ce13f7SAndroid Build Coastguard Worker }
54*03ce13f7SAndroid Build Coastguard Worker 
destroy(const VkAllocationCallbacks * pAllocator)55*03ce13f7SAndroid Build Coastguard Worker void Buffer::destroy(const VkAllocationCallbacks *pAllocator)
56*03ce13f7SAndroid Build Coastguard Worker {
57*03ce13f7SAndroid Build Coastguard Worker 	vk::freeHostMemory(queueFamilyIndices, pAllocator);
58*03ce13f7SAndroid Build Coastguard Worker }
59*03ce13f7SAndroid Build Coastguard Worker 
ComputeRequiredAllocationSize(const VkBufferCreateInfo * pCreateInfo)60*03ce13f7SAndroid Build Coastguard Worker size_t Buffer::ComputeRequiredAllocationSize(const VkBufferCreateInfo *pCreateInfo)
61*03ce13f7SAndroid Build Coastguard Worker {
62*03ce13f7SAndroid Build Coastguard Worker 	return (pCreateInfo->sharingMode == VK_SHARING_MODE_CONCURRENT) ? sizeof(uint32_t) * pCreateInfo->queueFamilyIndexCount : 0;
63*03ce13f7SAndroid Build Coastguard Worker }
64*03ce13f7SAndroid Build Coastguard Worker 
GetMemoryRequirements(VkDeviceSize size,VkBufferUsageFlags usage)65*03ce13f7SAndroid Build Coastguard Worker const VkMemoryRequirements Buffer::GetMemoryRequirements(VkDeviceSize size, VkBufferUsageFlags usage)
66*03ce13f7SAndroid Build Coastguard Worker {
67*03ce13f7SAndroid Build Coastguard Worker 	VkMemoryRequirements memoryRequirements = {};
68*03ce13f7SAndroid Build Coastguard Worker 
69*03ce13f7SAndroid Build Coastguard Worker 	memoryRequirements.size = size;
70*03ce13f7SAndroid Build Coastguard Worker 	memoryRequirements.alignment = vk::MEMORY_REQUIREMENTS_OFFSET_ALIGNMENT;
71*03ce13f7SAndroid Build Coastguard Worker 
72*03ce13f7SAndroid Build Coastguard Worker 	if(usage & (VK_BUFFER_USAGE_STORAGE_TEXEL_BUFFER_BIT | VK_BUFFER_USAGE_UNIFORM_TEXEL_BUFFER_BIT))
73*03ce13f7SAndroid Build Coastguard Worker 	{
74*03ce13f7SAndroid Build Coastguard Worker 		memoryRequirements.alignment = std::max(memoryRequirements.alignment, vk::MIN_TEXEL_BUFFER_OFFSET_ALIGNMENT);
75*03ce13f7SAndroid Build Coastguard Worker 	}
76*03ce13f7SAndroid Build Coastguard Worker 
77*03ce13f7SAndroid Build Coastguard Worker 	if(usage & VK_BUFFER_USAGE_STORAGE_BUFFER_BIT)
78*03ce13f7SAndroid Build Coastguard Worker 	{
79*03ce13f7SAndroid Build Coastguard Worker 		memoryRequirements.alignment = std::max(memoryRequirements.alignment, vk::MIN_STORAGE_BUFFER_OFFSET_ALIGNMENT);
80*03ce13f7SAndroid Build Coastguard Worker 	}
81*03ce13f7SAndroid Build Coastguard Worker 
82*03ce13f7SAndroid Build Coastguard Worker 	if(usage & VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT)
83*03ce13f7SAndroid Build Coastguard Worker 	{
84*03ce13f7SAndroid Build Coastguard Worker 		memoryRequirements.alignment = std::max(memoryRequirements.alignment, vk::MIN_UNIFORM_BUFFER_OFFSET_ALIGNMENT);
85*03ce13f7SAndroid Build Coastguard Worker 	}
86*03ce13f7SAndroid Build Coastguard Worker 
87*03ce13f7SAndroid Build Coastguard Worker 	memoryRequirements.memoryTypeBits = vk::MEMORY_TYPE_GENERIC_BIT;
88*03ce13f7SAndroid Build Coastguard Worker 
89*03ce13f7SAndroid Build Coastguard Worker 	return memoryRequirements;
90*03ce13f7SAndroid Build Coastguard Worker }
91*03ce13f7SAndroid Build Coastguard Worker 
getMemoryRequirements() const92*03ce13f7SAndroid Build Coastguard Worker const VkMemoryRequirements Buffer::getMemoryRequirements() const
93*03ce13f7SAndroid Build Coastguard Worker {
94*03ce13f7SAndroid Build Coastguard Worker 	return GetMemoryRequirements(size, usage);
95*03ce13f7SAndroid Build Coastguard Worker }
96*03ce13f7SAndroid Build Coastguard Worker 
canBindToMemory(DeviceMemory * pDeviceMemory) const97*03ce13f7SAndroid Build Coastguard Worker bool Buffer::canBindToMemory(DeviceMemory *pDeviceMemory) const
98*03ce13f7SAndroid Build Coastguard Worker {
99*03ce13f7SAndroid Build Coastguard Worker 	return pDeviceMemory->checkExternalMemoryHandleType(supportedExternalMemoryHandleTypes);
100*03ce13f7SAndroid Build Coastguard Worker }
101*03ce13f7SAndroid Build Coastguard Worker 
bind(DeviceMemory * pDeviceMemory,VkDeviceSize pMemoryOffset)102*03ce13f7SAndroid Build Coastguard Worker void Buffer::bind(DeviceMemory *pDeviceMemory, VkDeviceSize pMemoryOffset)
103*03ce13f7SAndroid Build Coastguard Worker {
104*03ce13f7SAndroid Build Coastguard Worker 	memory = pDeviceMemory->getOffsetPointer(pMemoryOffset);
105*03ce13f7SAndroid Build Coastguard Worker }
106*03ce13f7SAndroid Build Coastguard Worker 
copyFrom(const void * srcMemory,VkDeviceSize pSize,VkDeviceSize pOffset)107*03ce13f7SAndroid Build Coastguard Worker void Buffer::copyFrom(const void *srcMemory, VkDeviceSize pSize, VkDeviceSize pOffset)
108*03ce13f7SAndroid Build Coastguard Worker {
109*03ce13f7SAndroid Build Coastguard Worker 	ASSERT((pSize + pOffset) <= size);
110*03ce13f7SAndroid Build Coastguard Worker 
111*03ce13f7SAndroid Build Coastguard Worker 	memcpy(getOffsetPointer(pOffset), srcMemory, pSize);
112*03ce13f7SAndroid Build Coastguard Worker }
113*03ce13f7SAndroid Build Coastguard Worker 
copyTo(void * dstMemory,VkDeviceSize pSize,VkDeviceSize pOffset) const114*03ce13f7SAndroid Build Coastguard Worker void Buffer::copyTo(void *dstMemory, VkDeviceSize pSize, VkDeviceSize pOffset) const
115*03ce13f7SAndroid Build Coastguard Worker {
116*03ce13f7SAndroid Build Coastguard Worker 	ASSERT((pSize + pOffset) <= size);
117*03ce13f7SAndroid Build Coastguard Worker 
118*03ce13f7SAndroid Build Coastguard Worker 	memcpy(dstMemory, getOffsetPointer(pOffset), pSize);
119*03ce13f7SAndroid Build Coastguard Worker }
120*03ce13f7SAndroid Build Coastguard Worker 
copyTo(Buffer * dstBuffer,const VkBufferCopy2KHR & pRegion) const121*03ce13f7SAndroid Build Coastguard Worker void Buffer::copyTo(Buffer *dstBuffer, const VkBufferCopy2KHR &pRegion) const
122*03ce13f7SAndroid Build Coastguard Worker {
123*03ce13f7SAndroid Build Coastguard Worker 	copyTo(dstBuffer->getOffsetPointer(pRegion.dstOffset), pRegion.size, pRegion.srcOffset);
124*03ce13f7SAndroid Build Coastguard Worker }
125*03ce13f7SAndroid Build Coastguard Worker 
fill(VkDeviceSize dstOffset,VkDeviceSize fillSize,uint32_t data)126*03ce13f7SAndroid Build Coastguard Worker void Buffer::fill(VkDeviceSize dstOffset, VkDeviceSize fillSize, uint32_t data)
127*03ce13f7SAndroid Build Coastguard Worker {
128*03ce13f7SAndroid Build Coastguard Worker 	size_t bytes = (fillSize == VK_WHOLE_SIZE) ? (size - dstOffset) : fillSize;
129*03ce13f7SAndroid Build Coastguard Worker 
130*03ce13f7SAndroid Build Coastguard Worker 	ASSERT((bytes + dstOffset) <= size);
131*03ce13f7SAndroid Build Coastguard Worker 
132*03ce13f7SAndroid Build Coastguard Worker 	uint32_t *memToWrite = static_cast<uint32_t *>(getOffsetPointer(dstOffset));
133*03ce13f7SAndroid Build Coastguard Worker 
134*03ce13f7SAndroid Build Coastguard Worker 	// Vulkan 1.1 spec: "If VK_WHOLE_SIZE is used and the remaining size of the buffer is
135*03ce13f7SAndroid Build Coastguard Worker 	//                   not a multiple of 4, then the nearest smaller multiple is used."
136*03ce13f7SAndroid Build Coastguard Worker 	for(; bytes >= 4; bytes -= 4, memToWrite++)
137*03ce13f7SAndroid Build Coastguard Worker 	{
138*03ce13f7SAndroid Build Coastguard Worker 		*memToWrite = data;
139*03ce13f7SAndroid Build Coastguard Worker 	}
140*03ce13f7SAndroid Build Coastguard Worker }
141*03ce13f7SAndroid Build Coastguard Worker 
update(VkDeviceSize dstOffset,VkDeviceSize dataSize,const void * pData)142*03ce13f7SAndroid Build Coastguard Worker void Buffer::update(VkDeviceSize dstOffset, VkDeviceSize dataSize, const void *pData)
143*03ce13f7SAndroid Build Coastguard Worker {
144*03ce13f7SAndroid Build Coastguard Worker 	ASSERT((dataSize + dstOffset) <= size);
145*03ce13f7SAndroid Build Coastguard Worker 
146*03ce13f7SAndroid Build Coastguard Worker 	memcpy(getOffsetPointer(dstOffset), pData, dataSize);
147*03ce13f7SAndroid Build Coastguard Worker }
148*03ce13f7SAndroid Build Coastguard Worker 
getOffsetPointer(VkDeviceSize offset) const149*03ce13f7SAndroid Build Coastguard Worker void *Buffer::getOffsetPointer(VkDeviceSize offset) const
150*03ce13f7SAndroid Build Coastguard Worker {
151*03ce13f7SAndroid Build Coastguard Worker 	return reinterpret_cast<uint8_t *>(memory) + offset;
152*03ce13f7SAndroid Build Coastguard Worker }
153*03ce13f7SAndroid Build Coastguard Worker 
getOpaqueCaptureAddress() const154*03ce13f7SAndroid Build Coastguard Worker uint64_t Buffer::getOpaqueCaptureAddress() const
155*03ce13f7SAndroid Build Coastguard Worker {
156*03ce13f7SAndroid Build Coastguard Worker 	return (opaqueCaptureAddress != 0) ? opaqueCaptureAddress : static_cast<uint64_t>(reinterpret_cast<uintptr_t>(memory));
157*03ce13f7SAndroid Build Coastguard Worker }
158*03ce13f7SAndroid Build Coastguard Worker 
end() const159*03ce13f7SAndroid Build Coastguard Worker uint8_t *Buffer::end() const
160*03ce13f7SAndroid Build Coastguard Worker {
161*03ce13f7SAndroid Build Coastguard Worker 	return reinterpret_cast<uint8_t *>(getOffsetPointer(size + 1));
162*03ce13f7SAndroid Build Coastguard Worker }
163*03ce13f7SAndroid Build Coastguard Worker 
164*03ce13f7SAndroid Build Coastguard Worker }  // namespace vk
165