| #!/usr/bin/env python3 | 
 | # | 
 | # Copyright 2019 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. | 
 |  | 
 | """Generates the driver_gen.h and driver_gen.cpp. | 
 | """ | 
 |  | 
 | import os | 
 | import generator_common as gencom | 
 |  | 
 | # Extensions intercepted at vulkan::driver level. | 
 | _INTERCEPTED_EXTENSIONS = [ | 
 |     'VK_ANDROID_native_buffer', | 
 |     'VK_EXT_debug_report', | 
 |     'VK_EXT_hdr_metadata', | 
 |     'VK_EXT_swapchain_colorspace', | 
 |     'VK_GOOGLE_display_timing', | 
 |     'VK_KHR_android_surface', | 
 |     'VK_KHR_get_surface_capabilities2', | 
 |     'VK_KHR_incremental_present', | 
 |     'VK_KHR_shared_presentable_image', | 
 |     'VK_KHR_surface', | 
 |     'VK_KHR_swapchain', | 
 | ] | 
 |  | 
 | # Extensions known to vulkan::driver level. | 
 | _KNOWN_EXTENSIONS = _INTERCEPTED_EXTENSIONS + [ | 
 |     'VK_ANDROID_external_memory_android_hardware_buffer', | 
 |     'VK_KHR_bind_memory2', | 
 |     'VK_KHR_get_physical_device_properties2', | 
 |     'VK_KHR_device_group_creation', | 
 |     'VK_KHR_external_memory_capabilities', | 
 |     'VK_KHR_external_semaphore_capabilities', | 
 |     'VK_KHR_external_fence_capabilities', | 
 | ] | 
 |  | 
 | # Functions needed at vulkan::driver level. | 
 | _NEEDED_COMMANDS = [ | 
 |     # Create functions of dispatchable objects | 
 |     'vkCreateDevice', | 
 |     'vkGetDeviceQueue', | 
 |     'vkGetDeviceQueue2', | 
 |     'vkAllocateCommandBuffers', | 
 |  | 
 |     # Destroy functions of dispatchable objects | 
 |     'vkDestroyInstance', | 
 |     'vkDestroyDevice', | 
 |  | 
 |     # Enumeration of extensions | 
 |     'vkEnumerateDeviceExtensionProperties', | 
 |  | 
 |     # We cache physical devices in loader.cpp | 
 |     'vkEnumeratePhysicalDevices', | 
 |     'vkEnumeratePhysicalDeviceGroups', | 
 |  | 
 |     'vkGetInstanceProcAddr', | 
 |     'vkGetDeviceProcAddr', | 
 |  | 
 |     'vkQueueSubmit', | 
 |  | 
 |     # VK_KHR_swapchain->VK_ANDROID_native_buffer translation | 
 |     'vkCreateImage', | 
 |     'vkDestroyImage', | 
 |  | 
 |     'vkGetPhysicalDeviceProperties', | 
 |  | 
 |     # VK_KHR_swapchain v69 requirement | 
 |     'vkBindImageMemory2', | 
 |     'vkBindImageMemory2KHR', | 
 |  | 
 |     # For promoted VK_KHR_device_group_creation | 
 |     'vkEnumeratePhysicalDeviceGroupsKHR', | 
 |  | 
 |     # For promoted VK_KHR_get_physical_device_properties2 | 
 |     'vkGetPhysicalDeviceFeatures2', | 
 |     'vkGetPhysicalDeviceFeatures2KHR', | 
 |     'vkGetPhysicalDeviceProperties2', | 
 |     'vkGetPhysicalDeviceProperties2KHR', | 
 |     'vkGetPhysicalDeviceFormatProperties2', | 
 |     'vkGetPhysicalDeviceFormatProperties2KHR', | 
 |     'vkGetPhysicalDeviceImageFormatProperties2', | 
 |     'vkGetPhysicalDeviceImageFormatProperties2KHR', | 
 |     'vkGetPhysicalDeviceQueueFamilyProperties2', | 
 |     'vkGetPhysicalDeviceQueueFamilyProperties2KHR', | 
 |     'vkGetPhysicalDeviceMemoryProperties2', | 
 |     'vkGetPhysicalDeviceMemoryProperties2KHR', | 
 |     'vkGetPhysicalDeviceSparseImageFormatProperties2', | 
 |     'vkGetPhysicalDeviceSparseImageFormatProperties2KHR', | 
 |  | 
 |     # For promoted VK_KHR_external_memory_capabilities | 
 |     'vkGetPhysicalDeviceExternalBufferProperties', | 
 |     'vkGetPhysicalDeviceExternalBufferPropertiesKHR', | 
 |  | 
 |     # For promoted VK_KHR_external_semaphore_capabilities | 
 |     'vkGetPhysicalDeviceExternalSemaphoreProperties', | 
 |     'vkGetPhysicalDeviceExternalSemaphorePropertiesKHR', | 
 |  | 
 |     # For promoted VK_KHR_external_fence_capabilities | 
 |     'vkGetPhysicalDeviceExternalFenceProperties', | 
 |     'vkGetPhysicalDeviceExternalFencePropertiesKHR', | 
 | ] | 
 |  | 
 | # Functions intercepted at vulkan::driver level. | 
 | _INTERCEPTED_COMMANDS = [ | 
 |     # Create functions of dispatchable objects | 
 |     'vkCreateInstance', | 
 |     'vkCreateDevice', | 
 |     'vkEnumeratePhysicalDevices', | 
 |     'vkEnumeratePhysicalDeviceGroups', | 
 |     'vkGetDeviceQueue', | 
 |     'vkGetDeviceQueue2', | 
 |     'vkAllocateCommandBuffers', | 
 |  | 
 |     # Destroy functions of dispatchable objects | 
 |     'vkDestroyInstance', | 
 |     'vkDestroyDevice', | 
 |  | 
 |     # Enumeration of extensions | 
 |     'vkEnumerateInstanceExtensionProperties', | 
 |     'vkEnumerateDeviceExtensionProperties', | 
 |  | 
 |     'vkGetInstanceProcAddr', | 
 |     'vkGetDeviceProcAddr', | 
 |  | 
 |     'vkQueueSubmit', | 
 |  | 
 |     # VK_KHR_swapchain v69 requirement | 
 |     'vkBindImageMemory2', | 
 |     'vkBindImageMemory2KHR', | 
 |  | 
 |     # For promoted VK_KHR_get_physical_device_properties2 | 
 |     'vkGetPhysicalDeviceFeatures2', | 
 |     'vkGetPhysicalDeviceProperties2', | 
 |     'vkGetPhysicalDeviceFormatProperties2', | 
 |     'vkGetPhysicalDeviceImageFormatProperties2', | 
 |     'vkGetPhysicalDeviceQueueFamilyProperties2', | 
 |     'vkGetPhysicalDeviceMemoryProperties2', | 
 |     'vkGetPhysicalDeviceSparseImageFormatProperties2', | 
 |  | 
 |     # For promoted VK_KHR_external_memory_capabilities | 
 |     'vkGetPhysicalDeviceExternalBufferProperties', | 
 |  | 
 |     # For promoted VK_KHR_external_semaphore_capabilities | 
 |     'vkGetPhysicalDeviceExternalSemaphoreProperties', | 
 |  | 
 |     # For promoted VK_KHR_external_fence_capabilities | 
 |     'vkGetPhysicalDeviceExternalFenceProperties', | 
 | ] | 
 |  | 
 |  | 
 | def _is_driver_table_entry(cmd): | 
 |   """Returns true if a function is needed by vulkan::driver. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |   """ | 
 |   if gencom.is_function_supported(cmd): | 
 |     if cmd in _NEEDED_COMMANDS: | 
 |       return True | 
 |     if cmd in gencom.extension_dict: | 
 |       if (gencom.extension_dict[cmd] == 'VK_ANDROID_native_buffer' or | 
 |           gencom.extension_dict[cmd] == 'VK_EXT_debug_report'): | 
 |         return True | 
 |   return False | 
 |  | 
 |  | 
 | def _is_instance_driver_table_entry(cmd): | 
 |   """Returns true if a instance-dispatched function is needed by vulkan::driver. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |   """ | 
 |   return (_is_driver_table_entry(cmd) and | 
 |           gencom.is_instance_dispatched(cmd)) | 
 |  | 
 |  | 
 | def _is_device_driver_table_entry(cmd): | 
 |   """Returns true if a device-dispatched function is needed by vulkan::driver. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |   """ | 
 |   return (_is_driver_table_entry(cmd) and | 
 |           gencom.is_device_dispatched(cmd)) | 
 |  | 
 |  | 
 | def gen_h(): | 
 |   """Generates the driver_gen.h file. | 
 |   """ | 
 |   genfile = os.path.join(os.path.dirname(__file__), | 
 |                          '..', 'libvulkan', 'driver_gen.h') | 
 |  | 
 |   with open(genfile, 'w') as f: | 
 |     f.write(gencom.copyright_and_warning(2016)) | 
 |  | 
 |     f.write("""\ | 
 | #ifndef LIBVULKAN_DRIVER_GEN_H | 
 | #define LIBVULKAN_DRIVER_GEN_H | 
 |  | 
 | #include <vulkan/vk_android_native_buffer.h> | 
 | #include <vulkan/vulkan.h> | 
 |  | 
 | #include <bitset> | 
 | #include <optional> | 
 | #include <vector> | 
 |  | 
 | namespace vulkan { | 
 | namespace driver { | 
 |  | 
 | struct ProcHook { | 
 |     enum Type { | 
 |         GLOBAL, | 
 |         INSTANCE, | 
 |         DEVICE, | 
 |     }; | 
 |     enum Extension {\n""") | 
 |  | 
 |     for ext in _KNOWN_EXTENSIONS: | 
 |       f.write(gencom.indent(2) + gencom.base_ext_name(ext) + ',\n') | 
 |  | 
 |     f.write('\n') | 
 |     for version in gencom.version_code_list: | 
 |       f.write(gencom.indent(2) + 'EXTENSION_CORE_' + version + ',\n') | 
 |  | 
 |     # EXTENSION_COUNT must be the next enum after the highest API version. | 
 |     f.write("""\ | 
 |         EXTENSION_COUNT, | 
 |         EXTENSION_UNKNOWN, | 
 |     }; | 
 |  | 
 |     const char* name; | 
 |     Type type; | 
 |     Extension extension; | 
 |  | 
 |     PFN_vkVoidFunction proc; | 
 |     PFN_vkVoidFunction checked_proc;  // always nullptr for non-device hooks | 
 | }; | 
 |  | 
 | struct InstanceDriverTable { | 
 |     // clang-format off\n""") | 
 |  | 
 |     for cmd in gencom.command_list: | 
 |       if _is_instance_driver_table_entry(cmd): | 
 |         f.write(gencom.indent(1) + 'PFN_' + cmd + ' ' + | 
 |                 gencom.base_name(cmd) + ';\n') | 
 |  | 
 |     f.write("""\ | 
 |     // clang-format on | 
 | }; | 
 |  | 
 | struct DeviceDriverTable { | 
 |     // clang-format off\n""") | 
 |  | 
 |     for cmd in gencom.command_list: | 
 |       if _is_device_driver_table_entry(cmd): | 
 |         f.write(gencom.indent(1) + 'PFN_' + cmd + ' ' + | 
 |                 gencom.base_name(cmd) + ';\n') | 
 |  | 
 |     f.write("""\ | 
 |     // 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); | 
 |  | 
 | std::optional<uint32_t> GetInstanceExtensionPromotedVersion(const char* name); | 
 | uint32_t CountPromotedInstanceExtensions(uint32_t begin_version, | 
 |                                          uint32_t end_version); | 
 | std::vector<const char*> GetPromotedInstanceExtensions(uint32_t begin_version, | 
 |                                                        uint32_t end_version); | 
 |  | 
 | }  // namespace driver | 
 | }  // namespace vulkan | 
 |  | 
 | #endif  // LIBVULKAN_DRIVER_TABLE_H\n""") | 
 |  | 
 |     f.close() | 
 |   gencom.run_clang_format(genfile) | 
 |  | 
 |  | 
 | def _is_intercepted(cmd): | 
 |   """Returns true if a function is intercepted by vulkan::driver. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |   """ | 
 |   if gencom.is_function_supported(cmd): | 
 |     if cmd in _INTERCEPTED_COMMANDS: | 
 |       return True | 
 |  | 
 |     if cmd in gencom.extension_dict: | 
 |       return gencom.extension_dict[cmd] in _INTERCEPTED_EXTENSIONS | 
 |   return False | 
 |  | 
 |  | 
 | def _get_proc_hook_enum(cmd): | 
 |   """Returns the ProcHook enumeration for the corresponding core function. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |   """ | 
 |   assert cmd in gencom.version_dict | 
 |   for version in gencom.version_code_list: | 
 |     if gencom.version_dict[cmd] == 'VK_VERSION_' + version: | 
 |       return 'ProcHook::EXTENSION_CORE_' + version | 
 |  | 
 |  | 
 | def _need_proc_hook_stub(cmd): | 
 |   """Returns true if a function needs a ProcHook stub. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |   """ | 
 |   if _is_intercepted(cmd) and gencom.is_device_dispatched(cmd): | 
 |     if cmd in gencom.extension_dict: | 
 |       if not gencom.is_extension_internal(gencom.extension_dict[cmd]): | 
 |         return True | 
 |     elif gencom.version_dict[cmd] != 'VK_VERSION_1_0': | 
 |       return True | 
 |   return False | 
 |  | 
 |  | 
 | def _define_proc_hook_stub(cmd, f): | 
 |   """Emits a stub for ProcHook::checked_proc. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |     f: Output file handle. | 
 |   """ | 
 |   if _need_proc_hook_stub(cmd): | 
 |     return_type = gencom.return_type_dict[cmd] | 
 |  | 
 |     ext_name = '' | 
 |     ext_hook = '' | 
 |     if cmd in gencom.extension_dict: | 
 |       ext_name = gencom.extension_dict[cmd] | 
 |       ext_hook = 'ProcHook::' + gencom.base_ext_name(ext_name) | 
 |     else: | 
 |       ext_name = gencom.version_dict[cmd] | 
 |       ext_hook = _get_proc_hook_enum(cmd) | 
 |  | 
 |     handle = gencom.param_dict[cmd][0][1] | 
 |     param_types = ', '.join([''.join(i) for i in gencom.param_dict[cmd]]) | 
 |     param_names = ', '.join([''.join(i[1]) for i in gencom.param_dict[cmd]]) | 
 |  | 
 |     f.write('VKAPI_ATTR ' + return_type + ' checked' + gencom.base_name(cmd) + | 
 |             '(' + param_types + ') {\n') | 
 |     f.write(gencom.indent(1) + 'if (GetData(' + handle + ').hook_extensions[' + | 
 |             ext_hook + ']) {\n') | 
 |  | 
 |     f.write(gencom.indent(2)) | 
 |     if gencom.return_type_dict[cmd] != 'void': | 
 |       f.write('return ') | 
 |     f.write(gencom.base_name(cmd) + '(' + param_names + ');\n') | 
 |  | 
 |     f.write(gencom.indent(1) + '} else {\n') | 
 |     f.write(gencom.indent(2) + 'Logger(' + handle + ').Err(' + handle + ', \"' + | 
 |             ext_name + ' not enabled. ' + cmd + ' not executed.\");\n') | 
 |     if gencom.return_type_dict[cmd] != 'void': | 
 |       f.write(gencom.indent(2) + 'return VK_SUCCESS;\n') | 
 |     f.write(gencom.indent(1) + '}\n}\n\n') | 
 |  | 
 |  | 
 | def _define_global_proc_hook(cmd, f): | 
 |   """Emits definition of a global ProcHook. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |     f: Output file handle. | 
 |   """ | 
 |   assert cmd not in gencom.extension_dict | 
 |  | 
 |   f.write(gencom.indent(1) + '{\n') | 
 |   f.write(gencom.indent(2) + '\"' + cmd + '\",\n') | 
 |   f.write(gencom.indent(2) + 'ProcHook::GLOBAL,\n') | 
 |   f.write(gencom.indent(2) + _get_proc_hook_enum(cmd) + ',\n') | 
 |   f.write(gencom.indent(2) + 'reinterpret_cast<PFN_vkVoidFunction>(' + | 
 |           gencom.base_name(cmd) + '),\n') | 
 |   f.write(gencom.indent(2) + 'nullptr,\n') | 
 |   f.write(gencom.indent(1) + '},\n') | 
 |  | 
 |  | 
 | def _define_instance_proc_hook(cmd, f): | 
 |   """Emits definition of a instance ProcHook. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |     f: Output file handle. | 
 |   """ | 
 |   f.write(gencom.indent(1) + '{\n') | 
 |   f.write(gencom.indent(2) + '\"' + cmd + '\",\n') | 
 |   f.write(gencom.indent(2) + 'ProcHook::INSTANCE,\n') | 
 |  | 
 |   if cmd in gencom.extension_dict: | 
 |     ext_name = gencom.extension_dict[cmd] | 
 |     f.write(gencom.indent(2) + 'ProcHook::' + | 
 |             gencom.base_ext_name(ext_name) + ',\n') | 
 |  | 
 |     if gencom.is_extension_internal(ext_name): | 
 |       f.write("""\ | 
 |         nullptr, | 
 |         nullptr,\n""") | 
 |     else: | 
 |       f.write("""\ | 
 |         reinterpret_cast<PFN_vkVoidFunction>(""" + gencom.base_name(cmd) + """), | 
 |         nullptr,\n""") | 
 |   else: | 
 |     f.write(gencom.indent(2) + _get_proc_hook_enum(cmd) + ',\n') | 
 |     f.write("""\ | 
 |         reinterpret_cast<PFN_vkVoidFunction>(""" + gencom.base_name(cmd) + """), | 
 |         nullptr,\n""") | 
 |  | 
 |   f.write(gencom.indent(1) + '},\n') | 
 |  | 
 |  | 
 | def _define_device_proc_hook(cmd, f): | 
 |   """Emits definition of a device ProcHook. | 
 |  | 
 |   Args: | 
 |     cmd: Vulkan function name. | 
 |     f: Output file handle. | 
 |   """ | 
 |   f.write(gencom.indent(1) + '{\n') | 
 |   f.write(gencom.indent(2) + '\"' + cmd + '\",\n') | 
 |   f.write(gencom.indent(2) + 'ProcHook::DEVICE,\n') | 
 |  | 
 |   if (cmd in gencom.extension_dict or | 
 |       gencom.version_dict[cmd] != 'VK_VERSION_1_0'): | 
 |     ext_name = '' | 
 |     ext_hook = '' | 
 |     if cmd in gencom.extension_dict: | 
 |       ext_name = gencom.extension_dict[cmd] | 
 |       ext_hook = 'ProcHook::' + gencom.base_ext_name(ext_name) | 
 |     else: | 
 |       ext_name = gencom.version_dict[cmd] | 
 |       ext_hook = _get_proc_hook_enum(cmd) | 
 |     f.write(gencom.indent(2) + ext_hook + ',\n') | 
 |  | 
 |     if gencom.is_extension_internal(ext_name): | 
 |       f.write("""\ | 
 |         nullptr, | 
 |         nullptr,\n""") | 
 |     else: | 
 |       f.write("""\ | 
 |         reinterpret_cast<PFN_vkVoidFunction>(""" + gencom.base_name(cmd) + """), | 
 |         reinterpret_cast<PFN_vkVoidFunction>(checked""" + | 
 |               gencom.base_name(cmd) + '),\n') | 
 |  | 
 |   else: | 
 |     f.write(gencom.indent(2) + _get_proc_hook_enum(cmd) + ',\n') | 
 |     f.write("""\ | 
 |         reinterpret_cast<PFN_vkVoidFunction>(""" + gencom.base_name(cmd) + """), | 
 |         nullptr,\n""") | 
 |  | 
 |   f.write(gencom.indent(1) + '},\n') | 
 |  | 
 |  | 
 | def gen_cpp(): | 
 |   """Generates the driver_gen.cpp file. | 
 |   """ | 
 |   genfile = os.path.join(os.path.dirname(__file__), | 
 |                          '..', 'libvulkan', 'driver_gen.cpp') | 
 |  | 
 |   with open(genfile, 'w') as f: | 
 |     f.write(gencom.copyright_and_warning(2016)) | 
 |     f.write("""\ | 
 | #include <log/log.h> | 
 | #include <string.h> | 
 |  | 
 | #include <algorithm> | 
 |  | 
 | #include "driver.h" | 
 |  | 
 | namespace vulkan { | 
 | namespace driver { | 
 |  | 
 | namespace { | 
 |  | 
 | // clang-format off\n\n""") | 
 |  | 
 |     for cmd in gencom.command_list: | 
 |       _define_proc_hook_stub(cmd, f) | 
 |  | 
 |     f.write("""\ | 
 | // clang-format on | 
 |  | 
 | const ProcHook g_proc_hooks[] = { | 
 |     // clang-format off\n""") | 
 |  | 
 |     sorted_command_list = sorted(gencom.command_list) | 
 |     for cmd in sorted_command_list: | 
 |       if _is_intercepted(cmd): | 
 |         if gencom.is_globally_dispatched(cmd): | 
 |           _define_global_proc_hook(cmd, f) | 
 |         elif gencom.is_instance_dispatched(cmd): | 
 |           _define_instance_proc_hook(cmd, f) | 
 |         elif gencom.is_device_dispatched(cmd): | 
 |           _define_device_proc_hook(cmd, f) | 
 |  | 
 |     f.write("""\ | 
 |     // clang-format on | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | const ProcHook* GetProcHook(const char* name) { | 
 |     auto begin = std::cbegin(g_proc_hooks); | 
 |     auto end = std::cend(g_proc_hooks); | 
 |     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) { | 
 |     // clang-format off\n""") | 
 |  | 
 |     for ext in _KNOWN_EXTENSIONS: | 
 |       f.write(gencom.indent(1) + 'if (strcmp(name, \"' + ext + | 
 |               '\") == 0) return ProcHook::' + gencom.base_ext_name(ext) + ';\n') | 
 |  | 
 |     f.write("""\ | 
 |     // clang-format on | 
 |     return ProcHook::EXTENSION_UNKNOWN; | 
 | } | 
 |  | 
 | #define UNLIKELY(expr) __builtin_expect((expr), 0) | 
 |  | 
 | #define INIT_PROC(required, obj, proc)                                 \\ | 
 |     do {                                                               \\ | 
 |         data.driver.proc =                                             \\ | 
 |             reinterpret_cast<PFN_vk##proc>(get_proc(obj, "vk" #proc)); \\ | 
 |         if (UNLIKELY(required && !data.driver.proc)) {                 \\ | 
 |             ALOGE("missing " #obj " proc: vk" #proc);                  \\ | 
 |             success = false;                                           \\ | 
 |         }                                                              \\ | 
 |     } while (0) | 
 |  | 
 | #define INIT_PROC_EXT(ext, required, obj, proc) \\ | 
 |     do {                                        \\ | 
 |         if (extensions[ProcHook::ext])          \\ | 
 |             INIT_PROC(required, obj, proc);     \\ | 
 |     } while (0) | 
 |  | 
 | 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\n""") | 
 |  | 
 |     for cmd in gencom.command_list: | 
 |       if _is_instance_driver_table_entry(cmd): | 
 |         gencom.init_proc(cmd, f) | 
 |  | 
 |     f.write("""\ | 
 |     // 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\n""") | 
 |  | 
 |     for cmd in gencom.command_list: | 
 |       if _is_device_driver_table_entry(cmd): | 
 |         gencom.init_proc(cmd, f) | 
 |  | 
 |     f.write("""\ | 
 |     // clang-format on | 
 |  | 
 |     return success; | 
 | } | 
 |  | 
 | const std::pair<const char*, uint32_t> g_promoted_instance_extensions[] = { | 
 |     // clang-format off\n""") | 
 |  | 
 |     for key, value in sorted(gencom.promoted_inst_ext_dict.items()): | 
 |       f.write(gencom.indent(1) + 'std::make_pair("' + key + '", ' + value + '),\n') | 
 |  | 
 |     f.write("""\ | 
 |     // clang-format on | 
 | }; | 
 |  | 
 | std::optional<uint32_t> GetInstanceExtensionPromotedVersion(const char* name) { | 
 |     auto begin = std::cbegin(g_promoted_instance_extensions); | 
 |     auto end = std::cend(g_promoted_instance_extensions); | 
 |     auto iter = | 
 |         std::lower_bound(begin, end, name, | 
 |                          [](const std::pair<const char*, uint32_t>& e, | 
 |                             const char* n) { return strcmp(e.first, n) < 0; }); | 
 |     return (iter < end && strcmp(iter->first, name) == 0) | 
 |                ? std::optional<uint32_t>(iter->second) | 
 |                : std::nullopt; | 
 | } | 
 |  | 
 | uint32_t CountPromotedInstanceExtensions(uint32_t begin_version, | 
 |                                          uint32_t end_version) { | 
 |     auto begin = std::cbegin(g_promoted_instance_extensions); | 
 |     auto end = std::cend(g_promoted_instance_extensions); | 
 |     uint32_t count = 0; | 
 |  | 
 |     for (auto iter = begin; iter != end; iter++) | 
 |         if (iter->second > begin_version && iter->second <= end_version) | 
 |             count++; | 
 |  | 
 |     return count; | 
 | } | 
 |  | 
 | std::vector<const char*> GetPromotedInstanceExtensions(uint32_t begin_version, | 
 |                                                        uint32_t end_version) { | 
 |     auto begin = std::cbegin(g_promoted_instance_extensions); | 
 |     auto end = std::cend(g_promoted_instance_extensions); | 
 |     std::vector<const char*> extensions; | 
 |  | 
 |     for (auto iter = begin; iter != end; iter++) | 
 |         if (iter->second > begin_version && iter->second <= end_version) | 
 |             extensions.emplace_back(iter->first); | 
 |  | 
 |     return extensions; | 
 | } | 
 |  | 
 | }  // namespace driver | 
 | }  // namespace vulkan\n""") | 
 |  | 
 |     f.close() | 
 |   gencom.run_clang_format(genfile) |