anv_intel.c revision b8e80941
1/*
2 * Copyright © 2015 Intel Corporation
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
13 * Software.
14 *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21 * IN THE SOFTWARE.
22 */
23
24#include <assert.h>
25#include <stdbool.h>
26#include <string.h>
27#include <unistd.h>
28#include <fcntl.h>
29
30#include "anv_private.h"
31
32VkResult anv_CreateDmaBufImageINTEL(
33    VkDevice                                    _device,
34    const VkDmaBufImageCreateInfo*              pCreateInfo,
35    const VkAllocationCallbacks*                pAllocator,
36    VkDeviceMemory*                             pMem,
37    VkImage*                                    pImage)
38{
39   ANV_FROM_HANDLE(anv_device, device, _device);
40   struct anv_device_memory *mem;
41   struct anv_image *image;
42   VkResult result;
43   VkImage image_h;
44
45   assert(pCreateInfo->sType == VK_STRUCTURE_TYPE_DMA_BUF_IMAGE_CREATE_INFO_INTEL);
46
47   mem = vk_alloc2(&device->alloc, pAllocator, sizeof(*mem), 8,
48                    VK_SYSTEM_ALLOCATION_SCOPE_OBJECT);
49   if (mem == NULL)
50      return vk_error(VK_ERROR_OUT_OF_HOST_MEMORY);
51
52   result = anv_image_create(_device,
53      &(struct anv_image_create_info) {
54         .isl_tiling_flags = ISL_TILING_X_BIT,
55         .stride = pCreateInfo->strideInBytes,
56         .vk_info =
57      &(VkImageCreateInfo) {
58         .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
59         .imageType = VK_IMAGE_TYPE_2D,
60         .format = pCreateInfo->format,
61         .extent = pCreateInfo->extent,
62         .mipLevels = 1,
63         .arrayLayers = 1,
64         .samples = 1,
65         /* FIXME: Need a way to use X tiling to allow scanout */
66         .tiling = VK_IMAGE_TILING_OPTIMAL,
67         .usage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
68         .flags = 0,
69      }},
70      pAllocator, &image_h);
71   if (result != VK_SUCCESS)
72      goto fail;
73
74   image = anv_image_from_handle(image_h);
75
76   uint64_t bo_flags = ANV_BO_EXTERNAL;
77   if (device->instance->physicalDevice.supports_48bit_addresses)
78      bo_flags |= EXEC_OBJECT_SUPPORTS_48B_ADDRESS;
79   if (device->instance->physicalDevice.use_softpin)
80      bo_flags |= EXEC_OBJECT_PINNED;
81
82   result = anv_bo_cache_import(device, &device->bo_cache,
83                                pCreateInfo->fd, bo_flags, &mem->bo);
84   if (result != VK_SUCCESS)
85      goto fail_import;
86
87   VkDeviceSize aligned_image_size = align_u64(image->size, 4096);
88
89   if (mem->bo->size < aligned_image_size) {
90      result = vk_errorf(device->instance, device,
91                         VK_ERROR_INVALID_EXTERNAL_HANDLE,
92                         "dma-buf too small for image in "
93                         "vkCreateDmaBufImageINTEL: %"PRIu64"B < %"PRIu64"B",
94                         mem->bo->size, aligned_image_size);
95      anv_bo_cache_release(device, &device->bo_cache, mem->bo);
96      goto fail_import;
97   }
98
99   image->planes[0].address = (struct anv_address) {
100      .bo = mem->bo,
101      .offset = 0,
102   };
103
104   assert(image->extent.width > 0);
105   assert(image->extent.height > 0);
106   assert(image->extent.depth == 1);
107
108   *pMem = anv_device_memory_to_handle(mem);
109   *pImage = anv_image_to_handle(image);
110
111   close(pCreateInfo->fd);
112
113   return VK_SUCCESS;
114
115 fail_import:
116   vk_free2(&device->alloc, pAllocator, image);
117
118 fail:
119   vk_free2(&device->alloc, pAllocator, mem);
120
121   return result;
122}
123