//-------------------------------------------------------------------------------------- // File: Audio.h // // DirectXTK for Audio header // // THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF // ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO // THE IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A // PARTICULAR PURPOSE. // // Copyright (c) Microsoft Corporation. All rights reserved. // // http://go.microsoft.com/fwlink/?LinkId=248929 //-------------------------------------------------------------------------------------- #pragma once #include #include #include #if defined(_XBOX_ONE) && defined(_TITLE) #include #pragma comment(lib,"acphal.lib") #endif #if defined(WINAPI_FAMILY) && WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP #pragma comment(lib,"PhoneAudioSes.lib") #endif #ifndef XAUDIO2_HELPER_FUNCTIONS #define XAUDIO2_HELPER_FUNCTIONS #endif #if (_WIN32_WINNT >= 0x0602 /*_WIN32_WINNT_WIN8*/) #if defined(_MSC_VER) && (_MSC_VER < 1700) #error DirectX Tool Kit for Audio does not support VS 2010 without the DirectX SDK #endif #include #include #include #include #pragma comment(lib,"xaudio2.lib") #else // Using XAudio 2.7 requires the DirectX SDK #include #include #include #include #pragma warning(push) #pragma warning( disable : 4005 ) #include #pragma warning(pop) #pragma comment(lib,"x3daudio.lib") #pragma comment(lib,"xapofx.lib") #endif #include #pragma warning(push) #pragma warning(disable : 4005) #include #pragma warning(pop) #include #include #include #include // VS 2010 doesn't support explicit calling convention for std::function #ifndef DIRECTX_STD_CALLCONV #if defined(_MSC_VER) && (_MSC_VER < 1700) #define DIRECTX_STD_CALLCONV #else #define DIRECTX_STD_CALLCONV __cdecl #endif #endif // VS 2010/2012 do not support =default =delete #ifndef DIRECTX_CTOR_DEFAULT #if defined(_MSC_VER) && (_MSC_VER < 1800) #define DIRECTX_CTOR_DEFAULT {} #define DIRECTX_CTOR_DELETE ; #else #define DIRECTX_CTOR_DEFAULT =default; #define DIRECTX_CTOR_DELETE =delete; #endif #endif #pragma warning(push) #pragma warning(disable : 4481) // VS 2010 considers 'override' to be a extension, but it's part of C++11 as of VS 2012 namespace DirectX { #if (DIRECTX_MATH_VERSION < 305) && !defined(XM_CALLCONV) #define XM_CALLCONV __fastcall typedef const XMVECTOR& HXMVECTOR; typedef const XMMATRIX& FXMMATRIX; #endif class SoundEffectInstance; //---------------------------------------------------------------------------------- struct AudioStatistics { size_t playingOneShots; // Number of one-shot sounds currently playing size_t playingInstances; // Number of sound effect instances currently playing size_t allocatedInstances; // Number of SoundEffectInstance allocated size_t allocatedVoices; // Number of XAudio2 voices allocated (standard, 3D, one-shots, and idle one-shots) size_t allocatedVoices3d; // Number of XAudio2 voices allocated for 3D size_t allocatedVoicesOneShot; // Number of XAudio2 voices allocated for one-shot sounds size_t allocatedVoicesIdle; // Number of XAudio2 voices allocated for one-shot sounds but not currently in use size_t audioBytes; // Total wave data (in bytes) in SoundEffects and in-memory WaveBanks #if defined(_XBOX_ONE) && defined(_TITLE) size_t xmaAudioBytes; // Total wave data (in bytes) in SoundEffects and in-memory WaveBanks allocated with ApuAlloc #endif }; //---------------------------------------------------------------------------------- class IVoiceNotify { public: virtual void __cdecl OnBufferEnd() = 0; // Notfication that a voice buffer has finished // Note this is called from XAudio2's worker thread, so it should perform very minimal and thread-safe operations virtual void __cdecl OnCriticalError() = 0; // Notification that the audio engine encountered a critical error virtual void __cdecl OnReset() = 0; // Notification of an audio engine reset virtual void __cdecl OnUpdate() = 0; // Notification of an audio engine per-frame update (opt-in) virtual void __cdecl OnDestroyEngine() = 0; // Notification that the audio engine is being destroyed virtual void __cdecl OnTrim() = 0; // Notification of a request to trim the voice pool virtual void __cdecl GatherStatistics( AudioStatistics& stats ) const = 0; // Contribute to statistics request }; //---------------------------------------------------------------------------------- enum AUDIO_ENGINE_FLAGS { AudioEngine_Default = 0x0, AudioEngine_EnvironmentalReverb = 0x1, AudioEngine_ReverbUseFilters = 0x2, AudioEngine_UseMasteringLimiter = 0x4, AudioEngine_Debug = 0x10000, AudioEngine_ThrowOnNoAudioHW = 0x20000, AudioEngine_DisableVoiceReuse = 0x40000, }; inline AUDIO_ENGINE_FLAGS operator|(AUDIO_ENGINE_FLAGS a, AUDIO_ENGINE_FLAGS b) { return static_cast( static_cast(a) | static_cast(b) ); } enum SOUND_EFFECT_INSTANCE_FLAGS { SoundEffectInstance_Default = 0x0, SoundEffectInstance_Use3D = 0x1, SoundEffectInstance_ReverbUseFilters = 0x2, SoundEffectInstance_NoSetPitch = 0x4, SoundEffectInstance_UseRedirectLFE = 0x10000, }; inline SOUND_EFFECT_INSTANCE_FLAGS operator|(SOUND_EFFECT_INSTANCE_FLAGS a, SOUND_EFFECT_INSTANCE_FLAGS b) { return static_cast( static_cast(a) | static_cast(b) ); } enum AUDIO_ENGINE_REVERB { Reverb_Off, Reverb_Default, Reverb_Generic, Reverb_Forest, Reverb_PaddedCell, Reverb_Room, Reverb_Bathroom, Reverb_LivingRoom, Reverb_StoneRoom, Reverb_Auditorium, Reverb_ConcertHall, Reverb_Cave, Reverb_Arena, Reverb_Hangar, Reverb_CarpetedHallway, Reverb_Hallway, Reverb_StoneCorridor, Reverb_Alley, Reverb_City, Reverb_Mountains, Reverb_Quarry, Reverb_Plain, Reverb_ParkingLot, Reverb_SewerPipe, Reverb_Underwater, Reverb_SmallRoom, Reverb_MediumRoom, Reverb_LargeRoom, Reverb_MediumHall, Reverb_LargeHall, Reverb_Plate, Reverb_MAX }; enum SoundState { STOPPED = 0, PLAYING, PAUSED }; //---------------------------------------------------------------------------------- class AudioEngine { public: explicit AudioEngine( AUDIO_ENGINE_FLAGS flags = AudioEngine_Default, _In_opt_ const WAVEFORMATEX* wfx = nullptr, _In_opt_z_ const wchar_t* deviceId = nullptr, AUDIO_STREAM_CATEGORY category = AudioCategory_GameEffects ); AudioEngine(AudioEngine&& moveFrom); AudioEngine& operator= (AudioEngine&& moveFrom); virtual ~AudioEngine(); bool __cdecl Update(); // Performs per-frame processing for the audio engine, returns false if in 'silent mode' bool __cdecl Reset( _In_opt_ const WAVEFORMATEX* wfx = nullptr, _In_opt_z_ const wchar_t* deviceId = nullptr ); // Reset audio engine from critical error/silent mode using a new device; can also 'migrate' the graph // Returns true if succesfully reset, false if in 'silent mode' due to no default device // Note: One shots are lost, all SoundEffectInstances are in the STOPPED state after successful reset void __cdecl Suspend(); void __cdecl Resume(); // Suspend/resumes audio processing (i.e. global pause/resume) float __cdecl GetMasterVolume() const; void __cdecl SetMasterVolume( float volume ); // Master volume property for all sounds void __cdecl SetReverb( AUDIO_ENGINE_REVERB reverb ); void __cdecl SetReverb( _In_opt_ const XAUDIO2FX_REVERB_PARAMETERS* native ); // Sets environmental reverb for 3D positional audio (if active) void __cdecl SetMasteringLimit( int release, int loudness ); // Sets the mastering volume limiter properties (if active) AudioStatistics __cdecl GetStatistics() const; // Gathers audio engine statistics WAVEFORMATEXTENSIBLE __cdecl GetOutputFormat() const; // Returns the format consumed by the mastering voice (which is the same as the device output if defaults are used) uint32_t __cdecl GetChannelMask() const; // Returns the output channel mask int __cdecl GetOutputChannels() const; // Returns the number of output channels bool __cdecl IsAudioDevicePresent() const; // Returns true if the audio graph is operating normally, false if in 'silent mode' bool __cdecl IsCriticalError() const; // Returns true if the audio graph is halted due to a critical error (which also places the engine into 'silent mode') // Voice pool management. void __cdecl SetDefaultSampleRate( int sampleRate ); // Sample rate for voices in the reuse pool (defaults to 44100) void __cdecl SetMaxVoicePool( size_t maxOneShots, size_t maxInstances ); // Maximum number of voices to allocate for one-shots and instances // Note: one-shots over this limit are ignored; too many instance voices throws an exception void __cdecl TrimVoicePool(); // Releases any currently unused voices // Internal-use functions void __cdecl AllocateVoice( _In_ const WAVEFORMATEX* wfx, SOUND_EFFECT_INSTANCE_FLAGS flags, bool oneshot, _Outptr_result_maybenull_ IXAudio2SourceVoice** voice ); void __cdecl DestroyVoice( _In_ IXAudio2SourceVoice* voice ); // Should only be called for instance voices, not one-shots void __cdecl RegisterNotify( _In_ IVoiceNotify* notify, bool usesUpdate ); void __cdecl UnregisterNotify( _In_ IVoiceNotify* notify, bool usesOneShots, bool usesUpdate ); // XAudio2 interface access IXAudio2* __cdecl GetInterface() const; IXAudio2MasteringVoice* __cdecl GetMasterVoice() const; IXAudio2SubmixVoice* __cdecl GetReverbVoice() const; X3DAUDIO_HANDLE& __cdecl Get3DHandle() const; // Static functions struct RendererDetail { std::wstring deviceId; std::wstring description; }; static std::vector __cdecl GetRendererDetails(); // Returns a list of valid audio endpoint devices private: // Private implementation. class Impl; std::unique_ptr pImpl; // Prevent copying. AudioEngine(AudioEngine const&) DIRECTX_CTOR_DELETE AudioEngine& operator= (AudioEngine const&) DIRECTX_CTOR_DELETE }; //---------------------------------------------------------------------------------- class WaveBank { public: WaveBank( _In_ AudioEngine* engine, _In_z_ const wchar_t* wbFileName ); WaveBank(WaveBank&& moveFrom); WaveBank& operator= (WaveBank&& moveFrom); virtual ~WaveBank(); void __cdecl Play( int index ); void __cdecl Play( int index, float volume, float pitch, float pan ); void __cdecl Play( _In_z_ const char* name ); void __cdecl Play( _In_z_ const char* name, float volume, float pitch, float pan ); std::unique_ptr __cdecl CreateInstance( int index, SOUND_EFFECT_INSTANCE_FLAGS flags = SoundEffectInstance_Default ); std::unique_ptr __cdecl CreateInstance( _In_z_ const char* name, SOUND_EFFECT_INSTANCE_FLAGS flags = SoundEffectInstance_Default ); bool __cdecl IsPrepared() const; bool __cdecl IsInUse() const; bool __cdecl IsStreamingBank() const; size_t __cdecl GetSampleSizeInBytes( int index ) const; // Returns size of wave audio data size_t __cdecl GetSampleDuration( int index ) const; // Returns the duration in samples size_t __cdecl GetSampleDurationMS( int index ) const; // Returns the duration in milliseconds const WAVEFORMATEX* __cdecl GetFormat( int index, _Out_writes_bytes_(maxsize) WAVEFORMATEX* wfx, size_t maxsize ) const; int __cdecl Find( _In_z_ const char* name ) const; #if defined(_XBOX_ONE) || (_WIN32_WINNT < _WIN32_WINNT_WIN8) bool __cdecl FillSubmitBuffer( int index, _Out_ XAUDIO2_BUFFER& buffer, _Out_ XAUDIO2_BUFFER_WMA& wmaBuffer ) const; #else void __cdecl FillSubmitBuffer( int index, _Out_ XAUDIO2_BUFFER& buffer ) const; #endif private: // Private implementation. class Impl; std::unique_ptr pImpl; // Prevent copying. WaveBank(WaveBank const&) DIRECTX_CTOR_DELETE WaveBank& operator= (WaveBank const&) DIRECTX_CTOR_DELETE // Private interface void __cdecl UnregisterInstance( _In_ SoundEffectInstance* instance ); friend class SoundEffectInstance; }; //---------------------------------------------------------------------------------- class SoundEffect { public: SoundEffect( _In_ AudioEngine* engine, _In_z_ const wchar_t* waveFileName ); SoundEffect( _In_ AudioEngine* engine, _Inout_ std::unique_ptr& wavData, _In_ const WAVEFORMATEX* wfx, _In_reads_bytes_(audioBytes) const uint8_t* startAudio, size_t audioBytes ); SoundEffect( _In_ AudioEngine* engine, _Inout_ std::unique_ptr& wavData, _In_ const WAVEFORMATEX* wfx, _In_reads_bytes_(audioBytes) const uint8_t* startAudio, size_t audioBytes, uint32_t loopStart, uint32_t loopLength ); #if defined(_XBOX_ONE) || (_WIN32_WINNT < _WIN32_WINNT_WIN8) SoundEffect( _In_ AudioEngine* engine, _Inout_ std::unique_ptr& wavData, _In_ const WAVEFORMATEX* wfx, _In_reads_bytes_(audioBytes) const uint8_t* startAudio, size_t audioBytes, _In_reads_(seekCount) const uint32_t* seekTable, size_t seekCount ); #endif SoundEffect(SoundEffect&& moveFrom); SoundEffect& operator= (SoundEffect&& moveFrom); virtual ~SoundEffect(); void __cdecl Play(); void __cdecl Play(float volume, float pitch, float pan); std::unique_ptr __cdecl CreateInstance( SOUND_EFFECT_INSTANCE_FLAGS flags = SoundEffectInstance_Default ); bool __cdecl IsInUse() const; size_t __cdecl GetSampleSizeInBytes() const; // Returns size of wave audio data size_t __cdecl GetSampleDuration() const; // Returns the duration in samples size_t __cdecl GetSampleDurationMS() const; // Returns the duration in milliseconds const WAVEFORMATEX* __cdecl GetFormat() const; #if defined(_XBOX_ONE) || (_WIN32_WINNT < _WIN32_WINNT_WIN8) bool __cdecl FillSubmitBuffer( _Out_ XAUDIO2_BUFFER& buffer, _Out_ XAUDIO2_BUFFER_WMA& wmaBuffer ) const; #else void __cdecl FillSubmitBuffer( _Out_ XAUDIO2_BUFFER& buffer ) const; #endif private: // Private implementation. class Impl; std::unique_ptr pImpl; // Prevent copying. SoundEffect(SoundEffect const&) DIRECTX_CTOR_DELETE SoundEffect& operator= (SoundEffect const&) DIRECTX_CTOR_DELETE // Private interface void __cdecl UnregisterInstance( _In_ SoundEffectInstance* instance ); friend class SoundEffectInstance; }; //---------------------------------------------------------------------------------- struct AudioListener : public X3DAUDIO_LISTENER { AudioListener() { memset( this, 0, sizeof(X3DAUDIO_LISTENER) ); OrientFront.z = OrientTop.y = 1.f; } void XM_CALLCONV SetPosition( FXMVECTOR v ) { XMStoreFloat3( reinterpret_cast( &Position ), v ); } void __cdecl SetPosition( const XMFLOAT3& pos ) { Position.x = pos.x; Position.y = pos.y; Position.z = pos.z; } void XM_CALLCONV SetVelocity( FXMVECTOR v ) { XMStoreFloat3( reinterpret_cast( &Velocity ), v ); } void __cdecl SetVelocity( const XMFLOAT3& vel ) { Velocity.x = vel.x; Velocity.y = vel.y; Velocity.z = vel.z; } void XM_CALLCONV SetOrientation( FXMVECTOR forward, FXMVECTOR up ) { XMStoreFloat3( reinterpret_cast( &OrientFront ), forward ); XMStoreFloat3( reinterpret_cast( &OrientTop ), up ); } void __cdecl SetOrientation( const XMFLOAT3& forward, const XMFLOAT3& up ) { OrientFront.x = forward.x; OrientTop.x = up.x; OrientFront.y = forward.y; OrientTop.y = up.y; OrientFront.z = forward.z; OrientTop.z = up.z; } void XM_CALLCONV SetOrientationFromQuaternion( FXMVECTOR quat ) { XMVECTOR forward = XMVector3Rotate( g_XMIdentityR2, quat ); XMStoreFloat3( reinterpret_cast( &OrientFront ), forward ); XMVECTOR up = XMVector3Rotate( g_XMIdentityR1, quat ); XMStoreFloat3( reinterpret_cast( &OrientTop ), up ); } void XM_CALLCONV Update( FXMVECTOR newPos, XMVECTOR upDir, float dt ) // Updates velocity and orientation by tracking changes in position over time... { if ( dt > 0.f ) { XMVECTOR lastPos = XMLoadFloat3( reinterpret_cast( &Position ) ); XMVECTOR vDelta = ( newPos - lastPos ); XMVECTOR v = vDelta / dt; XMStoreFloat3( reinterpret_cast( &Velocity ), v ); vDelta = XMVector3Normalize( vDelta ); XMStoreFloat3( reinterpret_cast( &OrientFront ), vDelta ); v = XMVector3Cross( upDir, vDelta ); v = XMVector3Normalize( v ); v = XMVector3Cross( vDelta, v ); v = XMVector3Normalize( v ); XMStoreFloat3( reinterpret_cast( &OrientTop ), v ); XMStoreFloat3( reinterpret_cast( &Position ), newPos ); } } }; //---------------------------------------------------------------------------------- struct AudioEmitter : public X3DAUDIO_EMITTER { float EmitterAzimuths[XAUDIO2_MAX_AUDIO_CHANNELS]; AudioEmitter() { memset( this, 0, sizeof(X3DAUDIO_EMITTER) ); memset( EmitterAzimuths, 0, sizeof(EmitterAzimuths) ); OrientFront.z = OrientTop.y = ChannelRadius = CurveDistanceScaler = DopplerScaler = 1.f; ChannelCount = 1; pChannelAzimuths = EmitterAzimuths; InnerRadiusAngle = X3DAUDIO_PI / 4.0f; } void XM_CALLCONV SetPosition( FXMVECTOR v ) { XMStoreFloat3( reinterpret_cast( &Position ), v ); } void __cdecl SetPosition( const XMFLOAT3& pos ) { Position.x = pos.x; Position.y = pos.y; Position.z = pos.z; } void XM_CALLCONV SetVelocity( FXMVECTOR v ) { XMStoreFloat3( reinterpret_cast( &Velocity ), v ); } void __cdecl SetVelocity( const XMFLOAT3& vel ) { Velocity.x = vel.x; Velocity.y = vel.y; Velocity.z = vel.z; } void XM_CALLCONV SetOrientation( FXMVECTOR forward, FXMVECTOR up ) { XMStoreFloat3( reinterpret_cast( &OrientFront ), forward ); XMStoreFloat3( reinterpret_cast( &OrientTop ), up ); } void __cdecl SetOrientation( const XMFLOAT3& forward, const XMFLOAT3& up ) { OrientFront.x = forward.x; OrientTop.x = up.x; OrientFront.y = forward.y; OrientTop.y = up.y; OrientFront.z = forward.z; OrientTop.z = up.z; } void XM_CALLCONV SetOrientationFromQuaternion( FXMVECTOR quat ) { XMVECTOR forward = XMVector3Rotate( g_XMIdentityR2, quat ); XMStoreFloat3( reinterpret_cast( &OrientFront ), forward ); XMVECTOR up = XMVector3Rotate( g_XMIdentityR1, quat ); XMStoreFloat3( reinterpret_cast( &OrientTop ), up ); } void XM_CALLCONV Update( FXMVECTOR newPos, XMVECTOR upDir, float dt ) // Updates velocity and orientation by tracking changes in position over time... { if ( dt > 0.f ) { XMVECTOR lastPos = XMLoadFloat3( reinterpret_cast( &Position ) ); XMVECTOR vDelta = ( newPos - lastPos ); XMVECTOR v = vDelta / dt; XMStoreFloat3( reinterpret_cast( &Velocity ), v ); vDelta = XMVector3Normalize( vDelta ); XMStoreFloat3( reinterpret_cast( &OrientFront ), vDelta ); v = XMVector3Cross( upDir, vDelta ); v = XMVector3Normalize( v ); v = XMVector3Cross( vDelta, v ); v = XMVector3Normalize( v ); XMStoreFloat3( reinterpret_cast( &OrientTop ), v ); XMStoreFloat3( reinterpret_cast( &Position ), newPos ); } } }; //---------------------------------------------------------------------------------- class SoundEffectInstance { public: SoundEffectInstance(SoundEffectInstance&& moveFrom); SoundEffectInstance& operator= (SoundEffectInstance&& moveFrom); virtual ~SoundEffectInstance(); void __cdecl Play( bool loop = false ); void __cdecl Stop( bool immediate = true ); void __cdecl Pause(); void __cdecl Resume(); void __cdecl SetVolume( float volume ); void __cdecl SetPitch( float pitch ); void __cdecl SetPan( float pan ); void __cdecl Apply3D( const AudioListener& listener, const AudioEmitter& emitter ); bool __cdecl IsLooped() const; SoundState __cdecl GetState(); // Notifications. void __cdecl OnDestroyParent(); private: // Private implementation. class Impl; std::unique_ptr pImpl; // Private constructors SoundEffectInstance( _In_ AudioEngine* engine, _In_ SoundEffect* effect, SOUND_EFFECT_INSTANCE_FLAGS flags ); SoundEffectInstance( _In_ AudioEngine* engine, _In_ WaveBank* effect, int index, SOUND_EFFECT_INSTANCE_FLAGS flags ); friend std::unique_ptr __cdecl SoundEffect::CreateInstance( SOUND_EFFECT_INSTANCE_FLAGS ); friend std::unique_ptr __cdecl WaveBank::CreateInstance( int, SOUND_EFFECT_INSTANCE_FLAGS ); // Prevent copying. SoundEffectInstance(SoundEffectInstance const&) DIRECTX_CTOR_DELETE SoundEffectInstance& operator= (SoundEffectInstance const&) DIRECTX_CTOR_DELETE }; //---------------------------------------------------------------------------------- class DynamicSoundEffectInstance { public: DynamicSoundEffectInstance( _In_ AudioEngine* engine, _In_opt_ std::function bufferNeeded, int sampleRate, int channels, int sampleBits = 16, SOUND_EFFECT_INSTANCE_FLAGS flags = SoundEffectInstance_Default ); DynamicSoundEffectInstance(DynamicSoundEffectInstance&& moveFrom); DynamicSoundEffectInstance& operator= (DynamicSoundEffectInstance&& moveFrom); virtual ~DynamicSoundEffectInstance(); void __cdecl Play(); void __cdecl Stop( bool immediate = true ); void __cdecl Pause(); void __cdecl Resume(); void __cdecl SetVolume( float volume ); void __cdecl SetPitch( float pitch ); void __cdecl SetPan( float pan ); void __cdecl Apply3D( const AudioListener& listener, const AudioEmitter& emitter ); void __cdecl SubmitBuffer( _In_reads_bytes_(audioBytes) const uint8_t* pAudioData, size_t audioBytes ); void __cdecl SubmitBuffer( _In_reads_bytes_(audioBytes) const uint8_t* pAudioData, uint32_t offset, size_t audioBytes ); SoundState __cdecl GetState(); size_t __cdecl GetSampleDuration( size_t bytes ) const; // Returns duration in samples of a buffer of a given size size_t __cdecl GetSampleDurationMS( size_t bytes ) const; // Returns duration in milliseconds of a buffer of a given size size_t __cdecl GetSampleSizeInBytes( uint64_t duration ) const; // Returns size of a buffer for a duration given in milliseconds int __cdecl GetPendingBufferCount() const; const WAVEFORMATEX* __cdecl GetFormat() const; private: // Private implementation. class Impl; std::unique_ptr pImpl; // Prevent copying. DynamicSoundEffectInstance(DynamicSoundEffectInstance const&) DIRECTX_CTOR_DELETE DynamicSoundEffectInstance& operator= (DynamicSoundEffectInstance const&) DIRECTX_CTOR_DELETE }; } #pragma warning(pop)