195 lines
6.6 KiB
C
195 lines
6.6 KiB
C
|
#ifndef __MSPUTILS_H_
|
||
|
#define __MSPUTILS_H_
|
||
|
|
||
|
#if _ATL_VER >= 0x0300
|
||
|
#define DECLARE_VQI()
|
||
|
#else
|
||
|
#define DECLARE_VQI() STDMETHOD(QueryInterface)(REFIID iid,void **ppvObject) = 0; STDMETHOD_(ULONG,AddRef)() = 0; STDMETHOD_(ULONG,Release)() = 0;
|
||
|
#endif
|
||
|
|
||
|
#define MSP_(hr) (FAILED(hr)?MSP_ERROR:MSP_TRACE)
|
||
|
|
||
|
extern __inline WINBOOL IsValidAggregatedMediaType(DWORD dwAggregatedMediaType) {
|
||
|
const DWORD dwAllPossibleMediaTypes = TAPIMEDIATYPE_AUDIO | TAPIMEDIATYPE_VIDEO | TAPIMEDIATYPE_DATAMODEM | TAPIMEDIATYPE_G3FAX | TAPIMEDIATYPE_MULTITRACK;
|
||
|
WINBOOL bValidMediaType = FALSE;
|
||
|
if((0==(dwAggregatedMediaType & dwAllPossibleMediaTypes)) || (0!=(dwAggregatedMediaType & (~dwAllPossibleMediaTypes)))) {
|
||
|
bValidMediaType = FALSE;
|
||
|
} else {
|
||
|
bValidMediaType = TRUE;
|
||
|
}
|
||
|
return bValidMediaType;
|
||
|
}
|
||
|
|
||
|
extern __inline WINBOOL IsSingleMediaType(DWORD dwMediaType) { return !((dwMediaType==0) || ((dwMediaType & (dwMediaType - 1))!=0)); }
|
||
|
extern __inline WINBOOL IsValidSingleMediaType(DWORD dwMediaType,DWORD dwMask) { return IsSingleMediaType(dwMediaType) && ((dwMediaType & dwMask)==dwMediaType); }
|
||
|
|
||
|
const DWORD INITIAL = 8;
|
||
|
const DWORD DELTA = 8;
|
||
|
|
||
|
template <class T,DWORD dwInitial = INITIAL,DWORD dwDelta = DELTA> class CMSPArray {
|
||
|
protected:
|
||
|
T *m_aT;
|
||
|
int m_nSize;
|
||
|
int m_nAllocSize;
|
||
|
public:
|
||
|
CMSPArray() : m_aT(NULL),m_nSize(0),m_nAllocSize(0) { }
|
||
|
~CMSPArray() { RemoveAll(); }
|
||
|
int GetSize() const { return m_nSize; }
|
||
|
WINBOOL Grow() {
|
||
|
T *aT;
|
||
|
int nNewAllocSize = (m_nAllocSize==0) ? dwInitial : (m_nSize + DELTA);
|
||
|
aT = (T *)realloc(m_aT,nNewAllocSize *sizeof(T));
|
||
|
if(!aT) return FALSE;
|
||
|
m_nAllocSize = nNewAllocSize;
|
||
|
m_aT = aT;
|
||
|
return TRUE;
|
||
|
}
|
||
|
WINBOOL Add(T &t) {
|
||
|
if(m_nSize==m_nAllocSize) {
|
||
|
if(!Grow()) return FALSE;
|
||
|
}
|
||
|
m_nSize++;
|
||
|
SetAtIndex(m_nSize - 1,t);
|
||
|
return TRUE;
|
||
|
}
|
||
|
WINBOOL Remove(T &t) {
|
||
|
int nIndex = Find(t);
|
||
|
if(nIndex==-1) return FALSE;
|
||
|
return RemoveAt(nIndex);
|
||
|
}
|
||
|
WINBOOL RemoveAt(int nIndex) {
|
||
|
if(nIndex!=(m_nSize - 1))
|
||
|
memmove((void*)&m_aT[nIndex],(void*)&m_aT[nIndex + 1],(m_nSize - (nIndex + 1))*sizeof(T));
|
||
|
m_nSize--;
|
||
|
return TRUE;
|
||
|
}
|
||
|
void RemoveAll() {
|
||
|
if(m_nAllocSize > 0) {
|
||
|
free(m_aT);
|
||
|
m_aT = NULL;
|
||
|
m_nSize = 0;
|
||
|
m_nAllocSize = 0;
|
||
|
}
|
||
|
}
|
||
|
T &operator[] (int nIndex) const {
|
||
|
_ASSERTE(nIndex >= 0 && nIndex < m_nSize);
|
||
|
return m_aT[nIndex];
|
||
|
}
|
||
|
T *GetData() const { return m_aT; }
|
||
|
void SetAtIndex(int nIndex,T &t) {
|
||
|
_ASSERTE(nIndex >= 0 && nIndex < m_nSize);
|
||
|
m_aT[nIndex] = t;
|
||
|
}
|
||
|
int Find(T &t) const {
|
||
|
for(int i = 0;i < m_nSize;i++) {
|
||
|
if(m_aT[i]==t) return i;
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class CMSPCritSection {
|
||
|
private:
|
||
|
CRITICAL_SECTION m_CritSec;
|
||
|
public:
|
||
|
CMSPCritSection() { InitializeCriticalSection(&m_CritSec); }
|
||
|
~CMSPCritSection() { DeleteCriticalSection(&m_CritSec); }
|
||
|
void Lock() { EnterCriticalSection(&m_CritSec); }
|
||
|
WINBOOL TryLock() { return TryEnterCriticalSection(&m_CritSec); }
|
||
|
void Unlock() { LeaveCriticalSection(&m_CritSec); }
|
||
|
};
|
||
|
|
||
|
class CLock {
|
||
|
private:
|
||
|
CMSPCritSection &m_CriticalSection;
|
||
|
public:
|
||
|
CLock(CMSPCritSection &CriticalSection) : m_CriticalSection(CriticalSection) {
|
||
|
m_CriticalSection.Lock();
|
||
|
}
|
||
|
~CLock() { m_CriticalSection.Unlock(); }
|
||
|
};
|
||
|
|
||
|
class CCSLock {
|
||
|
private:
|
||
|
CRITICAL_SECTION *m_pCritSec;
|
||
|
public:
|
||
|
CCSLock(CRITICAL_SECTION *pCritSec) : m_pCritSec(pCritSec) {
|
||
|
EnterCriticalSection(m_pCritSec);
|
||
|
}
|
||
|
~CCSLock() { LeaveCriticalSection(m_pCritSec); }
|
||
|
};
|
||
|
|
||
|
#ifndef CONTAINING_RECORD
|
||
|
#define CONTAINING_RECORD(address,type,field) ((type *)((PCHAR)(address) - (ULONG_PTR)(&((type *)0)->field)))
|
||
|
#endif
|
||
|
|
||
|
#ifndef InitializeListHead
|
||
|
#define InitializeListHead(ListHead) ((ListHead)->Flink = (ListHead)->Blink = (ListHead))
|
||
|
#define IsListEmpty(ListHead) ((ListHead)->Flink==(ListHead))
|
||
|
#define RemoveHeadList(ListHead) (ListHead)->Flink; {RemoveEntryList((ListHead)->Flink)}
|
||
|
#define RemoveTailList(ListHead) (ListHead)->Blink; {RemoveEntryList((ListHead)->Blink)}
|
||
|
#define RemoveEntryList(Entry) { PLIST_ENTRY _EX_Blink; PLIST_ENTRY _EX_Flink; _EX_Flink = (Entry)->Flink; _EX_Blink = (Entry)->Blink; _EX_Blink->Flink = _EX_Flink; _EX_Flink->Blink = _EX_Blink; }
|
||
|
#define InsertTailList(ListHead,Entry) { PLIST_ENTRY _EX_Blink; PLIST_ENTRY _EX_ListHead; _EX_ListHead = (ListHead); _EX_Blink = _EX_ListHead->Blink; (Entry)->Flink = _EX_ListHead; (Entry)->Blink = _EX_Blink; _EX_Blink->Flink = (Entry); _EX_ListHead->Blink = (Entry); }
|
||
|
#define InsertHeadList(ListHead,Entry) { PLIST_ENTRY _EX_Flink; PLIST_ENTRY _EX_ListHead; _EX_ListHead = (ListHead); _EX_Flink = _EX_ListHead->Flink; (Entry)->Flink = _EX_Flink; (Entry)->Blink = _EX_ListHead; _EX_Flink->Blink = (Entry); _EX_ListHead->Flink = (Entry); }
|
||
|
|
||
|
WINBOOL IsNodeOnList(PLIST_ENTRY ListHead,PLIST_ENTRY Entry);
|
||
|
#endif
|
||
|
|
||
|
template <class T> ULONG MSPAddRefHelper (T *pMyThis) {
|
||
|
LOG((MSP_INFO,"MSPAddRefHelper - this = 0x%08x",pMyThis));
|
||
|
typedef CComAggObject<T> AggClass;
|
||
|
AggClass *p = CONTAINING_RECORD(pMyThis,AggClass,m_contained);
|
||
|
return p->AddRef();
|
||
|
}
|
||
|
|
||
|
template <class T> ULONG MSPReleaseHelper (T *pMyThis) {
|
||
|
LOG((MSP_INFO,"MSPReleaseHelper - this = 0x%08x",pMyThis));
|
||
|
typedef CComAggObject<T> AggClass;
|
||
|
AggClass *p = CONTAINING_RECORD(pMyThis,AggClass,m_contained);
|
||
|
return p->Release();
|
||
|
}
|
||
|
|
||
|
#include <objsafe.h>
|
||
|
|
||
|
class CMSPObjectSafetyImpl : public IObjectSafety {
|
||
|
public:
|
||
|
CMSPObjectSafetyImpl() : m_dwSafety(0) { }
|
||
|
enum {
|
||
|
SUPPORTED_SAFETY_OPTIONS = INTERFACESAFE_FOR_UNTRUSTED_CALLER | INTERFACESAFE_FOR_UNTRUSTED_DATA
|
||
|
};
|
||
|
STDMETHOD(SetInterfaceSafetyOptions)(REFIID riid,DWORD dwOptionSetMask,DWORD dwEnabledOptions) {
|
||
|
if((~SUPPORTED_SAFETY_OPTIONS & dwOptionSetMask)!=0) return E_FAIL;
|
||
|
IUnknown *pUnk = NULL;
|
||
|
HRESULT hr = QueryInterface(riid,(void**)&pUnk);
|
||
|
if(SUCCEEDED(hr)) {
|
||
|
pUnk->Release();
|
||
|
pUnk = NULL;
|
||
|
s_CritSection.Lock();
|
||
|
m_dwSafety = (dwEnabledOptions & dwOptionSetMask) | (m_dwSafety & ~dwOptionSetMask);
|
||
|
s_CritSection.Unlock();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
STDMETHOD(GetInterfaceSafetyOptions)(REFIID riid,DWORD *pdwSupportedOptions,DWORD *pdwEnabledOptions) {
|
||
|
if(IsBadWritePtr(pdwSupportedOptions,sizeof(DWORD)) || IsBadWritePtr(pdwEnabledOptions,sizeof(DWORD))) return E_POINTER;
|
||
|
*pdwSupportedOptions = 0;
|
||
|
*pdwEnabledOptions = 0;
|
||
|
IUnknown *pUnk = NULL;
|
||
|
HRESULT hr = QueryInterface(riid,(void**)&pUnk);
|
||
|
if(SUCCEEDED(hr)) {
|
||
|
pUnk->Release();
|
||
|
pUnk = NULL;
|
||
|
*pdwSupportedOptions = SUPPORTED_SAFETY_OPTIONS;
|
||
|
s_CritSection.Lock();
|
||
|
*pdwEnabledOptions = m_dwSafety;
|
||
|
s_CritSection.Unlock();
|
||
|
}
|
||
|
return hr;
|
||
|
}
|
||
|
private:
|
||
|
DWORD m_dwSafety;
|
||
|
static CMSPCritSection s_CritSection;
|
||
|
};
|
||
|
|
||
|
#endif
|