2009-12-15 22:39:18 +00:00
|
|
|
/* PROJECT: ReactOS sndrec32
|
2009-11-30 18:54:41 +00:00
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
|
|
* FILE: base/applications/sndrec32/audio_resampler_acm.cpp
|
2009-12-15 22:39:18 +00:00
|
|
|
* PURPOSE: Sound recording
|
|
|
|
* PROGRAMMERS: Marco Pagliaricci (irc: rendar)
|
2009-11-30 18:54:41 +00:00
|
|
|
*/
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
#include "stdafx.h"
|
|
|
|
#include "audio_resampler_acm.hpp"
|
2009-12-15 15:16:01 +00:00
|
|
|
#include <stdio.h>
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-03 19:00:41 +00:00
|
|
|
|
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
_AUDIO_NAMESPACE_START_
|
|
|
|
|
|
|
|
|
2009-12-03 19:00:41 +00:00
|
|
|
/////////////////////////////////////////
|
|
|
|
/////// Private Functions ////////
|
|
|
|
/////////////////////////////////////////
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-03 19:00:41 +00:00
|
|
|
void
|
|
|
|
audio_resampler_acm::init_( void )
|
2009-11-30 18:54:41 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Zeroing structures
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
ZeroMemory( &acm_header, sizeof( ACMSTREAMHEADER ));
|
|
|
|
ZeroMemory( &wformat_src, sizeof( WAVEFORMATEX ));
|
|
|
|
ZeroMemory( &wformat_dst, sizeof( WAVEFORMATEX ));
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Setting structures sizes
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
acm_header.cbStruct = sizeof( ACMSTREAMHEADER );
|
|
|
|
wformat_src.cbSize = sizeof( WAVEFORMATEX );
|
|
|
|
wformat_dst.cbSize = sizeof( WAVEFORMATEX );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Setting WAVEFORMATEX structure parameters
|
|
|
|
// according to `audio_format' in/out classes
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
wformat_src.wFormatTag = WAVE_FORMAT_PCM;
|
|
|
|
wformat_src.nSamplesPerSec = audfmt_in.sample_rate();
|
|
|
|
wformat_src.nChannels = audfmt_in.channels();
|
|
|
|
wformat_src.wBitsPerSample = audfmt_in.bits();
|
|
|
|
wformat_src.nAvgBytesPerSec = audfmt_in.byte_rate();
|
|
|
|
wformat_src.nBlockAlign = audfmt_in.block_align();
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
wformat_dst.wFormatTag = WAVE_FORMAT_PCM;
|
|
|
|
wformat_dst.nSamplesPerSec = audfmt_out.sample_rate();
|
|
|
|
wformat_dst.nChannels = audfmt_out.channels();
|
|
|
|
wformat_dst.wBitsPerSample = audfmt_out.bits();
|
|
|
|
wformat_dst.nAvgBytesPerSec = audfmt_out.byte_rate();
|
|
|
|
wformat_dst.nBlockAlign = audfmt_out.block_align();
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Init acm structures completed successfull
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////
|
|
|
|
/////// Public Functions ////////
|
|
|
|
/////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2009-12-03 19:00:41 +00:00
|
|
|
audio_resampler_acm::open( void )
|
2009-11-30 18:54:41 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
MMRESULT err;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Opens ACM stream
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
err = acmStreamOpen( &acm_stream, 0, &wformat_src, &wformat_dst,
|
|
|
|
0, 0, 0, ACM_STREAMOPENF_NONREALTIME );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( err != MMSYSERR_NOERROR )
|
|
|
|
{
|
|
|
|
//TODO: throw error
|
|
|
|
MessageBox( 0, _T("acmOpen error: %i"), _T("ERROR"), MB_ICONERROR );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
}
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Calcs source buffer lenght
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
src_buflen = ( unsigned int )
|
|
|
|
(( float )audfmt_in.byte_rate() * ( float )buf_secs );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Calcs destination source buffer lenght
|
|
|
|
// with help of ACM apis
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
err = acmStreamSize( acm_stream,
|
|
|
|
src_buflen, &dst_buflen, ACM_STREAMSIZEF_SOURCE );
|
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( err != MMSYSERR_NOERROR )
|
|
|
|
{
|
|
|
|
//TODO: throw error
|
|
|
|
MessageBox( 0, _T("acmStreamSize error"), _T("ERROR"), MB_ICONERROR );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
}
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-03 18:44:27 +00:00
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Initialize ACMSTREAMHEADER structure,
|
|
|
|
// and alloc memory for source and destination
|
|
|
|
// buffers.
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
acm_header.fdwStatus = 0;
|
|
|
|
acm_header.dwUser = 0;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
|
|
|
acm_header.pbSrc = ( LPBYTE ) new BYTE [ src_buflen ];
|
2009-11-30 18:54:41 +00:00
|
|
|
acm_header.cbSrcLength = src_buflen;
|
|
|
|
acm_header.cbSrcLengthUsed = 0;
|
|
|
|
acm_header.dwSrcUser = src_buflen;
|
|
|
|
|
|
|
|
|
|
|
|
acm_header.pbDst = ( LPBYTE ) new BYTE [ dst_buflen ];
|
|
|
|
acm_header.cbDstLength = dst_buflen;
|
|
|
|
acm_header.cbDstLengthUsed = 0;
|
|
|
|
acm_header.dwDstUser = dst_buflen;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Give ACMSTREAMHEADER initialized correctly to the
|
|
|
|
// driver.
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
err = acmStreamPrepareHeader( acm_stream, &acm_header, 0L );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( err != MMSYSERR_NOERROR )
|
|
|
|
{
|
|
|
|
//TODO: throw error
|
|
|
|
MessageBox( 0, _T("acmStreamPrepareHeader error"), _T("ERROR"), MB_ICONERROR );
|
2009-12-03 18:44:27 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
}
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// ACM stream successfully opened.
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
stream_opened = true;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2009-12-03 19:00:41 +00:00
|
|
|
audio_resampler_acm::close( void )
|
2009-11-30 18:54:41 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
MMRESULT err;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_stream )
|
|
|
|
{
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.fdwStatus & ACMSTREAMHEADER_STATUSF_PREPARED )
|
|
|
|
{
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
acm_header.cbSrcLength = src_buflen;
|
2009-11-30 18:54:41 +00:00
|
|
|
acm_header.cbDstLength = dst_buflen;
|
|
|
|
|
|
|
|
err = acmStreamUnprepareHeader( acm_stream, &acm_header, 0L );
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
|
|
|
if ( err != MMSYSERR_NOERROR )
|
|
|
|
{
|
|
|
|
|
|
|
|
//
|
|
|
|
// Free buffer memory
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.pbSrc != 0 )
|
|
|
|
delete[] acm_header.pbSrc;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.pbDst != 0 )
|
|
|
|
delete[] acm_header.pbDst;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Re-init structures
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
init_();
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Updating status
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
stream_opened = false;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//TODO: throw error
|
|
|
|
MessageBox( 0, _T("acmStreamUnPrepareHeader error"), _T("ERROR"), MB_ICONERROR );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2009-12-03 18:44:27 +00:00
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
err = acmStreamClose( acm_stream, 0 );
|
2009-11-30 18:54:41 +00:00
|
|
|
acm_stream = 0;
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( err != MMSYSERR_NOERROR )
|
|
|
|
{
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Free buffer memory
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.pbSrc != 0 )
|
|
|
|
delete[] acm_header.pbSrc;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.pbDst != 0 )
|
|
|
|
delete[] acm_header.pbDst;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Re-init structures
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
init_();
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Updating status
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
stream_opened = false;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//TODO: throw error!
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
MessageBox( 0, _T("acmStreamClose error"), _T("ERROR"), MB_ICONERROR );
|
2009-12-03 18:44:27 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
}
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
}//if acm_stream != 0
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Free buffer memory
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.pbSrc != 0 )
|
|
|
|
delete[] acm_header.pbSrc;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( acm_header.pbDst != 0 )
|
|
|
|
delete[] acm_header.pbDst;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Re-init structures
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
init_();
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Updating status
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
stream_opened = false;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// ACM sream successfully closed.
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
void
|
2009-12-03 19:00:41 +00:00
|
|
|
audio_resampler_acm::audio_receive( unsigned char * data, unsigned int size )
|
2009-11-30 18:54:41 +00:00
|
|
|
{
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
MMRESULT err;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Checking for acm stream opened
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( stream_opened )
|
|
|
|
{
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Copy audio data from extern to
|
|
|
|
// internal source buffer
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
memcpy( acm_header.pbSrc, data, size );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
acm_header.cbSrcLength = size;
|
|
|
|
acm_header.cbDstLengthUsed = 0;
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
err = acmStreamConvert( acm_stream, &acm_header, ACM_STREAMCONVERTF_BLOCKALIGN );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
if ( err != MMSYSERR_NOERROR )
|
|
|
|
{
|
|
|
|
//TODO: throw error
|
|
|
|
MessageBox( 0, _T("acmStreamConvert error"), _T("ERROR"), MB_ICONERROR );
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-03 18:44:27 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
}
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Wait for sound conversion
|
|
|
|
//
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
while(( ACMSTREAMHEADER_STATUSF_DONE & acm_header.fdwStatus ) == 0 );
|
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
2009-12-15 22:39:18 +00:00
|
|
|
//
|
|
|
|
// Copy resampled audio, to destination buffer.
|
|
|
|
//
|
|
|
|
|
|
|
|
//memcpy( pbOutputData, acm_header.pbDst, acm_header.cbDstLengthUsed );
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-12-03 19:00:41 +00:00
|
|
|
|
2009-11-30 18:54:41 +00:00
|
|
|
_AUDIO_NAMESPACE_END_
|