|
|
#include "pch.h"
#include "thisdll.h"
#include "wmwrap.h"
#include "MediaProp.h"
#include "ids.h"
// declr property set storage enum
class CMediaPropSetEnum : public IEnumSTATPROPSETSTG { public: CMediaPropSetEnum(const PROPSET_INFO *propsets, ULONG cpropset, ULONG pos);
// IUnknown
STDMETHODIMP QueryInterface(REFIID riid, void **ppv); STDMETHODIMP_(ULONG) AddRef(); STDMETHODIMP_(ULONG) Release();
// IEnumSTATPROPSETSTG
STDMETHODIMP Next(ULONG celt, STATPROPSETSTG *rgelt, ULONG *pceltFetched); STDMETHODIMP Skip(ULONG celt); STDMETHODIMP Reset(); STDMETHODIMP Clone(IEnumSTATPROPSETSTG **ppenum);
private: ~CMediaPropSetEnum(); LONG _cRef; ULONG _pos, _size; const PROPSET_INFO *_propsets; };
// property set storage enum
STDMETHODIMP_(ULONG) CMediaPropSetEnum::AddRef() { return InterlockedIncrement(&_cRef); }
STDMETHODIMP_(ULONG) CMediaPropSetEnum::Release() { if (InterlockedDecrement(&_cRef)) return _cRef; delete this; return 0; }
STDMETHODIMP CMediaPropSetEnum::QueryInterface(REFIID riid, void **ppv) { static const QITAB qit[] = { QITABENT(CMediaPropSetEnum, IEnumSTATPROPSETSTG), { 0 }, }; return QISearch(this, qit, riid, ppv); }
// IEnum
STDMETHODIMP CMediaPropSetEnum::Next(ULONG celt, STATPROPSETSTG *rgelt, ULONG *pceltFetched) { ULONG cFetched = 0; for (ULONG i = 0; i < celt && _pos < _size; i++) { ZeroMemory(&rgelt[i], sizeof(STATPROPSETSTG)); rgelt[i].fmtid = _propsets[_pos].fmtid; _pos++; cFetched++; } if (pceltFetched) *pceltFetched = cFetched; return cFetched == celt ? S_OK : S_FALSE; }
STDMETHODIMP CMediaPropSetEnum::Skip(ULONG celt) { HRESULT hr; if (_pos + celt > _size) { hr = S_FALSE; _pos = _size; } else { hr = S_OK; _pos += celt; } return hr; }
STDMETHODIMP CMediaPropSetEnum::Reset() { _pos = 0; return S_OK; }
STDMETHODIMP CMediaPropSetEnum::Clone(IEnumSTATPROPSETSTG **ppenum) { HRESULT hr; CMediaPropSetEnum *penum = new CMediaPropSetEnum(_propsets, _size, _pos); if (penum) { hr = penum->QueryInterface(IID_PPV_ARG(IEnumSTATPROPSETSTG, ppenum)); penum->Release(); } else hr = STG_E_INSUFFICIENTMEMORY; return hr; }
CMediaPropSetEnum::CMediaPropSetEnum(const PROPSET_INFO *propsets, ULONG cpropsets, ULONG pos) : _cRef(1), _propsets(propsets), _size(cpropsets), _pos(pos) { DllAddRef(); }
CMediaPropSetEnum::~CMediaPropSetEnum() { DllRelease(); }
HRESULT CMediaPropSetStg::_PopulateSlowProperties() { return S_OK; }
HRESULT CMediaPropSetStg::_PopulateProperty(const COLMAP *pPInfo, PROPVARIANT *pvar) { CMediaPropStorage *pps; HRESULT hr = _ResolveFMTID(pPInfo->pscid->fmtid, &pps); if (SUCCEEDED(hr)) { PROPSPEC spec; spec.ulKind = PRSPEC_PROPID; spec.propid = pPInfo->pscid->pid; hr = pps->SetProperty(&spec, pvar); }
PropVariantClear(pvar); return hr; }
// Internal enumeration class used when populating properties.
CEnumAllProps::CEnumAllProps(const PROPSET_INFO *pPropSets, UINT cPropSets) : _pPropSets(pPropSets), _cPropSets(cPropSets), _iPropSetPos(0), _iPropPos(0) { }
const COLMAP *CEnumAllProps::Next() { const COLMAP *pcmReturn = NULL; while (_iPropSetPos < _cPropSets) { if (_iPropPos < _pPropSets[_iPropSetPos].cNumProps) { // Go to next property.
pcmReturn = _pPropSets[_iPropSetPos].pcmProps[_iPropPos]; _iPropPos++; break; } else { // Go to next property set.
_iPropSetPos++; _iPropPos = 0; } }
return pcmReturn; }
// Base media property set storage
STDMETHODIMP_(ULONG) CMediaPropSetStg::AddRef() { return InterlockedIncrement(&_cRef); }
STDMETHODIMP_(ULONG) CMediaPropSetStg::Release() { if (InterlockedDecrement(&_cRef)) return _cRef; delete this; return 0; }
STDMETHODIMP CMediaPropSetStg::QueryInterface(REFIID riid, void **ppv) { static const QITAB qit[] = { QITABENT(CMediaPropSetStg, IPropertySetStorage), QITABENT(CMediaPropSetStg, IPersistFile), QITABENTMULTI(CMediaPropSetStg, IPersist, IPersistFile), QITABENT(CMediaPropSetStg, IWMReaderCallback), { 0 }, }; return QISearch(this, qit, riid, ppv); }
// IPersist
STDMETHODIMP CMediaPropSetStg::GetClassID(CLSID *pClassID) { return E_NOTIMPL; }
// IPersistFile
STDMETHODIMP CMediaPropSetStg::IsDirty(void) { return S_FALSE;// sniff for uncommitted changed?
}
#define STGM_OPENMODE (STGM_READ | STGM_WRITE | STGM_READWRITE)
// Any stuff we have to do at 'load time'
HRESULT CMediaPropSetStg::_PreCheck() { return S_OK; }
STDMETHODIMP CMediaPropSetStg::Load(LPCOLESTR pszFileName, DWORD dwMode) { // Allow Load only on existing files.
if (dwMode & (STGM_CREATE | STGM_CONVERT | STGM_FAILIFTHERE)) return STG_E_INVALIDFLAG;
EnterCriticalSection(&_cs);
DWORD dwFlags = dwMode & STGM_OPENMODE;
_dwMode = dwMode; StrCpyNW(_wszFile, pszFileName, ARRAYSIZE(_wszFile)); _bHasBeenPopulated = FALSE; _bSlowPropertiesExtracted = FALSE; _hrPopulated = S_OK; _bIsWritable = (dwFlags & (STGM_WRITE | STGM_READWRITE)); _ResetPropertySet();
HRESULT hr = _PreCheck();
LeaveCriticalSection(&_cs);
return hr; }
STDMETHODIMP CMediaPropSetStg::Save(LPCOLESTR pszFileName, BOOL fRemember) { return E_NOTIMPL; }
STDMETHODIMP CMediaPropSetStg::SaveCompleted(LPCOLESTR pszFileName) { return E_NOTIMPL; }
STDMETHODIMP CMediaPropSetStg::GetCurFile(LPOLESTR *ppszFileName) { EnterCriticalSection(&_cs);
HRESULT hr = SHStrDupW(_wszFile, ppszFileName);
LeaveCriticalSection(&_cs);
return hr; }
// IPropertySetStorage methods
STDMETHODIMP CMediaPropSetStg::Create(REFFMTID fmtid, const CLSID *pclsid, DWORD grfFlags, DWORD grfMode, IPropertyStorage** ppPropStg) { return E_NOTIMPL; }
STDMETHODIMP CMediaPropSetStg::Open(REFFMTID fmtid, DWORD grfMode, IPropertyStorage** ppPropStg) { EnterCriticalSection(&_cs);
HRESULT hr = _PopulatePropertySet(); if (SUCCEEDED(hr)) { DWORD dwPssMode = _dwMode & STGM_OPENMODE;
switch (grfMode & STGM_OPENMODE) { case STGM_READ: break; case STGM_WRITE: if (!_bIsWritable || (dwPssMode == STGM_READ)) hr = E_FAIL; break; case STGM_READWRITE: if (!_bIsWritable || (dwPssMode != STGM_READWRITE)) hr = E_FAIL; break;
default: hr = E_INVALIDARG; }
if (SUCCEEDED(hr)) { CMediaPropStorage *pps; hr = _ResolveFMTID(fmtid, &pps); if (SUCCEEDED(hr)) { hr = pps->Open(STGM_SHARE_EXCLUSIVE, grfMode & STGM_OPENMODE, ppPropStg); } } }
LeaveCriticalSection(&_cs);
return hr; }
STDMETHODIMP CMediaPropSetStg::Delete(REFFMTID fmtid) { return E_NOTIMPL; }
STDMETHODIMP CMediaPropSetStg::Enum(IEnumSTATPROPSETSTG** ppenum) { HRESULT hr; CMediaPropSetEnum *psenum = new CMediaPropSetEnum(_pPropStgInfo, _cPropertyStorages, 0); if (psenum) { hr = psenum->QueryInterface(IID_PPV_ARG(IEnumSTATPROPSETSTG, ppenum)); psenum->Release(); } else hr = STG_E_INSUFFICIENTMEMORY;
return hr; }
CMediaPropSetStg::CMediaPropSetStg() : _cRef(1), _bHasBeenPopulated(FALSE), _dwMode(STGM_READ), _propStg(NULL) { _wszFile[0] = 0; DllAddRef(); }
// The only place this is called is at creation time.
HRESULT CMediaPropSetStg::Init() { HRESULT hr = E_FAIL;
InitializeCriticalSection(&_cs);
_hFileOpenEvent = CreateEvent(NULL, TRUE, FALSE, NULL); if (_hFileOpenEvent != NULL) { hr = _CreatePropertyStorages(); if (SUCCEEDED(hr)) { hr = _ResetPropertySet(); } }
return hr; }
CMediaPropSetStg::~CMediaPropSetStg() { if (_propStg) { for (ULONG i = 0; i < _cPropertyStorages; i++) { ATOMICRELEASE(_propStg[i]); }
LocalFree(_propStg); }
if (_hFileOpenEvent) { CloseHandle(_hFileOpenEvent); }
DeleteCriticalSection(&_cs);
DllRelease(); }
BOOL CMediaPropSetStg::_IsSlowProperty(const COLMAP *pPInfo) { return FALSE; }
HRESULT CMediaPropSetStg::FlushChanges(REFFMTID fmtid, LONG cNumProps, const COLMAP **pcmapInfo, PROPVARIANT *pVarProps, BOOL *pbDirtyFlags) { return E_NOTIMPL; }
//returns success if we support that FMTID
// ppps and ppcmPropInfo are optional
HRESULT CMediaPropSetStg::_ResolveFMTID(REFFMTID fmtid, CMediaPropStorage **ppps) { for (ULONG i = 0; i < _cPropertyStorages; i++) { if (IsEqualGUID(_pPropStgInfo[i].fmtid, fmtid)) { if (ppps) *ppps = _propStg[i]; return S_OK; } } return E_FAIL; }
HRESULT CMediaPropSetStg::_ResetPropertySet() { for (ULONG i = 0; i < _cPropertyStorages; i++) { _propStg[i]->_ResetPropStorage(); } return S_OK; }
HRESULT CMediaPropSetStg::_CreatePropertyStorages() { HRESULT hr = E_OUTOFMEMORY;
ASSERT(NULL == _propStg);
_propStg = (CMediaPropStorage**)LocalAlloc(LPTR, sizeof(CMediaPropStorage*) * _cPropertyStorages); if (_propStg) { for (ULONG i = 0; i < _cPropertyStorages; i++) { ASSERTMSG(_pPropStgInfo[i].pcmProps != NULL, "CMediaPropSetStg::_CreatePropertyStorages: my COLMAP structure is null");
// We want to give each property storage a list of the COLMAPs that it supports.
// This information is contained in _pPropStgInfo[i].ppids and cpids.
// We'll make a new array of COLMAPS
_propStg[i] = new CMediaPropStorage(this, NULL, _pPropStgInfo[i].fmtid, _pPropStgInfo[i].pcmProps, _pPropStgInfo[i].cNumProps, _dwMode, &_cs); if (!_propStg[i]) break; }
hr = S_OK; } return hr; }
HRESULT CMediaPropSetStg::_PopulatePropertySet() { return E_NOTIMPL; }
STDMETHODIMP CMediaPropSetStg::OnStatus(WMT_STATUS Status, HRESULT hr, WMT_ATTR_DATATYPE dwType, BYTE *pValue, void *pvContext) { // This is callback from WMSDK while we're holding a critical section on the main thread,
// waiting for this event to be set.
switch(Status) { case WMT_OPENED: SetEvent(_hFileOpenEvent); break; } return S_OK; }
STDMETHODIMP CMediaPropSetStg::OnSample(DWORD dwOutputNum, QWORD cnsSampleTime, QWORD cnsSampleDuration, DWORD dwFlags, INSSBuffer *pSample, void* pcontext) { return S_OK; }
|