290 lines
13 KiB
C++
290 lines
13 KiB
C++
// Copyright (c) 2012-2023 Wojciech Figat. All rights reserved.
|
|
|
|
#if PLATFORM_TOOLS_GDK
|
|
|
|
#include "GDKPlatformTools.h"
|
|
#include "Engine/Platform/File.h"
|
|
#include "Engine/Platform/FileSystem.h"
|
|
#include "Engine/Platform/GDK/GDKPlatformSettings.h"
|
|
#include "Engine/Core/Types/StringBuilder.h"
|
|
#include "Engine/Core/Collections/Sorting.h"
|
|
#include "Engine/Core/Config/GameSettings.h"
|
|
#include "Engine/Graphics/PixelFormat.h"
|
|
#include "Editor/Editor.h"
|
|
#include "Editor/ProjectInfo.h"
|
|
#include "Editor/Utilities/EditorUtilities.h"
|
|
|
|
GDKPlatformTools::GDKPlatformTools()
|
|
{
|
|
// Find GDK
|
|
Platform::GetEnvironmentVariable(TEXT("GameDKLatest"), _gdkPath);
|
|
if (_gdkPath.IsEmpty() || !FileSystem::DirectoryExists(_gdkPath))
|
|
{
|
|
_gdkPath.Clear();
|
|
Platform::GetEnvironmentVariable(TEXT("GRDKLatest"), _gdkPath);
|
|
if (_gdkPath.IsEmpty() || !FileSystem::DirectoryExists(_gdkPath))
|
|
{
|
|
_gdkPath.Clear();
|
|
}
|
|
else
|
|
{
|
|
if (_gdkPath.EndsWith(TEXT("GRDK\\")))
|
|
_gdkPath.Remove(_gdkPath.Length() - 6);
|
|
else if (_gdkPath.EndsWith(TEXT("GRDK")))
|
|
_gdkPath.Remove(_gdkPath.Length() - 5);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GDKPlatformTools::UseAOT() const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool GDKPlatformTools::OnScriptsStepDone(CookingData& data)
|
|
{
|
|
// Override Newtonsoft.Json.dll for some platforms (that don't support runtime code generation)
|
|
const String customBinPath = data.GetPlatformBinariesRoot() / TEXT("Newtonsoft.Json.dll");
|
|
const String assembliesPath = data.ManagedCodeOutputPath;
|
|
if (FileSystem::CopyFile(assembliesPath / TEXT("Newtonsoft.Json.dll"), customBinPath))
|
|
{
|
|
data.Error(TEXT("Failed to copy deloy custom assembly."));
|
|
return true;
|
|
}
|
|
FileSystem::DeleteFile(assembliesPath / TEXT("Newtonsoft.Json.pdb"));
|
|
|
|
return false;
|
|
}
|
|
|
|
bool GDKPlatformTools::OnDeployBinaries(CookingData& data)
|
|
{
|
|
// Copy binaries
|
|
const Char* executableFilename = TEXT("FlaxGame.exe");
|
|
const auto binPath = data.GetGameBinariesPath();
|
|
FileSystem::CreateDirectory(data.NativeCodeOutputPath);
|
|
Array<String> files;
|
|
files.Add(binPath / executableFilename);
|
|
if (!FileSystem::FileExists(files[0]))
|
|
{
|
|
data.Error(TEXT("Missing executable file ({0})."), files[0]);
|
|
return true;
|
|
}
|
|
FileSystem::DirectoryGetFiles(files, binPath, TEXT("*.dll"), DirectorySearchOption::TopDirectoryOnly);
|
|
for (int32 i = 0; i < files.Count(); i++)
|
|
{
|
|
if (FileSystem::CopyFile(data.NativeCodeOutputPath / StringUtils::GetFileName(files[i]), files[i]))
|
|
{
|
|
data.Error(TEXT("Failed to setup output directory (file {0})."), files[i]);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void GDKPlatformTools::OnConfigureAOT(CookingData& data, AotConfig& config)
|
|
{
|
|
const auto platformDataPath = data.GetPlatformBinariesRoot();
|
|
const bool useInterpreter = true; // TODO: use Full AOT on GDK
|
|
const bool enableDebug = data.Configuration != BuildConfiguration::Release;
|
|
const Char* aotMode = useInterpreter ? TEXT("full,interp") : TEXT("full");
|
|
const Char* debugMode = enableDebug ? TEXT("soft-debug") : TEXT("nodebug");
|
|
config.AotCompilerArgs = String::Format(TEXT("--aot={0},verbose,stats,print-skipped,{1} -O=all"), aotMode, debugMode);
|
|
if (enableDebug)
|
|
config.AotCompilerArgs = TEXT("--debug ") + config.AotCompilerArgs;
|
|
config.AotCompilerPath = platformDataPath / TEXT("Tools/mono.exe");
|
|
}
|
|
|
|
bool GDKPlatformTools::OnPerformAOT(CookingData& data, AotConfig& config, const String& assemblyPath)
|
|
{
|
|
// Skip .dll.dll which could be a false result from the previous AOT which could fail
|
|
if (assemblyPath.EndsWith(TEXT(".dll.dll")))
|
|
{
|
|
LOG(Warning, "Skip AOT for file '{0}' as it can be a result from the previous task", assemblyPath);
|
|
return false;
|
|
}
|
|
|
|
// Check if skip this assembly (could be already processed)
|
|
const String filename = StringUtils::GetFileName(assemblyPath);
|
|
const String outputPath = config.AotCachePath / filename + TEXT(".dll");
|
|
if (FileSystem::FileExists(outputPath) && FileSystem::GetFileLastEditTime(assemblyPath) < FileSystem::GetFileLastEditTime(outputPath))
|
|
return false;
|
|
LOG(Info, "Calling AOT tool for \"{0}\"", assemblyPath);
|
|
|
|
// Cleanup temporary results (fromm the previous AT that fail or sth)
|
|
const String resultPath = assemblyPath + TEXT(".dll");
|
|
const String resultPathExp = resultPath + TEXT(".exp");
|
|
const String resultPathLib = resultPath + TEXT(".lib");
|
|
const String resultPathPdb = resultPath + TEXT(".pdb");
|
|
if (FileSystem::FileExists(resultPath))
|
|
FileSystem::DeleteFile(resultPath);
|
|
if (FileSystem::FileExists(resultPathExp))
|
|
FileSystem::DeleteFile(resultPathExp);
|
|
if (FileSystem::FileExists(resultPathLib))
|
|
FileSystem::DeleteFile(resultPathLib);
|
|
if (FileSystem::FileExists(resultPathPdb))
|
|
FileSystem::DeleteFile(resultPathPdb);
|
|
|
|
// Call tool
|
|
const String workingDir = StringUtils::GetDirectoryName(config.AotCompilerPath);
|
|
const String command = String::Format(TEXT("\"{0}\" {1} \"{2}\""), config.AotCompilerPath, config.AotCompilerArgs, assemblyPath);
|
|
const int32 result = Platform::RunProcess(command, workingDir, config.EnvVars);
|
|
if (result != 0)
|
|
{
|
|
data.Error(TEXT("AOT tool execution failed with result code {1} for assembly \"{0}\". See log for more info."), assemblyPath, result);
|
|
return true;
|
|
}
|
|
|
|
// Copy result
|
|
if (FileSystem::CopyFile(outputPath, resultPath))
|
|
{
|
|
data.Error(TEXT("Failed to copy the AOT tool result file. It can be missing."));
|
|
return true;
|
|
}
|
|
|
|
// Copy pdb file if exists
|
|
if (data.Configuration != BuildConfiguration::Release && FileSystem::FileExists(resultPathPdb))
|
|
{
|
|
FileSystem::CopyFile(config.AotCachePath / StringUtils::GetFileName(resultPathPdb), resultPathPdb);
|
|
}
|
|
|
|
// Clean intermediate results
|
|
if (FileSystem::DeleteFile(resultPath)
|
|
|| (FileSystem::FileExists(resultPathExp) && FileSystem::DeleteFile(resultPathExp))
|
|
|| (FileSystem::FileExists(resultPathLib) && FileSystem::DeleteFile(resultPathLib))
|
|
|| (FileSystem::FileExists(resultPathPdb) && FileSystem::DeleteFile(resultPathPdb))
|
|
)
|
|
{
|
|
LOG(Warning, "Failed to remove the AOT tool result file(s).");
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool GDKPlatformTools::OnPostProcess(CookingData& data, GDKPlatformSettings* platformSettings)
|
|
{
|
|
// Configuration
|
|
const auto gameSettings = GameSettings::Get();
|
|
const auto project = Editor::Project;
|
|
const Char* executableFilename = TEXT("FlaxGame.exe");
|
|
const String assetsFolder = data.DataOutputPath / TEXT("Assets");
|
|
if (!FileSystem::DirectoryExists(assetsFolder))
|
|
FileSystem::CreateDirectory(assetsFolder);
|
|
|
|
// Generate application icons
|
|
data.StepProgress(TEXT("Deploying icons"), 0);
|
|
if (EditorUtilities::ExportApplicationImage(platformSettings->Square150x150Logo, 150, 150, PixelFormat::B8G8R8A8_UNorm, assetsFolder / TEXT("Square150x150Logo.png")))
|
|
return true;
|
|
if (EditorUtilities::ExportApplicationImage(platformSettings->Square480x480Logo, 480, 480, PixelFormat::B8G8R8A8_UNorm, assetsFolder / TEXT("Square480x480Logo.png")))
|
|
return true;
|
|
if (EditorUtilities::ExportApplicationImage(platformSettings->Square44x44Logo, 44, 44, PixelFormat::B8G8R8A8_UNorm, assetsFolder / TEXT("Square44x44Logo.png")))
|
|
return true;
|
|
if (EditorUtilities::ExportApplicationImage(platformSettings->StoreLogo, 100, 100, PixelFormat::B8G8R8A8_UNorm, assetsFolder / TEXT("StoreLogo.png")))
|
|
return true;
|
|
if (EditorUtilities::ExportApplicationImage(platformSettings->SplashScreenImage, 1920, 1080, PixelFormat::B8G8R8X8_UNorm, assetsFolder / TEXT("SplashScreenImage.png"), EditorUtilities::ApplicationImageType::SplashScreen))
|
|
return true;
|
|
|
|
// Generate MicrosoftGame.config file
|
|
data.StepProgress(TEXT("Generating package meta"), 0.2f);
|
|
const String configFilePath = data.DataOutputPath / TEXT("MicrosoftGame.config");
|
|
LOG(Info, "Generating config file to \"{0}\"", configFilePath);
|
|
{
|
|
// Process name to be valid
|
|
StringBuilder sb;
|
|
Array<Char> validName;
|
|
const String& name = platformSettings->Name.HasChars() ? platformSettings->Name : gameSettings->ProductName;
|
|
for (int32 i = 0; i < name.Length() && validName.Count() <= 50; i++)
|
|
{
|
|
auto c = name[i];
|
|
switch (c)
|
|
{
|
|
case '.':
|
|
case '[':
|
|
case ']':
|
|
case '+':
|
|
case '-':
|
|
case '_':
|
|
validName.Add(c);
|
|
break;
|
|
default:
|
|
if (StringUtils::IsAlnum(c))
|
|
validName.Add(c);
|
|
}
|
|
}
|
|
validName.Add('\0');
|
|
|
|
sb.Append(TEXT("<?xml version=\"1.0\" encoding=\"utf-8\"?>\n"));
|
|
sb.Append(TEXT("<Game configVersion=\"0\">\n"));
|
|
sb.AppendFormat(TEXT(" <Identity Name=\"{0}\" Publisher=\"{1}\" Version=\"{2}\"/>\n"),
|
|
validName.Get(),
|
|
platformSettings->PublisherName.HasChars() ? platformSettings->PublisherName : TEXT("CN=") + gameSettings->CompanyName,
|
|
project->Version.ToString(4));
|
|
sb.Append(TEXT(" <ExecutableList>\n"));
|
|
sb.AppendFormat(TEXT(" <Executable Name=\"{0}\"\n"), executableFilename);
|
|
switch (data.Platform)
|
|
{
|
|
case BuildPlatform::XboxOne:
|
|
sb.Append(TEXT(" TargetDeviceFamily=\"XboxOne\"\n"));
|
|
break;
|
|
case BuildPlatform::XboxScarlett:
|
|
sb.Append(TEXT(" TargetDeviceFamily=\"Scarlett\"\n"));
|
|
break;
|
|
default:
|
|
sb.Append(TEXT(" TargetDeviceFamily=\"PC\"\n"));
|
|
break;
|
|
}
|
|
sb.Append(TEXT(" IsDevOnly=\"false\"\n"));
|
|
sb.Append(TEXT(" Id=\"Game\"\n"));
|
|
sb.Append(TEXT(" />"));
|
|
sb.Append(TEXT(" </ExecutableList>\n"));
|
|
sb.AppendFormat(TEXT(" <ShellVisuals DefaultDisplayName=\"{0}\"\n"), gameSettings->ProductName);
|
|
sb.AppendFormat(TEXT(" PublisherDisplayName=\"{0}\"\n"), platformSettings->PublisherDisplayName.HasChars() ? platformSettings->PublisherDisplayName : gameSettings->CompanyName);
|
|
sb.AppendFormat(TEXT(" BackgroundColor=\"#{0}\"\n"), platformSettings->BackgroundColor.ToHexString());
|
|
sb.AppendFormat(TEXT(" ForegroundText=\"{0}\"\n"), platformSettings->ForegroundText);
|
|
sb.Append(TEXT(" Square150x150Logo=\"Assets\\Square150x150Logo.png\"\n"));
|
|
sb.Append(TEXT(" Square480x480Logo=\"Assets\\Square480x480Logo.png\"\n"));
|
|
sb.Append(TEXT(" Square44x44Logo=\"Assets\\Square44x44Logo.png\"\n"));
|
|
sb.Append(TEXT(" StoreLogo=\"Assets\\StoreLogo.png\"\n"));
|
|
sb.Append(TEXT(" SplashScreenImage=\"Assets\\SplashScreenImage.png\"\n"));
|
|
sb.Append(TEXT(" />\n"));
|
|
if (platformSettings->TitleId.HasChars())
|
|
sb.AppendFormat(TEXT(" <TitleId>{0}</TitleId>\n"), platformSettings->TitleId);
|
|
if (platformSettings->StoreId.HasChars())
|
|
sb.AppendFormat(TEXT(" <StoreId>{0}</StoreId>\n"), platformSettings->StoreId);
|
|
sb.AppendFormat(TEXT(" <RequiresXboxLive>{0}</RequiresXboxLive>\n"), platformSettings->RequiresXboxLive);
|
|
sb.Append(TEXT(" <MediaCapture>\n"));
|
|
sb.AppendFormat(TEXT(" <GameDVRSystemComponent>{0}</GameDVRSystemComponent>\n"), platformSettings->GameDVRSystemComponent);
|
|
sb.AppendFormat(TEXT(" <BlockBroadcast>{0}</BlockBroadcast>\n"), platformSettings->BlockBroadcast);
|
|
sb.AppendFormat(TEXT(" <BlockGameDVR>{0}</BlockGameDVR>\n"), platformSettings->BlockGameDVR);
|
|
sb.Append(TEXT(" </MediaCapture>\n"));
|
|
sb.Append(TEXT("</Game>\n"));
|
|
|
|
if (File::WriteAllText(configFilePath, sb, Encoding::ANSI))
|
|
{
|
|
LOG(Error, "Failed to create config file.");
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Remove previous package file
|
|
const String packageOutputPath = data.DataOutputPath / TEXT("Package");
|
|
if (FileSystem::DirectoryExists(packageOutputPath))
|
|
FileSystem::DeleteDirectory(packageOutputPath);
|
|
|
|
// Generate package layout
|
|
data.StepProgress(TEXT("Generating package layout"), 0.3f);
|
|
const String gdkBinPath = _gdkPath / TEXT("../bin");
|
|
const String makePkgPath = gdkBinPath / TEXT("MakePkg.exe");
|
|
const String command = String::Format(TEXT("\"{0}\" genmap /f layout.xml /d \"{1}\""), makePkgPath, data.DataOutputPath);
|
|
const int32 result = Platform::RunProcess(command, data.DataOutputPath);
|
|
if (result != 0)
|
|
{
|
|
data.Error(TEXT("Failed to generate package layout."));
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
#endif
|