windows-nt/Source/XPSP1/NT/multimedia/dshow/streams/ddstream/austrm.cpp
2020-09-26 16:20:57 +08:00

314 lines
8.4 KiB
C++

// Copyright (c) 1997 - 1998 Microsoft Corporation. All Rights Reserved.
// austrm.cpp : Implementation of CAudioStream
#include "stdafx.h"
#include "project.h"
#include "austrm.h"
// Helper
void SetWaveFormatEx(
LPWAVEFORMATEX pFormat,
int nChannels,
int nBitsPerSample,
int nSamplesPerSecond
)
{
pFormat->wFormatTag = WAVE_FORMAT_PCM;
pFormat->nChannels = (WORD)nChannels;
pFormat->nSamplesPerSec = (DWORD)nSamplesPerSecond;
pFormat->nBlockAlign = (WORD)((nBitsPerSample * nChannels) / 8);
pFormat->nAvgBytesPerSec = (DWORD)(nSamplesPerSecond * pFormat->nBlockAlign);
pFormat->wBitsPerSample = (WORD)nBitsPerSample;
pFormat->cbSize = 0;
}
HRESULT ConvertWAVEFORMATEXToMediaType(
const WAVEFORMATEX *pFormat,
AM_MEDIA_TYPE **ppmt
)
{
AM_MEDIA_TYPE *pmt;
pmt = (AM_MEDIA_TYPE *)CoTaskMemAlloc(sizeof(*pmt));
if (pmt == NULL) {
return E_OUTOFMEMORY;
}
_ASSERTE(pFormat->wFormatTag == WAVE_FORMAT_PCM);
ZeroMemory(pmt, sizeof(*pmt));
pmt->majortype = MEDIATYPE_Audio;
pmt->formattype = FORMAT_WaveFormatEx;
pmt->bFixedSizeSamples = TRUE;
pmt->lSampleSize = pFormat->nBlockAlign;
pmt->cbFormat = sizeof(*pFormat);
pmt->pbFormat = (PBYTE)CoTaskMemAlloc(sizeof(*pFormat));
if (pmt->pbFormat == NULL) {
CoTaskMemFree(pmt);
return E_OUTOFMEMORY;
}
CopyMemory(pmt->pbFormat, pFormat, sizeof(*pFormat));
*ppmt = pmt;
return S_OK;
}
/////////////////////////////////////////////////////////////////////////////
// CAudioStream
CAudioStream::CAudioStream() :
m_fForceFormat(false)
{
// Set to mono 16bit PCM 11025Hz
SetWaveFormatEx(&m_Format, 1, 16, 11025);
}
STDMETHODIMP
CAudioStream::ReceiveConnection(
IPin * pConnector,
const AM_MEDIA_TYPE *pmt
)
{
AUTO_CRIT_LOCK;
//
// This helper function in CStream checks basic parameters for the Pin such as
// the connecting pin's direction (we need to check this -- Sometimes the filter
// graph will try to connect us to ourselves!) and other errors like already being
// connected, etc.
//
HRESULT hr = CheckReceiveConnectionPin(pConnector);
if (hr == NOERROR) {
/* Accept only the format we've been given. If we
haven't been given a format accept PCM only
*/
if (pmt->majortype != MEDIATYPE_Audio ||
pmt->formattype != FORMAT_WaveFormatEx ||
pmt->cbFormat < sizeof(WAVEFORMATEX)) {
hr = VFW_E_TYPE_NOT_ACCEPTED;
} else {
hr = InternalSetFormat((LPWAVEFORMATEX)pmt->pbFormat, true);
if (SUCCEEDED(hr)) {
CopyMediaType(&m_ConnectedMediaType, pmt);
m_pConnectedPin = pConnector;
}
}
}
return hr;
}
STDMETHODIMP CAudioStream::SetSameFormat(IMediaStream *pStream, DWORD dwFlags)
{
CComQIPtr<IAudioMediaStream, &IID_IAudioMediaStream> pSource(pStream);
if (!pSource) {
return MS_E_INCOMPATIBLE;
}
WAVEFORMATEX wfx;
HRESULT hr = pSource->GetFormat(&wfx);
if (SUCCEEDED(hr)) {
hr = SetFormat(&wfx);
}
return hr;
}
STDMETHODIMP CAudioStream::AllocateSample(DWORD dwFlags, IStreamSample **ppNewSample)
{
IAudioStreamSample *pSample = NULL;
IAudioData *pAudioData;
HRESULT hr = CoCreateInstance(CLSID_AMAudioData, NULL, CLSCTX_INPROC_SERVER,
IID_IAudioData, (void **)&pAudioData);
if (SUCCEEDED(hr)) {
// Pick a sensible buffer size - 1/10 second
DWORD dwBufferSize = m_Format.nAvgBytesPerSec / 10 +
m_Format.nBlockAlign - 1;
dwBufferSize -= dwBufferSize % m_Format.nBlockAlign;
pAudioData->SetBuffer(dwBufferSize, NULL, 0);
pAudioData->SetFormat(&m_Format);
hr = CreateSample(pAudioData, 0, &pSample);
}
*ppNewSample = pSample;
return hr;
}
STDMETHODIMP CAudioStream::CreateSharedSample(
/* [in] */ IStreamSample *pExistingSample,
DWORD dwFlags,
/* [out] */ IStreamSample **ppNewSample
)
{
AUTO_CRIT_LOCK;
// See if we can get the information we need from the existing
// sample
IAudioStreamSample *pAudioSample;
HRESULT hr = pExistingSample->QueryInterface(
IID_IAudioStreamSample,
(void **)&pAudioSample);
if (FAILED(hr)) {
return hr;
}
IAudioData *pAudioData;
hr = pAudioSample->GetAudioData(&pAudioData);
pAudioSample->Release();
if (FAILED(hr)) {
return hr;
}
IAudioStreamSample *pNewSample;
hr = CreateSample(pAudioData, 0, &pNewSample);
pAudioData->Release();
if (FAILED(hr)) {
return hr;
}
hr = pNewSample->QueryInterface(IID_IStreamSample, (void**)ppNewSample);
pNewSample->Release();
return hr;
}
STDMETHODIMP CAudioStream::SetFormat(const WAVEFORMATEX *pFormat)
{
if (pFormat == NULL) {
return E_POINTER;
}
AUTO_CRIT_LOCK;
return InternalSetFormat(pFormat, false);
}
STDMETHODIMP CAudioStream::GetFormat(LPWAVEFORMATEX pFormat)
{
if (pFormat == NULL) {
return E_POINTER;
}
if (!m_pConnectedPin) {
return MS_E_NOSTREAM;
}
*pFormat = m_Format;
return S_OK;
}
STDMETHODIMP CAudioStream::CreateSample(
/* [in] */ IAudioData *pAudioData,
/* [in] */ DWORD dwFlags,
/* [out] */ IAudioStreamSample **ppSample
)
{
if (dwFlags != 0) {
return E_INVALIDARG;
}
if (pAudioData == NULL || ppSample == NULL) {
return E_POINTER;
}
AUTO_CRIT_LOCK;
// Check the format
WAVEFORMATEX wfx;
HRESULT hr = pAudioData->GetFormat(&wfx);
if (FAILED(hr)) {
return hr;
}
hr = CheckFormat(&wfx);
if (FAILED(hr)) {
return hr;
}
typedef CComObject<CAudioStreamSample> _AudioSample;
_AudioSample *pSample = new _AudioSample;
if (pSample == NULL) {
return E_OUTOFMEMORY;
}
hr = pSample->Init(pAudioData);
if (FAILED(hr)) {
return hr;
}
pSample->InitSample(this, false);
return pSample->GetControllingUnknown()->QueryInterface(
IID_IAudioStreamSample, (void **)ppSample
);
}
HRESULT CAudioStream::CheckFormat(const WAVEFORMATEX *lpFormat, bool bForceFormat)
{
if (lpFormat->wFormatTag != WAVE_FORMAT_PCM ||
lpFormat->nBlockAlign == 0) {
return E_INVALIDARG;
}
if ((m_pConnectedPin || bForceFormat) &&
0 != memcmp(lpFormat, &m_Format, sizeof(m_Format)))
{
// Try reconnection!
return E_INVALIDARG;
}
return S_OK;
}
HRESULT CAudioStream::InternalSetFormat(const WAVEFORMATEX *lpFormat, bool bFromPin)
{
HRESULT hr = CheckFormat(lpFormat, m_fForceFormat);
if (FAILED(hr)) {
return hr;
}
m_Format = *lpFormat;
m_lBytesPerSecond = m_Format.nAvgBytesPerSec;
if(!bFromPin) {
m_fForceFormat = true;
}
return S_OK;
}
//
// Special CStream methods
//
HRESULT CAudioStream::GetMediaType(ULONG Index, AM_MEDIA_TYPE **ppMediaType)
{
if (Index != 0) {
return S_FALSE;
}
return ConvertWAVEFORMATEXToMediaType(&m_Format, ppMediaType);
}
//////////////////////////////////////////////////////////////////////
// CAudioData
CAudioData::CAudioData() :
m_cbSize(0),
m_pbData(0),
m_cbData(0),
m_bWeAllocatedData(false)
{
// Set to mono 16bit PCM 11025Hz
SetWaveFormatEx(&m_Format, 1, 16, 11025);
}
CAudioData::~CAudioData()
{
if (m_bWeAllocatedData) {
CoTaskMemFree(m_pbData);
}
}
STDMETHODIMP CAudioStream::GetProperties(ALLOCATOR_PROPERTIES* pProps)
{
AUTO_CRIT_LOCK;
// NB TAPI relies on this number as a max for now when
// we're connected to the AVI Mux which uses this size to
// create its own samples
pProps->cbBuffer = CAudioStream::GetChopSize();
// Default to 5 buffers (half a second at our default buffer size)
pProps->cBuffers = m_lRequestedBufferCount ? m_lRequestedBufferCount : 5;
pProps->cbAlign = 1;
pProps->cbPrefix = 0;
return NOERROR;
}
STDMETHODIMP CAudioStream::SetProperties(ALLOCATOR_PROPERTIES* pRequest, ALLOCATOR_PROPERTIES* pActual)
{
HRESULT hr;
AUTO_CRIT_LOCK;
ZeroMemory(pActual, sizeof(*pActual));
if (pRequest->cbAlign == 0) {
hr = VFW_E_BADALIGN;
} else {
if (m_bCommitted == TRUE) {
hr = VFW_E_ALREADY_COMMITTED;
} else {
m_lRequestedBufferCount = pRequest->cBuffers;
hr = GetProperties(pActual);
}
}
return hr;
}