mirror of
				https://github.com/RetroDECK/Duckstation.git
				synced 2025-04-10 19:15:14 +00:00 
			
		
		
		
	
		
			
	
	
		
			218 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			218 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// Copyright (C) 2014-2016 LunarG, Inc.
							 | 
						||
| 
								 | 
							
								// Copyright (C) 2018-2020 Google, Inc.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// All rights reserved.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// Redistribution and use in source and binary forms, with or without
							 | 
						||
| 
								 | 
							
								// modification, are permitted provided that the following conditions
							 | 
						||
| 
								 | 
							
								// are met:
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								//    Redistributions of source code must retain the above copyright
							 | 
						||
| 
								 | 
							
								//    notice, this list of conditions and the following disclaimer.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								//    Redistributions in binary form must reproduce the above
							 | 
						||
| 
								 | 
							
								//    copyright notice, this list of conditions and the following
							 | 
						||
| 
								 | 
							
								//    disclaimer in the documentation and/or other materials provided
							 | 
						||
| 
								 | 
							
								//    with the distribution.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								//    Neither the name of 3Dlabs Inc. Ltd. nor the names of its
							 | 
						||
| 
								 | 
							
								//    contributors may be used to endorse or promote products derived
							 | 
						||
| 
								 | 
							
								//    from this software without specific prior written permission.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
							 | 
						||
| 
								 | 
							
								// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
							 | 
						||
| 
								 | 
							
								// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
							 | 
						||
| 
								 | 
							
								// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
							 | 
						||
| 
								 | 
							
								// COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
							 | 
						||
| 
								 | 
							
								// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
							 | 
						||
| 
								 | 
							
								// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
							 | 
						||
| 
								 | 
							
								// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
							 | 
						||
| 
								 | 
							
								// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
							 | 
						||
| 
								 | 
							
								// LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
							 | 
						||
| 
								 | 
							
								// ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
							 | 
						||
| 
								 | 
							
								// POSSIBILITY OF SUCH DAMAGE.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// Call into SPIRV-Tools to disassemble, validate, and optimize.
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if ENABLE_OPT
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <cstdio>
							 | 
						||
| 
								 | 
							
								#include <iostream>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "SpvTools.h"
							 | 
						||
| 
								 | 
							
								#include "spirv-tools/optimizer.hpp"
							 | 
						||
| 
								 | 
							
								#include "spirv-tools/libspirv.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace glslang {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Translate glslang's view of target versioning to what SPIRV-Tools uses.
							 | 
						||
