mirror of
https://github.com/holub/mame
synced 2025-05-05 13:54:42 +03:00
305 lines
9.5 KiB
C++
305 lines
9.5 KiB
C++
/**
|
|
* This file has no copyright assigned and is placed in the Public Domain.
|
|
* This file is part of the mingw-w64 runtime package.
|
|
* No warranty is given; refer to the file DISCLAIMER.PD within this package.
|
|
*/
|
|
|
|
#ifndef _WRL_CLIENT_H_
|
|
#define _WRL_CLIENT_H_
|
|
|
|
#include <stddef.h>
|
|
#include <unknwn.h>
|
|
/* #include <weakreference.h> */
|
|
#include <roapi.h>
|
|
|
|
/* #include <wrl/def.h> */
|
|
#include <wrl/internal.h>
|
|
|
|
namespace Microsoft {
|
|
namespace WRL {
|
|
namespace Details {
|
|
template <typename T> class ComPtrRefBase {
|
|
protected:
|
|
T* ptr_;
|
|
|
|
public:
|
|
typedef typename T::InterfaceType InterfaceType;
|
|
|
|
#ifndef __WRL_CLASSIC_COM__
|
|
operator IInspectable**() const throw() {
|
|
static_assert(__is_base_of(IInspectable, InterfaceType), "Invalid cast");
|
|
return reinterpret_cast<IInspectable**>(ptr_->ReleaseAndGetAddressOf());
|
|
}
|
|
#endif
|
|
|
|
operator IUnknown**() const throw() {
|
|
static_assert(__is_base_of(IUnknown, InterfaceType), "Invalid cast");
|
|
return reinterpret_cast<IUnknown**>(ptr_->ReleaseAndGetAddressOf());
|
|
}
|
|
};
|
|
|
|
template <typename T> class ComPtrRef : public Details::ComPtrRefBase<T> {
|
|
public:
|
|
ComPtrRef(T *ptr) throw() {
|
|
ComPtrRefBase<T>::ptr_ = ptr;
|
|
}
|
|
|
|
operator void**() const throw() {
|
|
return reinterpret_cast<void**>(ComPtrRefBase<T>::ptr_->ReleaseAndGetAddressOf());
|
|
}
|
|
|
|
operator T*() throw() {
|
|
*ComPtrRefBase<T>::ptr_ = nullptr;
|
|
return ComPtrRefBase<T>::ptr_;
|
|
}
|
|
|
|
operator typename ComPtrRefBase<T>::InterfaceType**() throw() {
|
|
return ComPtrRefBase<T>::ptr_->ReleaseAndGetAddressOf();
|
|
}
|
|
|
|
typename ComPtrRefBase<T>::InterfaceType *operator*() throw() {
|
|
return ComPtrRefBase<T>::ptr_->Get();
|
|
}
|
|
|
|
typename ComPtrRefBase<T>::InterfaceType *const *GetAddressOf() const throw() {
|
|
return ComPtrRefBase<T>::ptr_->GetAddressOf();
|
|
}
|
|
|
|
typename ComPtrRefBase<T>::InterfaceType **ReleaseAndGetAddressOf() throw() {
|
|
return ComPtrRefBase<T>::ptr_->ReleaseAndGetAddressOf();
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
template<typename T> class ComPtr {
|
|
public:
|
|
typedef T InterfaceType;
|
|
|
|
ComPtr() throw() : ptr_(nullptr) {}
|
|
ComPtr(decltype(nullptr)) throw() : ptr_(nullptr) {}
|
|
|
|
ComPtr(const ComPtr &other) throw() : ptr_(other.ptr_) {
|
|
InternalAddRef();
|
|
}
|
|
|
|
template<class U>
|
|
ComPtr(const ComPtr<U> &other) throw() : ptr_(other.ptr_) {
|
|
InternalAddRef();
|
|
}
|
|
|
|
ComPtr(ComPtr &&other) throw() : ptr_(nullptr) {
|
|
if(this != reinterpret_cast<ComPtr*>(&reinterpret_cast<unsigned char&>(other)))
|
|
Swap(other);
|
|
}
|
|
|
|
template<class U>
|
|
ComPtr(ComPtr<U>&& other) throw() : ptr_(other.ptr_) {
|
|
other.ptr_ = nullptr;
|
|
}
|
|
|
|
~ComPtr() throw() {
|
|
InternalRelease();
|
|
}
|
|
|
|
ComPtr &operator=(decltype(nullptr)) throw() {
|
|
InternalRelease();
|
|
return *this;
|
|
}
|
|
|
|
ComPtr &operator=(InterfaceType *other) throw() {
|
|
if (ptr_ != other) {
|
|
InternalRelease();
|
|
ptr_ = other;
|
|
InternalAddRef();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template<typename U>
|
|
ComPtr &operator=(U *other) throw() {
|
|
if (ptr_ != other) {
|
|
InternalRelease();
|
|
ptr_ = other;
|
|
InternalAddRef();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
ComPtr& operator=(const ComPtr &other) throw() {
|
|
if (ptr_ != other.ptr_)
|
|
ComPtr(other).Swap(*this);
|
|
return *this;
|
|
}
|
|
|
|
template<class U>
|
|
ComPtr &operator=(const ComPtr<U> &other) throw() {
|
|
ComPtr(other).Swap(*this);
|
|
return *this;
|
|
}
|
|
|
|
ComPtr& operator=(ComPtr &&other) throw() {
|
|
ComPtr(other).Swap(*this);
|
|
return *this;
|
|
}
|
|
|
|
template<class U>
|
|
ComPtr& operator=(ComPtr<U> &&other) throw() {
|
|
ComPtr(other).Swap(*this);
|
|
return *this;
|
|
}
|
|
|
|
void Swap(ComPtr &&r) throw() {
|
|
InterfaceType *tmp = ptr_;
|
|
ptr_ = r.ptr_;
|
|
r.ptr_ = tmp;
|
|
}
|
|
|
|
void Swap(ComPtr &r) throw() {
|
|
InterfaceType *tmp = ptr_;
|
|
ptr_ = r.ptr_;
|
|
r.ptr_ = tmp;
|
|
}
|
|
|
|
operator Details::BoolType() const throw() {
|
|
return Get() != nullptr ? &Details::BoolStruct::Member : nullptr;
|
|
}
|
|
|
|
InterfaceType *Get() const throw() {
|
|
return ptr_;
|
|
}
|
|
|
|
InterfaceType *operator->() const throw() {
|
|
return ptr_;
|
|
}
|
|
|
|
Details::ComPtrRef<ComPtr<T>> operator&() throw() {
|
|
return Details::ComPtrRef<ComPtr<T>>(this);
|
|
}
|
|
|
|
const Details::ComPtrRef<const ComPtr<T>> operator&() const throw() {
|
|
return Details::ComPtrRef<const ComPtr<T>>(this);
|
|
}
|
|
|
|
InterfaceType *const *GetAddressOf() const throw() {
|
|
return &ptr_;
|
|
}
|
|
|
|
InterfaceType **GetAddressOf() throw() {
|
|
return &ptr_;
|
|
}
|
|
|
|
InterfaceType **ReleaseAndGetAddressOf() throw() {
|
|
InternalRelease();
|
|
return &ptr_;
|
|
}
|
|
|
|
InterfaceType *Detach() throw() {
|
|
T* ptr = ptr_;
|
|
ptr_ = nullptr;
|
|
return ptr;
|
|
}
|
|
|
|
void Attach(InterfaceType *other) throw() {
|
|
if (ptr_ != other) {
|
|
InternalRelease();
|
|
ptr_ = other;
|
|
InternalAddRef();
|
|
}
|
|
}
|
|
|
|
unsigned long Reset() {
|
|
return InternalRelease();
|
|
}
|
|
|
|
HRESULT CopyTo(InterfaceType **ptr) const throw() {
|
|
InternalAddRef();
|
|
*ptr = ptr_;
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CopyTo(REFIID riid, void **ptr) const throw() {
|
|
return ptr_->QueryInterface(riid, ptr);
|
|
}
|
|
|
|
template<typename U>
|
|
HRESULT CopyTo(U **ptr) const throw() {
|
|
return ptr_->QueryInterface(__uuidof(U), reinterpret_cast<void**>(ptr));
|
|
}
|
|
|
|
template<typename U>
|
|
HRESULT As(Details::ComPtrRef<ComPtr<U>> p) const throw() {
|
|
return ptr_->QueryInterface(__uuidof(U), p);
|
|
}
|
|
|
|
template<typename U>
|
|
HRESULT As(ComPtr<U> *p) const throw() {
|
|
return ptr_->QueryInterface(__uuidof(U), reinterpret_cast<void**>(p->ReleaseAndGetAddressOf()));
|
|
}
|
|
|
|
HRESULT AsIID(REFIID riid, ComPtr<IUnknown> *p) const throw() {
|
|
return ptr_->QueryInterface(riid, reinterpret_cast<void**>(p->ReleaseAndGetAddressOf()));
|
|
}
|
|
|
|
/*
|
|
HRESULT AsWeak(WeakRef *pWeakRef) const throw() {
|
|
return ::Microsoft::WRL::AsWeak(ptr_, pWeakRef);
|
|
}
|
|
*/
|
|
protected:
|
|
InterfaceType *ptr_;
|
|
|
|
void InternalAddRef() const throw() {
|
|
if(ptr_)
|
|
ptr_->AddRef();
|
|
}
|
|
|
|
unsigned long InternalRelease() throw() {
|
|
InterfaceType *tmp = ptr_;
|
|
if(!tmp)
|
|
return 0;
|
|
ptr_ = nullptr;
|
|
return tmp->Release();
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
void **IID_PPV_ARGS_Helper(::Microsoft::WRL::Details::ComPtrRef<T> pp) throw() {
|
|
static_assert(__is_base_of(IUnknown, typename T::InterfaceType), "Expected COM interface");
|
|
return pp;
|
|
}
|
|
|
|
namespace Windows {
|
|
namespace Foundation {
|
|
template<typename T>
|
|
inline HRESULT ActivateInstance(HSTRING classid, ::Microsoft::WRL::Details::ComPtrRef<T> instance) throw() {
|
|
return ActivateInstance(classid, instance.ReleaseAndGetAddressOf());
|
|
}
|
|
|
|
template<typename T>
|
|
inline HRESULT GetActivationFactory(HSTRING classid, ::Microsoft::WRL::Details::ComPtrRef<T> factory) throw() {
|
|
return RoGetActivationFactory(classid, IID_INS_ARGS(factory.ReleaseAndGetAddressOf()));
|
|
}
|
|
}
|
|
}
|
|
|
|
namespace ABI {
|
|
namespace Windows {
|
|
namespace Foundation {
|
|
template<typename T>
|
|
inline HRESULT ActivateInstance(HSTRING classid, ::Microsoft::WRL::Details::ComPtrRef<T> instance) throw() {
|
|
return ActivateInstance(classid, instance.ReleaseAndGetAddressOf());
|
|
}
|
|
|
|
template<typename T>
|
|
inline HRESULT GetActivationFactory(HSTRING classid, ::Microsoft::WRL::Details::ComPtrRef<T> factory) throw() {
|
|
return RoGetActivationFactory(classid, IID_INS_ARGS(factory.ReleaseAndGetAddressOf()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|