mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-05-13 10:51:35 +00:00
Bug: dawn:824 Change-Id: I01dda88caaf613092541b62ea1b8d92768d405e9 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/78724 Auto-Submit: Corentin Wallez <cwallez@chromium.org> Reviewed-by: Austin Eng <enga@chromium.org> Commit-Queue: Austin Eng <enga@chromium.org>
195 lines
7.3 KiB
C++
195 lines
7.3 KiB
C++
// Copyright 2017 The Dawn Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
#ifndef COMMON_VULKANPLATFORM_H_
|
|
#define COMMON_VULKANPLATFORM_H_
|
|
|
|
#if !defined(DAWN_ENABLE_BACKEND_VULKAN)
|
|
# error "vulkan_platform.h included without the Vulkan backend enabled"
|
|
#endif
|
|
#if defined(VULKAN_CORE_H_)
|
|
# error "vulkan.h included before vulkan_platform.h"
|
|
#endif
|
|
|
|
#include "common/Platform.h"
|
|
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
|
|
// vulkan.h defines non-dispatchable handles to opaque pointers on 64bit architectures and uint64_t
|
|
// on 32bit architectures. This causes a problem in 32bit where the handles cannot be used to
|
|
// distinguish between overloads of the same function.
|
|
// Change the definition of non-dispatchable handles to be opaque structures containing a uint64_t
|
|
// and overload the comparison operators between themselves and VK_NULL_HANDLE (which will be
|
|
// redefined to be nullptr). This keeps the type-safety of having the handles be different types
|
|
// (like vulkan.h on 64 bit) but makes sure the types are different on 32 bit architectures.
|
|
|
|
#if defined(DAWN_PLATFORM_64_BIT)
|
|
# define DAWN_DEFINE_NATIVE_NON_DISPATCHABLE_HANDLE(object) using object = struct object##_T*;
|
|
// This function is needed because MSVC doesn't accept reinterpret_cast from uint64_t from uint64_t
|
|
// TODO(cwallez@chromium.org): Remove this once we rework vulkan_platform.h
|
|
template <typename T>
|
|
T NativeNonDispatachableHandleFromU64(uint64_t u64) {
|
|
return reinterpret_cast<T>(u64);
|
|
}
|
|
#elif defined(DAWN_PLATFORM_32_BIT)
|
|
# define DAWN_DEFINE_NATIVE_NON_DISPATCHABLE_HANDLE(object) using object = uint64_t;
|
|
template <typename T>
|
|
T NativeNonDispatachableHandleFromU64(uint64_t u64) {
|
|
return u64;
|
|
}
|
|
#else
|
|
# error "Unsupported platform"
|
|
#endif
|
|
|
|
// Define a dummy Vulkan handle for use before we include vulkan.h
|
|
DAWN_DEFINE_NATIVE_NON_DISPATCHABLE_HANDLE(VkSomeHandle)
|
|
|
|
// Find out the alignment of native handles. Logically we would use alignof(VkSomeHandleNative) so
|
|
// why bother with the wrapper struct? It turns out that on Linux Intel x86 alignof(uint64_t) is 8
|
|
// but alignof(struct{uint64_t a;}) is 4. This is because this Intel ABI doesn't say anything about
|
|
// double-word alignment so for historical reasons compilers violated the standard and use an
|
|
// alignment of 4 for uint64_t (and double) inside structures.
|
|
// See https://stackoverflow.com/questions/44877185
|
|
// One way to get the alignment inside structures of a type is to look at the alignment of it
|
|
// wrapped in a structure. Hence VkSameHandleNativeWrappe
|
|
|
|
namespace dawn::native::vulkan {
|
|
|
|
namespace detail {
|
|
template <typename T>
|
|
struct WrapperStruct {
|
|
T member;
|
|
};
|
|
|
|
template <typename T>
|
|
static constexpr size_t AlignOfInStruct = alignof(WrapperStruct<T>);
|
|
|
|
static constexpr size_t kNativeVkHandleAlignment = AlignOfInStruct<VkSomeHandle>;
|
|
static constexpr size_t kUint64Alignment = AlignOfInStruct<uint64_t>;
|
|
|
|
// Simple handle types that supports "nullptr_t" as a 0 value.
|
|
template <typename Tag, typename HandleType>
|
|
class alignas(detail::kNativeVkHandleAlignment) VkHandle {
|
|
public:
|
|
// Default constructor and assigning of VK_NULL_HANDLE
|
|
VkHandle() = default;
|
|
VkHandle(std::nullptr_t) {
|
|
}
|
|
|
|
// Use default copy constructor/assignment
|
|
VkHandle(const VkHandle<Tag, HandleType>& other) = default;
|
|
VkHandle& operator=(const VkHandle<Tag, HandleType>&) = default;
|
|
|
|
// Comparisons between handles
|
|
bool operator==(VkHandle<Tag, HandleType> other) const {
|
|
return mHandle == other.mHandle;
|
|
}
|
|
bool operator!=(VkHandle<Tag, HandleType> other) const {
|
|
return mHandle != other.mHandle;
|
|
}
|
|
|
|
// Comparisons between handles and VK_NULL_HANDLE
|
|
bool operator==(std::nullptr_t) const {
|
|
return mHandle == 0;
|
|
}
|
|
bool operator!=(std::nullptr_t) const {
|
|
return mHandle != 0;
|
|
}
|
|
|
|
// Implicit conversion to real Vulkan types.
|
|
operator HandleType() const {
|
|
return GetHandle();
|
|
}
|
|
|
|
HandleType GetHandle() const {
|
|
return mHandle;
|
|
}
|
|
|
|
HandleType& operator*() {
|
|
return mHandle;
|
|
}
|
|
|
|
static VkHandle<Tag, HandleType> CreateFromHandle(HandleType handle) {
|
|
return VkHandle{handle};
|
|
}
|
|
|
|
private:
|
|
explicit VkHandle(HandleType handle) : mHandle(handle) {
|
|
}
|
|
|
|
HandleType mHandle = 0;
|
|
};
|
|
} // namespace detail
|
|
|
|
static constexpr std::nullptr_t VK_NULL_HANDLE = nullptr;
|
|
|
|
template <typename Tag, typename HandleType>
|
|
HandleType* AsVkArray(detail::VkHandle<Tag, HandleType>* handle) {
|
|
return reinterpret_cast<HandleType*>(handle);
|
|
}
|
|
|
|
} // namespace dawn::native::vulkan
|
|
|
|
#define VK_DEFINE_NON_DISPATCHABLE_HANDLE(object) \
|
|
DAWN_DEFINE_NATIVE_NON_DISPATCHABLE_HANDLE(object) \
|
|
namespace dawn::native::vulkan { \
|
|
using object = detail::VkHandle<struct VkTag##object, ::object>; \
|
|
static_assert(sizeof(object) == sizeof(uint64_t)); \
|
|
static_assert(alignof(object) == detail::kUint64Alignment); \
|
|
static_assert(sizeof(object) == sizeof(::object)); \
|
|
static_assert(alignof(object) == detail::kNativeVkHandleAlignment); \
|
|
} // namespace dawn::native::vulkan
|
|
|
|
// Import additional parts of Vulkan that are supported on our architecture and preemptively include
|
|
// headers that vulkan.h includes that we have "undefs" for.
|
|
#if defined(DAWN_PLATFORM_WINDOWS)
|
|
# define VK_USE_PLATFORM_WIN32_KHR
|
|
# include "common/windows_with_undefs.h"
|
|
#endif // DAWN_PLATFORM_WINDOWS
|
|
|
|
#if defined(DAWN_USE_X11)
|
|
# define VK_USE_PLATFORM_XLIB_KHR
|
|
# define VK_USE_PLATFORM_XCB_KHR
|
|
# include "common/xlib_with_undefs.h"
|
|
#endif // defined(DAWN_USE_X11)
|
|
|
|
#if defined(DAWN_ENABLE_BACKEND_METAL)
|
|
# define VK_USE_PLATFORM_METAL_EXT
|
|
#endif // defined(DAWN_ENABLE_BACKEND_METAL)
|
|
|
|
#if defined(DAWN_PLATFORM_ANDROID)
|
|
# define VK_USE_PLATFORM_ANDROID_KHR
|
|
#endif // defined(DAWN_PLATFORM_ANDROID)
|
|
|
|
#if defined(DAWN_PLATFORM_FUCHSIA)
|
|
# define VK_USE_PLATFORM_FUCHSIA
|
|
#endif // defined(DAWN_PLATFORM_FUCHSIA)
|
|
|
|
// The actual inclusion of vulkan.h!
|
|
#define VK_NO_PROTOTYPES
|
|
#include <vulkan/vulkan.h>
|
|
|
|
// Redefine VK_NULL_HANDLE for better type safety where possible.
|
|
#undef VK_NULL_HANDLE
|
|
#if defined(DAWN_PLATFORM_64_BIT)
|
|
static constexpr std::nullptr_t VK_NULL_HANDLE = nullptr;
|
|
#elif defined(DAWN_PLATFORM_32_BIT)
|
|
static constexpr uint64_t VK_NULL_HANDLE = 0;
|
|
#else
|
|
# error "Unsupported platform"
|
|
#endif
|
|
|
|
#endif // COMMON_VULKANPLATFORM_H_
|