aboutsummaryrefslogtreecommitdiffstats
path: root/src/native/windows/msofficecomm/UnknownImpl.h
blob: dcd6607f82faa48be4115bf7683080c9c2da016d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/*
 * Jitsi, the OpenSource Java VoIP and Instant Messaging client.
 *
 * Distributable under LGPL license.
 * See terms of license at gnu.org.
 */
#ifndef _JMSOFFICECOMM_UNKNOWNIMPL_H_
#define _JMSOFFICECOMM_UNKNOWNIMPL_H_

#include "Log.h"
#include "OutOfProcessServer.h"
#include "StringUtils.h"
#include "WeakReferenceSource.h"

#define STDMETHODIMP_E_NOTIMPL_STUB \
    { \
        Log::d(_T("%s\n"), __PRETTY_FUNCTION__); \
        return E_NOTIMPL; \
    }

/**
 * Represents a base implementation of the <tt>IUnknown</tt> interface.
 *
 * @author Lyubomir Marinov
 */
template <class T, REFIID IID_T>
class UnknownImpl
    : public T
{
public:
    // IUnknown
    STDMETHODIMP QueryInterface(REFIID iid, PVOID *obj)
        {
            HRESULT hr;

            if (!obj)
                hr = E_POINTER;
            else if (IID_IUnknown == iid)
            {
                AddRef();
                *obj = static_cast<LPUNKNOWN>(this);
                hr = S_OK;
            }
            else if (IID_T == iid)
            {
                AddRef();
                *obj = static_cast<T *>(this);
                hr = S_OK;
            }
            else if (IID_IWeakReferenceSource == iid)
            {
                if (!_weakReferenceSource)
                    _weakReferenceSource = new WeakReferenceSource(this);
                _weakReferenceSource->AddRef();
                *obj = static_cast<IWeakReferenceSource *>(_weakReferenceSource);
                hr = S_OK;
            }
            else
            {
                *obj = NULL;
                hr = E_NOINTERFACE;
            }

            if (FAILED(hr))
            {
                LPOLESTR olestr;

                if (SUCCEEDED(::StringFromIID(iid, &olestr)))
                {
                    LPTSTR tstr
#ifdef _UNICODE
                        = olestr;
#else
                        = StringUtils::WideCharToMultiByte(olestr);
#endif /* #ifdef _UNICODE */

                    if (tstr)
                    {
                        Log::d(
                                _T("UnknownImpl::QueryInterface: this=%p; iid=%s;\n"),
                                (PVOID) this,
                                tstr);
                        if (tstr != olestr)
                            ::free(tstr);
                    }
                    ::CoTaskMemFree(olestr);
                }
            }

            return hr;
        }

    STDMETHODIMP_(ULONG) AddRef() { return ++_refCount; }

    STDMETHODIMP_(ULONG) Release()
        {
            ULONG refCount = --_refCount;

            if (!refCount)
                delete this;
            Log::d(
                    _T("UnknownImpl::Release: this=%p; refCount=%lu;\n"),
                    (PVOID) this,
                    refCount);
            return refCount;
        }

protected:
    UnknownImpl()
            : _refCount(1),
              _weakReferenceSource(NULL)
        {
            OutOfProcessServer::addRef();
        }

    virtual ~UnknownImpl()
        {
            if (_weakReferenceSource)
                delete _weakReferenceSource;

            OutOfProcessServer::release();
        }

private:
    ULONG _refCount;
    WeakReferenceSource *_weakReferenceSource;
};

#endif /* #ifndef _JMSOFFICECOMM_UNKNOWNIMPL_H_ */