|  | #!/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. | 
|  |  | 
|  | """Provide the utilities for framework generation. | 
|  | """ | 
|  |  | 
|  | import os | 
|  | import subprocess | 
|  | import xml.etree.ElementTree as element_tree | 
|  |  | 
|  | # Extensions unsupported on Android. | 
|  | _BLOCKED_EXTENSIONS = [ | 
|  | 'VK_EXT_acquire_xlib_display', | 
|  | 'VK_EXT_direct_mode_display', | 
|  | 'VK_EXT_directfb_surface', | 
|  | 'VK_EXT_display_control', | 
|  | 'VK_EXT_display_surface_counter', | 
|  | 'VK_EXT_full_screen_exclusive', | 
|  | 'VK_EXT_headless_surface', | 
|  | 'VK_EXT_metal_surface', | 
|  | 'VK_FUCHSIA_imagepipe_surface', | 
|  | 'VK_GGP_stream_descriptor_surface', | 
|  | 'VK_KHR_display', | 
|  | 'VK_KHR_display_swapchain', | 
|  | 'VK_KHR_external_fence_win32', | 
|  | 'VK_KHR_external_memory_win32', | 
|  | 'VK_KHR_external_semaphore_win32', | 
|  | 'VK_KHR_mir_surface', | 
|  | 'VK_KHR_wayland_surface', | 
|  | 'VK_KHR_win32_keyed_mutex', | 
|  | 'VK_KHR_win32_surface', | 
|  | 'VK_KHR_xcb_surface', | 
|  | 'VK_KHR_xlib_surface', | 
|  | 'VK_MVK_ios_surface', | 
|  | 'VK_MVK_macos_surface', | 
|  | 'VK_NN_vi_surface', | 
|  | 'VK_NV_cooperative_matrix', | 
|  | 'VK_NV_coverage_reduction_mode', | 
|  | 'VK_NV_external_memory_win32', | 
|  | 'VK_NV_win32_keyed_mutex', | 
|  | 'VK_NVX_image_view_handle', | 
|  | ] | 
|  |  | 
|  | # Extensions having functions exported by the loader. | 
|  | _EXPORTED_EXTENSIONS = [ | 
|  | 'VK_ANDROID_external_memory_android_hardware_buffer', | 
|  | 'VK_KHR_android_surface', | 
|  | 'VK_KHR_surface', | 
|  | 'VK_KHR_swapchain', | 
|  | ] | 
|  |  | 
|  | # Functions optional on Android even if extension is advertised. | 
|  | _OPTIONAL_COMMANDS = [ | 
|  | 'vkGetSwapchainGrallocUsageANDROID', | 
|  | 'vkGetSwapchainGrallocUsage2ANDROID', | 
|  | ] | 
|  |  | 
|  | # Dict for mapping dispatch table to a type. | 
|  | _DISPATCH_TYPE_DICT = { | 
|  | 'VkInstance ': 'Instance', | 
|  | 'VkPhysicalDevice ': 'Instance', | 
|  | 'VkDevice ': 'Device', | 
|  | 'VkQueue ': 'Device', | 
|  | 'VkCommandBuffer ': 'Device' | 
|  | } | 
|  |  | 
|  | # Dict for mapping a function to its alias. | 
|  | alias_dict = {} | 
|  |  | 
|  | # List of all the Vulkan functions. | 
|  | command_list = [] | 
|  |  | 
|  | # Dict for mapping a function to an extension. | 
|  | extension_dict = {} | 
|  |  | 
|  | # Dict for mapping a function to all its parameters. | 
|  | param_dict = {} | 
|  |  | 
|  | # Dict for mapping a function to its return type. | 
|  | return_type_dict = {} | 
|  |  | 
|  | # List of the sorted Vulkan version codes. e.g. '1_0', '1_1'. | 
|  | version_code_list = [] | 
|  |  | 
|  | # Dict for mapping a function to the core Vulkan API version. | 
|  | version_dict = {} | 
|  |  | 
|  | # Dict for mapping a promoted instance extension to the core Vulkan API version. | 
|  | promoted_inst_ext_dict = {} | 
|  |  | 
|  |  | 
|  | def indent(num): | 
|  | """Returns the requested indents. | 
|  |  | 
|  | Args: | 
|  | num: Number of the 4-space indents. | 
|  | """ | 
|  | return '    ' * num | 
|  |  | 
|  |  | 
|  | def copyright_and_warning(year): | 
|  | """Returns the standard copyright and warning codes. | 
|  |  | 
|  | Args: | 
|  | year: An integer year for the copyright. | 
|  | """ | 
|  | return """\ | 
|  | /* | 
|  | * Copyright """ + str(year) + """ 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. | 
|  | */ | 
|  |  | 
|  | // WARNING: This file is generated. See ../README.md for instructions. | 
|  |  | 
|  | """ | 
|  |  | 
|  |  | 
|  | def run_clang_format(args): | 
|  | """Run clang format on the file. | 
|  |  | 
|  | Args: | 
|  | args: The file to be formatted. | 
|  | """ | 
|  | clang_call = ['clang-format', '--style', 'file', '-i', args] | 
|  | subprocess.check_call(clang_call) | 
|  |  | 
|  |  | 
|  | def is_extension_internal(ext): | 
|  | """Returns true if an extension is internal to the loader and drivers. | 
|  |  | 
|  | The loader should not enumerate this extension. | 
|  |  | 
|  | Args: | 
|  | ext: Vulkan extension name. | 
|  | """ | 
|  | return ext == 'VK_ANDROID_native_buffer' | 
|  |  | 
|  |  | 
|  | def base_name(cmd): | 
|  | """Returns a function name without the 'vk' prefix. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | return cmd[2:] | 
|  |  | 
|  |  | 
|  | def base_ext_name(ext): | 
|  | """Returns an extension name without the 'VK_' prefix. | 
|  |  | 
|  | Args: | 
|  | ext: Vulkan extension name. | 
|  | """ | 
|  | return ext[3:] | 
|  |  | 
|  |  | 
|  | def version_code(version): | 
|  | """Returns the version code from a version string. | 
|  |  | 
|  | Args: | 
|  | version: Vulkan version string. | 
|  | """ | 
|  | return version[11:] | 
|  |  | 
|  |  | 
|  | def version_2_api_version(version): | 
|  | """Returns the api version from a version string. | 
|  |  | 
|  | Args: | 
|  | version: Vulkan version string. | 
|  | """ | 
|  | return 'VK_API' + version[2:] | 
|  |  | 
|  |  | 
|  | def is_function_supported(cmd): | 
|  | """Returns true if a function is core or from a supportable extension. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | if cmd not in extension_dict: | 
|  | return True | 
|  | else: | 
|  | if extension_dict[cmd] not in _BLOCKED_EXTENSIONS: | 
|  | return True | 
|  | return False | 
|  |  | 
|  |  | 
|  | def get_dispatch_table_type(cmd): | 
|  | """Returns the dispatch table type for a function. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | if cmd not in param_dict: | 
|  | return None | 
|  |  | 
|  | if param_dict[cmd]: | 
|  | return _DISPATCH_TYPE_DICT.get(param_dict[cmd][0][0], 'Global') | 
|  | return 'Global' | 
|  |  | 
|  |  | 
|  | def is_globally_dispatched(cmd): | 
|  | """Returns true if the function is global, which is not dispatched. | 
|  |  | 
|  | Only global functions and functions handled in the loader top without calling | 
|  | into lower layers are not dispatched. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | return is_function_supported(cmd) and get_dispatch_table_type(cmd) == 'Global' | 
|  |  | 
|  |  | 
|  | def is_instance_dispatched(cmd): | 
|  | """Returns true for functions that can have instance-specific dispatch. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | return (is_function_supported(cmd) and | 
|  | get_dispatch_table_type(cmd) == 'Instance') | 
|  |  | 
|  |  | 
|  | def is_device_dispatched(cmd): | 
|  | """Returns true for functions that can have device-specific dispatch. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | return is_function_supported(cmd) and get_dispatch_table_type(cmd) == 'Device' | 
|  |  | 
|  |  | 
|  | def is_extension_exported(ext): | 
|  | """Returns true if an extension has functions exported by the loader. | 
|  |  | 
|  | E.g. applications can directly link to an extension function. | 
|  |  | 
|  | Args: | 
|  | ext: Vulkan extension name. | 
|  | """ | 
|  | return ext in _EXPORTED_EXTENSIONS | 
|  |  | 
|  |  | 
|  | def is_function_exported(cmd): | 
|  | """Returns true if a function is exported from the Android Vulkan library. | 
|  |  | 
|  | Functions in the core API and in loader extensions are exported. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | if is_function_supported(cmd): | 
|  | if cmd in extension_dict: | 
|  | return is_extension_exported(extension_dict[cmd]) | 
|  | return True | 
|  | return False | 
|  |  | 
|  |  | 
|  | def is_instance_dispatch_table_entry(cmd): | 
|  | """Returns true if a function is exported and instance-dispatched. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | if cmd == 'vkEnumerateDeviceLayerProperties': | 
|  | # deprecated, unused internally - @dbd33bc | 
|  | return False | 
|  | return is_function_exported(cmd) and is_instance_dispatched(cmd) | 
|  |  | 
|  |  | 
|  | def is_device_dispatch_table_entry(cmd): | 
|  | """Returns true if a function is exported and device-dispatched. | 
|  |  | 
|  | Args: | 
|  | cmd: Vulkan function name. | 
|  | """ | 
|  | return is_function_exported(cmd) and is_device_dispatched(cmd) | 
|  |  | 
|  |  | 
|  | def init_proc(name, f): | 
|  | """Emits code to invoke INIT_PROC or INIT_PROC_EXT. | 
|  |  | 
|  | Args: | 
|  | name: Vulkan function name. | 
|  | f: Output file handle. | 
|  | """ | 
|  | f.write(indent(1)) | 
|  | if name in extension_dict: | 
|  | f.write('INIT_PROC_EXT(' + base_ext_name(extension_dict[name]) + ', ') | 
|  | else: | 
|  | f.write('INIT_PROC(') | 
|  |  | 
|  | if name in _OPTIONAL_COMMANDS: | 
|  | f.write('false, ') | 
|  | elif version_dict[name] == 'VK_VERSION_1_0': | 
|  | f.write('true, ') | 
|  | else: | 
|  | f.write('false, ') | 
|  |  | 
|  | if is_instance_dispatched(name): | 
|  | f.write('instance, ') | 
|  | else: | 
|  | f.write('dev, ') | 
|  |  | 
|  | f.write(base_name(name) + ');\n') | 
|  |  | 
|  |  | 
|  | def parse_vulkan_registry(): | 
|  | """Parses Vulkan registry into the below global variables. | 
|  |  | 
|  | alias_dict | 
|  | command_list | 
|  | extension_dict | 
|  | param_dict | 
|  | return_type_dict | 
|  | version_code_list | 
|  | version_dict | 
|  | promoted_inst_ext_dict | 
|  | """ | 
|  | registry = os.path.join(os.path.dirname(__file__), '..', '..', '..', '..', | 
|  | 'external', 'vulkan-headers', 'registry', 'vk.xml') | 
|  | tree = element_tree.parse(registry) | 
|  | root = tree.getroot() | 
|  | for commands in root.iter('commands'): | 
|  | for command in commands: | 
|  | if command.tag == 'command': | 
|  | parameter_list = [] | 
|  | protoset = False | 
|  | cmd_name = '' | 
|  | cmd_type = '' | 
|  | if command.get('alias') is not None: | 
|  | alias = command.get('alias') | 
|  | cmd_name = command.get('name') | 
|  | alias_dict[cmd_name] = alias | 
|  | command_list.append(cmd_name) | 
|  | param_dict[cmd_name] = param_dict[alias].copy() | 
|  | return_type_dict[cmd_name] = return_type_dict[alias] | 
|  | for params in command: | 
|  | if params.tag == 'param': | 
|  | param_type = '' | 
|  | if params.text is not None and params.text.strip(): | 
|  | param_type = params.text.strip() + ' ' | 
|  | type_val = params.find('type') | 
|  | param_type = param_type + type_val.text | 
|  | if type_val.tail is not None: | 
|  | param_type += type_val.tail.strip() + ' ' | 
|  | pname = params.find('name') | 
|  | param_name = pname.text | 
|  | if pname.tail is not None and pname.tail.strip(): | 
|  | parameter_list.append( | 
|  | (param_type, param_name, pname.tail.strip())) | 
|  | else: | 
|  | parameter_list.append((param_type, param_name)) | 
|  | if params.tag == 'proto': | 
|  | for c in params: | 
|  | if c.tag == 'type': | 
|  | cmd_type = c.text | 
|  | if c.tag == 'name': | 
|  | cmd_name = c.text | 
|  | protoset = True | 
|  | command_list.append(cmd_name) | 
|  | return_type_dict[cmd_name] = cmd_type | 
|  | if protoset: | 
|  | param_dict[cmd_name] = parameter_list.copy() | 
|  |  | 
|  | for exts in root.iter('extensions'): | 
|  | for extension in exts: | 
|  | apiversion = 'VK_VERSION_1_0' | 
|  | if extension.tag == 'extension': | 
|  | extname = extension.get('name') | 
|  | if (extension.get('type') == 'instance' and | 
|  | extension.get('promotedto') is not None): | 
|  | promoted_inst_ext_dict[extname] = \ | 
|  | version_2_api_version(extension.get('promotedto')) | 
|  | for req in extension: | 
|  | if req.get('feature') is not None: | 
|  | apiversion = req.get('feature') | 
|  | for commands in req: | 
|  | if commands.tag == 'command': | 
|  | cmd_name = commands.get('name') | 
|  | if cmd_name not in extension_dict: | 
|  | extension_dict[cmd_name] = extname | 
|  | version_dict[cmd_name] = apiversion | 
|  |  | 
|  | for feature in root.iter('feature'): | 
|  | apiversion = feature.get('name') | 
|  | for req in feature: | 
|  | for command in req: | 
|  | if command.tag == 'command': | 
|  | cmd_name = command.get('name') | 
|  | if cmd_name in command_list: | 
|  | version_dict[cmd_name] = apiversion | 
|  |  | 
|  | version_code_set = set() | 
|  | for version in version_dict.values(): | 
|  | version_code_set.add(version_code(version)) | 
|  | for code in sorted(version_code_set): | 
|  | version_code_list.append(code) |