monado/scripts/generate_vk_helpers.py
2021-12-06 17:25:09 +00:00

472 lines
14 KiB
Python
Executable file

#!/usr/bin/env python3
# Copyright 2019-2021, Collabora, Ltd.
# SPDX-License-Identifier: BSL-1.0
"""Simple script to update vk_helpers.{c,h}."""
from pathlib import Path
from typing import Callable, Iterable, List, Optional
def get_device_cmds():
return [
Cmd("vkDestroyDevice"),
Cmd("vkDeviceWaitIdle"),
Cmd("vkAllocateMemory"),
Cmd("vkFreeMemory"),
Cmd("vkMapMemory"),
Cmd("vkUnmapMemory"),
None,
Cmd("vkCreateBuffer"),
Cmd("vkDestroyBuffer"),
Cmd("vkBindBufferMemory"),
None,
Cmd("vkCreateImage"),
Cmd("vkDestroyImage"),
Cmd("vkBindImageMemory"),
None,
Cmd("vkGetBufferMemoryRequirements"),
Cmd("vkFlushMappedMemoryRanges"),
Cmd("vkGetImageMemoryRequirements"),
Cmd(
"vkGetImageMemoryRequirements2KHR",
member_name="vkGetImageMemoryRequirements2",
),
Cmd("vkGetImageSubresourceLayout"),
None,
Cmd("vkCreateImageView"),
Cmd("vkDestroyImageView"),
None,
Cmd("vkCreateSampler"),
Cmd("vkDestroySampler"),
None,
Cmd("vkCreateShaderModule"),
Cmd("vkDestroyShaderModule"),
None,
Cmd("vkCreateCommandPool"),
Cmd("vkDestroyCommandPool"),
None,
Cmd("vkAllocateCommandBuffers"),
Cmd("vkBeginCommandBuffer"),
Cmd("vkCmdPipelineBarrier"),
Cmd("vkCmdBeginRenderPass"),
Cmd("vkCmdSetScissor"),
Cmd("vkCmdSetViewport"),
Cmd("vkCmdClearColorImage"),
Cmd("vkCmdEndRenderPass"),
Cmd("vkCmdBindDescriptorSets"),
Cmd("vkCmdBindPipeline"),
Cmd("vkCmdBindVertexBuffers"),
Cmd("vkCmdBindIndexBuffer"),
Cmd("vkCmdDraw"),
Cmd("vkCmdDrawIndexed"),
Cmd("vkCmdDispatch"),
Cmd("vkCmdCopyBuffer"),
Cmd("vkCmdCopyBufferToImage"),
Cmd("vkCmdCopyImage"),
Cmd("vkCmdCopyImageToBuffer"),
Cmd("vkEndCommandBuffer"),
Cmd("vkFreeCommandBuffers"),
None,
Cmd("vkCreateRenderPass"),
Cmd("vkDestroyRenderPass"),
None,
Cmd("vkCreateFramebuffer"),
Cmd("vkDestroyFramebuffer"),
None,
Cmd("vkCreatePipelineCache"),
Cmd("vkDestroyPipelineCache"),
None,
Cmd("vkResetDescriptorPool"),
Cmd("vkCreateDescriptorPool"),
Cmd("vkDestroyDescriptorPool"),
None,
Cmd("vkAllocateDescriptorSets"),
Cmd("vkFreeDescriptorSets"),
None,
Cmd("vkCreateComputePipelines"),
Cmd("vkCreateGraphicsPipelines"),
Cmd("vkDestroyPipeline"),
None,
Cmd("vkCreatePipelineLayout"),
Cmd("vkDestroyPipelineLayout"),
None,
Cmd("vkCreateDescriptorSetLayout"),
Cmd("vkUpdateDescriptorSets"),
Cmd("vkDestroyDescriptorSetLayout"),
None,
Cmd("vkGetDeviceQueue"),
Cmd("vkQueueSubmit"),
Cmd("vkQueueWaitIdle"),
None,
Cmd("vkCreateSemaphore"),
Cmd(
"vkSignalSemaphoreKHR",
member_name="vkSignalSemaphore",
requires=("VK_KHR_timeline_semaphore",),
),
Cmd(
"vkWaitSemaphoresKHR",
member_name="vkWaitSemaphores",
requires=("VK_KHR_timeline_semaphore",),
),
Cmd(
"vkGetSemaphoreCounterValueKHR",
member_name="vkGetSemaphoreCounterValue",
requires=("VK_KHR_timeline_semaphore",),
),
Cmd("vkDestroySemaphore"),
None,
Cmd("vkCreateFence"),
Cmd("vkWaitForFences"),
Cmd("vkGetFenceStatus"),
Cmd("vkDestroyFence"),
Cmd("vkResetFences"),
None,
Cmd("vkCreateSwapchainKHR"),
Cmd("vkDestroySwapchainKHR"),
Cmd("vkGetSwapchainImagesKHR"),
Cmd("vkAcquireNextImageKHR"),
Cmd("vkQueuePresentKHR"),
None,
Cmd("vkGetMemoryWin32HandleKHR", requires=("VK_USE_PLATFORM_WIN32_KHR",)),
Cmd("vkImportSemaphoreWin32HandleKHR", requires=("VK_USE_PLATFORM_WIN32_KHR",)),
Cmd("vkImportFenceWin32HandleKHR", requires=("VK_USE_PLATFORM_WIN32_KHR",)),
Cmd("vkGetMemoryFdKHR", requires=("!defined(VK_USE_PLATFORM_WIN32_KHR)",)),
None,
Cmd(
"vkImportSemaphoreFdKHR", requires=("!defined(VK_USE_PLATFORM_WIN32_KHR)",)
),
Cmd("vkGetSemaphoreFdKHR", requires=("!defined(VK_USE_PLATFORM_WIN32_KHR)",)),
None,
Cmd("vkImportFenceFdKHR", requires=("!defined(VK_USE_PLATFORM_WIN32_KHR)",)),
Cmd("vkGetFenceFdKHR", requires=("!defined(VK_USE_PLATFORM_WIN32_KHR)",)),
Cmd(
"vkGetMemoryAndroidHardwareBufferANDROID",
requires=("VK_USE_PLATFORM_ANDROID_KHR",),
),
Cmd(
"vkGetAndroidHardwareBufferPropertiesANDROID",
requires=("VK_USE_PLATFORM_ANDROID_KHR",),
),
None,
Cmd("vkGetPastPresentationTimingGOOGLE"),
]
def get_instance_cmds():
return [
Cmd("vkDestroyInstance"),
Cmd("vkGetDeviceProcAddr"),
Cmd("vkCreateDevice"),
Cmd("vkDestroySurfaceKHR"),
None,
Cmd("vkCreateDebugReportCallbackEXT"),
Cmd("vkDestroyDebugReportCallbackEXT"),
None,
Cmd("vkEnumeratePhysicalDevices"),
Cmd("vkGetPhysicalDeviceProperties"),
Cmd("vkGetPhysicalDeviceProperties2"),
Cmd("vkGetPhysicalDeviceFeatures2"),
Cmd("vkGetPhysicalDeviceMemoryProperties"),
Cmd("vkGetPhysicalDeviceQueueFamilyProperties"),
Cmd("vkGetPhysicalDeviceSurfaceCapabilitiesKHR"),
Cmd("vkGetPhysicalDeviceSurfaceFormatsKHR"),
Cmd("vkGetPhysicalDeviceSurfacePresentModesKHR"),
Cmd("vkGetPhysicalDeviceSurfaceSupportKHR"),
Cmd("vkGetPhysicalDeviceFormatProperties"),
Cmd("vkEnumerateDeviceExtensionProperties"),
Cmd("vkGetPhysicalDeviceImageFormatProperties2"),
None,
Cmd(
"vkCreateDisplayPlaneSurfaceKHR", requires=("VK_USE_PLATFORM_DISPLAY_KHR",)
),
Cmd(
"vkGetDisplayPlaneCapabilitiesKHR",
requires=("VK_USE_PLATFORM_DISPLAY_KHR",),
),
Cmd(
"vkGetPhysicalDeviceDisplayPropertiesKHR",
requires=("VK_USE_PLATFORM_DISPLAY_KHR",),
),
Cmd(
"vkGetPhysicalDeviceDisplayPlanePropertiesKHR",
requires=("VK_USE_PLATFORM_DISPLAY_KHR",),
),
Cmd("vkGetDisplayModePropertiesKHR", requires=("VK_USE_PLATFORM_DISPLAY_KHR",)),
Cmd("vkReleaseDisplayEXT", requires=("VK_USE_PLATFORM_DISPLAY_KHR",)),
None,
Cmd("vkCreateXcbSurfaceKHR", requires=("VK_USE_PLATFORM_XCB_KHR",)),
None,
Cmd("vkCreateWaylandSurfaceKHR", requires=("VK_USE_PLATFORM_WAYLAND_KHR",)),
None,
Cmd(
"vkAcquireDrmDisplayEXT",
requires=("VK_USE_PLATFORM_WAYLAND_KHR", "VK_EXT_acquire_drm_display"),
),
Cmd(
"vkGetDrmDisplayEXT",
requires=("VK_USE_PLATFORM_WAYLAND_KHR", "VK_EXT_acquire_drm_display"),
),
None,
Cmd(
"vkGetRandROutputDisplayEXT", requires=("VK_USE_PLATFORM_XLIB_XRANDR_EXT",)
),
Cmd("vkAcquireXlibDisplayEXT", requires=("VK_USE_PLATFORM_XLIB_XRANDR_EXT",)),
None,
Cmd("vkCreateAndroidSurfaceKHR", requires=("VK_USE_PLATFORM_ANDROID_KHR",)),
None,
Cmd("vkCreateWin32SurfaceKHR", requires=("VK_USE_PLATFORM_WIN32_KHR",)),
]
EXTENSIONS_TO_CHECK = [
"VK_KHR_timeline_semaphore",
"VK_EXT_global_priority",
"VK_EXT_robustness2",
"VK_GOOGLE_display_timing",
]
ROOT = Path(__file__).resolve().parent.parent
DIR = ROOT / "src" / "xrt" / "auxiliary" / "vk"
HEADER_FN = DIR / "vk_helpers.h"
IMPL_FN = DIR / "vk_helpers.c"
BEGIN_TEMPLATE = "\t// beginning of GENERATED %s code - do not modify - used by scripts"
END_TEMPLATE = "\t// end of GENERATED %s code - do not modify - used by scripts"
class Cmd:
def __init__(
self,
name: str,
member_name: Optional[str] = None,
*,
requires: Optional[Iterable[str]] = None,
):
self.name = name
if not member_name:
member_name = name
self.member_name = member_name
if not requires:
# normalize empty lists to None
requires = None
self.requires = requires
def __repr__(self) -> str:
args = [repr(self.name)]
if self.member_name != self.name:
args.append(repr(self.member_name))
if self.requires:
args.append(f"requires={repr(self.requires)}")
return "Function({})".format(", ".join(args))
def wrap_condition(condition):
if "defined" in condition:
return condition
return "defined({})".format(condition)
def compute_condition(pp_conditions):
if not pp_conditions:
return None
return " && ".join(wrap_condition(x) for x in pp_conditions)
class ConditionalGenerator:
"""Keep track of conditions to avoid unneeded repetition of ifdefs."""
def __init__(self):
self.current_condition = None
def process_condition(self, new_condition: Optional[str]) -> Optional[str]:
"""Return a line (or lines) to yield if required based on the new condition state."""
lines = []
if self.current_condition and new_condition != self.current_condition:
# Close current condition if required.
lines.append("#endif // {}".format(self.current_condition))
# empty line
lines.append("")
self.current_condition = None
if new_condition != self.current_condition:
# Open new condition if required
lines.append("#if {}".format(new_condition))
self.current_condition = new_condition
if lines:
return "\n".join(lines)
def finish(self) -> Optional[str]:
"""Return a line (or lines) to yield if required at the end of the loop."""
return self.process_condition(None)
def generate_per_command(
commands: List[Cmd], per_command_handler: Callable[[Cmd], str]
):
conditional = ConditionalGenerator()
for cmd in commands:
if not cmd:
# empty line
yield ""
continue
condition = compute_condition(cmd.requires)
condition_line = conditional.process_condition(condition)
if condition_line:
yield condition_line
yield per_command_handler(cmd)
# close any trailing conditions
condition_line = conditional.finish()
if condition_line:
yield condition_line
def generate_structure_members(commands: List[Cmd]):
def per_command(cmd: Cmd):
return "\tPFN_{} {};".format(cmd.name, cmd.member_name)
return generate_per_command(commands, per_command)
def generate_proc_macro(macro: str, commands: List[Cmd]):
name_width = max([len(cmd.member_name) for cmd in commands if cmd])
def per_command(cmd: Cmd) -> str:
return "\tvk->{} = {}(vk, {});".format(
cmd.member_name.ljust(name_width), macro, cmd.name
)
return generate_per_command(
commands,
per_command,
)
def make_ext_member_name(ext: str):
return "has_{}".format(ext[3:])
def make_ext_name_define(ext: str):
return "{}_EXTENSION_NAME".format(ext.upper()).replace("2", "_2")
def generate_ext_members():
for ext in EXTENSIONS_TO_CHECK:
yield "\tbool {};".format(make_ext_member_name(ext))
def generate_ext_check():
yield "\t// Reset before filling out."
for ext in EXTENSIONS_TO_CHECK:
yield "\tvk->{} = false;".format(make_ext_member_name(ext))
yield ""
yield "\tfor (uint32_t i = 0; i < device_extension_count; i++) {"
yield "\t\tconst char *ext = device_extensions[i];"
yield ""
conditional = ConditionalGenerator()
for ext in EXTENSIONS_TO_CHECK:
condition_line = conditional.process_condition(compute_condition((ext,)))
if condition_line:
yield condition_line
yield "\t\tif (strcmp(ext, {}) == 0) {{".format(make_ext_name_define(ext))
yield "\t\t\tvk->{} = true;".format(make_ext_member_name(ext))
yield "\t\t\tcontinue;"
yield "\t\t}"
# close any trailing conditions
condition_line = conditional.finish()
if condition_line:
yield condition_line
yield "\t}"
def replace_middle(
lines: List[str], start_sentinel: str, end_sentinel: str, new_middle: List[str]
) -> List[str]:
middle_start = lines.index(start_sentinel) + 1
middle_end = lines.index(end_sentinel)
return lines[:middle_start] + new_middle + lines[middle_end:]
DEVICE_TEMPLATES = {
"BEGIN": BEGIN_TEMPLATE % "device loader",
"END": END_TEMPLATE % "device loader",
}
INSTANCE_TEMPLATES = {
"BEGIN": BEGIN_TEMPLATE % "instance loader",
"END": END_TEMPLATE % "instance loader",
}
EXT_TEMPLATES = {
"BEGIN": BEGIN_TEMPLATE % "extension",
"END": END_TEMPLATE % "extension",
}
def process_header():
with open(str(HEADER_FN), "r", encoding="utf-8") as fp:
lines = [line.rstrip() for line in fp.readlines()]
lines = replace_middle(
lines,
INSTANCE_TEMPLATES["BEGIN"],
INSTANCE_TEMPLATES["END"],
list(generate_structure_members(get_instance_cmds())),
)
lines = replace_middle(
lines,
DEVICE_TEMPLATES["BEGIN"],
DEVICE_TEMPLATES["END"],
list(generate_structure_members(get_device_cmds())),
)
lines = replace_middle(
lines,
EXT_TEMPLATES["BEGIN"],
EXT_TEMPLATES["END"],
list(generate_ext_members()),
)
with open(str(HEADER_FN), "w", encoding="utf-8") as fp:
fp.write("\n".join(lines))
fp.write("\n")
def process_impl():
with open(str(IMPL_FN), "r", encoding="utf-8") as fp:
lines = [line.rstrip() for line in fp.readlines()]
lines = replace_middle(
lines,
INSTANCE_TEMPLATES["BEGIN"],
INSTANCE_TEMPLATES["END"],
list(generate_proc_macro("GET_INS_PROC", get_instance_cmds())),
)
lines = replace_middle(
lines,
DEVICE_TEMPLATES["BEGIN"],
DEVICE_TEMPLATES["END"],
list(generate_proc_macro("GET_DEV_PROC", get_device_cmds())),
)
lines = replace_middle(
lines,
EXT_TEMPLATES["BEGIN"],
EXT_TEMPLATES["END"],
list(generate_ext_check()),
)
with open(str(IMPL_FN), "w", encoding="utf-8") as fp:
fp.write("\n".join(lines))
fp.write("\n")
if __name__ == "__main__":
process_header()
process_impl()