mirror of
https://github.com/id-Software/DOOM-3-BFG.git
synced 2025-01-22 09:21:12 +00:00
488 lines
13 KiB
C++
488 lines
13 KiB
C++
/*
|
|
===========================================================================
|
|
|
|
Doom 3 BFG Edition GPL Source Code
|
|
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
|
|
Copyright (C) 2013 Robert Beckebans
|
|
Copyright (c) 2010 by Chris Robinson <chris.kcat@gmail.com> (OpenAL Info Utility)
|
|
|
|
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
|
|
|
|
Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
|
|
|
|
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
|
|
|
|
===========================================================================
|
|
*/
|
|
#pragma hdrstop
|
|
#include "precompiled.h"
|
|
#include "../snd_local.h"
|
|
#include "../../../doomclassic/doom/i_sound.h"
|
|
|
|
idCVar s_showLevelMeter( "s_showLevelMeter", "0", CVAR_BOOL | CVAR_ARCHIVE, "Show VU meter" );
|
|
idCVar s_meterTopTime( "s_meterTopTime", "1000", CVAR_INTEGER | CVAR_ARCHIVE, "How long (in milliseconds) peaks are displayed on the VU meter" );
|
|
idCVar s_meterPosition( "s_meterPosition", "100 100 20 200", CVAR_ARCHIVE, "VU meter location (x y w h)" );
|
|
idCVar s_device( "s_device", "-1", CVAR_INTEGER | CVAR_ARCHIVE, "Which audio device to use (listDevices to list, -1 for default)" );
|
|
idCVar s_showPerfData( "s_showPerfData", "0", CVAR_BOOL, "Show XAudio2 Performance data" );
|
|
extern idCVar s_volume_dB;
|
|
|
|
|
|
/*
|
|
========================
|
|
idSoundHardware_OpenAL::idSoundHardware_OpenAL
|
|
========================
|
|
*/
|
|
idSoundHardware_OpenAL::idSoundHardware_OpenAL()
|
|
{
|
|
openalDevice = NULL;
|
|
openalContext = NULL;
|
|
|
|
//vuMeterRMS = NULL;
|
|
//vuMeterPeak = NULL;
|
|
|
|
//outputChannels = 0;
|
|
//channelMask = 0;
|
|
|
|
voices.SetNum( 0 );
|
|
zombieVoices.SetNum( 0 );
|
|
freeVoices.SetNum( 0 );
|
|
|
|
lastResetTime = 0;
|
|
}
|
|
|
|
void idSoundHardware_OpenAL::PrintDeviceList( const char* list )
|
|
{
|
|
if( !list || *list == '\0' )
|
|
{
|
|
idLib::Printf( " !!! none !!!\n" );
|
|
}
|
|
else
|
|
{
|
|
do
|
|
{
|
|
idLib::Printf( " %s\n", list );
|
|
list += strlen( list ) + 1;
|
|
}
|
|
while( *list != '\0' );
|
|
}
|
|
}
|
|
|
|
void idSoundHardware_OpenAL::PrintALCInfo( ALCdevice* device )
|
|
{
|
|
ALCint major, minor;
|
|
|
|
if( device )
|
|
{
|
|
const ALCchar* devname = NULL;
|
|
idLib::Printf( "\n" );
|
|
if( alcIsExtensionPresent( device, "ALC_ENUMERATE_ALL_EXT" ) != AL_FALSE )
|
|
{
|
|
devname = alcGetString( device, ALC_ALL_DEVICES_SPECIFIER );
|
|
}
|
|
|
|
if( CheckALCErrors( device ) != ALC_NO_ERROR || !devname )
|
|
{
|
|
devname = alcGetString( device, ALC_DEVICE_SPECIFIER );
|
|
}
|
|
|
|
idLib::Printf( "** Info for device \"%s\" **\n", devname );
|
|
}
|
|
alcGetIntegerv( device, ALC_MAJOR_VERSION, 1, &major );
|
|
alcGetIntegerv( device, ALC_MINOR_VERSION, 1, &minor );
|
|
|
|
if( CheckALCErrors( device ) == ALC_NO_ERROR )
|
|
idLib::Printf( "ALC version: %d.%d\n", major, minor );
|
|
|
|
if( device )
|
|
{
|
|
idLib::Printf( "OpenAL extensions: %s", alGetString( AL_EXTENSIONS ) );
|
|
|
|
//idLib::Printf("ALC extensions:");
|
|
//printList(alcGetString(device, ALC_EXTENSIONS), ' ');
|
|
CheckALCErrors( device );
|
|
}
|
|
}
|
|
|
|
void idSoundHardware_OpenAL::PrintALInfo()
|
|
{
|
|
idLib::Printf( "OpenAL vendor string: %s\n", alGetString( AL_VENDOR ) );
|
|
idLib::Printf( "OpenAL renderer string: %s\n", alGetString( AL_RENDERER ) );
|
|
idLib::Printf( "OpenAL version string: %s\n", alGetString( AL_VERSION ) );
|
|
idLib::Printf( "OpenAL extensions: %s", alGetString( AL_EXTENSIONS ) );
|
|
//PrintList(alGetString(AL_EXTENSIONS), ' ');
|
|
CheckALErrors();
|
|
}
|
|
|
|
void listDevices_f( const idCmdArgs& args )
|
|
{
|
|
idLib::Printf( "Available playback devices:\n" );
|
|
if( alcIsExtensionPresent( NULL, "ALC_ENUMERATE_ALL_EXT" ) != AL_FALSE )
|
|
{
|
|
idSoundHardware_OpenAL::PrintDeviceList( alcGetString( NULL, ALC_ALL_DEVICES_SPECIFIER ) );
|
|
}
|
|
else
|
|
{
|
|
idSoundHardware_OpenAL::PrintDeviceList( alcGetString( NULL, ALC_DEVICE_SPECIFIER ) );
|
|
}
|
|
|
|
//idLib::Printf("Available capture devices:\n");
|
|
//printDeviceList(alcGetString(NULL, ALC_CAPTURE_DEVICE_SPECIFIER));
|
|
|
|
if( alcIsExtensionPresent( NULL, "ALC_ENUMERATE_ALL_EXT" ) != AL_FALSE )
|
|
{
|
|
idLib::Printf( "Default playback device: %s\n", alcGetString( NULL, ALC_DEFAULT_ALL_DEVICES_SPECIFIER ) );
|
|
}
|
|
else
|
|
{
|
|
idLib::Printf( "Default playback device: %s\n", alcGetString( NULL, ALC_DEFAULT_DEVICE_SPECIFIER ) );
|
|
}
|
|
|
|
//idLib::Printf("Default capture device: %s\n", alcGetString(NULL, ALC_CAPTURE_DEFAULT_DEVICE_SPECIFIER));
|
|
|
|
idSoundHardware_OpenAL::PrintALCInfo( NULL );
|
|
|
|
idSoundHardware_OpenAL::PrintALCInfo( ( ALCdevice* )soundSystem->GetOpenALDevice() );
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idSoundHardware_OpenAL::Init
|
|
========================
|
|
*/
|
|
void idSoundHardware_OpenAL::Init()
|
|
{
|
|
cmdSystem->AddCommand( "listDevices", listDevices_f, 0, "Lists the connected sound devices", NULL );
|
|
|
|
common->Printf( "Setup OpenAL device and context... " );
|
|
|
|
openalDevice = alcOpenDevice( NULL );
|
|
if( openalDevice == NULL )
|
|
{
|
|
common->FatalError( "idSoundHardware_OpenAL::Init: alcOpenDevice() failed\n" );
|
|
return;
|
|
}
|
|
|
|
openalContext = alcCreateContext( openalDevice, NULL );
|
|
if( alcMakeContextCurrent( openalContext ) == 0 )
|
|
{
|
|
common->FatalError( "idSoundHardware_OpenAL::Init: alcMakeContextCurrent( %p) failed\n", openalContext );
|
|
return;
|
|
}
|
|
|
|
common->Printf( "Done.\n" );
|
|
|
|
common->Printf( "OpenAL vendor: %s\n", alGetString( AL_VENDOR ) );
|
|
common->Printf( "OpenAL renderer: %s\n", alGetString( AL_RENDERER ) );
|
|
common->Printf( "OpenAL version: %s\n", alGetString( AL_VERSION ) );
|
|
common->Printf( "OpenAL extensions: %s\n", alGetString( AL_EXTENSIONS ) );
|
|
|
|
//pMasterVoice->SetVolume( DBtoLinear( s_volume_dB.GetFloat() ) );
|
|
|
|
//outputChannels = deviceDetails.OutputFormat.Format.nChannels;
|
|
//channelMask = deviceDetails.OutputFormat.dwChannelMask;
|
|
|
|
//idSoundVoice::InitSurround( outputChannels, channelMask );
|
|
|
|
// ---------------------
|
|
// Initialize the Doom classic sound system.
|
|
// ---------------------
|
|
I_InitSoundHardware( voices.Max(), 0 );
|
|
|
|
// ---------------------
|
|
// Create VU Meter Effect
|
|
// ---------------------
|
|
/*
|
|
IUnknown* vuMeter = NULL;
|
|
XAudio2CreateVolumeMeter( &vuMeter, 0 );
|
|
|
|
XAUDIO2_EFFECT_DESCRIPTOR descriptor;
|
|
descriptor.InitialState = true;
|
|
descriptor.OutputChannels = outputChannels;
|
|
descriptor.pEffect = vuMeter;
|
|
|
|
XAUDIO2_EFFECT_CHAIN chain;
|
|
chain.EffectCount = 1;
|
|
chain.pEffectDescriptors = &descriptor;
|
|
|
|
pMasterVoice->SetEffectChain( &chain );
|
|
|
|
vuMeter->Release();
|
|
*/
|
|
|
|
// ---------------------
|
|
// Create VU Meter Graph
|
|
// ---------------------
|
|
|
|
/*
|
|
vuMeterRMS = console->CreateGraph( outputChannels );
|
|
vuMeterPeak = console->CreateGraph( outputChannels );
|
|
vuMeterRMS->Enable( false );
|
|
vuMeterPeak->Enable( false );
|
|
|
|
memset( vuMeterPeakTimes, 0, sizeof( vuMeterPeakTimes ) );
|
|
|
|
vuMeterPeak->SetFillMode( idDebugGraph::GRAPH_LINE );
|
|
vuMeterPeak->SetBackgroundColor( idVec4( 0.0f, 0.0f, 0.0f, 0.0f ) );
|
|
|
|
vuMeterRMS->AddGridLine( 0.500f, idVec4( 0.5f, 0.5f, 0.5f, 1.0f ) );
|
|
vuMeterRMS->AddGridLine( 0.250f, idVec4( 0.5f, 0.5f, 0.5f, 1.0f ) );
|
|
vuMeterRMS->AddGridLine( 0.125f, idVec4( 0.5f, 0.5f, 0.5f, 1.0f ) );
|
|
|
|
const char* channelNames[] = { "L", "R", "C", "S", "Lb", "Rb", "Lf", "Rf", "Cb", "Ls", "Rs" };
|
|
for( int i = 0, ci = 0; ci < sizeof( channelNames ) / sizeof( channelNames[0] ); ci++ )
|
|
{
|
|
if( ( channelMask & BIT( ci ) ) == 0 )
|
|
{
|
|
continue;
|
|
}
|
|
vuMeterRMS->SetLabel( i, channelNames[ ci ] );
|
|
i++;
|
|
}
|
|
*/
|
|
|
|
// OpenAL doesn't really impose a maximum number of sources
|
|
voices.SetNum( voices.Max() );
|
|
freeVoices.SetNum( voices.Max() );
|
|
zombieVoices.SetNum( 0 );
|
|
for( int i = 0; i < voices.Num(); i++ )
|
|
{
|
|
freeVoices[i] = &voices[i];
|
|
}
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idSoundHardware_OpenAL::Shutdown
|
|
========================
|
|
*/
|
|
void idSoundHardware_OpenAL::Shutdown()
|
|
{
|
|
for( int i = 0; i < voices.Num(); i++ )
|
|
{
|
|
voices[ i ].DestroyInternal();
|
|
}
|
|
voices.Clear();
|
|
freeVoices.Clear();
|
|
zombieVoices.Clear();
|
|
|
|
// ---------------------
|
|
// Shutdown the Doom classic sound system.
|
|
// ---------------------
|
|
I_ShutdownSoundHardware();
|
|
|
|
alcMakeContextCurrent( NULL );
|
|
|
|
alcDestroyContext( openalContext );
|
|
openalContext = NULL;
|
|
|
|
alcCloseDevice( openalDevice );
|
|
openalDevice = NULL;
|
|
|
|
/*
|
|
if( vuMeterRMS != NULL )
|
|
{
|
|
console->DestroyGraph( vuMeterRMS );
|
|
vuMeterRMS = NULL;
|
|
}
|
|
if( vuMeterPeak != NULL )
|
|
{
|
|
console->DestroyGraph( vuMeterPeak );
|
|
vuMeterPeak = NULL;
|
|
}
|
|
*/
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idSoundHardware_OpenAL::AllocateVoice
|
|
========================
|
|
*/
|
|
idSoundVoice* idSoundHardware_OpenAL::AllocateVoice( const idSoundSample* leadinSample, const idSoundSample* loopingSample )
|
|
{
|
|
if( leadinSample == NULL )
|
|
{
|
|
return NULL;
|
|
}
|
|
if( loopingSample != NULL )
|
|
{
|
|
if( ( leadinSample->format.basic.formatTag != loopingSample->format.basic.formatTag ) || ( leadinSample->format.basic.numChannels != loopingSample->format.basic.numChannels ) )
|
|
{
|
|
idLib::Warning( "Leadin/looping format mismatch: %s & %s", leadinSample->GetName(), loopingSample->GetName() );
|
|
loopingSample = NULL;
|
|
}
|
|
}
|
|
|
|
// Try to find a free voice that matches the format
|
|
// But fallback to the last free voice if none match the format
|
|
idSoundVoice* voice = NULL;
|
|
for( int i = 0; i < freeVoices.Num(); i++ )
|
|
{
|
|
if( freeVoices[i]->IsPlaying() )
|
|
{
|
|
continue;
|
|
}
|
|
voice = ( idSoundVoice* )freeVoices[i];
|
|
if( voice->CompatibleFormat( ( idSoundSample_OpenAL* )leadinSample ) )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if( voice != NULL )
|
|
{
|
|
voice->Create( leadinSample, loopingSample );
|
|
freeVoices.Remove( voice );
|
|
return voice;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idSoundHardware_OpenAL::FreeVoice
|
|
========================
|
|
*/
|
|
void idSoundHardware_OpenAL::FreeVoice( idSoundVoice* voice )
|
|
{
|
|
voice->Stop();
|
|
|
|
// Stop() is asyncronous, so we won't flush bufferes until the
|
|
// voice on the zombie channel actually returns !IsPlaying()
|
|
zombieVoices.Append( voice );
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idSoundHardware_OpenAL::Update
|
|
========================
|
|
*/
|
|
void idSoundHardware_OpenAL::Update()
|
|
{
|
|
if( openalDevice == NULL )
|
|
{
|
|
int nowTime = Sys_Milliseconds();
|
|
if( lastResetTime + 1000 < nowTime )
|
|
{
|
|
lastResetTime = nowTime;
|
|
Init();
|
|
}
|
|
return;
|
|
}
|
|
|
|
if( soundSystem->IsMuted() )
|
|
{
|
|
alListenerf( AL_GAIN, 0.0f );
|
|
}
|
|
else
|
|
{
|
|
alListenerf( AL_GAIN, DBtoLinear( s_volume_dB.GetFloat() ) );
|
|
}
|
|
|
|
// IXAudio2SourceVoice::Stop() has been called for every sound on the
|
|
// zombie list, but it is documented as asyncronous, so we have to wait
|
|
// until it actually reports that it is no longer playing.
|
|
for( int i = 0; i < zombieVoices.Num(); i++ )
|
|
{
|
|
zombieVoices[i]->FlushSourceBuffers();
|
|
if( !zombieVoices[i]->IsPlaying() )
|
|
{
|
|
freeVoices.Append( zombieVoices[i] );
|
|
zombieVoices.RemoveIndexFast( i );
|
|
i--;
|
|
}
|
|
else
|
|
{
|
|
static int playingZombies;
|
|
playingZombies++;
|
|
}
|
|
}
|
|
|
|
/*
|
|
if( s_showPerfData.GetBool() )
|
|
{
|
|
XAUDIO2_PERFORMANCE_DATA perfData;
|
|
pXAudio2->GetPerformanceData( &perfData );
|
|
idLib::Printf( "Voices: %d/%d CPU: %.2f%% Mem: %dkb\n", perfData.ActiveSourceVoiceCount, perfData.TotalSourceVoiceCount, perfData.AudioCyclesSinceLastQuery / ( float )perfData.TotalCyclesSinceLastQuery, perfData.MemoryUsageInBytes / 1024 );
|
|
}
|
|
*/
|
|
|
|
/*
|
|
if( vuMeterRMS == NULL )
|
|
{
|
|
// Init probably hasn't been called yet
|
|
return;
|
|
}
|
|
|
|
vuMeterRMS->Enable( s_showLevelMeter.GetBool() );
|
|
vuMeterPeak->Enable( s_showLevelMeter.GetBool() );
|
|
|
|
if( !s_showLevelMeter.GetBool() )
|
|
{
|
|
pMasterVoice->DisableEffect( 0 );
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
pMasterVoice->EnableEffect( 0 );
|
|
}
|
|
|
|
float peakLevels[ 8 ];
|
|
float rmsLevels[ 8 ];
|
|
|
|
XAUDIO2FX_VOLUMEMETER_LEVELS levels;
|
|
levels.ChannelCount = outputChannels;
|
|
levels.pPeakLevels = peakLevels;
|
|
levels.pRMSLevels = rmsLevels;
|
|
|
|
if( levels.ChannelCount > 8 )
|
|
{
|
|
levels.ChannelCount = 8;
|
|
}
|
|
|
|
pMasterVoice->GetEffectParameters( 0, &levels, sizeof( levels ) );
|
|
|
|
int currentTime = Sys_Milliseconds();
|
|
for( int i = 0; i < outputChannels; i++ )
|
|
{
|
|
if( vuMeterPeakTimes[i] < currentTime )
|
|
{
|
|
vuMeterPeak->SetValue( i, vuMeterPeak->GetValue( i ) * 0.9f, colorRed );
|
|
}
|
|
}
|
|
|
|
float width = 20.0f;
|
|
float height = 200.0f;
|
|
float left = 100.0f;
|
|
float top = 100.0f;
|
|
|
|
sscanf( s_meterPosition.GetString(), "%f %f %f %f", &left, &top, &width, &height );
|
|
|
|
vuMeterRMS->SetPosition( left, top, width * levels.ChannelCount, height );
|
|
vuMeterPeak->SetPosition( left, top, width * levels.ChannelCount, height );
|
|
|
|
for( uint32 i = 0; i < levels.ChannelCount; i++ )
|
|
{
|
|
vuMeterRMS->SetValue( i, rmsLevels[ i ], idVec4( 0.5f, 1.0f, 0.0f, 1.00f ) );
|
|
if( peakLevels[ i ] >= vuMeterPeak->GetValue( i ) )
|
|
{
|
|
vuMeterPeak->SetValue( i, peakLevels[ i ], colorRed );
|
|
vuMeterPeakTimes[i] = currentTime + s_meterTopTime.GetInteger();
|
|
}
|
|
}
|
|
*/
|
|
}
|
|
|
|
|