#!/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("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_GOOGLE_display_timing", "VK_EXT_global_priority", "VK_EXT_robustness2", ] 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 < num_device_extensions; 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()