source: trunk/Utilities/GeneratePcmWavFile/GeneratePcmWavFile.cpp @ 9

Last change on this file since 9 was 9, checked in by roman, 10 years ago
  • Property svn:keywords set to Id
File size: 20.6 KB
Line 
1////////////////////////////////////////////////////////////
2// Copyright (C) Roman Ryltsov, 2008-2011
3// Created by Roman Ryltsov roman@alax.info
4//
5// $Id: GeneratePcmWavFile.cpp 9 2011-08-28 11:57:19Z roman $
6
7#include "stdafx.h"
8#include "rodshow.h"
9
10////////////////////////////////////////////////////////////
11// CModule
12
13class CModule : 
14        public CAtlExeModuleT<CModule>
15{
16        typedef CThreadT<CModule> CThread;
17
18public:
19
20        ////////////////////////////////////////////////////////
21        // CGenericFilterGraph
22
23        class CGenericFilterGraph
24        {
25        public:
26#if _DEVELOPMENT
27                BOOL m_bShowDestructorMessageBox;
28#endif // _DEVELOPMENT
29                CComPtr<IFilterGraph2> m_pFilterGraph;
30                CComPtr<IMediaControl> m_pMediaControl;
31                CComPtr<IMediaEventEx> m_pMediaEventEx;
32                CComPtr<IMediaFilter> m_pMediaFilter;
33                CComPtr<IMediaPosition> m_pMediaPosition;
34
35        public:
36        // CGenericFilterGraph
37                CGenericFilterGraph(BOOL bShowDestructorMessageBox = FALSE) throw()
38                {
39                        bShowDestructorMessageBox;
40#if _DEVELOPMENT
41                        m_bShowDestructorMessageBox = bShowDestructorMessageBox;
42#endif // _DEVELOPMENT
43                }
44                ~CGenericFilterGraph() throw()
45                {
46#if _DEVELOPMENT
47                        if(m_pFilterGraph && m_bShowDestructorMessageBox)
48                                AtlMessageBox(GetActiveWindow(), _T("DirectShow Filter Graph is about to be Released - Break In"), _T("Debug"), MB_ICONWARNING | MB_OK);
49#endif // _DEVELOPMENT
50                }
51                VOID CoCreateInstance(const CLSID& ClassIdentifier = CLSID_FilterGraph)
52                {
53                        __C(m_pFilterGraph.CoCreateInstance(ClassIdentifier));
54                        m_pMediaControl = CComQIPtr<IMediaControl>(m_pFilterGraph);
55                        m_pMediaEventEx = CComQIPtr<IMediaEventEx>(m_pFilterGraph);
56                        m_pMediaFilter = CComQIPtr<IMediaFilter>(m_pFilterGraph);
57                        m_pMediaPosition = CComQIPtr<IMediaPosition>(m_pFilterGraph);
58                        __D(m_pMediaControl && m_pMediaEventEx && m_pMediaFilter && m_pMediaPosition, E_NOINTERFACE);
59                }
60                VOID SetShowDestructorMessageBox(BOOL bShowDestructorMessageBox = TRUE)
61                {
62                        bShowDestructorMessageBox;
63#if _DEVELOPMENT
64                        m_bShowDestructorMessageBox = bShowDestructorMessageBox;
65#endif // _DEVELOPMENT
66                }
67                operator const CComPtr<IFilterGraph2>& () const throw()
68                {
69                        return m_pFilterGraph;
70                }
71                const CComPtr<IFilterGraph2>& operator -> () const throw()
72                {
73                        return m_pFilterGraph;
74                }
75        };
76
77        ////////////////////////////////////////////////////////
78        // CSourceFilter
79
80        class ATL_NO_VTABLE CSourceFilter :
81                public CComObjectRootEx<CComMultiThreadModelNoCS>,
82                public CComCoClass<CSourceFilter>,
83                public CPushSourceFilterT<CSourceFilter>,
84                public CBasePersistT<CSourceFilter>,
85                public CAmFilterMiscFlagsT<CSourceFilter, AM_FILTER_MISC_FLAGS_IS_SOURCE>
86        {
87        public:
88
89        DECLARE_NO_REGISTRY()
90
91        DECLARE_PROTECT_FINAL_CONSTRUCT()
92
93        //DECLARE_QI_TRACE(CSourceFilter)
94
95        BEGIN_COM_MAP(CSourceFilter)
96                COM_INTERFACE_ENTRY(IBaseFilter)
97                COM_INTERFACE_ENTRY(IMediaFilter)
98                COM_INTERFACE_ENTRY_IID(__uuidof(IPersist), IBaseFilter)
99                COM_INTERFACE_ENTRY(IAMFilterMiscFlags)
100        END_COM_MAP()
101
102        public:
103
104                ////////////////////////////////////////////////////////
105                // CThreadContext
106
107                class COutputPin;
108
109                class CThreadContext :
110                        public CPushSourceFilterT<CSourceFilter>::CThreadContext
111                {
112                public:
113                        CEvent m_MediaSampleTimeEvent;
114                        REFERENCE_TIME m_nMediaSampleTime;
115                        SIZE_T m_nCurrentSampleIndex;
116
117                public:
118                // CThreadContext
119                        CThreadContext(CEvent& TerminationEvent) throw() :
120                                CPushSourceFilter::CThreadContext(TerminationEvent),
121                                m_nMediaSampleTime(0),
122                                m_nCurrentSampleIndex(0)
123                        {
124                        }
125                };
126
127                ////////////////////////////////////////////////////////
128                // COutputPin
129
130                class ATL_NO_VTABLE COutputPin :
131                        public CComObjectRootEx<CComMultiThreadModelNoCS>,
132                        public CPushSourceFilterT<CSourceFilter>::COutputPinT<COutputPin, CSourceFilter, CThreadContext>,
133                        public CAmPushSourceT<COutputPin, 0>
134                {
135                public:
136
137                //DECLARE_QI_TRACE(CSourceFilter::COutputPin)
138
139                BEGIN_COM_MAP(COutputPin)
140                        COM_INTERFACE_ENTRY(IPin)
141                        COM_INTERFACE_ENTRY(IAMPushSource)
142                        COM_INTERFACE_ENTRY(IAMLatency)
143                END_COM_MAP()
144
145                public:
146                        CMediaType m_pDataMediaType;
147                        REFERENCE_TIME m_nDataLength;
148                        DOUBLE m_fSignalPeriod;
149                        DOUBLE m_fSignalAmplitude;
150                        DOUBLE m_fNoiseAmplitude;
151
152                public:
153                // COutputPin
154                        COutputPin() throw() :
155                                m_nDataLength(0),
156                                m_fSignalPeriod(0),
157                                m_fSignalAmplitude(0),
158                                m_fNoiseAmplitude(0)
159                        {
160                                _Z4(atlTraceRefcount, 4, _T("this 0x%p\n"), this);
161                        }
162                        ~COutputPin() throw()
163                        {
164                                _Z4(atlTraceRefcount, 4, _T("this 0x%p\n"), this);
165                        }
166                        VOID EnumerateMediaTypes(CAtlList<CMediaType>& MediaTypeList)
167                        {
168                                CRoCriticalSectionLock DataLock(GetDataCriticalSection());
169                                _W(MediaTypeList.AddTail(m_pDataMediaType));
170                        }
171                        BOOL CheckMediaType(const CMediaType& pMediaType) const throw()
172                        {
173                                _A(pMediaType);
174                                //if(pMediaType->majortype != MEDIATYPE_Audio || pMediaType->subtype != MEDIASUBTYPE_PCM)
175                                //      return FALSE;
176                                //const CWaveFormatEx* pWaveFormatEx = pMediaType.GetWaveFormatEx();
177                                //if(!pWaveFormatEx)
178                                //      return FALSE;
179                                return m_pDataMediaType.Compare(pMediaType);
180                        }
181                        BOOL DecideMemAllocatorProperties(IMemAllocator* pMemAllocator, ALLOCATOR_PROPERTIES Properties) throw()
182                        {
183                                CRoCriticalSectionLock DataLock(GetDataCriticalSection());
184                                const CMediaType& pMediaType = GetMediaTypeReference();
185                                const CWaveFormatEx* pWaveFormatEx = pMediaType.GetWaveFormatEx();
186                                if(!pWaveFormatEx)
187                                        return FALSE;
188                                if(!__super::DecideMemAllocatorProperties(pMemAllocator, Properties))
189                                        return FALSE;
190                                static const ULONG g_nDefaultBufferLength = 500; // 500 ms
191                                const SIZE_T nDefaultBufferBlockCount = (pWaveFormatEx->nAvgBytesPerSec * g_nDefaultBufferLength / 1000 + pWaveFormatEx->nBlockAlign - 1) / pWaveFormatEx->nBlockAlign;
192                                const SIZE_T nDefaultBufferSize = nDefaultBufferBlockCount * pWaveFormatEx->nBlockAlign;
193                                if(Properties.cbBuffer < (LONG) nDefaultBufferSize)
194                                        Properties.cbBuffer = (LONG) nDefaultBufferSize;
195                                if(Properties.cbAlign < pWaveFormatEx->nBlockAlign)
196                                        Properties.cbAlign = pWaveFormatEx->nBlockAlign;
197                                if(!SetMemAllocatorBufferSize(pMemAllocator, Properties))
198                                        return FALSE;
199                                const SIZE_T nSampleCount = Properties.cbBuffer / pWaveFormatEx->nBlockAlign;
200                                const SIZE_T nEffectiveBufferSize = nSampleCount * pWaveFormatEx->nBlockAlign;
201                                const REFERENCE_TIME nBufferTime = (REFERENCE_TIME) (1000 * 10000) * nEffectiveBufferSize / pWaveFormatEx->nAvgBytesPerSec;
202                                SetLatency(nBufferTime);
203                                return TRUE;
204                        }
205                        VOID InitializeThread(CThreadContext& ThreadContext) throw()
206                        {
207                                CRoCriticalSectionLock DataLock(GetDataCriticalSection());
208                                __super::InitializeThread(ThreadContext);
209                                if(m_fSignalPeriod > 0)
210                                {
211                                        __D(m_pDataMediaType.GetWaveFormatEx()->wBitsPerSample == 16, E_NOTIMPL);
212                                        ThreadContext.m_nCurrentSampleIndex = 0;
213                                }
214                        }
215                        BOOL ComposeMediaSample(CThreadContext& ThreadContext, IMediaSample* pMediaSample)
216                        {
217                                CMediaSampleProperties Properties(pMediaSample);
218                                _A(!Properties.pMediaType);
219                                if(ThreadContext.m_nMediaSampleTime >= m_nDataLength)
220                                        return FALSE; // Finished
221                                CRoCriticalSectionLock DataLock(GetDataCriticalSection());
222                                const CWaveFormatEx* pWaveFormatEx = GetMediaTypeReference().GetWaveFormatEx();
223                                _A(pWaveFormatEx);
224                                const SIZE_T nRemainedDataSize = (SIZE_T) ((m_nDataLength - ThreadContext.m_nMediaSampleTime) * pWaveFormatEx->nAvgBytesPerSec / (1000 * 10000i64));
225                                SIZE_T nDataSize = Properties.cbBuffer;
226                                if(nDataSize > nRemainedDataSize)
227                                        nDataSize = nRemainedDataSize;
228                                nDataSize = (nDataSize / pWaveFormatEx->nBlockAlign) * pWaveFormatEx->nBlockAlign;
229                                if(pWaveFormatEx->wBitsPerSample == 8)
230                                {
231                                        FillMemory(Properties.pbBuffer, nDataSize, 0x80);
232                                } else
233                                {
234                                        ZeroMemory(Properties.pbBuffer, nDataSize);
235                                        #pragma region Sine Wave Data
236                                        if(m_fSignalPeriod > 0)
237                                        {
238                                                for(SIZE_T nIndex = 0; nIndex < nDataSize; nIndex += pWaveFormatEx->nBlockAlign)
239                                                {
240                                                        const SHORT nValue = (SHORT) (m_fSignalAmplitude * sin(2 * M_PI * (ThreadContext.m_nCurrentSampleIndex + nIndex) / m_fSignalPeriod));
241                                                        SHORT* pnSampleData = (SHORT*) (Properties.pbBuffer + nIndex);
242                                                        for(WORD nChannelIndex = 0; nChannelIndex < pWaveFormatEx->nChannels; nChannelIndex++)
243                                                                pnSampleData[nChannelIndex] = nValue;
244                                                }
245                                        }
246                                        #pragma endregion
247                                        #pragma region Noise Wave Data
248                                        if(m_fNoiseAmplitude > 0)
249                                        {
250                                                for(SIZE_T nIndex = 0; nIndex < nDataSize; nIndex += pWaveFormatEx->nBlockAlign)
251                                                {
252                                                        SHORT* pnSampleData = (SHORT*) (Properties.pbBuffer + nIndex);
253                                                        for(WORD nChannelIndex = 0; nChannelIndex < pWaveFormatEx->nChannels; nChannelIndex++)
254                                                                pnSampleData[nChannelIndex] += (SHORT) (m_fNoiseAmplitude * (DOUBLE) (rand() - RAND_MAX / 2) / (RAND_MAX / 2));
255                                                }
256                                        }
257                                        #pragma endregion
258                                }
259                                ThreadContext.m_nMediaSampleTime += nDataSize * (1000 * 10000i64) / pWaveFormatEx->nAvgBytesPerSec;
260                                ThreadContext.m_nCurrentSampleIndex += nDataSize / pWaveFormatEx->nBlockAlign;
261                                Properties.lActual = (LONG) nDataSize;
262                                Properties.Set();
263                                return TRUE;
264                        }
265                        VOID InitializeData(const CWaveFormatEx* pWaveFormatEx, REFERENCE_TIME nLength)
266                        {
267                                _A(pWaveFormatEx);
268                                _A(nLength > 0);
269                                __D(!m_pDataMediaType, E_UNNAMED);
270                                m_pDataMediaType.AllocateWaveFormatEx(pWaveFormatEx);
271                                m_nDataLength = nLength;
272                        }
273                        VOID InitializeSignal(DOUBLE fSignalPeriod, DOUBLE fSignalAmplitude, DOUBLE fNoiseAmplitude)
274                        {
275                                __D(fSignalPeriod >= 0 && fSignalAmplitude >= 0 && fNoiseAmplitude >= 0, E_INVALIDARG);
276                                m_fSignalPeriod = fSignalPeriod;
277                                m_fSignalAmplitude = fSignalAmplitude;
278                                m_fNoiseAmplitude = fNoiseAmplitude;
279                        }
280                };
281
282        private:
283                CObjectPtr<COutputPin> m_pOutputPin;
284
285        public:
286        // CSourceFilter
287                CSourceFilter() throw() :
288                        CBasePersistT<CSourceFilter>(GetDataCriticalSection())
289                {
290                        _Z4(atlTraceRefcount, 4, _T("this 0x%p\n"), this);
291                }
292                ~CSourceFilter() throw()
293                {
294                        _Z4(atlTraceRefcount, 4, _T("this 0x%p\n"), this);
295                }
296                HRESULT FinalConstruct() throw()
297                {
298                        _ATLTRY
299                        {
300                                m_pOutputPin.Construct()->Initialize(this, L"Output");
301                                AddPin(m_pOutputPin);
302                        }
303                        _ATLCATCH(Exception)
304                        {
305                                _C(Exception);
306                        }
307                        return S_OK;
308                }
309                VOID FinalRelease() throw()
310                {
311                        m_pOutputPin = NULL;
312                }
313                VOID DeliverBeginFlush(IPin*)
314                {
315                        m_pOutputPin->DeliverBeginFlush();
316                }
317                VOID DeliverEndFlush(IPin*)
318                {
319                        m_pOutputPin->DeliverEndFlush();
320                }
321                VOID DeliverNewSegment(IPin*, REFERENCE_TIME nStartTime, REFERENCE_TIME nStopTime, DOUBLE fRate)
322                {
323                        m_pOutputPin->DeliverNewSegment(nStartTime, nStopTime, fRate);
324                }
325                static BOOL CanCue() throw()
326                {
327                        return FALSE;
328                }
329                VOID CueFilter()
330                {
331                        m_pOutputPin->CuePin();
332                }
333                VOID RunFilter(REFERENCE_TIME nStartTime)
334                {
335                        m_pOutputPin->RunPin(nStartTime);
336                }
337                VOID PauseFilter() throw()
338                {
339                        m_pOutputPin->PausePin();
340                }
341                VOID StopFilter() throw()
342                {
343                        m_pOutputPin->StopPin();
344                }
345                const CObjectPtr<COutputPin>& GetOutputPin() const throw()
346                {
347                        return m_pOutputPin;
348                }
349                VOID Initialize(const CWaveFormatEx* pWaveFormatEx, REFERENCE_TIME nLength)
350                {
351                        m_pOutputPin->InitializeData(pWaveFormatEx, nLength);
352                }
353                VOID InitializeSignal(DOUBLE fSignalAmplitude, DOUBLE fSignalPeriod, DOUBLE fNoiseAmplitude)
354                {
355                        m_pOutputPin->InitializeSignal(fSignalAmplitude, fSignalPeriod, fNoiseAmplitude);
356                }
357        };
358
359private:
360        CWaveFormatEx m_WaveFormatEx;
361        REFERENCE_TIME m_nLength;
362        UINT m_nSignalFrequency;
363        UINT m_nSignalLoudness;
364        UINT m_nNoiseLoudness;
365        CPath m_sPath;
366
367public:
368// CModule
369        CModule() throw()
370        {
371#if defined(_DEBUG)
372                AtlTraceLoadSettings(NULL);
373#endif // defined(_DEBUG)
374                _Z4(atlTraceRefcount, 4, _T("this 0x%p\n"), this);
375                ZeroMemory(&m_WaveFormatEx, sizeof m_WaveFormatEx);
376                m_nLength = 0;
377                m_nSignalFrequency = 0;
378                m_nSignalLoudness = 0;
379                m_nNoiseLoudness = 0;
380        }
381        ~CModule() throw()
382        {
383                _Z4(atlTraceRefcount, 4, _T("this 0x%p\n"), this);
384        }
385        VOID SetSampleRate(UINT nSampleRate)
386        {
387                __D(nSampleRate > 0, E_INVALIDARG);
388                m_WaveFormatEx.nSamplesPerSec = (DWORD) nSampleRate;
389        }
390        VOID SetChannelCount(UINT nChannelCount)
391        {
392                __D(nChannelCount > 0, E_INVALIDARG);
393                m_WaveFormatEx.nChannels = (WORD) nChannelCount;
394        }
395        VOID SetSampleBitCount(UINT nSampleBitCount)
396        {
397                __D(nSampleBitCount == 8 || nSampleBitCount == 16, E_INVALIDARG);
398                m_WaveFormatEx.wBitsPerSample = (WORD) nSampleBitCount;
399        }
400        VOID SetLength(REFERENCE_TIME nLength)
401        {
402                __D(nLength > 0, E_INVALIDARG);
403                m_nLength = nLength;
404        }
405        VOID SetSignalFrequency(UINT nSignalFrequency)
406        {
407                __D(nSignalFrequency > 0, E_INVALIDARG);
408                m_nSignalFrequency = nSignalFrequency;
409        }
410        VOID SetSignalLoudness(UINT nSignalLoudness)
411        {
412                __D(nSignalLoudness >= 0, E_INVALIDARG);
413                m_nSignalLoudness = nSignalLoudness;
414        }
415        VOID SetNoiseLoudness(UINT nNoiseLoudness)
416        {
417                __D(nNoiseLoudness >= 0, E_INVALIDARG);
418                m_nNoiseLoudness = nNoiseLoudness;
419        }
420        VOID SetPath(LPCTSTR pszPath)
421        {
422                __D(_tcslen(pszPath), E_INVALIDARG);
423                __D(!_tcslen(m_sPath), E_INVALIDARG);
424                m_sPath = pszPath;
425        }
426        HRESULT PreMessageLoop(INT nShowCommand)
427        {
428                __C(__super::PreMessageLoop(nShowCommand));
429                return S_OK;
430        }
431        VOID RunMessageLoop()
432        {
433                #pragma region Input Validation and Syntax
434                if(!m_WaveFormatEx.nSamplesPerSec || !m_WaveFormatEx.nChannels || !m_WaveFormatEx.wBitsPerSample || !m_nLength || !_tcslen(m_sPath))
435                {
436                        _tprintf(_T("Syntax: GeneratePcmWavFile <options> <output-path>\n"));
437                        _tprintf(_T("  /s:N: Sampling Rate N, Hz\n"));
438                        _tprintf(_T("  /c:N: Channel Count N\n"));
439                        _tprintf(_T("  /b:N: Sample Bit Count N, 8 or 16\n"));
440                        _tprintf(_T("  /t:N: Length N, seconds\n"));
441                        _tprintf(_T("  /f:N: Sine Signal Frequency N, Hz\n"));
442                        _tprintf(_T("  /l:N: Sine Signal Loudness N, dB below full scale\n"));
443                        _tprintf(_T("  /n:N: Noise Signal Loudness N, dB below full scale\n"));
444                        __C(S_FALSE);
445                }
446                #pragma endregion
447                #pragma region Complete Audio Format
448                m_WaveFormatEx.wFormatTag = WAVE_FORMAT_PCM;
449                m_WaveFormatEx.nBlockAlign = m_WaveFormatEx.nChannels * ((m_WaveFormatEx.wBitsPerSample + 7) >> 3);
450                m_WaveFormatEx.nAvgBytesPerSec = m_WaveFormatEx.nSamplesPerSec * m_WaveFormatEx.nBlockAlign;
451                CWaveFormatEx* pWaveFormatEx = &m_WaveFormatEx;
452                // SUGG: Replace with WAVEFORMATEXTENSIBLE where appropriate
453                #pragma endregion
454                #pragma region Filter Graph
455                CGenericFilterGraph GenericFilterGraph(TRUE);
456                GenericFilterGraph.CoCreateInstance();
457                CObjectPtr<CSourceFilter> pSourceFilter;
458                pSourceFilter.Construct()->Initialize(pWaveFormatEx, m_nLength);
459                if(m_nSignalFrequency && m_nSignalLoudness || m_nNoiseLoudness)
460                {
461                        __D(m_WaveFormatEx.wBitsPerSample == 16, E_NOTIMPL);
462                        DOUBLE fSignalPeriod = 0, fSignalAmplitude = 0;
463                        if(m_nSignalFrequency && m_nSignalLoudness)
464                        {
465                                fSignalPeriod = (DOUBLE) pWaveFormatEx->nSamplesPerSec / m_nSignalFrequency;
466                                fSignalAmplitude = 32767.0 / pow(10.0, m_nSignalLoudness / 20.0);
467                        }
468                        DOUBLE fNoiseAmplitude = 0;
469                        if(m_nNoiseLoudness)
470                        {
471                                fNoiseAmplitude = 32767.0 / pow(10.0, m_nNoiseLoudness / 20.0);
472                        }
473                        pSourceFilter->InitializeSignal(fSignalPeriod, fSignalAmplitude, fNoiseAmplitude);
474                }
475                __C(GenericFilterGraph->AddFilter(pSourceFilter, CT2CW(_T("Source"))));
476                CComPtr<IPin> pCurrentOutputPin = pSourceFilter->GetOutputPin();
477                #pragma region WAV Dest
478                CComPtr<IBaseFilter> pWavDestBaseFilter;
479                _ATLTRY
480                {
481                        __C(pWavDestBaseFilter.CoCreateInstance(_PersistHelper::ClassIdentifierFromString(CT2CW(_T("{3C78B8E2-6C4D-11D1-ADE2-0000F8754B99}")))));
482                        __C(GenericFilterGraph->AddFilter(pWavDestBaseFilter, CT2CW(_T(".WAV Multiplexer"))));
483                        __C(GenericFilterGraph->Connect(pCurrentOutputPin, _FilterGraphHelper::GetFilterPin(pWavDestBaseFilter, PINDIR_INPUT)));
484                        pCurrentOutputPin = _FilterGraphHelper::GetFilterPin(pWavDestBaseFilter, PINDIR_OUTPUT);
485                }
486                _ATLCATCH(Exception)
487                {
488                        _tprintf(_T("There was an error trying to instantiate WAV Dest filter, Exception 0x%08X\n"), (HRESULT) Exception);
489                        _ATLRETHROW;
490                }
491                #pragma endregion
492                #pragma region File Writer
493                CComPtr<IBaseFilter> pFileWriterBaseFilter;
494                __C(pFileWriterBaseFilter.CoCreateInstance(CLSID_FileWriter));
495                CComQIPtr<IFileSinkFilter2> pFileSinkFilter2 = pFileWriterBaseFilter;
496                __D(pFileSinkFilter2, E_NOINTERFACE);
497                _tprintf(_T("Writing into \"%s\"...\n"), m_sPath);
498                __C(pFileSinkFilter2->SetFileName(CT2CW(m_sPath), NULL));
499                __C(pFileSinkFilter2->SetMode(AM_FILE_OVERWRITE));
500                __C(GenericFilterGraph->AddFilter(pFileWriterBaseFilter, CT2CW(_T("File Writer"))));
501                __C(GenericFilterGraph->Connect(pCurrentOutputPin, _FilterGraphHelper::GetFilterPin(pFileWriterBaseFilter, PINDIR_INPUT)));
502                pCurrentOutputPin.Release();
503                #pragma endregion
504                _A(!pCurrentOutputPin);
505                _A(GenericFilterGraph.m_pMediaFilter);
506                __C(GenericFilterGraph.m_pMediaFilter->SetSyncSource(NULL));
507                #pragma endregion
508                _tprintf(_T("Running...\n"));
509                _A(GenericFilterGraph.m_pMediaControl);
510                __C(GenericFilterGraph.m_pMediaControl->Run());
511                GenericFilterGraph.SetShowDestructorMessageBox(FALSE);
512                _A(GenericFilterGraph.m_pMediaEventEx);
513                LONG nEventCode;
514                __C(GenericFilterGraph.m_pMediaEventEx->WaitForCompletion(INFINITE, &nEventCode));
515                _tprintf(_T("Complete, nEventCode 0x%02X\n"), nEventCode);
516                __C(GenericFilterGraph.m_pMediaControl->Stop());
517        }
518};
519
520////////////////////////////////////////////////////////////
521// Main
522
523int _tmain(int argc, _TCHAR* argv[])
524{
525        _ATLTRY
526        {
527                CModule Module;
528                #pragma region Parse Command Line
529                for(INT nIndex = 1; nIndex < argc; nIndex++)
530                {
531                        CString sArgument = argv[nIndex];
532                        _A(!sArgument.IsEmpty());
533                        #pragma region Switches
534                        if(_tcschr(_T("-/"), sArgument[0]))
535                        {
536                                sArgument.Delete(0);
537                                #pragma region Switch Value/Specification
538                                CString sArgumentValue;
539                                if(sArgument.GetLength() > 1)
540                                {
541                                        SIZE_T nIndex = 1;
542                                        if(sArgument[1] == _T(':'))
543                                                nIndex++;
544                                        sArgumentValue = (LPCTSTR) sArgument + nIndex;
545                                }
546                                INT nIntegerArgumentValue = 0;
547                                const BOOL bIntegerArgumentValueValid = !sArgumentValue.IsEmpty() ? AtlStringToInteger(sArgumentValue, nIntegerArgumentValue) : FALSE;
548                                #pragma endregion
549                                if(_tcschr(_T("Ss"), sArgument[0])) // Sample Rate
550                                {
551                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
552                                        //_tprintf(_T("Option: Sample Rate, %d\n"), nIntegerArgumentValue);
553                                        Module.SetSampleRate(nIntegerArgumentValue);
554                                } else
555                                if(_tcschr(_T("Cc"), sArgument[0])) // Channel Count
556                                {
557                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
558                                        //_tprintf(_T("Option: Channel Count, %d\n"), nIntegerArgumentValue);
559                                        Module.SetChannelCount(nIntegerArgumentValue);
560                                } else
561                                if(_tcschr(_T("Bb"), sArgument[0])) // Sample Bit Count
562                                {
563                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
564                                        //_tprintf(_T("Option: Sample Bit Count, %d\n"), nIntegerArgumentValue);
565                                        Module.SetSampleBitCount(nIntegerArgumentValue);
566                                } else
567                                if(_tcschr(_T("Tt"), sArgument[0])) // Length, seconds
568                                {
569                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
570                                        //_tprintf(_T("Option: Length, %d\n"), nIntegerArgumentValue);
571                                        Module.SetLength(nIntegerArgumentValue * 1000 * 10000i64);
572                                } else
573                                if(_tcschr(_T("Ff"), sArgument[0])) // Sine Signal Frequency, Hz
574                                {
575                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
576                                        //_tprintf(_T("Option: Sine Signal Frequency, %d\n"), nIntegerArgumentValue);
577                                        Module.SetSignalFrequency(nIntegerArgumentValue);
578                                } else
579                                if(_tcschr(_T("Ll"), sArgument[0])) // Sine Signal Loudness, dB below FS
580                                {
581                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
582                                        //_tprintf(_T("Option: Sine Signal Loudness, %d\n"), nIntegerArgumentValue);
583                                        Module.SetSignalLoudness(nIntegerArgumentValue);
584                                } else
585                                if(_tcschr(_T("Nn"), sArgument[0])) // Noise Signal Loudness, dB below FS
586                                {
587                                        __D(bIntegerArgumentValueValid, E_INVALIDARG);
588                                        //_tprintf(_T("Option: Noise Signal Loudness, %d\n"), nIntegerArgumentValue);
589                                        Module.SetNoiseLoudness(nIntegerArgumentValue);
590                                }
591                                continue;
592                        }
593                        #pragma endregion
594                        if(sArgument.GetLength() >= 2 && sArgument[0] == _T('"') && sArgument[sArgument.GetLength() - 1] == _T('"'))
595                                sArgument = sArgument.Mid(1, sArgument.GetLength() - 2);
596                        Module.SetPath(sArgument);
597                }
598                #pragma endregion
599                Module.WinMain(SW_SHOWNORMAL);
600        }
601        _ATLCATCH(Exception)
602        {
603                if(FAILED((HRESULT) Exception))
604                        _tprintf(_T("Error 0x%08x: %s\n"), (HRESULT) Exception, AtlFormatSystemMessage(Exception).TrimRight(_T("\t\n\r .")));
605                return (INT) (HRESULT) Exception;
606        }
607        _ATLCATCHALL()
608        {
609                _tprintf(_T("Fatal Error\n"));
610                return (INT) E_FAIL;
611        }
612        return 0;
613}
Note: See TracBrowser for help on using the repository browser.