blob: 03520461f6f623332e519b0762e3e27b197bba34 [file] [log] [blame]
David 'Digit' Turnere52e2db2019-09-09 17:58:46 +02001// Copyright 2019 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 "VkStringify.hpp"
16
17#include "System/Debug.hpp"
18
19#include <zircon/process.h>
20#include <zircon/syscalls.h>
21
22namespace zircon {
23
24class VmoExternalMemory : public vk::DeviceMemory::ExternalBase
25{
26public:
27 // Helper struct to parse the VkMemoryAllocateInfo.pNext chain and
28 // extract relevant information related to the handle type supported
29 // by this DeviceMemory::ExternalBase subclass.
30 struct AllocateInfo
31 {
32 bool importHandle = false;
33 bool exportHandle = false;
34 zx_handle_t handle = ZX_HANDLE_INVALID;
35
36 AllocateInfo() = default;
37
38 // Parse the VkMemoryAllocateInfo->pNext chain to initialize a AllocateInfo.
39 AllocateInfo(const VkMemoryAllocateInfo *pAllocateInfo)
40 {
41 const auto *extInfo = reinterpret_cast<const VkBaseInStructure *>(pAllocateInfo->pNext);
42 while(extInfo)
43 {
44 switch(extInfo->sType)
45 {
46 case VK_STRUCTURE_TYPE_TEMP_IMPORT_MEMORY_ZIRCON_HANDLE_INFO_FUCHSIA:
47 {
48 const auto *importInfo = reinterpret_cast<const VkImportMemoryZirconHandleInfoFUCHSIA *>(extInfo);
49
50 if(importInfo->handleType != VK_EXTERNAL_MEMORY_HANDLE_TYPE_TEMP_ZIRCON_VMO_BIT_FUCHSIA)
51 {
52 UNSUPPORTED("importInfo->handleType");
53 }
54 importHandle = true;
55 handle = importInfo->handle;
56 break;
57 }
58 case VK_STRUCTURE_TYPE_EXPORT_MEMORY_ALLOCATE_INFO:
59 {
60 const auto *exportInfo = reinterpret_cast<const VkExportMemoryAllocateInfo *>(extInfo);
61
62 if(exportInfo->handleTypes != VK_EXTERNAL_MEMORY_HANDLE_TYPE_TEMP_ZIRCON_VMO_BIT_FUCHSIA)
63 {
64 UNSUPPORTED("exportInfo->handleTypes");
65 }
66 exportHandle = true;
67 break;
68 }
69
70 default:
71 WARN("VkMemoryAllocateInfo->pNext sType = %s", vk::Stringify(extInfo->sType).c_str());
72 }
73 extInfo = extInfo->pNext;
74 }
75 }
76 };
77
78 static const VkExternalMemoryHandleTypeFlagBits typeFlagBit = VK_EXTERNAL_MEMORY_HANDLE_TYPE_TEMP_ZIRCON_VMO_BIT_FUCHSIA;
79
80 static bool supportsAllocateInfo(const VkMemoryAllocateInfo *pAllocateInfo)
81 {
82 AllocateInfo info(pAllocateInfo);
83 return info.importHandle || info.exportHandle;
84 }
85
86 explicit VmoExternalMemory(const VkMemoryAllocateInfo *pAllocateInfo)
87 : allocateInfo(pAllocateInfo)
88 {
89 }
90
91 ~VmoExternalMemory()
92 {
93 closeVmo();
94 }
95
96 VkResult allocate(size_t size, void **pBuffer) override
97 {
98 if(allocateInfo.importHandle)
99 {
100 // NOTE: handle ownership is passed to the VkDeviceMemory.
101 vmoHandle = allocateInfo.handle;
102 }
103 else
104 {
105 ASSERT(allocateInfo.exportHandle);
106 zx_status_t status = zx_vmo_create(size, 0, &vmoHandle);
107 if(status != ZX_OK)
108 {
109 TRACE("zx_vmo_create() returned %d", status);
110 return VK_ERROR_OUT_OF_DEVICE_MEMORY;
111 }
112 }
113
114 // Now map it directly.
115 zx_vaddr_t addr = 0;
116 zx_status_t status = zx_vmar_map(zx_vmar_root_self(),
117 ZX_VM_PERM_READ | ZX_VM_PERM_WRITE,
118 0, // vmar_offset
119 vmoHandle,
120 0, // vmo_offset
121 size,
122 &addr);
123 if(status != ZX_OK)
124 {
125 TRACE("zx_vmar_map() failed with %d", status);
126 return VK_ERROR_MEMORY_MAP_FAILED;
127 }
128 *pBuffer = reinterpret_cast<void *>(addr);
129 return VK_SUCCESS;
130 }
131
132 void deallocate(void *buffer, size_t size) override
133 {
134 zx_status_t status = zx_vmar_unmap(zx_vmar_root_self(),
135 reinterpret_cast<zx_vaddr_t>(buffer),
136 size);
137 if(status != ZX_OK)
138 {
139 TRACE("zx_vmar_unmap() failed with %d", status);
140 }
141 closeVmo();
142 }
143
144 VkExternalMemoryHandleTypeFlagBits getFlagBit() const override
145 {
146 return typeFlagBit;
147 }
148
149 VkResult exportHandle(zx_handle_t *pHandle) const override
150 {
151 if(vmoHandle == ZX_HANDLE_INVALID)
152 {
153 return VK_ERROR_INVALID_EXTERNAL_HANDLE;
154 }
155 zx_status_t status = zx_handle_duplicate(vmoHandle, ZX_RIGHT_SAME_RIGHTS, pHandle);
156 if(status != ZX_OK)
157 {
158 TRACE("zx_handle_duplicate() returned %d", status);
159 return VK_ERROR_INVALID_EXTERNAL_HANDLE;
160 }
161 return VK_SUCCESS;
162 }
163
164private:
165 void closeVmo()
166 {
167 if(vmoHandle != ZX_HANDLE_INVALID)
168 {
169 zx_handle_close(vmoHandle);
170 vmoHandle = ZX_HANDLE_INVALID;
171 }
172 }
173
174 zx_handle_t vmoHandle = ZX_HANDLE_INVALID;
175 AllocateInfo allocateInfo;
176};
177
178} // namespace zircon