Remove the TODO on the max Wayland image size.
[platform/core/uifw/vulkan-wsi-tizen.git] / wsi / wayland / surface_properties.cpp
1 /*
2  * Copyright (c) 2017-2019, 2021 Arm Limited.
3  *
4  * SPDX-License-Identifier: MIT
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a copy
7  * of this software and associated documentation files (the "Software"), to
8  * deal in the Software without restriction, including without limitation the
9  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10  * sell copies of the Software, and to permit persons to whom the Software is
11  * furnished to do so, subject to the following conditions:
12  *
13  * The above copyright notice and this permission notice shall be included in all
14  * copies or substantial portions of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22  * SOFTWARE.
23  */
24
25 #define VK_USE_PLATFORM_WAYLAND_KHR 1
26
27 #include <wayland-client.h>
28 #include <linux-dmabuf-unstable-v1-client-protocol.h>
29
30 #include <cassert>
31 #include <cstdlib>
32 #include <algorithm>
33 #include <array>
34 #include <cstring>
35 #include "surface_properties.hpp"
36 #include "surface.hpp"
37 #include "layer/private_data.hpp"
38 #include "wl_helpers.hpp"
39 #include "wl_object_owner.hpp"
40 #include "util/drm/drm_utils.hpp"
41 #include "util/log.hpp"
42
43 #define NELEMS(x) (sizeof(x) / sizeof(x[0]))
44
45 namespace wsi
46 {
47 namespace wayland
48 {
49
50 surface_properties::surface_properties(surface &wsi_surface, const util::allocator &allocator)
51    : specific_surface(&wsi_surface)
52    , supported_formats(allocator)
53 {
54 }
55
56 surface_properties::surface_properties()
57    : specific_surface(nullptr)
58    , supported_formats(util::allocator::get_generic())
59 {
60 }
61
62 surface_properties &surface_properties::get_instance()
63 {
64    static surface_properties instance;
65    return instance;
66 }
67
68 VkResult surface_properties::get_surface_capabilities(VkPhysicalDevice physical_device, VkSurfaceKHR surface,
69                                                       VkSurfaceCapabilitiesKHR *pSurfaceCapabilities)
70 {
71    /* Image count limits */
72    pSurfaceCapabilities->minImageCount = 2;
73    pSurfaceCapabilities->maxImageCount = MAX_SWAPCHAIN_IMAGE_COUNT;
74
75    /* Surface extents */
76    pSurfaceCapabilities->currentExtent = { 0xffffffff, 0xffffffff };
77    pSurfaceCapabilities->minImageExtent = { 1, 1 };
78
79    VkPhysicalDeviceProperties dev_props;
80    layer::instance_private_data::get(physical_device).disp.GetPhysicalDeviceProperties(physical_device, &dev_props);
81
82    pSurfaceCapabilities->maxImageExtent = { dev_props.limits.maxImageDimension2D,
83                                             dev_props.limits.maxImageDimension2D };
84    pSurfaceCapabilities->maxImageArrayLayers = 1;
85
86    /* Surface transforms */
87    pSurfaceCapabilities->supportedTransforms = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
88    pSurfaceCapabilities->currentTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
89
90    /* TODO: Composite alpha */
91    pSurfaceCapabilities->supportedCompositeAlpha = static_cast<VkCompositeAlphaFlagBitsKHR>(
92       VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR | VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR |
93       VK_COMPOSITE_ALPHA_PRE_MULTIPLIED_BIT_KHR | VK_COMPOSITE_ALPHA_POST_MULTIPLIED_BIT_KHR);
94
95    /* Image usage flags */
96    pSurfaceCapabilities->supportedUsageFlags =
97       VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT | VK_IMAGE_USAGE_SAMPLED_BIT |
98       VK_IMAGE_USAGE_STORAGE_BIT | VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_INPUT_ATTACHMENT_BIT;
99
100    return VK_SUCCESS;
101 }
102
103 static VkResult get_vk_supported_formats(const util::vector<drm_format_pair> &drm_supported_formats,
104                                          vk_format_set &vk_supported_formats)
105 {
106    for (const auto &drm_format : drm_supported_formats)
107    {
108       const VkFormat vk_format = util::drm::drm_to_vk_format(drm_format.fourcc);
109       if (vk_format != VK_FORMAT_UNDEFINED)
110       {
111          auto it = vk_supported_formats.try_insert(vk_format);
112          if (!it.has_value())
113          {
114             return VK_ERROR_OUT_OF_HOST_MEMORY;
115          }
116       }
117       const VkFormat srgb_vk_format = util::drm::drm_to_vk_srgb_format(drm_format.fourcc);
118       if (srgb_vk_format != VK_FORMAT_UNDEFINED)
119       {
120          auto it = vk_supported_formats.try_insert(srgb_vk_format);
121          if (!it.has_value())
122          {
123             return VK_ERROR_OUT_OF_HOST_MEMORY;
124          }
125       }
126    }
127    return VK_SUCCESS;
128 }
129
130 VkResult surface_properties::get_surface_formats(VkPhysicalDevice physical_device, VkSurfaceKHR surface,
131                                                  uint32_t *surfaceFormatCount, VkSurfaceFormatKHR *surfaceFormats)
132 {
133    assert(specific_surface);
134    if (!supported_formats.size())
135    {
136       VkResult res = get_vk_supported_formats(specific_surface->get_formats(), supported_formats);
137       if (res != VK_SUCCESS)
138       {
139          return res;
140       }
141    }
142
143    assert(surfaceFormatCount != nullptr);
144    if (nullptr == surfaceFormats)
145    {
146       *surfaceFormatCount = supported_formats.size();
147       return VK_SUCCESS;
148    }
149
150    VkResult res = VK_SUCCESS;
151
152    if (supported_formats.size() > *surfaceFormatCount)
153    {
154       res = VK_INCOMPLETE;
155    }
156
157    uint32_t format_count = 0;
158    for (const auto &format : supported_formats)
159    {
160       if (format_count >= *surfaceFormatCount)
161       {
162          break;
163       }
164       surfaceFormats[format_count].format = format;
165       surfaceFormats[format_count++].colorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
166    }
167    *surfaceFormatCount = format_count;
168
169    return res;
170 }
171
172 VkResult surface_properties::get_surface_present_modes(VkPhysicalDevice physical_device, VkSurfaceKHR surface,
173                                                        uint32_t *pPresentModeCount, VkPresentModeKHR *pPresentModes)
174 {
175
176    VkResult res = VK_SUCCESS;
177
178    static std::array<const VkPresentModeKHR, 2> modes = {
179       VK_PRESENT_MODE_FIFO_KHR,
180       VK_PRESENT_MODE_MAILBOX_KHR,
181    };
182
183    assert(pPresentModeCount != nullptr);
184
185    if (nullptr == pPresentModes)
186    {
187       *pPresentModeCount = modes.size();
188    }
189    else
190    {
191       if (modes.size() > *pPresentModeCount)
192       {
193          res = VK_INCOMPLETE;
194       }
195       *pPresentModeCount = std::min(*pPresentModeCount, static_cast<uint32_t>(modes.size()));
196       for (uint32_t i = 0; i < *pPresentModeCount; ++i)
197       {
198          pPresentModes[i] = modes[i];
199       }
200    }
201
202    return res;
203 }
204
205 static const char *required_device_extensions[] = {
206    VK_EXT_IMAGE_DRM_FORMAT_MODIFIER_EXTENSION_NAME,
207    VK_KHR_BIND_MEMORY_2_EXTENSION_NAME,
208    VK_KHR_IMAGE_FORMAT_LIST_EXTENSION_NAME,
209    VK_KHR_SAMPLER_YCBCR_CONVERSION_EXTENSION_NAME,
210    VK_KHR_MAINTENANCE1_EXTENSION_NAME,
211    VK_KHR_GET_MEMORY_REQUIREMENTS_2_EXTENSION_NAME,
212    VK_EXT_EXTERNAL_MEMORY_DMA_BUF_EXTENSION_NAME,
213    VK_KHR_EXTERNAL_MEMORY_FD_EXTENSION_NAME,
214    VK_KHR_EXTERNAL_MEMORY_EXTENSION_NAME,
215    VK_KHR_EXTERNAL_FENCE_EXTENSION_NAME,
216    VK_KHR_EXTERNAL_FENCE_FD_EXTENSION_NAME,
217 };
218
219 VkResult surface_properties::get_required_device_extensions(util::extension_list &extension_list)
220 {
221    return extension_list.add(required_device_extensions, NELEMS(required_device_extensions));
222 }
223
224 /* TODO: Check for zwp_linux_dmabuf_v1 protocol in display */
225 VkBool32 GetPhysicalDeviceWaylandPresentationSupportKHR(VkPhysicalDevice physical_device, uint32_t queue_index,
226                                                         struct wl_display *display)
227 {
228    bool dev_supports_sync =
229       sync_fd_fence_sync::is_supported(layer::instance_private_data::get(physical_device), physical_device);
230    if (!dev_supports_sync)
231    {
232       return VK_FALSE;
233    }
234
235    return VK_TRUE;
236 }
237
238 extern "C" VkResult CreateWaylandSurfaceKHR(VkInstance instance, const VkWaylandSurfaceCreateInfoKHR *pCreateInfo,
239                                             const VkAllocationCallbacks *pAllocator, VkSurfaceKHR *pSurface)
240 {
241    auto &instance_data = layer::instance_private_data::get(instance);
242    util::allocator allocator{ instance_data.get_allocator(), VK_SYSTEM_ALLOCATION_SCOPE_OBJECT, pAllocator };
243    auto wsi_surface = surface::make_surface(allocator, pCreateInfo->display, pCreateInfo->surface);
244    if (wsi_surface == nullptr)
245    {
246       return VK_ERROR_OUT_OF_HOST_MEMORY;
247    }
248
249    VkResult res = instance_data.disp.CreateWaylandSurfaceKHR(instance, pCreateInfo, pAllocator, pSurface);
250    if (res == VK_SUCCESS)
251    {
252       auto surface_base = util::unique_ptr<wsi::surface>(std::move(wsi_surface));
253       res = instance_data.add_surface(*pSurface, surface_base);
254       if (res != VK_SUCCESS)
255       {
256          instance_data.disp.DestroySurfaceKHR(instance, *pSurface, pAllocator);
257       }
258    }
259    return res;
260 }
261
262 PFN_vkVoidFunction surface_properties::get_proc_addr(const char *name)
263 {
264    if (strcmp(name, "vkGetPhysicalDeviceWaylandPresentationSupportKHR") == 0)
265    {
266       return reinterpret_cast<PFN_vkVoidFunction>(GetPhysicalDeviceWaylandPresentationSupportKHR);
267    }
268    else if (strcmp(name, "vkCreateWaylandSurfaceKHR") == 0)
269    {
270       return reinterpret_cast<PFN_vkVoidFunction>(CreateWaylandSurfaceKHR);
271    }
272    return nullptr;
273 }
274
275 } // namespace wayland
276 } // namespace wsi