e913b690b9
Change-Id: Ibba27351a81b0b132ce702c1dfd49d56f23bd8c1 Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
553 lines
18 KiB
C++
553 lines
18 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Copyright (C) 2016 Intel Corporation.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of the QtDBus module of the Qt Toolkit.
|
|
**
|
|
** $QT_BEGIN_LICENSE:LGPL$
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.LGPL3 included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU Lesser General Public License version 3 requirements
|
|
** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 2.0 or (at your option) the GNU General
|
|
** Public license version 3 or any later version approved by the KDE Free
|
|
** Qt Foundation. The licenses are as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-2.0.html and
|
|
** https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
** $QT_END_LICENSE$
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "qdbuspendingcall.h"
|
|
#include "qdbuspendingcall_p.h"
|
|
|
|
#include "qdbusconnection_p.h"
|
|
#include "qdbusmetatype_p.h"
|
|
#include "qdbusutil_p.h"
|
|
#include "qcoreapplication.h"
|
|
#include "qcoreevent.h"
|
|
#include <private/qobject_p.h>
|
|
#include <private/qlocking_p.h>
|
|
|
|
#ifndef QT_NO_DBUS
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
/*!
|
|
\class QDBusPendingCall
|
|
\inmodule QtDBus
|
|
\ingroup shared
|
|
\since 4.5
|
|
|
|
\brief The QDBusPendingCall class refers to one pending asynchronous call.
|
|
|
|
A QDBusPendingCall object is a reference to a method call that was
|
|
sent over D-Bus without waiting for a reply. QDBusPendingCall is an
|
|
opaque type, meant to be used as a handle for a pending reply.
|
|
|
|
In most programs, the QDBusPendingCall class will not be used
|
|
directly. It can be safely replaced with the template-based
|
|
QDBusPendingReply, in order to access the contents of the reply or
|
|
wait for it to be complete.
|
|
|
|
The QDBusPendingCallWatcher class allows one to connect to a signal
|
|
that will indicate when the reply has arrived or if the call has
|
|
timed out. It also provides the
|
|
QDBusPendingCallWatcher::waitForFinished() method which will suspend
|
|
the execution of the program until the reply has arrived.
|
|
|
|
\note If you create a copy of a QDBusPendingCall object, all
|
|
information will be shared among the many copies. Therefore,
|
|
QDBusPendingCall is an explicitly-shared object and does not
|
|
provide a method of detaching the copies (since they refer
|
|
to the same pending call)
|
|
|
|
\sa QDBusPendingReply, QDBusPendingCallWatcher,
|
|
QDBusAbstractInterface::asyncCall()
|
|
*/
|
|
|
|
/*!
|
|
\class QDBusPendingCallWatcher
|
|
\inmodule QtDBus
|
|
\since 4.5
|
|
|
|
\brief The QDBusPendingCallWatcher class provides a convenient way for
|
|
waiting for asynchronous replies.
|
|
|
|
The QDBusPendingCallWatcher provides the finished() signal that will be
|
|
emitted when a reply arrives.
|
|
|
|
It is usually used like the following example:
|
|
|
|
\snippet code/src_qdbus_qdbuspendingcall.cpp 0
|
|
|
|
Note that it is not necessary to keep the original QDBusPendingCall
|
|
object around since QDBusPendingCallWatcher inherits from that class
|
|
too.
|
|
|
|
The slot connected to by the above code could be something similar
|
|
to the following:
|
|
|
|
\snippet code/src_qdbus_qdbuspendingcall.cpp 1
|
|
|
|
Note the use of QDBusPendingReply to validate the argument types in
|
|
the reply. If the reply did not contain exactly two arguments
|
|
(one string and one QByteArray), QDBusPendingReply::isError() will
|
|
return true.
|
|
|
|
\sa QDBusPendingReply, QDBusAbstractInterface::asyncCall()
|
|
*/
|
|
|
|
/*!
|
|
\fn void QDBusPendingCallWatcher::finished(QDBusPendingCallWatcher *self)
|
|
|
|
This signal is emitted when the pending call has finished and its
|
|
reply is available. The \a self parameter is a pointer to the
|
|
object itself, passed for convenience so that the slot can access
|
|
the properties and determine the contents of the reply.
|
|
*/
|
|
|
|
void QDBusPendingCallWatcherHelper::add(QDBusPendingCallWatcher *watcher)
|
|
{
|
|
connect(this, SIGNAL(finished()), watcher, SLOT(_q_finished()), Qt::QueuedConnection);
|
|
}
|
|
|
|
QDBusPendingCallPrivate::~QDBusPendingCallPrivate()
|
|
{
|
|
if (pending) {
|
|
q_dbus_pending_call_cancel(pending);
|
|
q_dbus_pending_call_unref(pending);
|
|
}
|
|
delete watcherHelper;
|
|
}
|
|
|
|
bool QDBusPendingCallPrivate::setReplyCallback(QObject *target, const char *member)
|
|
{
|
|
receiver = target;
|
|
metaTypes.clear();
|
|
methodIdx = -1;
|
|
if (!target)
|
|
return true;; // unsetting
|
|
|
|
if (!member || !*member) {
|
|
// would not be able to deliver a reply
|
|
qWarning("QDBusPendingCall::setReplyCallback: error: cannot deliver a reply to %s::%s (%s)",
|
|
target ? target->metaObject()->className() : "(null)",
|
|
member ? member + 1 : "(null)",
|
|
target ? qPrintable(target->objectName()) : "no name");
|
|
return false;
|
|
}
|
|
|
|
methodIdx = QDBusConnectionPrivate::findSlot(target, member + 1, metaTypes);
|
|
if (methodIdx == -1) {
|
|
QByteArray normalizedName = QMetaObject::normalizedSignature(member + 1);
|
|
methodIdx = QDBusConnectionPrivate::findSlot(target, normalizedName, metaTypes);
|
|
}
|
|
if (methodIdx == -1) {
|
|
// would not be able to deliver a reply
|
|
qWarning("QDBusPendingCall::setReplyCallback: error: cannot deliver a reply to %s::%s (%s)",
|
|
target->metaObject()->className(),
|
|
member + 1, qPrintable(target->objectName()));
|
|
return false;
|
|
}
|
|
|
|
// success
|
|
// construct the expected signature
|
|
int count = metaTypes.count() - 1;
|
|
if (count == 1 && metaTypes.at(1) == QDBusMetaTypeId::message()) {
|
|
// wildcard slot, can receive anything, so don't set the signature
|
|
return true;
|
|
}
|
|
|
|
if (metaTypes.at(count) == QDBusMetaTypeId::message())
|
|
--count;
|
|
|
|
setMetaTypes(count, count ? metaTypes.constData() + 1 : 0);
|
|
return true;
|
|
}
|
|
|
|
void QDBusPendingCallPrivate::setMetaTypes(int count, const int *types)
|
|
{
|
|
if (count == 0) {
|
|
expectedReplySignature = QLatin1String(""); // not null
|
|
return;
|
|
}
|
|
|
|
QByteArray sig;
|
|
sig.reserve(count + count / 2);
|
|
for (int i = 0; i < count; ++i) {
|
|
const char *typeSig = QDBusMetaType::typeToSignature(types[i]);
|
|
if (Q_UNLIKELY(!typeSig)) {
|
|
qFatal("QDBusPendingReply: type %s is not registered with QtDBus",
|
|
QMetaType::typeName(types[i]));
|
|
}
|
|
sig += typeSig;
|
|
}
|
|
|
|
expectedReplySignature = QString::fromLatin1(sig);
|
|
}
|
|
|
|
void QDBusPendingCallPrivate::checkReceivedSignature()
|
|
{
|
|
// MUST BE CALLED WITH A LOCKED MUTEX!
|
|
|
|
if (replyMessage.type() == QDBusMessage::InvalidMessage)
|
|
return; // not yet finished - no message to
|
|
// validate against
|
|
if (replyMessage.type() == QDBusMessage::ErrorMessage)
|
|
return; // we don't have to check the signature of an error reply
|
|
|
|
if (expectedReplySignature.isNull())
|
|
return; // no signature to validate against
|
|
|
|
// can't use startsWith here because a null string doesn't start or end with an empty string
|
|
if (replyMessage.signature().indexOf(expectedReplySignature) != 0) {
|
|
const auto errorMsg = QLatin1String("Unexpected reply signature: got \"%1\", "
|
|
"expected \"%2\"");
|
|
replyMessage = QDBusMessage::createError(
|
|
QDBusError::InvalidSignature,
|
|
errorMsg.arg(replyMessage.signature(), expectedReplySignature));
|
|
|
|
}
|
|
}
|
|
|
|
void QDBusPendingCallPrivate::waitForFinished()
|
|
{
|
|
const auto locker = qt_scoped_lock(mutex);
|
|
|
|
if (replyMessage.type() != QDBusMessage::InvalidMessage)
|
|
return; // already finished
|
|
|
|
waitForFinishedCondition.wait(&mutex);
|
|
}
|
|
|
|
/*!
|
|
Creates a copy of the \a other pending asynchronous call. Note
|
|
that both objects will refer to the same pending call.
|
|
*/
|
|
QDBusPendingCall::QDBusPendingCall(const QDBusPendingCall &other)
|
|
: d(other.d)
|
|
{
|
|
}
|
|
|
|
/*!
|
|
\internal
|
|
*/
|
|
QDBusPendingCall::QDBusPendingCall(QDBusPendingCallPrivate *dd)
|
|
: d(dd)
|
|
{
|
|
if (dd) {
|
|
bool r = dd->ref.deref();
|
|
Q_ASSERT(r);
|
|
Q_UNUSED(r);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
Destroys this copy of the QDBusPendingCall object. If this copy is
|
|
also the last copy of a pending asynchronous call, the call will
|
|
be canceled and no further notifications will be received. There
|
|
will be no way of accessing the reply's contents when it arrives.
|
|
*/
|
|
QDBusPendingCall::~QDBusPendingCall()
|
|
{
|
|
// d deleted by QExplicitlySharedDataPointer
|
|
}
|
|
|
|
|
|
/*!
|
|
Creates a copy of the \a other pending asynchronous call and drops
|
|
the reference to the previously-referenced call. Note that both
|
|
objects will refer to the same pending call after this function.
|
|
|
|
If this object contained the last reference of a pending
|
|
asynchronous call, the call will be canceled and no further
|
|
notifications will be received. There will be no way of accessing
|
|
the reply's contents when it arrives.
|
|
*/
|
|
QDBusPendingCall &QDBusPendingCall::operator=(const QDBusPendingCall &other)
|
|
{
|
|
d = other.d;
|
|
return *this;
|
|
}
|
|
|
|
/*!
|
|
\fn void QDBusPendingCall::swap(QDBusPendingCall &other)
|
|
\since 5.0
|
|
|
|
Swaps this pending call instance with \a other. This function is
|
|
very fast and never fails.
|
|
*/
|
|
|
|
/*!
|
|
\fn bool QDBusPendingCallWatcher::isFinished() const
|
|
|
|
Returns \c true if the pending call has finished processing and the
|
|
reply has been received.
|
|
|
|
Note that this function only changes state if you call
|
|
waitForFinished() or if an external D-Bus event happens, which in
|
|
general only happens if you return to the event loop execution.
|
|
|
|
\sa QDBusPendingReply::isFinished()
|
|
*/
|
|
/*!
|
|
\fn template <typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8> bool QDBusPendingReply<T1, T2, T3, T4, T5, T6, T7, T8>::isFinished() const
|
|
|
|
Returns \c true if the pending call has finished processing and the
|
|
reply has been received. If this function returns \c true, the
|
|
isError(), error() and reply() methods should return valid
|
|
information.
|
|
|
|
Note that this function only changes state if you call
|
|
waitForFinished() or if an external D-Bus event happens, which in
|
|
general only happens if you return to the event loop execution.
|
|
|
|
\sa QDBusPendingCallWatcher::isFinished()
|
|
*/
|
|
|
|
bool QDBusPendingCall::isFinished() const
|
|
{
|
|
if (!d)
|
|
return true; // considered finished
|
|
|
|
const auto locker = qt_scoped_lock(d->mutex);
|
|
return d->replyMessage.type() != QDBusMessage::InvalidMessage;
|
|
}
|
|
|
|
void QDBusPendingCall::waitForFinished()
|
|
{
|
|
if (d) d->waitForFinished();
|
|
}
|
|
|
|
/*!
|
|
\fn template<typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8> bool QDBusPendingReply<T1, T2, T3, T4, T5, T6, T7, T8>::isValid() const
|
|
|
|
Returns \c true if the reply contains a normal reply message, false
|
|
if it contains anything else.
|
|
|
|
If the pending call has not finished processing, this function
|
|
return false.
|
|
*/
|
|
bool QDBusPendingCall::isValid() const
|
|
{
|
|
if (!d)
|
|
return false;
|
|
const auto locker = qt_scoped_lock(d->mutex);
|
|
return d->replyMessage.type() == QDBusMessage::ReplyMessage;
|
|
}
|
|
|
|
/*!
|
|
\fn template<typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8> bool QDBusPendingReply<T1, T2, T3, T4, T5, T6, T7, T8>::isError() const
|
|
|
|
Returns \c true if the reply contains an error message, false if it
|
|
contains a normal method reply.
|
|
|
|
If the pending call has not finished processing, this function
|
|
also returns \c true.
|
|
*/
|
|
bool QDBusPendingCall::isError() const
|
|
{
|
|
if (!d)
|
|
return true; // considered finished and an error
|
|
const auto locker = qt_scoped_lock(d->mutex);
|
|
return d->replyMessage.type() == QDBusMessage::ErrorMessage;
|
|
}
|
|
|
|
/*!
|
|
\fn template<typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8> QDBusError QDBusPendingReply<T1, T2, T3, T4, T5, T6, T7, T8>::error() const
|
|
|
|
Retrieves the error content of the reply message, if it has
|
|
finished processing. If the reply message has not finished
|
|
processing or if it contains a normal reply message (non-error),
|
|
this function returns an invalid QDBusError.
|
|
*/
|
|
QDBusError QDBusPendingCall::error() const
|
|
{
|
|
if (d) {
|
|
const auto locker = qt_scoped_lock(d->mutex);
|
|
return QDBusError(d->replyMessage);
|
|
}
|
|
|
|
// not connected, return an error
|
|
QDBusError err = QDBusError(QDBusError::Disconnected,
|
|
QDBusUtil::disconnectedErrorMessage());
|
|
return err;
|
|
}
|
|
|
|
/*!
|
|
\fn template<typename T1, typename T2, typename T3, typename T4, typename T5, typename T6, typename T7, typename T8> QDBusMessage QDBusPendingReply<T1, T2, T3, T4, T5, T6, T7, T8>::reply() const
|
|
|
|
Retrieves the reply message received for the asynchronous call
|
|
that was sent, if it has finished processing. If the pending call
|
|
is not finished, this function returns a QDBusMessage of type
|
|
QDBusMessage::InvalidMessage.
|
|
|
|
After it has finished processing, the message type will either be
|
|
an error message or a normal method reply message.
|
|
*/
|
|
QDBusMessage QDBusPendingCall::reply() const
|
|
{
|
|
if (!d)
|
|
return QDBusMessage::createError(error());
|
|
const auto locker = qt_scoped_lock(d->mutex);
|
|
return d->replyMessage;
|
|
}
|
|
|
|
#if 0
|
|
/*
|
|
Sets the slot \a member in object \a target to be called when the
|
|
reply arrives. The slot's parameter list must match the reply
|
|
message's arguments for it to be called.
|
|
|
|
It may, optionally, contain a QDBusMessage final parameter. If it
|
|
is present, the parameter will contain the reply message object.
|
|
|
|
The callback will not be called if the reply is an error message.
|
|
|
|
This function returns \c true if it could set the callback, false
|
|
otherwise. It is not a guarantee that the callback will be
|
|
called.
|
|
|
|
\warning QDBusPendingCall only supports one callback per pending
|
|
asynchronous call, even if multiple QDBusPendingCall
|
|
objects are referencing the same pending call.
|
|
*/
|
|
bool QDBusPendingCall::setReplyCallback(QObject *target, const char *member)
|
|
{
|
|
if (!d)
|
|
return false;
|
|
|
|
return d->setReplyCallback(target, member);
|
|
}
|
|
#endif
|
|
|
|
/*!
|
|
\since 4.6
|
|
Creates a QDBusPendingCall object based on the error condition
|
|
\a error. The resulting pending call object will be in the
|
|
"finished" state and QDBusPendingReply<T1, T2, T3, T4, T5, T6, T7, T8>::isError() will return true.
|
|
|
|
\sa fromCompletedCall()
|
|
*/
|
|
QDBusPendingCall QDBusPendingCall::fromError(const QDBusError &error)
|
|
{
|
|
return fromCompletedCall(QDBusMessage::createError(error));
|
|
}
|
|
|
|
/*!
|
|
\since 4.6
|
|
Creates a QDBusPendingCall object based on the message \a msg.
|
|
The message must be of type QDBusMessage::ErrorMessage or
|
|
QDBusMessage::ReplyMessage (that is, a message that is typical
|
|
of a completed call).
|
|
|
|
This function is useful for code that requires simulating a pending
|
|
call, but that has already finished.
|
|
|
|
\sa fromError()
|
|
*/
|
|
QDBusPendingCall QDBusPendingCall::fromCompletedCall(const QDBusMessage &msg)
|
|
{
|
|
QDBusPendingCallPrivate *d = 0;
|
|
if (msg.type() == QDBusMessage::ErrorMessage ||
|
|
msg.type() == QDBusMessage::ReplyMessage) {
|
|
d = new QDBusPendingCallPrivate(QDBusMessage(), 0);
|
|
d->replyMessage = msg;
|
|
d->ref.storeRelaxed(1);
|
|
}
|
|
|
|
return QDBusPendingCall(d);
|
|
}
|
|
|
|
|
|
class QDBusPendingCallWatcherPrivate: public QObjectPrivate
|
|
{
|
|
public:
|
|
void _q_finished();
|
|
|
|
Q_DECLARE_PUBLIC(QDBusPendingCallWatcher)
|
|
};
|
|
|
|
inline void QDBusPendingCallWatcherPrivate::_q_finished()
|
|
{
|
|
Q_Q(QDBusPendingCallWatcher);
|
|
emit q->finished(q);
|
|
}
|
|
|
|
/*!
|
|
Creates a QDBusPendingCallWatcher object to watch for replies on the
|
|
asynchronous pending call \a call and sets this object's parent
|
|
to \a parent.
|
|
*/
|
|
QDBusPendingCallWatcher::QDBusPendingCallWatcher(const QDBusPendingCall &call, QObject *parent)
|
|
: QObject(*new QDBusPendingCallWatcherPrivate, parent), QDBusPendingCall(call)
|
|
{
|
|
if (d) { // QDBusPendingCall::d
|
|
const auto locker = qt_scoped_lock(d->mutex);
|
|
if (!d->watcherHelper) {
|
|
d->watcherHelper = new QDBusPendingCallWatcherHelper;
|
|
if (d->replyMessage.type() != QDBusMessage::InvalidMessage) {
|
|
// cause a signal emission anyways
|
|
QMetaObject::invokeMethod(d->watcherHelper, "finished", Qt::QueuedConnection);
|
|
}
|
|
}
|
|
d->watcherHelper->add(this);
|
|
}
|
|
}
|
|
|
|
/*!
|
|
Destroys this object. If this QDBusPendingCallWatcher object was the
|
|
last reference to the unfinished pending call, the call will be
|
|
canceled.
|
|
*/
|
|
QDBusPendingCallWatcher::~QDBusPendingCallWatcher()
|
|
{
|
|
}
|
|
|
|
/*!
|
|
\fn void QDBusPendingCallWatcher::waitForFinished()
|
|
|
|
Suspends the execution of the calling thread until the reply is
|
|
received and processed. After this function returns, isFinished()
|
|
should return true, indicating the reply's contents are ready to
|
|
be processed.
|
|
|
|
\sa QDBusPendingReply::waitForFinished()
|
|
*/
|
|
void QDBusPendingCallWatcher::waitForFinished()
|
|
{
|
|
if (d) {
|
|
d->waitForFinished();
|
|
|
|
// our signals were queued, so deliver them
|
|
QCoreApplication::sendPostedEvents(d->watcherHelper, QEvent::MetaCall);
|
|
QCoreApplication::sendPostedEvents(this, QEvent::MetaCall);
|
|
}
|
|
}
|
|
QT_END_NAMESPACE
|
|
|
|
#endif // QT_NO_DBUS
|
|
|
|
#include "moc_qdbuspendingcall.cpp"
|