blob: f9a4670c4eb66ae1d212d314470db0e68ab202f2 [file] [log] [blame]
{{define "Copyright"}}
/*
•* Copyright 2016 The Android Open Source Project
•*
•* 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.
•*/
¶{{end}}
{{Include "../api/templates/vulkan_common.tmpl"}}
{{Global "clang-format" (Strings "clang-format" "-style=file")}}
{{Macro "DefineGlobals" $}}
{{$ | Macro "api_gen.h" | Format (Global "clang-format") | Write "api_gen.h" }}
{{$ | Macro "api_gen.cpp" | Format (Global "clang-format") | Write "api_gen.cpp"}}
{{$ | Macro "driver_gen.h" | Format (Global "clang-format") | Write "driver_gen.h"}}
{{$ | Macro "driver_gen.cpp" | Format (Global "clang-format") | Write "driver_gen.cpp"}}
{{/*
-------------------------------------------------------------------------------
api_gen.h
-------------------------------------------------------------------------------
*/}}
{{define "api_gen.h"}}
{{Macro "Copyright"}}
// WARNING: This file is generated. See ../README.md for instructions.
#ifndef LIBVULKAN_API_GEN_H
#define LIBVULKAN_API_GEN_H
#include <bitset>
#include <vulkan/vulkan.h>
#include "driver_gen.h"
namespace vulkan
namespace api
struct InstanceDispatchTable {
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "api.IsInstanceDispatchTableEntry" $f)}}
{{Macro "C++.DeclareTableEntry" $f}};
{{end}}
{{end}}
// clang-format on
};
struct DeviceDispatchTable {
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "api.IsDeviceDispatchTableEntry" $f)}}
{{Macro "C++.DeclareTableEntry" $f}};
{{end}}
{{end}}
// clang-format on
};
bool InitDispatchTable(
VkInstance instance,
PFN_vkGetInstanceProcAddr get_proc,
const std::bitset<driver::ProcHook::EXTENSION_COUNT> &extensions);
bool InitDispatchTable(
VkDevice dev,
PFN_vkGetDeviceProcAddr get_proc,
const std::bitset<driver::ProcHook::EXTENSION_COUNT> &extensions);
»} // namespace api
»} // namespace vulkan
#endif // LIBVULKAN_API_GEN_H
¶{{end}}
{{/*
-------------------------------------------------------------------------------
api_gen.cpp
-------------------------------------------------------------------------------
*/}}
{{define "api_gen.cpp"}}
{{Macro "Copyright"}}
// WARNING: This file is generated. See ../README.md for instructions.
#include <string.h>
#include <algorithm>
#include <log/log.h>
// to catch mismatches between vulkan.h and this file
#undef VK_NO_PROTOTYPES
#include "api.h"
namespace vulkan
namespace api
{{Macro "C++.DefineInitProcMacro" "dispatch"}}
{{Macro "api.C++.DefineInitProcExtMacro"}}
namespace
// clang-format off
{{range $f := AllCommands $}}
{{Macro "api.C++.DefineExtensionStub" $f}}
{{end}}
// clang-format on
»} // anonymous
bool InitDispatchTable(
VkInstance instance,
PFN_vkGetInstanceProcAddr get_proc,
const std::bitset<driver::ProcHook::EXTENSION_COUNT> &extensions) {
auto& data = GetData(instance);
bool success = true;
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "api.IsInstanceDispatchTableEntry" $f)}}
{{Macro "C++.InitProc" $f}}
{{end}}
{{end}}
// clang-format on
return success;
}
bool InitDispatchTable(
VkDevice dev,
PFN_vkGetDeviceProcAddr get_proc,
const std::bitset<driver::ProcHook::EXTENSION_COUNT> &extensions) {
auto& data = GetData(dev);
bool success = true;
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "api.IsDeviceDispatchTableEntry" $f)}}
{{Macro "C++.InitProc" $f}}
{{end}}
{{end}}
// clang-format on
return success;
}
// clang-format off
namespace
// forward declarations needed by GetInstanceProcAddr and GetDeviceProcAddr
{{range $f := AllCommands $}}
{{if and (Macro "IsFunctionExported" $f) (not (Macro "api.IsIntercepted" $f))}}
VKAPI_ATTR {{Node "Type" $f.Return}} {{Macro "BaseName" $f}}({{Macro "Parameters" $f}});
{{end}}
{{end}}
{{range $f := AllCommands $}}
{{if and (Macro "IsFunctionExported" $f) (not (Macro "api.IsIntercepted" $f))}}
VKAPI_ATTR {{Node "Type" $f.Return}} {{Macro "BaseName" $f}}({{Macro "Parameters" $f}}) {
{{ if eq $f.Name "vkGetInstanceProcAddr"}}
{{Macro "api.C++.InterceptInstanceProcAddr" $}}
{{else if eq $f.Name "vkGetDeviceProcAddr"}}
{{Macro "api.C++.InterceptDeviceProcAddr" $}}
{{end}}
{{Macro "api.C++.Dispatch" $f}}
}
{{end}}
{{end}}
»} // anonymous namespace
// clang-format on
»} // namespace api
»} // namespace vulkan
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "IsFunctionExported" $f)}}
__attribute__((visibility("default")))
VKAPI_ATTR {{Node "Type" $f.Return}} {{$f.Name}}({{Macro "Parameters" $f}}) {
{{if not (IsVoid $f.Return.Type)}}return §{{end}}
vulkan::api::{{Macro "BaseName" $f}}({{Macro "Arguments" $f}});
}
{{end}}
{{end}}
// clang-format on
¶{{end}}
{{/*
-------------------------------------------------------------------------------
driver_gen.h
-------------------------------------------------------------------------------
*/}}
{{define "driver_gen.h"}}
{{Macro "Copyright"}}
// WARNING: This file is generated. See ../README.md for instructions.
#ifndef LIBVULKAN_DRIVER_GEN_H
#define LIBVULKAN_DRIVER_GEN_H
#include <bitset>
#include <vulkan/vulkan.h>
#include <vulkan/vk_android_native_buffer.h>
namespace vulkan
namespace driver
{{Macro "driver.C++.DefineProcHookType"}}
struct InstanceDriverTable {
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "driver.IsInstanceDriverTableEntry" $f)}}
{{Macro "C++.DeclareTableEntry" $f}};
{{end}}
{{end}}
// clang-format on
};
struct DeviceDriverTable {
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "driver.IsDeviceDriverTableEntry" $f)}}
{{Macro "C++.DeclareTableEntry" $f}};
{{end}}
{{end}}
// clang-format on
};
const ProcHook* GetProcHook(const char* name);
ProcHook::Extension GetProcHookExtension(const char* name);
bool InitDriverTable(VkInstance instance, PFN_vkGetInstanceProcAddr get_proc,
const std::bitset<ProcHook::EXTENSION_COUNT> &extensions);
bool InitDriverTable(VkDevice dev, PFN_vkGetDeviceProcAddr get_proc,
const std::bitset<ProcHook::EXTENSION_COUNT> &extensions);
»} // namespace driver
»} // namespace vulkan
#endif // LIBVULKAN_DRIVER_TABLE_H
¶{{end}}
{{/*
-------------------------------------------------------------------------------
driver_gen.cpp
-------------------------------------------------------------------------------
*/}}
{{define "driver_gen.cpp"}}
{{Macro "Copyright"}}
// WARNING: This file is generated. See ../README.md for instructions.
#include <string.h>
#include <algorithm>
#include <log/log.h>
#include "driver.h"
namespace vulkan
namespace driver
namespace
// clang-format off
{{range $f := AllCommands $}}
{{Macro "driver.C++.DefineProcHookStub" $f}}
{{end}}
// clang-format on
const ProcHook g_proc_hooks[] = {
// clang-format off
{{range $f := SortBy (AllCommands $) "FunctionName"}}
{{if (Macro "driver.IsIntercepted" $f)}}
{{ if (Macro "IsGloballyDispatched" $f)}}
{{Macro "driver.C++.DefineGlobalProcHook" $f}}
{{else if (Macro "IsInstanceDispatched" $f)}}
{{Macro "driver.C++.DefineInstanceProcHook" $f}}
{{else if (Macro "IsDeviceDispatched" $f)}}
{{Macro "driver.C++.DefineDeviceProcHook" $f}}
{{end}}
{{end}}
{{end}}
// clang-format on
};
»} // anonymous
const ProcHook* GetProcHook(const char* name) {
const auto& begin = g_proc_hooks;
const auto& end = g_proc_hooks +
sizeof(g_proc_hooks) / sizeof(g_proc_hooks[0]);
const auto hook = std::lower_bound(begin, end, name,
[](const ProcHook& e, const char* n) { return strcmp(e.name, n) < 0; });
return (hook < end && strcmp(hook->name, name) == 0) ? hook : nullptr;
}
ProcHook::Extension GetProcHookExtension(const char* name) {
{{$exts := Strings (Macro "driver.InterceptedExtensions") | SplitOn "\n"}}
// clang-format off
{{range $e := $exts}}
if (strcmp(name, "{{$e}}") == 0) return ProcHook::{{TrimPrefix "VK_" $e}};
{{end}}
// clang-format on
return ProcHook::EXTENSION_UNKNOWN;
}
{{Macro "C++.DefineInitProcMacro" "driver"}}
{{Macro "driver.C++.DefineInitProcExtMacro"}}
bool InitDriverTable(VkInstance instance, PFN_vkGetInstanceProcAddr get_proc,
const std::bitset<ProcHook::EXTENSION_COUNT> &extensions)
{
auto& data = GetData(instance);
bool success = true;
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "driver.IsInstanceDriverTableEntry" $f)}}
{{Macro "C++.InitProc" $f}}
{{end}}
{{end}}
// clang-format on
return success;
}
bool InitDriverTable(VkDevice dev, PFN_vkGetDeviceProcAddr get_proc,
const std::bitset<ProcHook::EXTENSION_COUNT> &extensions)
{
auto& data = GetData(dev);
bool success = true;
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "driver.IsDeviceDriverTableEntry" $f)}}
{{Macro "C++.InitProc" $f}}
{{end}}
{{end}}
// clang-format on
return success;
}
»} // namespace driver
»} // namespace vulkan
// clang-format on
¶{{end}}
{{/*
------------------------------------------------------------------------------
Emits a declaration of a dispatch/driver table entry.
------------------------------------------------------------------------------
*/}}
{{define "C++.DeclareTableEntry"}}
{{AssertType $ "Function"}}
{{Macro "FunctionPtrName" $}} {{Macro "BaseName" $}}
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits INIT_PROC macro.
-------------------------------------------------------------------------------
*/}}
{{define "C++.DefineInitProcMacro"}}
#define UNLIKELY(expr) __builtin_expect((expr), 0)
#define INIT_PROC(obj, proc) do { \
data.{{$}}.proc = reinterpret_cast<PFN_vk ## proc>( \
get_proc(obj, "vk" # proc)); \
if (UNLIKELY(!data.{{$}}.proc)) { \
ALOGE("missing " # obj " proc: vk" # proc); \
success = false; \
} \
} while(0)
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits code to invoke INIT_PROC or INIT_PROC_EXT.
-------------------------------------------------------------------------------
*/}}
{{define "C++.InitProc"}}
{{AssertType $ "Function"}}
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
INIT_PROC_EXT({{Macro "BaseName" $ext}}, §
{{else}}
INIT_PROC
{{end}}
{{if (Macro "IsInstanceDispatched" $)}}
instance, §
{{else}}
dev, §
{{end}}
{{Macro "BaseName" $}});
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if a function is exported and instance-dispatched.
------------------------------------------------------------------------------
*/}}
{{define "api.IsInstanceDispatchTableEntry"}}
{{AssertType $ "Function"}}
{{if and (Macro "IsFunctionExported" $) (Macro "IsInstanceDispatched" $)}}
{{/* deprecated and unused internally */}}
{{if not (eq $.Name "vkEnumerateDeviceLayerProperties")}}
true
{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if a function is exported and device-dispatched.
------------------------------------------------------------------------------
*/}}
{{define "api.IsDeviceDispatchTableEntry"}}
{{AssertType $ "Function"}}
{{if and (Macro "IsFunctionExported" $) (Macro "IsDeviceDispatched" $)}}
true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if a function is intercepted by vulkan::api.
------------------------------------------------------------------------------
*/}}
{{define "api.IsIntercepted"}}
{{AssertType $ "Function"}}
{{if (Macro "IsFunctionSupported" $)}}
{{/* Global functions cannot be dispatched at all */}}
{{ if (Macro "IsGloballyDispatched" $)}}true
{{/* VkPhysicalDevice functions that manage device layers */}}
{{else if eq $.Name "vkCreateDevice"}}true
{{else if eq $.Name "vkEnumerateDeviceLayerProperties"}}true
{{else if eq $.Name "vkEnumerateDeviceExtensionProperties"}}true
{{/* Destroy functions of dispatchable objects */}}
{{else if eq $.Name "vkDestroyInstance"}}true
{{else if eq $.Name "vkDestroyDevice"}}true
{{end}}
{{end}}
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits INIT_PROC_EXT macro for vulkan::api.
-------------------------------------------------------------------------------
*/}}
{{define "api.C++.DefineInitProcExtMacro"}}
// Exported extension functions may be invoked even when their extensions
// are disabled. Dispatch to stubs when that happens.
#define INIT_PROC_EXT(ext, obj, proc) do { \
if (extensions[driver::ProcHook::ext]) \
INIT_PROC(obj, proc); \
else \
data.dispatch.proc = disabled ## proc; \
} while(0)
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits a stub for an exported extension function.
-------------------------------------------------------------------------------
*/}}
{{define "api.C++.DefineExtensionStub"}}
{{AssertType $ "Function"}}
{{$ext := GetAnnotation $ "extension"}}
{{if and $ext (Macro "IsFunctionExported" $)}}
{{$ext_name := index $ext.Arguments 0}}
{{$base := (Macro "BaseName" $)}}
{{$p0 := (index $.CallParameters 0)}}
{{$ptail := (Tail 1 $.CallParameters)}}
{{$first_type := (Macro "Parameter" $p0)}}
{{$tail_types := (ForEach $ptail "ParameterType" | JoinWith ", ")}}
VKAPI_ATTR {{Node "Type" $.Return}} disabled{{$base}}({{$first_type}}, {{$tail_types}}) {
driver::Logger({{$p0.Name}}).Err({{$p0.Name}}, §
"{{$ext_name}} not enabled. Exported {{$.Name}} not executed.");
{{if not (IsVoid $.Return.Type)}}return VK_SUCCESS;{{end}}
}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits code for vkGetInstanceProcAddr for function interception.
------------------------------------------------------------------------------
*/}}
{{define "api.C++.InterceptInstanceProcAddr"}}
{{AssertType $ "API"}}
// global functions
if (instance == VK_NULL_HANDLE) {
{{range $f := AllCommands $}}
{{if (Macro "IsGloballyDispatched" $f)}}
if (strcmp(pName, "{{$f.Name}}") == 0) return §
reinterpret_cast<PFN_vkVoidFunction>({{Macro "BaseName" $f}});
{{end}}
{{end}}
ALOGE("invalid vkGetInstanceProcAddr(VK_NULL_HANDLE, \"%s\") call", pName);
return nullptr;
}
static const struct Hook {
const char* name;
PFN_vkVoidFunction proc;
} hooks[] = {
{{range $f := SortBy (AllCommands $) "FunctionName"}}
{{if (Macro "IsFunctionExported" $f)}}
{{/* hide global functions */}}
{{if (Macro "IsGloballyDispatched" $f)}}
{ "{{$f.Name}}", nullptr },
{{/* redirect intercepted functions */}}
{{else if (Macro "api.IsIntercepted" $f)}}
{ "{{$f.Name}}", reinterpret_cast<PFN_vkVoidFunction>(§
{{Macro "BaseName" $f}}) },
{{/* redirect vkGetInstanceProcAddr to itself */}}
{{else if eq $f.Name "vkGetInstanceProcAddr"}}
{ "{{$f.Name}}", reinterpret_cast<PFN_vkVoidFunction>({{Macro "BaseName" $f}}) },
{{/* redirect device functions to themselves as a workaround for
layers that do not intercept in their vkGetInstanceProcAddr */}}
{{else if (Macro "IsDeviceDispatched" $f)}}
{ "{{$f.Name}}", reinterpret_cast<PFN_vkVoidFunction>({{Macro "BaseName" $f}}) },
{{end}}
{{end}}
{{end}}
};
// clang-format on
constexpr size_t count = sizeof(hooks) / sizeof(hooks[0]);
auto hook = std::lower_bound(
hooks, hooks + count, pName,
[](const Hook& h, const char* n) { return strcmp(h.name, n) < 0; });
if (hook < hooks + count && strcmp(hook->name, pName) == 0) {
if (!hook->proc) {
vulkan::driver::Logger(instance).Err(
instance, "invalid vkGetInstanceProcAddr(%p, \"%s\") call",
instance, pName);
}
return hook->proc;
}
// clang-format off
{{end}}
{{/*
------------------------------------------------------------------------------
Emits code for vkGetDeviceProcAddr for function interception.
------------------------------------------------------------------------------
*/}}
{{define "api.C++.InterceptDeviceProcAddr"}}
{{AssertType $ "API"}}
if (device == VK_NULL_HANDLE) {
ALOGE("invalid vkGetDeviceProcAddr(VK_NULL_HANDLE, ...) call");
return nullptr;
}
static const char* const known_non_device_names[] = {
{{range $f := SortBy (AllCommands $) "FunctionName"}}
{{if (Macro "IsFunctionSupported" $f)}}
{{if not (Macro "IsDeviceDispatched" $f)}}
"{{$f.Name}}",
{{end}}
{{end}}
{{end}}
};
// clang-format on
constexpr size_t count = sizeof(known_non_device_names) /
sizeof(known_non_device_names[0]);
if (!pName ||
std::binary_search(
known_non_device_names, known_non_device_names + count, pName,
[](const char* a, const char* b) { return (strcmp(a, b) < 0); })) {
vulkan::driver::Logger(device).Err
device, "invalid vkGetDeviceProcAddr(%p, \"%s\") call", device
(pName) ? pName : "(null)");
return nullptr;
}
// clang-format off
{{range $f := AllCommands $}}
{{if (Macro "IsDeviceDispatched" $f)}}
{{ if (Macro "api.IsIntercepted" $f)}}
if (strcmp(pName, "{{$f.Name}}") == 0) return §
reinterpret_cast<PFN_vkVoidFunction>(§
{{Macro "BaseName" $f}});
{{else if eq $f.Name "vkGetDeviceProcAddr"}}
if (strcmp(pName, "{{$f.Name}}") == 0) return §
reinterpret_cast<PFN_vkVoidFunction>(§
{{Macro "BaseName" $f}});
{{end}}
{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits code to dispatch a function.
------------------------------------------------------------------------------
*/}}
{{define "api.C++.Dispatch"}}
{{AssertType $ "Function"}}
{{if (Macro "api.IsIntercepted" $)}}
{{Error "$.Name should not be generated"}}
{{end}}
{{if not (IsVoid $.Return.Type)}}return §{{end}}
{{$p0 := index $.CallParameters 0}}
GetData({{$p0.Name}}).dispatch
{{Macro "BaseName" $}}({{Macro "Arguments" $}});
{{end}}
{{/*
------------------------------------------------------------------------------
Emits a list of extensions intercepted by vulkan::driver.
------------------------------------------------------------------------------
*/}}
{{define "driver.InterceptedExtensions"}}
VK_ANDROID_native_buffer
VK_EXT_debug_report
VK_KHR_android_surface
VK_KHR_surface
VK_KHR_swapchain
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if an extension is intercepted by vulkan::driver.
------------------------------------------------------------------------------
*/}}
{{define "driver.IsExtensionIntercepted"}}
{{$ext_name := index $.Arguments 0}}
{{$filters := Strings (Macro "driver.InterceptedExtensions") | SplitOn "\n"}}
{{range $f := $filters}}
{{if eq $ext_name $f}}true{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if a function is intercepted by vulkan::driver.
------------------------------------------------------------------------------
*/}}
{{define "driver.IsIntercepted"}}
{{AssertType $ "Function"}}
{{if (Macro "IsFunctionSupported" $)}}
{{/* Create functions of dispatchable objects */}}
{{ if eq $.Name "vkCreateInstance"}}true
{{else if eq $.Name "vkCreateDevice"}}true
{{else if eq $.Name "vkEnumeratePhysicalDevices"}}true
{{else if eq $.Name "vkGetDeviceQueue"}}true
{{else if eq $.Name "vkAllocateCommandBuffers"}}true
{{/* Destroy functions of dispatchable objects */}}
{{else if eq $.Name "vkDestroyInstance"}}true
{{else if eq $.Name "vkDestroyDevice"}}true
{{/* Enumeration of extensions */}}
{{else if eq $.Name "vkEnumerateInstanceExtensionProperties"}}true
{{else if eq $.Name "vkEnumerateDeviceExtensionProperties"}}true
{{else if eq $.Name "vkGetInstanceProcAddr"}}true
{{else if eq $.Name "vkGetDeviceProcAddr"}}true
{{end}}
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
{{Macro "driver.IsExtensionIntercepted" $ext}}
{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if a function needs a ProcHook stub.
------------------------------------------------------------------------------
*/}}
{{define "driver.NeedProcHookStub"}}
{{AssertType $ "Function"}}
{{if and (Macro "driver.IsIntercepted" $) (Macro "IsDeviceDispatched" $)}}
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
{{if not (Macro "IsExtensionInternal" $ext)}}true{{end}}
{{end}}
{{end}}
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits definition of struct ProcHook.
-------------------------------------------------------------------------------
*/}}
{{define "driver.C++.DefineProcHookType"}}
struct ProcHook {
enum Type {
GLOBAL,
INSTANCE,
DEVICE,
};
enum Extension {
{{$exts := Strings (Macro "driver.InterceptedExtensions") | SplitOn "\n"}}
{{range $e := $exts}}
{{TrimPrefix "VK_" $e}},
{{end}}
EXTENSION_CORE, // valid bit
EXTENSION_COUNT,
EXTENSION_UNKNOWN,
};
const char* name;
Type type;
Extension extension;
PFN_vkVoidFunction proc;
PFN_vkVoidFunction checked_proc; // always nullptr for non-device hooks
};
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits INIT_PROC_EXT macro for vulkan::driver.
-------------------------------------------------------------------------------
*/}}
{{define "driver.C++.DefineInitProcExtMacro"}}
#define INIT_PROC_EXT(ext, obj, proc) do { \
if (extensions[ProcHook::ext]) \
INIT_PROC(obj, proc); \
} while(0)
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits a stub for ProcHook::checked_proc.
-------------------------------------------------------------------------------
*/}}
{{define "driver.C++.DefineProcHookStub"}}
{{AssertType $ "Function"}}
{{if (Macro "driver.NeedProcHookStub" $)}}
{{$ext := GetAnnotation $ "extension"}}
{{$ext_name := index $ext.Arguments 0}}
{{$base := (Macro "BaseName" $)}}
VKAPI_ATTR {{Node "Type" $.Return}} checked{{$base}}({{Macro "Parameters" $}}) {
{{$p0 := index $.CallParameters 0}}
{{$ext_hook := Strings ("ProcHook::") (Macro "BaseName" $ext)}}
if (GetData({{$p0.Name}}).hook_extensions[{{$ext_hook}}]) {
{{if not (IsVoid $.Return.Type)}}return §{{end}}
{{$base}}({{Macro "Arguments" $}});
} else {
Logger({{$p0.Name}}).Err({{$p0.Name}}, "{{$ext_name}} not enabled. {{$.Name}} not executed.");
{{if not (IsVoid $.Return.Type)}}return VK_SUCCESS;{{end}}
}
}
{{end}}
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits definition of a global ProcHook.
-------------------------------------------------------------------------------
*/}}
{{define "driver.C++.DefineGlobalProcHook"}}
{{AssertType $ "Function"}}
{{$base := (Macro "BaseName" $)}}
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
{{Error "invalid global extension"}}
{{end}}
{
"{{$.Name}}",
ProcHook::GLOBAL,
ProcHook::EXTENSION_CORE,
reinterpret_cast<PFN_vkVoidFunction>({{$base}}),
nullptr,
},
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits definition of an instance ProcHook.
-------------------------------------------------------------------------------
*/}}
{{define "driver.C++.DefineInstanceProcHook"}}
{{AssertType $ "Function"}}
{{$base := (Macro "BaseName" $)}}
{
"{{$.Name}}",
ProcHook::INSTANCE,
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
ProcHook::{{Macro "BaseName" $ext}},
{{if (Macro "IsExtensionInternal" $ext)}}
nullptr,
nullptr,
{{else}}
reinterpret_cast<PFN_vkVoidFunction>({{$base}}),
nullptr,
{{end}}
{{else}}
ProcHook::EXTENSION_CORE,
reinterpret_cast<PFN_vkVoidFunction>({{$base}}),
nullptr,
{{end}}
},
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits definition of a device ProcHook.
-------------------------------------------------------------------------------
*/}}
{{define "driver.C++.DefineDeviceProcHook"}}
{{AssertType $ "Function"}}
{{$base := (Macro "BaseName" $)}}
{
"{{$.Name}}",
ProcHook::DEVICE,
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
ProcHook::{{Macro "BaseName" $ext}},
{{if (Macro "IsExtensionInternal" $ext)}}
nullptr,
nullptr,
{{else}}
reinterpret_cast<PFN_vkVoidFunction>({{$base}}),
reinterpret_cast<PFN_vkVoidFunction>(checked{{$base}}),
{{end}}
{{else}}
ProcHook::EXTENSION_CORE,
reinterpret_cast<PFN_vkVoidFunction>({{$base}}),
nullptr,
{{end}}
},
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits true if a function is needed by vulkan::driver.
-------------------------------------------------------------------------------
*/}}
{{define "driver.IsDriverTableEntry"}}
{{AssertType $ "Function"}}
{{if (Macro "IsFunctionSupported" $)}}
{{/* Create functions of dispatchable objects */}}
{{ if eq $.Name "vkCreateDevice"}}true
{{else if eq $.Name "vkGetDeviceQueue"}}true
{{else if eq $.Name "vkAllocateCommandBuffers"}}true
{{/* Destroy functions of dispatchable objects */}}
{{else if eq $.Name "vkDestroyInstance"}}true
{{else if eq $.Name "vkDestroyDevice"}}true
{{/* Enumeration of extensions */}}
{{else if eq $.Name "vkEnumerateDeviceExtensionProperties"}}true
{{/* We cache physical devices in loader.cpp */}}
{{else if eq $.Name "vkEnumeratePhysicalDevices"}}true
{{else if eq $.Name "vkGetInstanceProcAddr"}}true
{{else if eq $.Name "vkGetDeviceProcAddr"}}true
{{/* VK_KHR_swapchain->VK_ANDROID_native_buffer translation */}}
{{else if eq $.Name "vkCreateImage"}}true
{{else if eq $.Name "vkDestroyImage"}}true
{{end}}
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
{{$ext_name := index $ext.Arguments 0}}
{{ if eq $ext_name "VK_ANDROID_native_buffer"}}true
{{else if eq $ext_name "VK_EXT_debug_report"}}true
{{end}}
{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if an instance-dispatched function is needed by vulkan::driver.
------------------------------------------------------------------------------
*/}}
{{define "driver.IsInstanceDriverTableEntry"}}
{{AssertType $ "Function"}}
{{if and (Macro "driver.IsDriverTableEntry" $) (Macro "IsInstanceDispatched" $)}}
true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emits true if a device-dispatched function is needed by vulkan::driver.
------------------------------------------------------------------------------
*/}}
{{define "driver.IsDeviceDriverTableEntry"}}
{{AssertType $ "Function"}}
{{if and (Macro "driver.IsDriverTableEntry" $) (Macro "IsDeviceDispatched" $)}}
true
{{end}}
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits a function/extension name without the "vk"/"VK_" prefix.
-------------------------------------------------------------------------------
*/}}
{{define "BaseName"}}
{{ if IsFunction $}}{{TrimPrefix "vk" $.Name}}
{{else if eq $.Name "extension"}}{{TrimPrefix "VK_" (index $.Arguments 0)}}
{{else}}{{Error "invalid use of BaseName"}}
{{end}}
{{end}}
{{/*
-------------------------------------------------------------------------------
Emits a comma-separated list of C parameter names for the given command.
-------------------------------------------------------------------------------
*/}}
{{define "Arguments"}}
{{AssertType $ "Function"}}
{{ForEach $.CallParameters "ParameterName" | JoinWith ", "}}
{{end}}
{{/*
------------------------------------------------------------------------------
------------------------------------------------------------------------------
*/}}
{{define "IsGloballyDispatched"}}
{{AssertType $ "Function"}}
{{if and (Macro "IsFunctionSupported" $) (eq (Macro "Vtbl" $) "Global")}}
true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emit "true" for supported functions that undergo table dispatch. Only global
functions and functions handled in the loader top without calling into
lower layers are not dispatched.
------------------------------------------------------------------------------
*/}}
{{define "IsInstanceDispatched"}}
{{AssertType $ "Function"}}
{{if and (Macro "IsFunctionSupported" $) (eq (Macro "Vtbl" $) "Instance")}}
true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emit "true" for supported functions that can have device-specific dispatch.
------------------------------------------------------------------------------
*/}}
{{define "IsDeviceDispatched"}}
{{AssertType $ "Function"}}
{{if and (Macro "IsFunctionSupported" $) (eq (Macro "Vtbl" $) "Device")}}
true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emit "true" if a function is core or from a supportable extension.
------------------------------------------------------------------------------
*/}}
{{define "IsFunctionSupported"}}
{{AssertType $ "Function"}}
{{if not (GetAnnotation $ "pfn")}}
{{$ext := GetAnnotation $ "extension"}}
{{if not $ext}}true
{{else if not (Macro "IsExtensionBlacklisted" $ext)}}true
{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Decides whether a function should be exported from the Android Vulkan
library. Functions in the core API and in loader extensions are exported.
------------------------------------------------------------------------------
*/}}
{{define "IsFunctionExported"}}
{{AssertType $ "Function"}}
{{if (Macro "IsFunctionSupported" $)}}
{{$ext := GetAnnotation $ "extension"}}
{{if $ext}}
{{Macro "IsExtensionExported" $ext}}
{{else}}
true
{{end}}
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Emit "true" if an extension is unsupportable on Android.
------------------------------------------------------------------------------
*/}}
{{define "IsExtensionBlacklisted"}}
{{$ext := index $.Arguments 0}}
{{ if eq $ext "VK_KHR_display"}}true
{{else if eq $ext "VK_KHR_display_swapchain"}}true
{{else if eq $ext "VK_KHR_xlib_surface"}}true
{{else if eq $ext "VK_KHR_xcb_surface"}}true
{{else if eq $ext "VK_KHR_wayland_surface"}}true
{{else if eq $ext "VK_KHR_mir_surface"}}true
{{else if eq $ext "VK_KHR_win32_surface"}}true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Reports whether an extension is implemented entirely by the loader,
so drivers should not enumerate it.
------------------------------------------------------------------------------
*/}}
{{define "IsExtensionExported"}}
{{$ext := index $.Arguments 0}}
{{ if eq $ext "VK_KHR_surface"}}true
{{else if eq $ext "VK_KHR_swapchain"}}true
{{else if eq $ext "VK_KHR_android_surface"}}true
{{end}}
{{end}}
{{/*
------------------------------------------------------------------------------
Reports whether an extension is internal to the loader and drivers,
so the loader should not enumerate it.
------------------------------------------------------------------------------
*/}}
{{define "IsExtensionInternal"}}
{{$ext := index $.Arguments 0}}
{{ if eq $ext "VK_ANDROID_native_buffer"}}true
{{end}}
{{end}}