You've already forked UnrealEngineUWP
mirror of
https://github.com/izzy2lost/UnrealEngineUWP.git
synced 2026-03-26 18:15:20 -07:00
#rnx #preflight 64767efb4b1ead7c7f428c7a [CL 25693857 by joe kirchoff in ue5-main branch]
298 lines
12 KiB
C#
298 lines
12 KiB
C#
// Copyright Epic Games, Inc. All Rights Reserved.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text.RegularExpressions;
|
|
using EpicGames.Core;
|
|
using Microsoft.Extensions.Logging;
|
|
using UnrealBuildBase;
|
|
|
|
namespace UnrealBuildTool
|
|
{
|
|
abstract class UEToolChain
|
|
{
|
|
protected readonly ILogger Logger;
|
|
|
|
// Return the extension for response files
|
|
public static string ResponseExt => ".rsp";
|
|
|
|
public UEToolChain(ILogger InLogger)
|
|
{
|
|
Logger = InLogger;
|
|
}
|
|
|
|
public virtual void SetEnvironmentVariables()
|
|
{
|
|
}
|
|
|
|
public virtual void GetVersionInfo(List<string> Lines)
|
|
{
|
|
}
|
|
|
|
public virtual void GetExternalDependencies(HashSet<FileItem> ExternalDependencies)
|
|
{
|
|
}
|
|
|
|
public static DirectoryReference GetModuleInterfaceDir(DirectoryReference OutputDir)
|
|
{
|
|
return DirectoryReference.Combine(OutputDir, "Ifc");
|
|
}
|
|
|
|
// Return the path to the cpp compiler that will be used by this toolchain.
|
|
public virtual FileReference? GetCppCompilerPath()
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public virtual FileItem? CopyDebuggerVisualizer(FileItem SourceFile, DirectoryReference IntermediateDirectory, IActionGraphBuilder Graph)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
public virtual FileItem? LinkDebuggerVisualizer(FileItem SourceFile, DirectoryReference IntermediateDirectory)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
protected abstract CPPOutput CompileCPPFiles(CppCompileEnvironment CompileEnvironment, List<FileItem> InputFiles, DirectoryReference OutputDir, string ModuleName, IActionGraphBuilder Graph);
|
|
|
|
public CPPOutput CompileAllCPPFiles(CppCompileEnvironment CompileEnvironment, List<FileItem> InputFiles, DirectoryReference OutputDir, string ModuleName, IActionGraphBuilder Graph)
|
|
{
|
|
CPPOutput Result;
|
|
|
|
UnrealArchitectureConfig ArchConfig = UnrealArchitectureConfig.ForPlatform(CompileEnvironment.Platform);
|
|
// compile architectures separately if needed
|
|
if (ArchConfig.Mode == UnrealArchitectureMode.SingleTargetCompileSeparately || ArchConfig.Mode == UnrealArchitectureMode.SingleTargetLinkSeparately)
|
|
{
|
|
Result = new CPPOutput();
|
|
foreach (UnrealArch Arch in CompileEnvironment.Architectures.Architectures)
|
|
{
|
|
// determine the output location of intermediates (so, if OutputDir had the arch name in it, like Intermediate/x86+arm64, we would replace it with either emptry string
|
|
// or a single arch name depending on if the platform uses architecture directories for the architecture)
|
|
// @todo Add ArchitectureConfig.RequiresArchitectureFilenames but for directory -- or can we just use GetFolderNameForArch?!?!?
|
|
// string ArchReplacement = (Arch == ArchitectureWithoutMarkup()) ? "" : ArchConfig.GetFolderNameForArchitecture(Arch);
|
|
|
|
string PlatformArchitecturesString = ArchConfig.GetFolderNameForArchitectures(CompileEnvironment.Architectures);
|
|
DirectoryReference ArchOutputDir = new(OutputDir.FullName.Replace(PlatformArchitecturesString, ArchConfig.GetFolderNameForArchitecture(Arch)));
|
|
|
|
CppCompileEnvironment ArchEnvironment = new(CompileEnvironment, Arch);
|
|
CPPOutput ArchResult = CompileCPPFiles(ArchEnvironment, InputFiles, ArchOutputDir, ModuleName, Graph);
|
|
Result.Merge(ArchResult, Arch);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Result = CompileCPPFiles(CompileEnvironment, InputFiles, OutputDir, ModuleName, Graph);
|
|
}
|
|
|
|
return Result;
|
|
}
|
|
|
|
public virtual CPPOutput CompileRCFiles(CppCompileEnvironment Environment, List<FileItem> InputFiles, DirectoryReference OutputDir, IActionGraphBuilder Graph)
|
|
{
|
|
CPPOutput Result = new CPPOutput();
|
|
return Result;
|
|
}
|
|
|
|
public virtual CPPOutput CompileISPCFiles(CppCompileEnvironment Environment, List<FileItem> InputFiles, DirectoryReference OutputDir, IActionGraphBuilder Graph)
|
|
{
|
|
CPPOutput Result = new CPPOutput();
|
|
return Result;
|
|
}
|
|
|
|
public virtual CPPOutput GenerateISPCHeaders(CppCompileEnvironment Environment, List<FileItem> InputFiles, DirectoryReference OutputDir, IActionGraphBuilder Graph)
|
|
{
|
|
CPPOutput Result = new CPPOutput();
|
|
return Result;
|
|
}
|
|
|
|
public virtual void GenerateTypeLibraryHeader(CppCompileEnvironment CompileEnvironment, ModuleRules.TypeLibrary TypeLibrary, FileReference OutputFile, IActionGraphBuilder Graph)
|
|
{
|
|
throw new NotSupportedException("This platform does not support type libraries.");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Allows a toolchain to decide to create an import library if needed for this Environment
|
|
/// </summary>
|
|
/// <param name="LinkEnvironment"></param>
|
|
/// <param name="Graph"></param>
|
|
/// <returns></returns>
|
|
public virtual FileItem[] LinkImportLibrary(LinkEnvironment LinkEnvironment, IActionGraphBuilder Graph)
|
|
{
|
|
// by default doing nothing
|
|
return new FileItem[] { };
|
|
}
|
|
|
|
public abstract FileItem? LinkFiles(LinkEnvironment LinkEnvironment, bool bBuildImportLibraryOnly, IActionGraphBuilder Graph);
|
|
public virtual FileItem[] LinkAllFiles(LinkEnvironment LinkEnvironment, bool bBuildImportLibraryOnly, IActionGraphBuilder Graph)
|
|
{
|
|
List<FileItem> Result = new();
|
|
|
|
// compile architectures separately if needed
|
|
UnrealArchitectureConfig ArchConfig = UnrealArchitectureConfig.ForPlatform(LinkEnvironment.Platform);
|
|
if (ArchConfig.Mode == UnrealArchitectureMode.SingleTargetLinkSeparately)
|
|
{
|
|
foreach (UnrealArch Arch in LinkEnvironment.Architectures.Architectures)
|
|
{
|
|
LinkEnvironment ArchEnvironment = new LinkEnvironment(LinkEnvironment, Arch);
|
|
|
|
// determine the output location of intermediates (so, if OutputDir had the arch name in it, like Intermediate/x86+arm64, we would replace it with either emptry string
|
|
// or a single arch name
|
|
//string ArchReplacement = Arch == ArchitectureWithoutMarkup() ? "" : ArchConfig.GetFolderNameForArchitecture(Arch);
|
|
string PlatformArchitecturesString = ArchConfig.GetFolderNameForArchitectures(LinkEnvironment.Architectures);
|
|
|
|
ArchEnvironment.OutputFilePaths = LinkEnvironment.OutputFilePaths.Select(x => new FileReference(x.FullName.Replace(PlatformArchitecturesString, ArchConfig.GetFolderNameForArchitecture(Arch)))).ToList();
|
|
|
|
FileItem? LinkFile = LinkFiles(ArchEnvironment, bBuildImportLibraryOnly, Graph);
|
|
if (LinkFile != null)
|
|
{
|
|
Result.Add(LinkFile);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FileItem? LinkFile = LinkFiles(LinkEnvironment, bBuildImportLibraryOnly, Graph);
|
|
if (LinkFile != null)
|
|
{
|
|
Result.Add(LinkFile);
|
|
}
|
|
}
|
|
return Result.ToArray();
|
|
}
|
|
|
|
public virtual CppCompileEnvironment CreateSharedResponseFile(CppCompileEnvironment CompileEnvironment, FileReference OutResponseFile, IActionGraphBuilder Graph)
|
|
{
|
|
return CompileEnvironment;
|
|
}
|
|
|
|
public virtual void CreateSpecificFileAction(CppCompileEnvironment CompileEnvironment, DirectoryReference SourceDir, DirectoryReference OutputDir, IActionGraphBuilder Graph)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the name of the response file for the current compile environment and output file
|
|
/// </summary>
|
|
/// <param name="CompileEnvironment"></param>
|
|
/// <param name="OutputFile"></param>
|
|
/// <returns></returns>
|
|
public static FileReference GetResponseFileName(CppCompileEnvironment CompileEnvironment, FileItem OutputFile)
|
|
{
|
|
// Construct a relative path for the intermediate response file
|
|
return OutputFile.Location.ChangeExtension(OutputFile.Location.GetExtension() + ResponseExt);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get the name of the response file for the current linker environment and output file
|
|
/// </summary>
|
|
/// <param name="LinkEnvironment"></param>
|
|
/// <param name="OutputFile"></param>
|
|
/// <returns></returns>
|
|
public static FileReference GetResponseFileName(LinkEnvironment LinkEnvironment, FileItem OutputFile)
|
|
{
|
|
// Construct a relative path for the intermediate response file
|
|
return FileReference.Combine(LinkEnvironment.IntermediateDirectory!, OutputFile.Location.GetFileName() + ResponseExt);
|
|
}
|
|
|
|
public virtual ICollection<FileItem> PostBuild(ReadOnlyTargetRules Target, FileItem Executable, LinkEnvironment ExecutableLinkEnvironment, IActionGraphBuilder Graph)
|
|
{
|
|
return new List<FileItem>();
|
|
}
|
|
|
|
public virtual ICollection<FileItem> PostBuild(ReadOnlyTargetRules Target, FileItem[] Executables, LinkEnvironment ExecutableLinkEnvironment, IActionGraphBuilder Graph)
|
|
{
|
|
// by default, run PostBuild for exe Exe and merge results
|
|
return Executables.SelectMany(x => PostBuild(Target, x, ExecutableLinkEnvironment, Graph)).ToList();
|
|
}
|
|
|
|
public virtual void SetUpGlobalEnvironment(ReadOnlyTargetRules Target)
|
|
{
|
|
}
|
|
|
|
public virtual void ModifyBuildProducts(ReadOnlyTargetRules Target, UEBuildBinary Binary, List<string> Libraries, List<UEBuildBundleResource> BundleResources, Dictionary<FileReference, BuildProductType> BuildProducts)
|
|
{
|
|
}
|
|
|
|
public virtual void ModifyTargetReceipt(TargetReceipt Receipt)
|
|
{
|
|
}
|
|
|
|
public virtual void FinalizeOutput(ReadOnlyTargetRules Target, TargetMakefileBuilder MakefileBuilder)
|
|
{
|
|
}
|
|
|
|
public virtual void PrepareRuntimeDependencies(List<RuntimeDependency> RuntimeDependencies, Dictionary<FileReference, FileReference> TargetFileToSourceFile, DirectoryReference ExeDir)
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds a build product and its associated debug file to a receipt.
|
|
/// </summary>
|
|
/// <param name="OutputFile">Build product to add</param>
|
|
/// <param name="OutputType">The type of build product</param>
|
|
public virtual bool ShouldAddDebugFileToReceipt(FileReference OutputFile, BuildProductType OutputType)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
public virtual FileReference GetDebugFile(FileReference OutputFile, string DebugExtension)
|
|
{
|
|
// by default, just change the extension to the debug extension
|
|
return OutputFile.ChangeExtension(DebugExtension);
|
|
}
|
|
|
|
public virtual void SetupBundleDependencies(ReadOnlyTargetRules Target, List<UEBuildBinary> Binaries, string GameName)
|
|
{
|
|
}
|
|
|
|
public virtual string GetSDKVersion()
|
|
{
|
|
return "Not Applicable";
|
|
}
|
|
|
|
/// <summary>
|
|
/// Runs the provided tool and argument. Returns the output, using a rexex capture if one is provided
|
|
/// </summary>
|
|
/// <param name="Command">Full path to the tool to run</param>
|
|
/// <param name="ToolArg">Argument that will be passed to the tool</param>
|
|
/// <param name="Expression">null, or a Regular expression to capture in the output</param>
|
|
/// <returns></returns>
|
|
protected string? RunToolAndCaptureOutput(FileReference Command, string ToolArg, string? Expression = null)
|
|
{
|
|
string ProcessOutput = Utils.RunLocalProcessAndReturnStdOut(Command.FullName, ToolArg, Logger);
|
|
|
|
if (String.IsNullOrEmpty(Expression))
|
|
{
|
|
return ProcessOutput;
|
|
}
|
|
|
|
Match M = Regex.Match(ProcessOutput, Expression);
|
|
return M.Success ? M.Groups[1].ToString() : null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Runs the provided tool and argument and parses the output to retrieve the version
|
|
/// </summary>
|
|
/// <param name="Command">Full path to the tool to run</param>
|
|
/// <param name="VersionArg">Argument that will result in the version string being shown (it's ok this is a byproduct of a command that returns an error)</param>
|
|
/// <param name="VersionExpression">Regular expression to capture the version. By default we look for four integers separated by periods, with the last two optional</param>
|
|
/// <returns></returns>
|
|
public Version RunToolAndCaptureVersion(FileReference Command, string VersionArg, string VersionExpression = @"(\d+\.\d+(\.\d+)?(\.\d+)?)")
|
|
{
|
|
string? ProcessOutput = RunToolAndCaptureOutput(Command, VersionArg, VersionExpression);
|
|
|
|
Version? ToolVersion;
|
|
|
|
if (Version.TryParse(ProcessOutput, out ToolVersion))
|
|
{
|
|
return ToolVersion;
|
|
}
|
|
|
|
Logger.LogWarning("Unable to retrieve version from {Command} {Arg}", Command, VersionArg);
|
|
|
|
return new Version(0, 0);
|
|
}
|
|
};
|
|
}
|