| 
								 | 
							
								spv_target_env MapToSpirvToolsEnv(const SpvVersion& spvVersion, spv::SpvBuildLogger* logger)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    switch (spvVersion.vulkan) {
							 | 
						||
| 
								 | 
							
								    case glslang::EShTargetVulkan_1_0:
							 | 
						||
| 
								 | 
							
								        return spv_target_env::SPV_ENV_VULKAN_1_0;
							 | 
						||
| 
								 | 
							
								    case glslang::EShTargetVulkan_1_1:
							 | 
						||
| 
								 | 
							
								        switch (spvVersion.spv) {
							 | 
						||
| 
								 | 
							
								        case EShTargetSpv_1_0:
							 | 
						||
| 
								 | 
							
								        case EShTargetSpv_1_1:
							 | 
						||
| 
								 | 
							
								        case EShTargetSpv_1_2:
							 | 
						||
| 
								 | 
							
								        case EShTargetSpv_1_3:
							 | 
						||
| 
								 | 
							
								            return spv_target_env::SPV_ENV_VULKAN_1_1;
							 | 
						||
| 
								 | 
							
								        case EShTargetSpv_1_4:
							 | 
						||
| 
								 | 
							
								            return spv_target_env::SPV_ENV_VULKAN_1_1_SPIRV_1_4;
							 | 
						||
| 
								 | 
							
								        default:
							 | 
						||
| 
								 | 
							
								            logger->missingFunctionality("Target version for SPIRV-Tools validator");
							 | 
						||
| 
								 | 
							
								            return spv_target_env::SPV_ENV_VULKAN_1_1;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    case glslang::EShTargetVulkan_1_2:
							 | 
						||
| 
								 | 
							
								        return spv_target_env::SPV_ENV_VULKAN_1_2;
							 | 
						||
| 
								 | 
							
								    default:
							 | 
						||
| 
								 | 
							
								        break;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (spvVersion.openGl > 0)
							 | 
						||
| 
								 | 
							
								        return spv_target_env::SPV_ENV_OPENGL_4_5;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    logger->missingFunctionality("Target version for SPIRV-Tools validator");
							 | 
						||
| 
								 | 
							
								    return spv_target_env::SPV_ENV_UNIVERSAL_1_0;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Use the SPIRV-Tools disassembler to print SPIR-V.
							 | 
						||
| 
								 | 
							
								void SpirvToolsDisassemble(std::ostream& out, const std::vector<unsigned int>& spirv)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    // disassemble
							 | 
						||
| 
								 | 
							
								    spv_context context = spvContextCreate(SPV_ENV_UNIVERSAL_1_3);
							 | 
						||
| 
								 | 
							
								    spv_text text;
							 | 
						||
| 
								 | 
							
								    spv_diagnostic diagnostic = nullptr;
							 | 
						||
| 
								 | 
							
								    spvBinaryToText(context, spirv.data(), spirv.size(),
							 | 
						||
| 
								 | 
							
								        SPV_BINARY_TO_TEXT_OPTION_FRIENDLY_NAMES | SPV_BINARY_TO_TEXT_OPTION_INDENT,
							 | 
						||
| 
								 | 
							
								        &text, &diagnostic);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // dump
							 | 
						||
| 
								 | 
							
								    if (diagnostic == nullptr)
							 | 
						||
| 
								 | 
							
								        out << text->str;
							 | 
						||
| 
								 | 
							
								    else
							 | 
						||
| 
								 | 
							
								        spvDiagnosticPrint(diagnostic);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // teardown
							 | 
						||
| 
								 | 
							
								    spvDiagnosticDestroy(diagnostic);
							 | 
						||
| 
								 | 
							
								    spvContextDestroy(context);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Apply the SPIRV-Tools validator to generated SPIR-V.
							 | 
						||
| 
								 | 
							
								void SpirvToolsValidate(const glslang::TIntermediate& intermediate, std::vector<unsigned int>& spirv,
							 | 
						||
| 
								 | 
							
								                        spv::SpvBuildLogger* logger, bool prelegalization)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    // validate
							 | 
						||
| 
								 | 
							
								    spv_context context = spvContextCreate(MapToSpirvToolsEnv(intermediate.getSpv(), logger));
							 | 
						||
| 
								 | 
							
								    spv_const_binary_t binary = { spirv.data(), spirv.size() };
							 | 
						||
| 
								 | 
							
								    spv_diagnostic diagnostic = nullptr;
							 | 
						||
| 
								 | 
							
								    spv_validator_options options = spvValidatorOptionsCreate();
							 | 
						||
| 
								 | 
							
								    spvValidatorOptionsSetRelaxBlockLayout(options, intermediate.usingHlslOffsets());
							 | 
						||
| 
								 | 
							
								    spvValidatorOptionsSetBeforeHlslLegalization(options, prelegalization);
							 | 
						||
| 
								 | 
							
								    spvValidateWithOptions(context, options, &binary, &diagnostic);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // report
							 | 
						||
| 
								 | 
							
								    if (diagnostic != nullptr) {
							 | 
						||
| 
								 | 
							
								        logger->error("SPIRV-Tools Validation Errors");
							 | 
						||
| 
								 | 
							
								        logger->error(diagnostic->error);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // tear down
							 | 
						||
| 
								 | 
							
								    spvValidatorOptionsDestroy(options);
							 | 
						||
| 
								 | 
							
								    spvDiagnosticDestroy(diagnostic);
							 | 
						||
| 
								 | 
							
								    spvContextDestroy(context);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Apply the SPIRV-Tools optimizer to generated SPIR-V, for the purpose of
							 | 
						||
| 
								 | 
							
								// legalizing HLSL SPIR-V.
							 | 
						||
| 
								 | 
							
								void SpirvToolsLegalize(const glslang::TIntermediate& intermediate, std::vector<unsigned int>& spirv,
							 | 
						||
| 
								 | 
							
								                        spv::SpvBuildLogger* logger, const SpvOptions* options)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    spv_target_env target_env = SPV_ENV_UNIVERSAL_1_2;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    spvtools::Optimizer optimizer(target_env);
							 | 
						||
| 
								 | 
							
								    optimizer.SetMessageConsumer(
							 | 
						||
| 
								 | 
							
								        [](spv_message_level_t level, const char *source, const spv_position_t &position, const char *message) {
							 | 
						||
| 
								 | 
							
								            auto &out = std::cerr;
							 | 
						||
| 
								 | 
							
								            switch (level)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								            case SPV_MSG_FATAL:
							 | 
						||
| 
								 | 
							
								            case SPV_MSG_INTERNAL_ERROR:
							 | 
						||
| 
								 | 
							
								            case SPV_MSG_ERROR:
							 | 
						||
| 
								 | 
							
								                out << "error: ";
							 | 
						||
| 
								 | 
							
								                break;
							 | 
						||
| 
								 | 
							
								            case SPV_MSG_WARNING:
							 | 
						||
| 
								 | 
							
								                out << "warning: ";
							 | 
						||
| 
								 | 
							
								                break;
							 | 
						||
| 
								 | 
							
								            case SPV_MSG_INFO:
							 | 
						||
| 
								 | 
							
								            case SPV_MSG_DEBUG:
							 | 
						||
| 
								 | 
							
								                out << "info: ";
							 | 
						||
| 
								 | 
							
								                break;
							 | 
						||
| 
								 | 
							
								            default:
							 | 
						||
| 
								 | 
							
								                break;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            if (source)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                out << source << ":";
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            out << position.line << ":" << position.column << ":" << position.index << ":";
							 | 
						||
| 
								 | 
							
								            if (message)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                out << " " << message;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            out << std::endl;
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // If debug (specifically source line info) is being generated, propagate
							 | 
						||
| 
								 | 
							
								    // line information into all SPIR-V instructions. This avoids loss of
							 | 
						||
| 
								 | 
							
								    // information when instructions are deleted or moved. Later, remove
							 | 
						||
| 
								 | 
							
								    // redundant information to minimize final SPRIR-V size.
							 | 
						||
| 
								 | 
							
								    if (options->generateDebugInfo) {
							 | 
						||
| 
								 | 
							
								        optimizer.RegisterPass(spvtools::CreatePropagateLineInfoPass());
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateWrapOpKillPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateDeadBranchElimPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateMergeReturnPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateInlineExhaustivePass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateEliminateDeadFunctionsPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateScalarReplacementPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateLocalAccessChainConvertPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateLocalSingleBlockLoadStoreElimPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateLocalSingleStoreElimPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateSimplificationPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateVectorDCEPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateDeadInsertElimPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateDeadBranchElimPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateBlockMergePass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateLocalMultiStoreElimPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateIfConversionPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateSimplificationPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateVectorDCEPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateDeadInsertElimPass());
							 | 
						||
| 
								 | 
							
								    if (options->optimizeSize) {
							 | 
						||
| 
								 | 
							
								        optimizer.RegisterPass(spvtools::CreateRedundancyEliminationPass());
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
							 | 
						||
| 
								 | 
							
								    optimizer.RegisterPass(spvtools::CreateCFGCleanupPass());
							 | 
						||
| 
								 | 
							
								    if (options->generateDebugInfo) {
							 | 
						||
| 
								 | 
							
								        optimizer.RegisterPass(spvtools::CreateRedundantLineInfoElimPass());
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    spvtools::OptimizerOptions spvOptOptions;
							 | 
						||
| 
								 | 
							
								    optimizer.SetTargetEnv(MapToSpirvToolsEnv(intermediate.getSpv(), logger));
							 | 
						||
| 
								 | 
							
								    spvOptOptions.set_run_validator(false); // The validator may run as a separate step later on
							 | 
						||
| 
								 | 
							
								    optimizer.Run(spirv.data(), spirv.size(), &spirv, spvOptOptions);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								}; // end namespace glslang
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif
							 |