| // Copyright 2013 The Flutter Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <filesystem> |
| #include <system_error> |
| |
| #include "flutter/fml/backtrace.h" |
| #include "flutter/fml/command_line.h" |
| #include "flutter/fml/file.h" |
| #include "flutter/fml/mapping.h" |
| #include "impeller/compiler/compiler.h" |
| #include "impeller/compiler/shader_bundle.h" |
| #include "impeller/compiler/source_options.h" |
| #include "impeller/compiler/switches.h" |
| #include "impeller/compiler/types.h" |
| #include "impeller/compiler/utilities.h" |
| |
| namespace impeller { |
| namespace compiler { |
| |
| /// Run the shader compiler to geneate SkSL. |
| /// If there is an error, prints error text and returns `nullptr`. |
| static std::shared_ptr<fml::Mapping> CompileSkSL( |
| std::shared_ptr<fml::Mapping> source_file_mapping, |
| SourceOptions& options, |
| Reflector::Options& reflector_options) { |
| SourceOptions sksl_options = options; |
| sksl_options.target_platform = TargetPlatform::kSkSL; |
| |
| Reflector::Options sksl_reflector_options = reflector_options; |
| sksl_reflector_options.target_platform = TargetPlatform::kSkSL; |
| |
| Compiler sksl_compiler = Compiler(std::move(source_file_mapping), |
| sksl_options, sksl_reflector_options); |
| if (!sksl_compiler.IsValid()) { |
| std::cerr << "Compilation to SkSL failed." << std::endl; |
| std::cerr << sksl_compiler.GetErrorMessages() << std::endl; |
| return nullptr; |
| } |
| return sksl_compiler.GetSLShaderSource(); |
| } |
| |
| /// Outputs artifacts for a single compiler invocation and option configuration. |
| /// If there is an error, prints error text and returns `false`. |
| static bool OutputArtifacts(Compiler& compiler, |
| Switches& switches, |
| std::shared_ptr<fml::Mapping> source_file_mapping, |
| SourceOptions& options, |
| Reflector::Options& reflector_options) { |
| // -------------------------------------------------------------------------- |
| /// 1. Invoke the compiler to generate SkSL if needed. |
| /// |
| |
| std::shared_ptr<fml::Mapping> sksl_mapping; |
| if (switches.iplr && TargetPlatformBundlesSkSL(switches.target_platform)) { |
| sksl_mapping = |
| CompileSkSL(std::move(source_file_mapping), options, reflector_options); |
| if (!sksl_mapping) { |
| return false; |
| } |
| } |
| |
| // -------------------------------------------------------------------------- |
| /// 2. Output the source file. When in IPLR/RuntimeStage mode, output the |
| /// serialized IPLR flatbuffer. |
| /// |
| |
| auto sl_file_name = std::filesystem::absolute( |
| std::filesystem::current_path() / switches.sl_file_name); |
| if (switches.iplr) { |
| auto reflector = compiler.GetReflector(); |
| if (reflector == nullptr) { |
| std::cerr << "Could not create reflector." << std::endl; |
| return false; |
| } |
| auto stage_data = reflector->GetRuntimeStageData(); |
| if (!stage_data) { |
| std::cerr << "Runtime stage information was nil." << std::endl; |
| return false; |
| } |
| if (sksl_mapping) { |
| stage_data->SetSkSLData(std::move(sksl_mapping)); |
| } |
| auto stage_data_mapping = options.json_format |
| ? stage_data->CreateJsonMapping() |
| : stage_data->CreateMapping(); |
| if (!stage_data_mapping) { |
| std::cerr << "Runtime stage data could not be created." << std::endl; |
| return false; |
| } |
| if (!fml::WriteAtomically(*switches.working_directory, // |
| Utf8FromPath(sl_file_name).c_str(), // |
| *stage_data_mapping // |
| )) { |
| std::cerr << "Could not write file to " << switches.sl_file_name |
| << std::endl; |
| return false; |
| } |
| // Tools that consume the runtime stage data expect the access mode to |
| // be 0644. |
| if (!SetPermissiveAccess(sl_file_name)) { |
| return false; |
| } |
| } else { |
| if (!fml::WriteAtomically(*switches.working_directory, |
| Utf8FromPath(sl_file_name).c_str(), |
| *compiler.GetSLShaderSource())) { |
| std::cerr << "Could not write file to " << switches.sl_file_name |
| << std::endl; |
| return false; |
| } |
| } |
| |
| // -------------------------------------------------------------------------- |
| /// 3. Output shader reflection data. |
| /// May include a JSON file, a C++ header, and/or a C++ TU. |
| /// |
| |
| if (TargetPlatformNeedsReflection(options.target_platform)) { |
| if (!switches.reflection_json_name.empty()) { |
| auto reflection_json_name = std::filesystem::absolute( |
| std::filesystem::current_path() / switches.reflection_json_name); |
| if (!fml::WriteAtomically( |
| *switches.working_directory, |
| Utf8FromPath(reflection_json_name).c_str(), |
| *compiler.GetReflector()->GetReflectionJSON())) { |
| std::cerr << "Could not write reflection json to " |
| << switches.reflection_json_name << std::endl; |
| return false; |
| } |
| } |
| |
| if (!switches.reflection_header_name.empty()) { |
| auto reflection_header_name = |
| std::filesystem::absolute(std::filesystem::current_path() / |
| switches.reflection_header_name.c_str()); |
| if (!fml::WriteAtomically( |
| *switches.working_directory, |
| Utf8FromPath(reflection_header_name).c_str(), |
| *compiler.GetReflector()->GetReflectionHeader())) { |
| std::cerr << "Could not write reflection header to " |
| << switches.reflection_header_name << std::endl; |
| return false; |
| } |
| } |
| |
| if (!switches.reflection_cc_name.empty()) { |
| auto reflection_cc_name = |
| std::filesystem::absolute(std::filesystem::current_path() / |
| switches.reflection_cc_name.c_str()); |
| if (!fml::WriteAtomically(*switches.working_directory, |
| Utf8FromPath(reflection_cc_name).c_str(), |
| *compiler.GetReflector()->GetReflectionCC())) { |
| std::cerr << "Could not write reflection CC to " |
| << switches.reflection_cc_name << std::endl; |
| return false; |
| } |
| } |
| } |
| |
| // -------------------------------------------------------------------------- |
| /// 4. Output a depfile. |
| /// |
| |
| if (!switches.depfile_path.empty()) { |
| std::string result_file; |
| switch (switches.target_platform) { |
| case TargetPlatform::kMetalDesktop: |
| case TargetPlatform::kMetalIOS: |
| case TargetPlatform::kOpenGLES: |
| case TargetPlatform::kOpenGLDesktop: |
| case TargetPlatform::kRuntimeStageMetal: |
| case TargetPlatform::kRuntimeStageGLES: |
| case TargetPlatform::kRuntimeStageVulkan: |
| case TargetPlatform::kSkSL: |
| case TargetPlatform::kVulkan: |
| result_file = switches.sl_file_name; |
| break; |
| case TargetPlatform::kUnknown: |
| result_file = switches.spirv_file_name; |
| break; |
| } |
| auto depfile_path = std::filesystem::absolute( |
| std::filesystem::current_path() / switches.depfile_path.c_str()); |
| if (!fml::WriteAtomically(*switches.working_directory, |
| Utf8FromPath(depfile_path).c_str(), |
| *compiler.CreateDepfileContents({result_file}))) { |
| std::cerr << "Could not write depfile to " << switches.depfile_path |
| << std::endl; |
| return false; |
| } |
| } |
| |
| return true; |
| } |
| |
| bool Main(const fml::CommandLine& command_line) { |
| fml::InstallCrashHandler(); |
| if (command_line.HasOption("help")) { |
| Switches::PrintHelp(std::cout); |
| return true; |
| } |
| |
| Switches switches(command_line); |
| if (!switches.AreValid(std::cerr)) { |
| std::cerr << "Invalid flags specified." << std::endl; |
| Switches::PrintHelp(std::cerr); |
| return false; |
| } |
| SourceOptions options; |
| options.target_platform = switches.target_platform; |
| options.source_language = switches.source_language; |
| if (switches.input_type == SourceType::kUnknown) { |
| options.type = SourceTypeFromFileName(switches.source_file_name); |
| } else { |
| options.type = switches.input_type; |
| } |
| options.working_directory = switches.working_directory; |
| options.file_name = switches.source_file_name; |
| options.include_dirs = switches.include_directories; |
| options.defines = switches.defines; |
| options.entry_point_name = EntryPointFunctionNameFromSourceName( |
| switches.source_file_name, options.type, options.source_language, |
| switches.entry_point); |
| options.json_format = switches.json_format; |
| options.gles_language_version = switches.gles_language_version; |
| options.metal_version = switches.metal_version; |
| options.use_half_textures = switches.use_half_textures; |
| options.require_framebuffer_fetch = switches.require_framebuffer_fetch; |
| |
| if (!switches.shader_bundle.empty()) { |
| // Invoke the compiler multiple times to build a shader bundle with the |
| // given shader_bundle spec. |
| return GenerateShaderBundle(switches, options); |
| } |
| |
| std::shared_ptr<fml::FileMapping> source_file_mapping = |
| fml::FileMapping::CreateReadOnly(switches.source_file_name); |
| if (!source_file_mapping) { |
| std::cerr << "Could not open input file." << std::endl; |
| return false; |
| } |
| |
| // Invoke the compiler and generate reflection data for a single shader or |
| // runtime stage IPLR. |
| |
| Reflector::Options reflector_options; |
| reflector_options.target_platform = switches.target_platform; |
| reflector_options.entry_point_name = options.entry_point_name; |
| reflector_options.shader_name = |
| InferShaderNameFromPath(switches.source_file_name); |
| reflector_options.header_file_name = Utf8FromPath( |
| std::filesystem::path{switches.reflection_header_name}.filename()); |
| |
| Compiler compiler(source_file_mapping, options, reflector_options); |
| if (!compiler.IsValid()) { |
| std::cerr << "Compilation failed." << std::endl; |
| std::cerr << compiler.GetErrorMessages() << std::endl; |
| return false; |
| } |
| |
| auto spriv_file_name = std::filesystem::absolute( |
| std::filesystem::current_path() / switches.spirv_file_name); |
| if (!fml::WriteAtomically(*switches.working_directory, |
| Utf8FromPath(spriv_file_name).c_str(), |
| *compiler.GetSPIRVAssembly())) { |
| std::cerr << "Could not write file to " << switches.spirv_file_name |
| << std::endl; |
| return false; |
| } |
| |
| if (!OutputArtifacts(compiler, switches, std::move(source_file_mapping), |
| options, reflector_options)) { |
| return false; |
| } |
| |
| return true; |
| } |
| |
| } // namespace compiler |
| } // namespace impeller |
| |
| int main(int argc, char const* argv[]) { |
| return impeller::compiler::Main( |
| fml::CommandLineFromPlatformOrArgcArgv(argc, argv)) |
| ? EXIT_SUCCESS |
| : EXIT_FAILURE; |
| } |