Files
UnrealEngineUWP/Engine/Plugins/Runtime/Metasound/Source/MetasoundEngine/Private/MetasoundEngineModule.cpp

539 lines
19 KiB
C++
Raw Normal View History

// Copyright Epic Games, Inc. All Rights Reserved.
#include "MetasoundEngineModule.h"
#include "AssetRegistry/AssetData.h"
#include "AssetRegistry/AssetRegistryModule.h"
#include "Metasound.h"
#include "MetasoundAssetSubsystem.h"
#include "MetasoundAudioBus.h"
#include "MetasoundBuilderSubsystem.h"
#include "MetasoundDataReference.h"
#include "MetasoundDataTypeRegistrationMacro.h"
#include "MetasoundDocumentBuilderRegistry.h"
#include "MetasoundDocumentInterface.h"
#include "MetasoundFrontendDocumentBuilder.h"
#include "MetasoundGlobals.h"
#include "MetasoundLog.h"
#include "MetasoundOutputSubsystem.h"
#include "MetasoundSettings.h"
#include "MetasoundSource.h"
#include "MetasoundTrace.h"
#include "MetasoundUObjectRegistry.h"
#include "MetasoundWave.h"
#include "MetasoundWaveTable.h"
#include "Analysis/MetasoundFrontendAnalyzerRegistry.h"
#include "Analysis/MetasoundFrontendVertexAnalyzerAudioBuffer.h"
#include "Analysis/MetasoundFrontendVertexAnalyzerEnvelopeFollower.h"
#include "Analysis/MetasoundFrontendVertexAnalyzerForwardValue.h"
#include "Analysis/MetasoundFrontendVertexAnalyzerTriggerDensity.h"
#include "Analysis/MetasoundFrontendVertexAnalyzerTriggerToTime.h"
#include "Analysis/MetasoundVertexAnalyzerAudioBusWriter.h"
#include "Interfaces/MetasoundDeprecatedInterfaces.h"
#include "Interfaces/MetasoundInterface.h"
#include "Interfaces/MetasoundInterfaceBindingsPrivate.h"
#include "Modules/ModuleManager.h"
#include "Sound/AudioSettings.h"
namespace Metasound
{
// Enable send/receive node registration for data types which existed before
// send/receive were deprecated in order to support old UMetaSound assets.
template<>
struct TEnableTransmissionNodeRegistration<FWaveAsset>
{
static constexpr bool Value = true;
};
}
REGISTER_METASOUND_DATATYPE(Metasound::FAudioBusAsset, "AudioBusAsset", Metasound::ELiteralType::UObjectProxy, UAudioBus);
REGISTER_METASOUND_DATATYPE(Metasound::FWaveAsset, "WaveAsset", Metasound::ELiteralType::UObjectProxy, USoundWave);
REGISTER_METASOUND_DATATYPE(WaveTable::FWaveTable, "WaveTable", Metasound::ELiteralType::FloatArray)
REGISTER_METASOUND_DATATYPE(Metasound::FWaveTableBankAsset, "WaveTableBankAsset", Metasound::ELiteralType::UObjectProxy, UWaveTableBank);
namespace Metasound::Engine
{
#if WITH_EDITOR
namespace MetasoundEngineModulePrivate
{
int32 EnableMetaSoundEditorAssetValidation = 1;
FAutoConsoleVariableRef CVarEnableMetaSoundEditorAssetValidation(
TEXT("au.MetaSound.Editor.EnableAssetValidation"),
EnableMetaSoundEditorAssetValidation,
TEXT("Enables MetaSound specific asset validation.\n")
TEXT("Default: 1 (Enabled)"),
ECVF_Default);
} // namespace MetasoundEngineModulePrivate
#endif // WITH_EDITOR
void FModule::FModule::StartupModule()
{
using namespace Frontend;
METASOUND_LLM_SCOPE;
FModuleManager::Get().LoadModuleChecked("MetasoundGraphCore");
FModuleManager::Get().LoadModuleChecked("MetasoundFrontend");
FModuleManager::Get().LoadModuleChecked("MetasoundStandardNodes");
FModuleManager::Get().LoadModuleChecked("MetasoundGenerator");
FModuleManager::Get().LoadModuleChecked("WaveTable");
InitializeAssetManager();
IDocumentBuilderRegistry::Initialize(MakeUnique<FDocumentBuilderRegistry>());
// Set GCObject referencer for metasound frontend node registry. The MetaSound
// frontend does not have access to Engine GC tools and must have them
// supplied externally.
FMetasoundFrontendRegistryContainer::Get()->SetObjectReferencer(MakeUnique<FObjectReferencer>());
// Register engine-level parameter interfaces if not done already.
// (Potentially not already called if plugin is loaded while cooking.)
UAudioSettings* AudioSettings = GetMutableDefault<UAudioSettings>();
check(AudioSettings);
AudioSettings->RegisterParameterInterfaces();
IMetasoundUObjectRegistry::Get().RegisterUClass(MakeUnique<TMetasoundUObjectRegistryEntry<UMetaSoundBuilderDocument>>());
IMetasoundUObjectRegistry::Get().RegisterUClass(MakeUnique<TMetasoundUObjectRegistryEntry<UMetaSoundPatch>>());
IMetasoundUObjectRegistry::Get().RegisterUClass(MakeUnique<TMetasoundUObjectRegistryEntry<UMetaSoundSource>>());
RegisterDeprecatedInterfaces();
RegisterInterfaces();
RegisterInternalInterfaceBindings();
// Flush node registration queue
FMetasoundFrontendRegistryContainer::Get()->RegisterPendingNodes();
// Register Analyzers
// TODO: Determine if we can move this registration to Frontend where it likely belongs
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerAudioBuffer)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerEnvelopeFollower)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerForwardBool)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerForwardFloat)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerForwardInt)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerForwardTime)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerForwardString)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerTriggerDensity)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Frontend::FVertexAnalyzerTriggerToTime)
METASOUND_REGISTER_VERTEX_ANALYZER_FACTORY(Engine::FVertexAnalyzerAudioBusWriter)
// Register passthrough output analyzers
UMetasoundGeneratorHandle::RegisterPassthroughAnalyzerForType(
GetMetasoundDataTypeName<float>(),
Frontend::FVertexAnalyzerForwardFloat::GetAnalyzerName(),
Frontend::FVertexAnalyzerForwardFloat::FOutputs::GetValue().Name);
UMetasoundGeneratorHandle::RegisterPassthroughAnalyzerForType(
GetMetasoundDataTypeName<int32>(),
Frontend::FVertexAnalyzerForwardInt::GetAnalyzerName(),
Frontend::FVertexAnalyzerForwardInt::FOutputs::GetValue().Name);
UMetasoundGeneratorHandle::RegisterPassthroughAnalyzerForType(
GetMetasoundDataTypeName<bool>(),
Frontend::FVertexAnalyzerForwardBool::GetAnalyzerName(),
Frontend::FVertexAnalyzerForwardBool::FOutputs::GetValue().Name);
UMetasoundGeneratorHandle::RegisterPassthroughAnalyzerForType(
GetMetasoundDataTypeName<FString>(),
Frontend::FVertexAnalyzerForwardString::GetAnalyzerName(),
Frontend::FVertexAnalyzerForwardString::FOutputs::GetValue().Name);
UMetasoundGeneratorHandle::RegisterPassthroughAnalyzerForType(
GetMetasoundDataTypeName<FTime>(),
Frontend::FVertexAnalyzerForwardTime::GetAnalyzerName(),
Frontend::FVertexAnalyzerForwardTime::FOutputs::GetValue().Name);
UMetasoundGeneratorHandle::RegisterPassthroughAnalyzerForType(
GetMetasoundDataTypeName<FTrigger>(),
Frontend::FVertexAnalyzerTriggerToTime::GetAnalyzerName(),
Frontend::FVertexAnalyzerTriggerToTime::FOutputs::GetValue().Name);
#if WITH_EDITOR
FAssetRegistryModule& AssetRegistryModule = FModuleManager::LoadModuleChecked<FAssetRegistryModule>("AssetRegistry");
if (AssetRegistryModule.Get().IsLoadingAssets())
{
AssetScanStatus = EAssetScanStatus::InProgress;
AssetRegistryModule.Get().OnFilesLoaded().AddRaw(this, &FModule::OnAssetScanFinished);
}
else
{
AssetScanStatus = EAssetScanStatus::Complete;
}
#endif // WITH_EDITOR
UE_LOG(LogMetaSound, Log, TEXT("MetaSound Engine Initialized"));
}
void FModule::FModule::ShutdownModule()
{
#if WITH_EDITOR
if (NodeClassRegistryPrimeStatus != ENodeClassRegistryPrimeStatus::NotRequested && NodeClassRegistryPrimeStatus != ENodeClassRegistryPrimeStatus::Complete)
{
NodeClassRegistryPrimeStatus = ENodeClassRegistryPrimeStatus::Canceled;
}
ShutdownAssetClassRegistry();
#endif // WITH_EDITOR
DeinitializeAssetManager();
Frontend::IDocumentBuilderRegistry::Deinitialize();
}
#if WITH_EDITOR
void FModule::LoadAndRegisterAsset(const FAssetData& InAssetData)
{
// Ignore requests if graphs cannot be executed, as registration that results in IGraph generation is not supported.
if (!Metasound::CanEverExecuteGraph())
{
return;
}
Frontend::FMetaSoundAssetRegistrationOptions RegOptions;
RegOptions.bForceReregister = false;
if (const UMetaSoundSettings* Settings = GetDefault<UMetaSoundSettings>())
{
RegOptions.bAutoUpdateLogWarningOnDroppedConnection = Settings->bAutoUpdateLogWarningOnDroppedConnection;
}
if (InAssetData.IsAssetLoaded())
{
if (UObject* AssetObject = InAssetData.GetAsset())
{
OnGraphRegister.ExecuteIfBound(*AssetObject, ERegistrationAssetContext::None);
}
}
else
{
if (NodeClassRegistryPrimeStatus == ENodeClassRegistryPrimeStatus::NotRequested || NodeClassRegistryPrimeStatus == ENodeClassRegistryPrimeStatus::Canceled)
{
return;
}
ActiveAsyncAssetLoadRequests++;
FSoftObjectPath AssetPath = InAssetData.ToSoftObjectPath();
auto LoadAndRegister = [this, ObjectPath = AssetPath, RegOptions](const FName& PackageName, UPackage* Package, EAsyncLoadingResult::Type Result)
{
if (NodeClassRegistryPrimeStatus == ENodeClassRegistryPrimeStatus::Canceled)
{
return;
}
if (Result == EAsyncLoadingResult::Succeeded)
{
UObject* MetaSoundObj = ObjectPath.ResolveObject();
FMetasoundAssetBase* MetaSoundAsset = IMetasoundUObjectRegistry::Get().GetObjectAsAssetBase(MetaSoundObj);
check(MetaSoundAsset);
if (!MetaSoundAsset->IsRegistered())
{
OnGraphRegister.ExecuteIfBound(*MetaSoundObj, ERegistrationAssetContext::None);
}
}
ActiveAsyncAssetLoadRequests--;
if (NodeClassRegistryPrimeStatus == ENodeClassRegistryPrimeStatus::InProgress && ActiveAsyncAssetLoadRequests == 0)
{
NodeClassRegistryPrimeStatus = ENodeClassRegistryPrimeStatus::Complete;
}
};
LoadPackageAsync(AssetPath.GetLongPackageName(), FLoadPackageAsyncDelegate::CreateLambda(LoadAndRegister));
}
}
void FModule::AddClassRegistryAsset(const FAssetData& InAssetData)
{
using namespace Frontend;
// If an object's class could not be found, ignore this asset. This can hit for non-MetaSound assets
// and it is up to the system in charge of interacting with that asset or the loading behavior to
// report the failed load of the class.
if (const UClass* AssetClass = InAssetData.GetClass())
{
// Don't add temporary assets used for diffing
if (InAssetData.HasAnyPackageFlags(PKG_ForDiffing))
{
return;
}
const bool bIsRegisteredClass = IMetasoundUObjectRegistry::Get().IsRegisteredClass(*AssetClass);
if (bIsRegisteredClass)
{
const FNodeRegistryKey RegistryKey = IMetaSoundAssetManager::GetChecked().AddOrUpdateAsset(InAssetData);
// Can be invalid if being called for the first time on an asset before its class name is generated
if (RegistryKey.IsValid())
{
const bool bPrimeRequested = NodeClassRegistryPrimeStatus > ENodeClassRegistryPrimeStatus::NotRequested;
const bool bIsRegistered = FMetasoundFrontendRegistryContainer::Get()->IsNodeRegistered(RegistryKey);
if (bPrimeRequested && !bIsRegistered)
{
LoadAndRegisterAsset(InAssetData);
}
}
}
}
}
void FModule::UpdateClassRegistryAsset(const FAssetData& InAssetData)
{
using namespace Frontend;
// If an object's class could not be found, ignore this asset. This can hit for non-MetaSound assets
// and it is up to the system in charge of interacting with that asset or the loading behavior to
// report the failed load of the class.
if (const UClass* AssetClass = InAssetData.GetClass())
{
const bool bIsRegisteredClass = IMetasoundUObjectRegistry::Get().IsRegisteredClass(*AssetClass);
if (bIsRegisteredClass)
{
const FNodeRegistryKey RegistryKey = IMetaSoundAssetManager::GetChecked().AddOrUpdateAsset(InAssetData);
const bool bPrimeRequested = NodeClassRegistryPrimeStatus > ENodeClassRegistryPrimeStatus::NotRequested;
const bool bIsRegistered = FMetasoundFrontendRegistryContainer::Get()->IsNodeRegistered(RegistryKey);
// Have to re-register even if prime was not requested to avoid registry desync.
if (bPrimeRequested || bIsRegistered)
{
LoadAndRegisterAsset(InAssetData);
}
}
}
}
void FModule::OnPackageReloaded(const EPackageReloadPhase InPackageReloadPhase, FPackageReloadedEvent* InPackageReloadedEvent)
{
using namespace Metasound;
using namespace Metasound::Frontend;
if (!InPackageReloadedEvent)
{
return;
}
if (InPackageReloadPhase != EPackageReloadPhase::OnPackageFixup)
{
return;
}
auto IsAssetMetaSound = [](const UObject* Obj)
{
check(Obj);
if (const UClass* AssetClass = Obj->GetClass())
{
return IMetasoundUObjectRegistry::Get().IsRegisteredClass(*AssetClass);
}
return false;
};
for (const TPair<UObject*, UObject*>& Pair : InPackageReloadedEvent->GetRepointedObjects())
{
if (UObject* Obj = Pair.Key)
{
if (IsAssetMetaSound(Obj))
{
OnGraphUnregister.ExecuteIfBound(*Obj, ERegistrationAssetContext::Reloading);
IMetaSoundAssetManager::GetChecked().RemoveAsset(*Pair.Key);
}
}
if (UObject* Obj = Pair.Value)
{
if (IsAssetMetaSound(Obj))
{
IMetaSoundAssetManager::GetChecked().AddOrUpdateAsset(*Pair.Value);
OnGraphRegister.ExecuteIfBound(*Obj, ERegistrationAssetContext::Reloading);
}
}
}
}
void FModule::OnAssetScanFinished()
{
if (IsRunningCookCommandlet())
{
return;
}
AssetScanStatus = EAssetScanStatus::Complete;
if (NodeClassRegistryPrimeStatus == ENodeClassRegistryPrimeStatus::Requested)
{
PrimeAssetRegistryAsync();
}
FAssetRegistryModule& AssetRegistryModule = FModuleManager::LoadModuleChecked<FAssetRegistryModule>("AssetRegistry");
AssetRegistryModule.Get().OnAssetAdded().AddRaw(this, &FModule::AddClassRegistryAsset);
AssetRegistryModule.Get().OnAssetUpdated().AddRaw(this, &FModule::UpdateClassRegistryAsset);
AssetRegistryModule.Get().OnAssetRemoved().AddRaw(this, &FModule::RemoveAssetFromClassRegistry);
AssetRegistryModule.Get().OnAssetRenamed().AddRaw(this, &FModule::RenameAssetInClassRegistry);
AssetRegistryModule.Get().OnFilesLoaded().RemoveAll(this);
FCoreUObjectDelegates::OnPackageReloaded.AddRaw(this, &FModule::OnPackageReloaded);
}
void FModule::RemoveAssetFromClassRegistry(const FAssetData& InAssetData)
{
using namespace Frontend;
if (const UClass* AssetClass = InAssetData.GetClass())
{
const bool bIsRegisteredClass = IMetasoundUObjectRegistry::Get().IsRegisteredClass(*AssetClass);
if (bIsRegisteredClass)
{
// Use the editor version of UnregisterWithFrontend so it refreshes any open MetaSound editors
// Doesn't use AssetData::GetAsset() as this can result in attempting to reload the object.
// If this call is hit after the asset is removed, the assumption is unregistration already
// occurred on object destroy.
if (UObject* AssetObject = InAssetData.GetSoftObjectPath().ResolveObject())
{
OnGraphUnregister.ExecuteIfBound(*AssetObject, ERegistrationAssetContext::Removing);
}
IMetaSoundAssetManager::GetChecked().RemoveAsset(InAssetData);
}
}
}
void FModule::RenameAssetInClassRegistry(const FAssetData& InAssetData, const FString& InOldObjectPath)
{
using namespace Frontend;
if (const UClass* AssetClass = InAssetData.GetClass())
{
const bool bIsRegisteredClass = IMetasoundUObjectRegistry::Get().IsRegisteredClass(*AssetClass);
if (bIsRegisteredClass)
{
IMetaSoundAssetManager& AssetManager = IMetaSoundAssetManager::GetChecked();
// Unregister using the new asset data even though the old object was last to be registered
// as the old asset is no longer accessible by the time rename is called. The asset at this
// point is identical however to its prior counterpart.
UObject* AssetObject = InAssetData.GetAsset();
check(AssetObject);
FMetasoundAssetBase* AssetBase = AssetManager.GetAsAsset(*AssetObject);
check(AssetBase);
bool bIsRegistered = AssetBase->IsRegistered();
if (bIsRegistered)
{
OnGraphUnregister.ExecuteIfBound(*AssetObject, ERegistrationAssetContext::Renaming);
}
IMetaSoundAssetManager::GetChecked().RenameAsset(InAssetData, InOldObjectPath);
if (bIsRegistered)
{
OnGraphRegister.ExecuteIfBound(*AssetObject, ERegistrationAssetContext::Renaming);
}
}
}
}
void FModule::ShutdownAssetClassRegistry()
{
if (FAssetRegistryModule* AssetRegistryModule = static_cast<FAssetRegistryModule*>(FModuleManager::Get().GetModule("AssetRegistry")))
{
AssetRegistryModule->Get().OnAssetAdded().RemoveAll(this);
AssetRegistryModule->Get().OnAssetUpdated().RemoveAll(this);
AssetRegistryModule->Get().OnAssetRemoved().RemoveAll(this);
AssetRegistryModule->Get().OnAssetRenamed().RemoveAll(this);
AssetRegistryModule->Get().OnFilesLoaded().RemoveAll(this);
FCoreUObjectDelegates::OnPackageReloaded.RemoveAll(this);
}
}
void FModule::PrimeAssetRegistryAsync()
{
// Ignore step if still loading assets from initial scan but set prime status as requested.
if (AssetScanStatus <= EAssetScanStatus::InProgress)
{
NodeClassRegistryPrimeStatus = ENodeClassRegistryPrimeStatus::Requested;
return;
}
// Prime both asset manager and node class registry
if (NodeClassRegistryPrimeStatus != ENodeClassRegistryPrimeStatus::InProgress)
{
NodeClassRegistryPrimeStatus = ENodeClassRegistryPrimeStatus::InProgress;
TArray<FTopLevelAssetPath> ClassNames;
IMetasoundUObjectRegistry::Get().IterateRegisteredUClasses([&ClassNames](UClass& InClass)
{
ClassNames.Add(InClass.GetClassPathName());
});
FARFilter Filter;
Filter.ClassPaths = ClassNames;
FAssetRegistryModule& AssetRegistryModule = FModuleManager::LoadModuleChecked<FAssetRegistryModule>("AssetRegistry");
uint32 AssetCount = 0;
AssetRegistryModule.Get().EnumerateAssets(Filter, [this, &AssetCount](const FAssetData& AssetData)
{
AddClassRegistryAsset(AssetData);
AssetCount++;
return true;
});
// Node class registry prime is complete if there are no assets to process
if (AssetCount == 0 || ActiveAsyncAssetLoadRequests == 0)
{
NodeClassRegistryPrimeStatus = ENodeClassRegistryPrimeStatus::Complete;
}
// Asset manager prime also occurred as part of AddClassRegistryAsset
bAssetManagerPrimed = true;
}
}
void FModule::PrimeAssetManager()
{
using namespace Frontend;
if (AssetScanStatus <= EAssetScanStatus::InProgress)
{
UE_LOG(LogMetaSound, Display, TEXT("MetaSound Asset Manager prime requested before Asset Registry scan completed."));
return;
}
if (!IsAssetManagerPrimed())
{
TArray<FTopLevelAssetPath> ClassNames;
IMetasoundUObjectRegistry::Get().IterateRegisteredUClasses([&ClassNames](UClass& InClass)
{
ClassNames.Add(InClass.GetClassPathName());
});
FARFilter Filter;
Filter.ClassPaths = ClassNames;
FAssetRegistryModule& AssetRegistryModule = FModuleManager::LoadModuleChecked<FAssetRegistryModule>("AssetRegistry");
AssetRegistryModule.Get().EnumerateAssets(Filter, [this](const FAssetData& AssetData)
{
IMetaSoundAssetManager::GetChecked().AddOrUpdateAsset(AssetData);
return true;
});
bAssetManagerPrimed = true;
}
}
ENodeClassRegistryPrimeStatus FModule::GetNodeClassRegistryPrimeStatus() const
{
return NodeClassRegistryPrimeStatus;
}
EAssetScanStatus FModule::GetAssetRegistryScanStatus() const
{
return AssetScanStatus;
}
bool FModule::IsAssetManagerPrimed() const
{
return bAssetManagerPrimed;
}
FOnMetasoundGraphRegister& FModule::GetOnGraphRegisteredDelegate()
{
return OnGraphRegister;
}
FOnMetasoundGraphUnregister& FModule::GetOnGraphUnregisteredDelegate()
{
return OnGraphUnregister;
}
#endif // WITH_EDITOR
} // namespace Metasound::Engine
IMPLEMENT_MODULE(Metasound::Engine::FModule, MetasoundEngine);