blob: 0812e50586c281da0989b4a48f0189ae83a1a391 [file] [log] [blame]
#!/usr/bin/python3 -i
#
# Copyright (c) 2015-2024 The Khronos Group Inc.
# Copyright (c) 2015-2024 Valve Corporation
# Copyright (c) 2015-2024 LunarG, Inc.
#
# 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.
import os
from generators.base_generator import BaseGenerator
from generators.generator_utils import PlatformGuardHelper
# If there is another success code other than VK_SUCCESS
def hasNonVkSuccess(successCodes: list[str]) -> bool:
if successCodes is None or len(successCodes) == 0:
return False
return len(successCodes) > 1 or 'VK_SUCCESS' not in successCodes
class BestPracticesOutputGenerator(BaseGenerator):
def __init__(self):
BaseGenerator.__init__(self)
# Commands which are not autogenerated but still intercepted
# More are added from a first pass over the functions
self.no_autogen_list = [
'vkEnumerateInstanceVersion',
'vkCreateValidationCacheEXT',
'vkDestroyValidationCacheEXT',
'vkMergeValidationCachesEXT',
'vkGetValidationCacheDataEXT',
]
# Commands that require an extra parameter for state sharing between validate/record steps
self.extra_parameter_map = {
'vkCreateShaderModule' : 'chassis::CreateShaderModule',
'vkCreateShadersEXT' : 'chassis::ShaderObject',
'vkAllocateDescriptorSets' : 'vvl::AllocateDescriptorSetsData',
}
self.pipeline_parameter_map = {
'vkCreateGraphicsPipelines' : 'chassis::CreateGraphicsPipelines&',
'vkCreateComputePipelines' : 'chassis::CreateComputePipelines&',
'vkCreateRayTracingPipelinesNV' : 'chassis::CreateRayTracingPipelinesNV&',
'vkCreateRayTracingPipelinesKHR' : 'std::shared_ptr<chassis::CreateRayTracingPipelinesKHR>',
}
# Commands that have a manually written post-call-record step which needs to be called from the autogen'd fcn
self.manual_postcallrecord_list = [
'vkAllocateDescriptorSets',
'vkQueuePresentKHR',
'vkQueueBindSparse',
'vkCreateGraphicsPipelines',
'vkGetPhysicalDeviceSurfaceCapabilitiesKHR',
'vkGetPhysicalDeviceSurfaceCapabilities2KHR',
'vkGetPhysicalDeviceSurfaceCapabilities2EXT',
'vkGetPhysicalDeviceSurfacePresentModesKHR',
'vkGetPhysicalDeviceSurfaceFormatsKHR',
'vkGetPhysicalDeviceSurfaceFormats2KHR',
'vkGetPhysicalDeviceDisplayPlanePropertiesKHR',
'vkGetSwapchainImagesKHR',
'vkBindBufferMemory2',
'vkBindImageMemory2',
# AMD tracked
'vkCreateComputePipelines',
'vkCmdPipelineBarrier',
'vkQueueSubmit',
]
self.extension_info = dict()
def generate(self):
self.write(f'''// *** THIS FILE IS GENERATED - DO NOT EDIT ***
// See {os.path.basename(__file__)} for modifications
/***************************************************************************
*
* Copyright (c) 2015-2024 The Khronos Group Inc.
* Copyright (c) 2015-2024 Valve Corporation
* Copyright (c) 2015-2024 LunarG, Inc.
*
* 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.
****************************************************************************/\n''')
self.write('// NOLINTBEGIN') # Wrap for clang-tidy to ignore
# Build additional pass of functions to ignore
for name, command in self.vk.commands.items():
if command.returnType != 'VkResult':
self.no_autogen_list.append(name)
# This is just to remove un-used commands from be generated
# This can be removed if another use for these commands are needed
if (command.errorCodes is None and not hasNonVkSuccess(command.successCodes)):
self.no_autogen_list.append(name)
if self.filename == 'best_practices.h':
self.generateHeader()
elif self.filename == 'best_practices.cpp':
self.generateSource()
else:
self.write(f'\nFile name {self.filename} has no code to generate\n')
self.write('// NOLINTEND') # Wrap for clang-tidy to ignore
def generateHeader(self):
out = []
guard_helper = PlatformGuardHelper()
# List all Function declarations
for command in [x for x in self.vk.commands.values() if x.name not in self.no_autogen_list]:
out.extend(guard_helper.add_guard(command.protect))
prototype = command.cPrototype.split("VKAPI_CALL ")[1]
prototype = f'void PostCallRecord{prototype[2:]}'
prototype = prototype.replace(');', ', const RecordObject& record_obj) {\n')
prototype = prototype.replace(') {', ') override;\n')
if command.name in self.extra_parameter_map:
prototype = prototype.replace(')', f', {self.extra_parameter_map[command.name]}& chassis_state)')
elif command.name in self.pipeline_parameter_map:
prototype = prototype.replace(')', f', PipelineStates& pipeline_states, {self.pipeline_parameter_map[command.name]} chassis_state)')
out.append(prototype)
out.extend(guard_helper.add_guard(None))
self.write("".join(out))
def generateSource(self):
guard_helper = PlatformGuardHelper()
out = []
out.append('''
#include "best_practices/best_practices_validation.h"
DeprecationData GetDeprecatedData(vvl::Extension extension_name) {
static const DeprecationData empty_deprecated_data{DeprecationReason::Empty, vvl::Extension::Empty};
static const vvl::unordered_map<vvl::Extension, DeprecationData> deprecated_extensions = {
''')
for extension in self.vk.extensions.values():
target = None
reason = None
if extension.promotedTo is not None:
reason = 'DeprecationReason::Promoted'
target = extension.promotedTo
elif extension.obsoletedBy is not None:
reason = 'DeprecationReason::Obsoleted'
target = extension.obsoletedBy
elif extension.deprecatedBy is not None:
reason = 'DeprecationReason::Deprecated'
target = extension.deprecatedBy
else:
continue
if len(target) == 0:
target = 'vvl::Extension::Empty'
elif 'VERSION' in target:
target = f'vvl::Version::_{target}'
else:
target = f'vvl::Extension::_{target}'
out.append(f' {{vvl::Extension::_{extension.name}, {{{reason}, {{{target}}}}}}},\n')
out.append(''' };
auto it = deprecated_extensions.find(extension_name);
return (it == deprecated_extensions.end()) ? empty_deprecated_data : it->second;
}
std::string GetSpecialUse(vvl::Extension extension_name) {
const vvl::unordered_map<vvl::Extension, std::string> special_use_extensions = {
''')
for extension in self.vk.extensions.values():
if extension.specialUse is not None:
out.append(f' {{vvl::Extension::_{extension.name}, "{", ".join(extension.specialUse)}"}},\n')
out.append(''' };
auto it = special_use_extensions.find(extension_name);
return (it == special_use_extensions.end()) ? "" : it->second;
}
''')
for command in [x for x in self.vk.commands.values() if x.name not in self.no_autogen_list]:
paramList = [param.name for param in command.params]
paramList.append('record_obj')
if command.name in self.extra_parameter_map:
paramList.append('chassis_state')
elif command.name in self.pipeline_parameter_map:
paramList.append('pipeline_states')
paramList.append('chassis_state')
params = ', '.join(paramList)
out.extend(guard_helper.add_guard(command.protect, extra_newline=True))
prototype = command.cPrototype.split("VKAPI_CALL ")[1]
prototype = f'void BestPractices::PostCallRecord{prototype[2:]}'
prototype = prototype.replace(');', ', const RecordObject& record_obj) {\n')
if command.name in self.extra_parameter_map:
prototype = prototype.replace(')', f', {self.extra_parameter_map[command.name]}& chassis_state)')
elif command.name in self.pipeline_parameter_map:
prototype = prototype.replace(')', f', PipelineStates& pipeline_states, {self.pipeline_parameter_map[command.name]} chassis_state)')
out.append(prototype)
if command.alias:
# For alias that are promoted, just point to new function, RecordObject will allow us to distinguish the caller
paramList = [param.name for param in command.params]
paramList.append('record_obj')
params = ', '.join(paramList)
out.append(f'PostCallRecord{command.alias[2:]}({params});')
else:
out.append(f'BaseClass::PostCallRecord{command.name[2:]}({params});\n')
if command.name in self.manual_postcallrecord_list:
out.append(f'ManualPostCallRecord{command.name[2:]}({params});\n')
if hasNonVkSuccess(command.successCodes):
out.append('''
if (record_obj.result > VK_SUCCESS) {
LogPositiveSuccessCode(record_obj);
return;
}''')
if command.errorCodes is not None:
out.append('''
if (record_obj.result < VK_SUCCESS) {
LogErrorCode(record_obj);
}''')
out.append('}\n')
out.extend(guard_helper.add_guard(None, extra_newline=True))
self.write(''.join(out))