//========= Copyright 1996-2009, Valve Corporation, All rights reserved. ============// // // Purpose: // // $NoKeywords: $ // //=============================================================================// // This module implements the voice record and compression functions //#include "audio_pch.h" //#include "voice.h" #include "tier0/platform.h" #include "ivoicerecord.h" #include "tier0/dbg.h" #include "tier0/threadtools.h" #include <assert.h> #include <SDL_audio.h> #define RECORDING_BUFFER_SECONDS 3 #define SAMPLE_COUNT 2048 // ------------------------------------------------------------------------------ // VoiceRecord_SDL // ------------------------------------------------------------------------------ struct AudioBuf { int Read(char *out, int len) { int nAvalible = (size + (writePtr - readPtr)) % size; if( nAvalible == 0 ) return 0; if( len > nAvalible ) len = nAvalible; int diff = (data + size) - readPtr; if( len > diff ) { memcpy(out, readPtr, diff ); memcpy(out+diff, data, len-diff ); } else memcpy(out, readPtr, len); readPtr += len; if( readPtr >= data + size ) readPtr -= size; return len; } void Write(char *in, int len) { int diff = (data + size) - writePtr; if( len > diff ) { memcpy(writePtr, in, diff ); memcpy(data, in+diff, len-diff ); } else memcpy(writePtr, in, len); writePtr += len; if (writePtr >= (data + size)) writePtr -= size; } int size; char *data; char *readPtr; char *writePtr; }; class VoiceRecord_SDL : public IVoiceRecord { protected: virtual ~VoiceRecord_SDL(); public: VoiceRecord_SDL(); virtual void Release(); virtual bool RecordStart(); virtual void RecordStop(); // Initialize. The format of the data we expect from the provider is // 8-bit signed mono at the specified sample rate. virtual bool Init(int sampleRate); virtual void Idle() {}; // Stub void RenderBuffer( char *pszBuf, int size ); // Get the most recent N samples. virtual int GetRecordedData(short *pOut, int nSamplesWanted ); SDL_AudioSpec m_ReceivedRecordingSpec; int m_BytesPerSample; // Да кому нужна эта ваша инкапсуляция? int m_nSampleRate; private: bool InitalizeInterfaces(); // Initialize the openal capture buffers and other interfaces void ReleaseInterfaces(); // Release openal buffers and other interfaces void ClearInterfaces(); // Clear members. private: SDL_AudioDeviceID m_Device; AudioBuf m_AudioBuffer; }; void audioRecordingCallback( void *userdata, uint8 *stream, int len ) { VoiceRecord_SDL *voice = (VoiceRecord_SDL*)userdata; voice->RenderBuffer( (char*)stream, len ); } VoiceRecord_SDL::VoiceRecord_SDL() : m_nSampleRate( 0 ) ,m_Device( 0 ) { m_AudioBuffer.data = NULL; m_AudioBuffer.readPtr = NULL; m_AudioBuffer.writePtr = NULL; ClearInterfaces(); } VoiceRecord_SDL::~VoiceRecord_SDL() { ReleaseInterfaces(); ClearInterfaces(); } void VoiceRecord_SDL::Release() { ReleaseInterfaces(); ClearInterfaces(); delete this; } bool VoiceRecord_SDL::RecordStart() { if ( !m_Device ) InitalizeInterfaces(); if ( !m_Device ) return false; SDL_PauseAudioDevice( m_Device, SDL_FALSE ); return true; } void VoiceRecord_SDL::RecordStop() { // Stop capturing. if ( m_Device ) SDL_PauseAudioDevice( m_Device, SDL_TRUE ); // Release the capture buffer interface and any other resources that are no // longer needed ReleaseInterfaces(); } bool VoiceRecord_SDL::InitalizeInterfaces() { //Default audio spec SDL_AudioSpec desiredRecordingSpec; SDL_zero(desiredRecordingSpec); desiredRecordingSpec.freq = m_nSampleRate; desiredRecordingSpec.format = AUDIO_S16; desiredRecordingSpec.channels = 1; desiredRecordingSpec.samples = SAMPLE_COUNT; desiredRecordingSpec.callback = audioRecordingCallback; desiredRecordingSpec.userdata = (void*)this; //Open recording device m_Device = SDL_OpenAudioDevice( NULL, SDL_TRUE, &desiredRecordingSpec, &m_ReceivedRecordingSpec, 0 ); if( m_Device != 0 ) { //Calculate per sample bytes m_BytesPerSample = m_ReceivedRecordingSpec.channels * ( SDL_AUDIO_BITSIZE( m_ReceivedRecordingSpec.format ) / 8 ); //Calculate bytes per second int bytesPerSecond = m_ReceivedRecordingSpec.freq * m_BytesPerSample; //Allocate and initialize byte buffer m_AudioBuffer.size = RECORDING_BUFFER_SECONDS * bytesPerSecond; if( !m_AudioBuffer.data ) m_AudioBuffer.data = (char *)malloc( m_AudioBuffer.size ); m_AudioBuffer.readPtr = m_AudioBuffer.data; m_AudioBuffer.writePtr = m_AudioBuffer.data + SAMPLE_COUNT*m_BytesPerSample*2; memset( m_AudioBuffer.data, 0, m_AudioBuffer.size ); return true; } else return false; } bool VoiceRecord_SDL::Init(int sampleRate) { m_nSampleRate = sampleRate; ReleaseInterfaces(); return true; } void VoiceRecord_SDL::ReleaseInterfaces() { if( m_Device != 0 ) SDL_CloseAudioDevice( m_Device ); m_Device = 0; } void VoiceRecord_SDL::ClearInterfaces() { if( m_AudioBuffer.data ) { free( m_AudioBuffer.data ); m_AudioBuffer.data = NULL; m_AudioBuffer.readPtr = NULL; m_AudioBuffer.writePtr = NULL; } m_Device = 0; } void VoiceRecord_SDL::RenderBuffer( char *pszBuf, int size ) { m_AudioBuffer.Write( pszBuf, size ); } int VoiceRecord_SDL::GetRecordedData(short *pOut, int nSamples ) { if ( !m_AudioBuffer.data || nSamples == 0 ) return 0; int cbSamples = nSamples * m_BytesPerSample; return m_AudioBuffer.Read( (char*)pOut, cbSamples )/m_BytesPerSample; } IVoiceRecord* CreateVoiceRecord_SDL(int sampleRate) { VoiceRecord_SDL *pRecord = new VoiceRecord_SDL; if ( pRecord && pRecord->Init(sampleRate) ) return pRecord; else if( pRecord ) pRecord->Release(); return NULL; }