summaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorLiang Qi <liang.qi@qt.io>2017-01-24 13:34:35 +0100
committerLiang Qi <liang.qi@qt.io>2017-01-24 13:34:35 +0100
commitbada5f8ea74d6d9d2a60b35f4d271da143aced51 (patch)
tree7b936bc66691fec2efee0a2f8685b06fdac36497 /src
parentc550735ebeb937a36b82c161f0fed4e35b8776bb (diff)
parent555527b6992374f8a5139674a316e819f9d5685f (diff)
Merge remote-tracking branch 'origin/5.8' into devv5.9.0-alpha1
Conflicts: .qmake.conf src/activeqt/control/qaxserverbase.cpp Change-Id: Ic47ed12f7acf788b923571d1e3515ececd986a19
Diffstat (limited to 'src')
-rw-r--r--src/activeqt/container/qaxbase.cpp2
-rw-r--r--src/activeqt/container/qaxwidget.cpp16
-rw-r--r--src/activeqt/control/control.pro3
-rw-r--r--src/activeqt/control/qaxserver.cpp8
-rw-r--r--src/activeqt/control/qaxserverbase.cpp295
-rw-r--r--src/activeqt/control/qaxserverdll.cpp3
-rw-r--r--src/activeqt/control/qaxservermain.cpp85
-rw-r--r--src/activeqt/control/qclassfactory_p.h100
8 files changed, 332 insertions, 180 deletions
diff --git a/src/activeqt/container/qaxbase.cpp b/src/activeqt/container/qaxbase.cpp
index 36bf4e6..4c0bae7 100644
--- a/src/activeqt/container/qaxbase.cpp
+++ b/src/activeqt/container/qaxbase.cpp
@@ -3396,7 +3396,7 @@ static bool checkHRESULT(HRESULT hres, EXCEPINFO *exc, QAxBase *that, const QStr
case DISP_E_EXCEPTION:
{
bool printWarning = true;
- unsigned short code = ushort(-1);
+ unsigned int code = uint(-1);
QString source, desc, help;
const QMetaObject *mo = that->metaObject();
int exceptionSignal = mo->indexOfSignal("exception(int,QString,QString,QString)");
diff --git a/src/activeqt/container/qaxwidget.cpp b/src/activeqt/container/qaxwidget.cpp
index 8c74159..3e927f4 100644
--- a/src/activeqt/container/qaxwidget.cpp
+++ b/src/activeqt/container/qaxwidget.cpp
@@ -759,8 +759,20 @@ void QAxClientSite::releaseAll()
void QAxClientSite::deactivate()
{
- if (m_spInPlaceObject) m_spInPlaceObject->InPlaceDeactivate();
- // if this assertion fails the control didn't call OnInPlaceDeactivate
+ if (!m_spInPlaceObject)
+ return;
+
+ // InPlaceDeactivate should trigger an OnInPlaceDeactivate callback
+ HRESULT hr = m_spInPlaceObject->InPlaceDeactivate();
+
+ // call fails if an out-of-process control crashes
+ if (FAILED(hr)) {
+ // Call OnInPlaceDeactivate directly to clean up
+ OnInPlaceDeactivate();
+ // schedule release of QAxClientSite references that were held by the control
+ CoDisconnectObject(static_cast<IUnknown *>(static_cast<IDispatch *>(this)), 0);
+ }
+
Q_ASSERT(m_spInPlaceObject == 0);
}
diff --git a/src/activeqt/control/control.pro b/src/activeqt/control/control.pro
index b87c30e..e55bdab 100644
--- a/src/activeqt/control/control.pro
+++ b/src/activeqt/control/control.pro
@@ -10,6 +10,7 @@ mingw: DEFINES += QT_NEEDS_QMAIN
HEADERS = qaxaggregated.h \
qaxbindable.h \
qaxfactory.h \
+ qclassfactory_p.h \
../shared/qaxtypes.h
SOURCES = qaxaggregated.cpp \
@@ -22,6 +23,8 @@ SOURCES = qaxaggregated.cpp \
qaxmain.cpp \
../shared/qaxtypes.cpp
+LIBS += -lshell32
+
MODULE = axserver
MODULE_DEFINES = QAXSERVER
MODULE_CONFIG = idcidl force_import_plugins
diff --git a/src/activeqt/control/qaxserver.cpp b/src/activeqt/control/qaxserver.cpp
index 97756c3..b3c11db 100644
--- a/src/activeqt/control/qaxserver.cpp
+++ b/src/activeqt/control/qaxserver.cpp
@@ -490,10 +490,10 @@ static const char* const type_map[][2] =
{ "QVariant", "VARIANT" },
{ "QVariantList", "SAFEARRAY(VARIANT)" },
{ "QList<QVariant>", "SAFEARRAY(VARIANT)" },
- { "quint64", "CY" },
- { "qint64", "CY" },
- { "qulonglong", "CY" },
- { "qlonglong", "CY" },
+ { "quint64", "unsigned hyper" },
+ { "qint64", "hyper" },
+ { "qulonglong", "unsigned hyper" },
+ { "qlonglong", "hyper" },
{ "QByteArray", "SAFEARRAY(BYTE)" },
{ "QStringList", "SAFEARRAY(BSTR)" },
// Userdefined Qt datatypes - some not on Borland though
diff --git a/src/activeqt/control/qaxserverbase.cpp b/src/activeqt/control/qaxserverbase.cpp
index 7be78d2..9b7fab8 100644
--- a/src/activeqt/control/qaxserverbase.cpp
+++ b/src/activeqt/control/qaxserverbase.cpp
@@ -65,6 +65,9 @@
#include <qpa/qplatformnativeinterface.h>
#include <qabstractnativeeventfilter.h>
+#include <qcoreapplication.h>
+#include <private/qthread_p.h>
+
#include "qaxfactory.h"
#include "qaxbindable.h"
#include "qaxaggregated.h"
@@ -72,6 +75,8 @@
#include "../shared/qaxtypes.h"
#include "../shared/qaxutils_p.h"
+#include "qclassfactory_p.h"
+
#if defined Q_CC_GNU
# include <w32api.h>
#endif
@@ -831,182 +836,180 @@ Q_GLOBAL_STATIC(QAxWinEventFilter, qax_winEventFilter);
// COM Factory class, mapping COM requests to ActiveQt requests.
// One instance of this class for each ActiveX the server can provide.
-class QClassFactory : public IClassFactory2
+QClassFactory::QClassFactory(CLSID clsid)
+ : ref(0), licensed(false)
{
-public:
- QClassFactory(CLSID clsid)
- : ref(0), licensed(false)
- {
- InitializeCriticalSection(&refCountSection);
-
- // COM only knows the CLSID, but QAxFactory is class name based...
- const QStringList keys = qAxFactory()->featureList();
- for (const QString &key : keys) {
- if (qAxFactory()->classID(key) == clsid) {
- className = key;
- break;
- }
- }
+ InitializeCriticalSection(&refCountSection);
- const QMetaObject *mo = qAxFactory()->metaObject(className);
- if (mo) {
- classKey = QLatin1String(mo->classInfo(mo->indexOfClassInfo("LicenseKey")).value());
- licensed = !classKey.isEmpty();
+ // COM only knows the CLSID, but QAxFactory is class name based...
+ const QStringList keys = qAxFactory()->featureList();
+ for (const QString &key : keys) {
+ if (qAxFactory()->classID(key) == clsid) {
+ className = key;
+ break;
}
}
- virtual ~QClassFactory()
- {
- DeleteCriticalSection(&refCountSection);
+ const QMetaObject *mo = qAxFactory()->metaObject(className);
+ if (mo) {
+ classKey = QLatin1String(mo->classInfo(mo->indexOfClassInfo("LicenseKey")).value());
+ licensed = !classKey.isEmpty();
}
+}
- // IUnknown
- unsigned long WINAPI AddRef()
- {
- return InterlockedIncrement(&ref);
- }
- unsigned long WINAPI Release()
- {
- LONG refCount = InterlockedDecrement(&ref);
- if (!refCount)
- delete this;
+QClassFactory::~QClassFactory()
+{
+ DeleteCriticalSection(&refCountSection);
+}
- return refCount;
- }
- HRESULT WINAPI QueryInterface(REFIID iid, LPVOID *iface)
- {
- *iface = 0;
- if (iid == IID_IUnknown)
- *iface = static_cast<IUnknown *>(this);
- else if (iid == IID_IClassFactory)
- *iface = static_cast<IClassFactory *>(this);
- else if (iid == IID_IClassFactory2 && licensed)
- *iface = static_cast<IClassFactory2 *>(this);
- else
- return E_NOINTERFACE;
+// IUnknown
+unsigned long QClassFactory::AddRef()
+{
+ return InterlockedIncrement(&ref);
+}
- AddRef();
- return S_OK;
- }
+unsigned long QClassFactory::Release()
+{
+ LONG refCount = InterlockedDecrement(&ref);
+ if (!refCount)
+ delete this;
- HRESULT WINAPI CreateInstanceHelper(IUnknown *pUnkOuter, REFIID iid, void **ppObject)
- {
- if (pUnkOuter) {
- if (iid != IID_IUnknown)
- return CLASS_E_NOAGGREGATION;
- const QMetaObject *mo = qAxFactory()->metaObject(className);
- if (mo && !qstricmp(mo->classInfo(mo->indexOfClassInfo("Aggregatable")).value(), "no"))
- return CLASS_E_NOAGGREGATION;
- }
+ return refCount;
+}
- // Make sure a QApplication instance is present (inprocess case)
- if (!qApp) {
- qax_ownQApp = true;
- int argc = 0;
- new QApplication(argc, 0);
- }
- QGuiApplication::setQuitOnLastWindowClosed(false);
+HRESULT QClassFactory::QueryInterface(REFIID iid, LPVOID *iface)
+{
+ *iface = 0;
+ if (iid == IID_IUnknown)
+ *iface = static_cast<IUnknown *>(this);
+ else if (iid == IID_IClassFactory)
+ *iface = static_cast<IClassFactory *>(this);
+ else if (iid == IID_IClassFactory2 && licensed)
+ *iface = static_cast<IClassFactory2 *>(this);
+ else
+ return E_NOINTERFACE;
- if (qAxOutProcServer)
- QAbstractEventDispatcher::instance()->installNativeEventFilter(qax_winEventFilter());
- else
- QApplication::instance()->d_func()->in_exec = true;
+ AddRef();
+ return S_OK;
+}
- // hook into eventloop; this allows a server to create his own QApplication object
- if (!qax_hhook && qax_ownQApp) {
- qax_hhook = SetWindowsHookEx(WH_GETMESSAGE, axs_FilterProc, 0, GetCurrentThreadId());
- }
+HRESULT QClassFactory::CreateInstanceHelper(IUnknown *pUnkOuter, REFIID iid, void **ppObject)
+{
+ if (pUnkOuter) {
+ if (iid != IID_IUnknown)
+ return CLASS_E_NOAGGREGATION;
+ const QMetaObject *mo = qAxFactory()->metaObject(className);
+ if (mo && !qstricmp(mo->classInfo(mo->indexOfClassInfo("Aggregatable")).value(), "no"))
+ return CLASS_E_NOAGGREGATION;
+ }
- // If we created QApplication instance, ensure native event loop starts properly
- // by calling processEvents.
- if (qax_ownQApp)
- QCoreApplication::processEvents();
-
- HRESULT res;
- // Create the ActiveX wrapper - aggregate if requested
- if (pUnkOuter) {
- QAxServerAggregate *aggregate = new QAxServerAggregate(className, pUnkOuter);
- res = aggregate->QueryInterface(iid, ppObject);
- if (FAILED(res))
- delete aggregate;
- } else {
- QAxServerBase *activeqt = new QAxServerBase(className, pUnkOuter);
- res = activeqt->QueryInterface(iid, ppObject);
- if (FAILED(res))
- delete activeqt;
- else
- activeqt->registerActiveObject((IUnknown*)(IDispatch*)activeqt);
- }
- return res;
+ // Make sure a QApplication instance is present (inprocess case)
+ if (!qApp) {
+ qax_ownQApp = true;
+ int argc = 0;
+ new QApplication(argc, 0);
}
+ QGuiApplication::setQuitOnLastWindowClosed(false);
- // IClassFactory
- HRESULT WINAPI CreateInstance(IUnknown *pUnkOuter, REFIID iid, void **ppObject)
- {
- // class is licensed
- if (licensed && !qAxFactory()->validateLicenseKey(className, QString()))
- return CLASS_E_NOTLICENSED;
+ if (qAxOutProcServer)
+ QAbstractEventDispatcher::instance()->installNativeEventFilter(qax_winEventFilter());
+ else
+ QApplication::instance()->d_func()->in_exec = true;
- return CreateInstanceHelper(pUnkOuter, iid, ppObject);
+ // hook into eventloop; this allows a server to create his own QApplication object
+ if (!qax_hhook && qax_ownQApp) {
+ qax_hhook = SetWindowsHookEx(WH_GETMESSAGE, axs_FilterProc, 0, GetCurrentThreadId());
}
- HRESULT WINAPI LockServer(BOOL fLock)
- {
- if (fLock)
- qAxLock();
- else
- qAxUnlock();
- return S_OK;
+ // If we created QApplication instance, ensure native event loop starts properly
+ // by calling processEvents.
+ if (qax_ownQApp)
+ QCoreApplication::processEvents();
+
+ HRESULT res;
+ // Create the ActiveX wrapper - aggregate if requested
+ if (pUnkOuter) {
+ QAxServerAggregate *aggregate = new QAxServerAggregate(className, pUnkOuter);
+ res = aggregate->QueryInterface(iid, ppObject);
+ if (FAILED(res))
+ delete aggregate;
+ } else {
+ QAxServerBase *activeqt = new QAxServerBase(className, pUnkOuter);
+ res = activeqt->QueryInterface(iid, ppObject);
+ if (FAILED(res))
+ delete activeqt;
+ else
+ activeqt->registerActiveObject((IUnknown*)(IDispatch*)activeqt);
}
+ return res;
+}
- // IClassFactory2
- HRESULT WINAPI RequestLicKey(DWORD, BSTR *pKey)
- {
- if (!pKey)
- return E_POINTER;
- *pKey = 0;
+// IClassFactory
+HRESULT QClassFactory::CreateInstance(IUnknown *pUnkOuter, REFIID iid, void **ppObject)
+{
+ // class is licensed
+ if (licensed && !qAxFactory()->validateLicenseKey(className, QString()))
+ return CLASS_E_NOTLICENSED;
- // This of course works only on fully licensed machines
- if (!qAxFactory()->validateLicenseKey(className, QString()))
- return CLASS_E_NOTLICENSED;
+ return CreateInstanceHelper(pUnkOuter, iid, ppObject);
+}
- *pKey = QStringToBSTR(classKey);
- return S_OK;
- }
+HRESULT QClassFactory::LockServer(BOOL fLock)
+{
+ if (fLock)
+ qAxLock();
+ else
+ qAxUnlock();
- HRESULT WINAPI GetLicInfo(LICINFO *pLicInfo)
- {
- if (!pLicInfo)
- return E_POINTER;
- pLicInfo->cbLicInfo = sizeof(LICINFO);
+ return S_OK;
+}
- // class specific license key?
- const QMetaObject *mo = qAxFactory()->metaObject(className);
- const char *key = mo->classInfo(mo->indexOfClassInfo("LicenseKey")).value();
- pLicInfo->fRuntimeKeyAvail = key && key[0];
+// IClassFactory2
+HRESULT QClassFactory::RequestLicKey(DWORD, BSTR *pKey)
+{
+ if (!pKey)
+ return E_POINTER;
+ *pKey = 0;
- // machine fully licensed?
- pLicInfo->fLicVerified = qAxFactory()->validateLicenseKey(className, QString());
+ // This of course works only on fully licensed machines
+ if (!qAxFactory()->validateLicenseKey(className, QString()))
+ return CLASS_E_NOTLICENSED;
- return S_OK;
- }
+ *pKey = QStringToBSTR(classKey);
+ return S_OK;
+}
- HRESULT WINAPI CreateInstanceLic(IUnknown *pUnkOuter, IUnknown * /* pUnkReserved */, REFIID iid, BSTR bKey, PVOID *ppObject)
- {
- QString licenseKey = QString::fromWCharArray(bKey);
- if (!qAxFactory()->validateLicenseKey(className, licenseKey))
- return CLASS_E_NOTLICENSED;
- return CreateInstanceHelper(pUnkOuter, iid, ppObject);
- }
+HRESULT QClassFactory::GetLicInfo(LICINFO *pLicInfo)
+{
+ if (!pLicInfo)
+ return E_POINTER;
+ pLicInfo->cbLicInfo = sizeof(LICINFO);
- QString className;
+ // class specific license key?
+ const QMetaObject *mo = qAxFactory()->metaObject(className);
+ const char *key = mo->classInfo(mo->indexOfClassInfo("LicenseKey")).value();
+ pLicInfo->fRuntimeKeyAvail = key && key[0];
+
+ // machine fully licensed?
+ pLicInfo->fLicVerified = qAxFactory()->validateLicenseKey(className, QString());
+
+ return S_OK;
+}
+
+HRESULT QClassFactory::CreateInstanceLic(IUnknown *pUnkOuter, IUnknown * /* pUnkReserved */, REFIID iid, BSTR bKey, PVOID *ppObject)
+{
+ QString licenseKey = QString::fromWCharArray(bKey);
+ if (!qAxFactory()->validateLicenseKey(className, licenseKey))
+ return CLASS_E_NOTLICENSED;
+ return CreateInstanceHelper(pUnkOuter, iid, ppObject);
+}
+
+void QClassFactory::cleanupCreatedApplication(QCoreApplication &app)
+{
+ // Cleanup similar to QCoreApplication::exec()
+ app.d_func()->execCleanup();
+}
-protected:
- CRITICAL_SECTION refCountSection;
- LONG ref;
- bool licensed;
- QString classKey;
-};
// Create a QClassFactory object for class \a iid
HRESULT GetClassObject(REFIID clsid, REFIID iid, void **ppUnk)
diff --git a/src/activeqt/control/qaxserverdll.cpp b/src/activeqt/control/qaxserverdll.cpp
index 7117cbb..b0584d5 100644
--- a/src/activeqt/control/qaxserverdll.cpp
+++ b/src/activeqt/control/qaxserverdll.cpp
@@ -45,6 +45,8 @@
#include <qt_windows.h>
+#include "qclassfactory_p.h"
+
QT_BEGIN_NAMESPACE
bool qax_ownQApp = false;
@@ -111,6 +113,7 @@ STDAPI DllCanUnloadNow()
if (qax_hhook)
UnhookWindowsHookEx(qax_hhook);
+ QClassFactory::cleanupCreatedApplication(*qApp);
delete qApp;
qax_ownQApp = false;
diff --git a/src/activeqt/control/qaxservermain.cpp b/src/activeqt/control/qaxservermain.cpp
index de1c5a5..e6ddd52 100644
--- a/src/activeqt/control/qaxservermain.cpp
+++ b/src/activeqt/control/qaxservermain.cpp
@@ -43,6 +43,8 @@
#include "qaxfactory.h"
+#include <string.h>
+
#include <qt_windows.h>
QT_BEGIN_NAMESPACE
@@ -173,8 +175,6 @@ bool qax_stopServer()
return true;
}
-extern void qWinMain(HINSTANCE, HINSTANCE, LPSTR, int, int &, QVector<char *> &);
-
QT_END_NAMESPACE
#if defined(QT_NEEDS_QMAIN)
@@ -184,8 +184,47 @@ int qMain(int, char **);
extern "C" int main(int, char **);
#endif
+static inline QStringList commandLineArguments()
+{
+ QStringList result;
+ int size;
+ if (wchar_t **argv = CommandLineToArgvW(GetCommandLine(), &size)) {
+ result.reserve(size);
+ wchar_t **argvEnd = argv + size;
+ for (wchar_t **a = argv; a < argvEnd; ++a)
+ result.append(QString::fromWCharArray(*a));
+ LocalFree(argv);
+ }
+ return result;
+}
-EXTERN_C int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR, int nShowCmd)
+static inline bool matchesOption(const QString &arg, const char *option)
+{
+ return (arg.startsWith(QLatin1Char('/')) || arg.startsWith(QLatin1Char('-')))
+ && arg.rightRef(arg.size() - 1).compare(QLatin1String(option), Qt::CaseInsensitive) == 0;
+}
+
+namespace {
+struct Arg {
+ explicit Arg(const QStringList &args)
+ {
+ argv.reserve(args.size() + 1);
+ for (const QString &a : args)
+ argv.append(_strdup(a.toLocal8Bit().constData()));
+ argv.append(nullptr);
+ }
+
+ ~Arg()
+ {
+ for (int i = 0, last = argv.size() - 1; i < last; ++i)
+ free(argv.at(i));
+ }
+
+ QVector<char *> argv;
+};
+} // namespace
+
+EXTERN_C int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE /* hPrevInstance */, LPSTR, int /* nShowCmd */)
{
QT_USE_NAMESPACE
@@ -193,47 +232,47 @@ EXTERN_C int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR,
GetModuleFileName(0, qAxModuleFilename, MAX_PATH);
qAxInstance = hInstance;
- QByteArray cmdParam = QString::fromWCharArray(GetCommandLine()).toLocal8Bit();
- QList<QByteArray> cmds = cmdParam.split(' ');
- QByteArray unprocessed;
+ const QStringList cmds = commandLineArguments();
+ QStringList unprocessed;
+ unprocessed.reserve(cmds.size());
int nRet = 0;
bool run = true;
bool runServer = false;
for (int i = 0; i < cmds.count(); ++i) {
- QByteArray cmd = cmds.at(i).toLower();
- if (cmd == "-activex" || cmd == "/activex" || cmd == "-embedding" || cmd == "/embedding") {
+ const QString &cmd = cmds.at(i);
+ if (matchesOption(cmd, "activex") || matchesOption(cmd, "embedding")) {
runServer = true;
- } else if (cmd == "-unregserver" || cmd == "/unregserver") {
+ } else if (matchesOption(cmd, "unregserver")) {
nRet = UpdateRegistry(false);
run = false;
break;
- } else if (cmd == "-regserver" || cmd == "/regserver") {
+ } else if (matchesOption(cmd, "regserver")) {
nRet = UpdateRegistry(true);
run = false;
break;
- } else if (cmd == "-dumpidl" || cmd == "/dumpidl") {
+ } else if (matchesOption(cmd, "dumpidl")) {
++i;
if (i < cmds.count()) {
- QByteArray outfile = cmds.at(i);
+ const QString &outfile = cmds.at(i);
++i;
- QByteArray version;
- if (i < cmds.count() && (cmds.at(i) == "-version" || cmds.at(i) == "/version")) {
+ QString version;
+ if (i < cmds.count() && matchesOption(cmds.at(i), "version")) {
++i;
if (i < cmds.count())
version = cmds.at(i);
else
- version = "1.0";
+ version = QStringLiteral("1.0");
}
- nRet = DumpIDL(QString::fromLatin1(outfile.constData()), QString::fromLatin1(version.constData()));
+ nRet = DumpIDL(outfile, version);
} else {
qWarning("Wrong commandline syntax: <app> -dumpidl <idl file> [-version <x.y.z>]");
}
run = false;
break;
} else {
- unprocessed += cmds.at(i) + ' ';
+ unprocessed.append(cmds.at(i));
}
}
@@ -255,19 +294,11 @@ EXTERN_C int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR,
}
{
- struct Arg {
- int c;
- QVector<char*> v;
-
- Arg() : v(8) {}
- ~Arg() { Q_FOREACH (char *arg, v) free(arg); }
- } arg;
-
- qWinMain(hInstance, hPrevInstance, unprocessed.data(), nShowCmd, arg.c, arg.v);
+ Arg args(unprocessed);
qAxInit();
if (runServer)
QAxFactory::startServer();
- nRet = ::main(arg.c, arg.v.data());
+ nRet = ::main(args.argv.size() - 1, args.argv.data());
QAxFactory::stopServer();
qAxCleanup();
}
diff --git a/src/activeqt/control/qclassfactory_p.h b/src/activeqt/control/qclassfactory_p.h
new file mode 100644
index 0000000..633c6f1
--- /dev/null
+++ b/src/activeqt/control/qclassfactory_p.h
@@ -0,0 +1,100 @@
+/****************************************************************************
+**
+** Copyright (C) 2016 The Qt Company Ltd.
+** Contact: http://www.qt.io/licensing/
+**
+** This file is part of the ActiveQt framework of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:BSD$
+** You may use this file under the terms of the BSD license as follows:
+**
+** "Redistribution and use in source and binary forms, with or without
+** modification, are permitted provided that the following conditions are
+** met:
+** * Redistributions of source code must retain the above copyright
+** notice, this list of conditions and the following disclaimer.
+** * Redistributions in binary form must reproduce the above copyright
+** notice, this list of conditions and the following disclaimer in
+** the documentation and/or other materials provided with the
+** distribution.
+** * Neither the name of The Qt Company Ltd nor the names of its
+** contributors may be used to endorse or promote products derived
+** from this software without specific prior written permission.
+**
+**
+** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+#ifndef QCLASSFACTORY_P_H
+#define QCLASSFACTORY_P_H
+
+//
+// W A R N I N G
+// -------------
+//
+// This file is not part of the Qt API. It exists for the convenience
+// of other Qt classes. This header file may change from version to
+// version without notice, or even be removed.
+//
+// We mean it.
+//
+
+#include <qcoreapplication.h>
+#include <ocidl.h>
+
+QT_BEGIN_NAMESPACE
+
+// COM Factory class, mapping COM requests to ActiveQt requests.
+// One instance of this class for each ActiveX the server can provide.
+class QClassFactory : public IClassFactory2
+{
+public:
+ QClassFactory(CLSID clsid);
+
+ virtual ~QClassFactory();
+
+ // IUnknown
+ unsigned long WINAPI AddRef();
+ unsigned long WINAPI Release();
+ HRESULT WINAPI QueryInterface(REFIID iid, LPVOID *iface);
+
+ HRESULT WINAPI CreateInstanceHelper(IUnknown *pUnkOuter, REFIID iid, void **ppObject);
+
+ // IClassFactory
+ HRESULT WINAPI CreateInstance(IUnknown *pUnkOuter, REFIID iid, void **ppObject);
+
+ HRESULT WINAPI LockServer(BOOL fLock);
+ // IClassFactory2
+ HRESULT WINAPI RequestLicKey(DWORD, BSTR *pKey);
+
+ HRESULT WINAPI GetLicInfo(LICINFO *pLicInfo);
+
+ HRESULT WINAPI CreateInstanceLic(IUnknown *pUnkOuter, IUnknown * /* pUnkReserved */, REFIID iid, BSTR bKey, PVOID *ppObject);
+
+ static void cleanupCreatedApplication(QCoreApplication &app);
+
+ QString className;
+
+protected:
+ CRITICAL_SECTION refCountSection;
+ LONG ref;
+ bool licensed;
+ QString classKey;
+};
+
+QT_END_NAMESPACE
+
+#endif // QCLASSFACTORY_P_H