diff options
author | toma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2009-11-25 17:56:58 +0000 |
---|---|---|
committer | toma <toma@283d02a7-25f6-0310-bc7c-ecb5cbfe19da> | 2009-11-25 17:56:58 +0000 |
commit | ce4a32fe52ef09d8f5ff1dd22c001110902b60a2 (patch) | |
tree | 5ac38a06f3dde268dc7927dc155896926aaf7012 /dcop/dcopclient.cpp | |
download | tdelibs-ce4a32fe52ef09d8f5ff1dd22c001110902b60a2.tar.gz tdelibs-ce4a32fe52ef09d8f5ff1dd22c001110902b60a2.zip |
Copy the KDE 3.5 branch to branches/trinity for new KDE 3.5 features.
BUG:215923
git-svn-id: svn://anonsvn.kde.org/home/kde/branches/trinity/kdelibs@1054174 283d02a7-25f6-0310-bc7c-ecb5cbfe19da
Diffstat (limited to 'dcop/dcopclient.cpp')
-rw-r--r-- | dcop/dcopclient.cpp | 2283 |
1 files changed, 2283 insertions, 0 deletions
diff --git a/dcop/dcopclient.cpp b/dcop/dcopclient.cpp new file mode 100644 index 000000000..2b39ecb55 --- /dev/null +++ b/dcop/dcopclient.cpp @@ -0,0 +1,2283 @@ +/***************************************************************** + +Copyright (c) 1999 Preston Brown <pbrown@kde.org> +Copyright (c) 1999 Matthias Ettrich <ettrich@kde.org> + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN +AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + +******************************************************************/ + +// qt <-> dcop integration +#include <qobjectlist.h> +#include <qmetaobject.h> +#include <qvariant.h> +#include <qtimer.h> +#include <qintdict.h> +#include <qeventloop.h> +// end of qt <-> dcop integration + +#include "config.h" + +#include <config.h> +#include <dcopref.h> + +#include <sys/time.h> +#include <sys/types.h> +#include <sys/stat.h> +#include <sys/file.h> +#include <sys/socket.h> +#include <fcntl.h> +#include <unistd.h> + +#include <ctype.h> +#include <unistd.h> +#include <stdlib.h> +#include <assert.h> +#include <string.h> + +#ifndef QT_CLEAN_NAMESPACE +#define QT_CLEAN_NAMESPACE +#endif +#include <qguardedptr.h> +#include <qtextstream.h> +#include <qfile.h> +#include <qdir.h> +#include <qapplication.h> +#include <qsocketnotifier.h> +#include <qregexp.h> + +#include <private/qucomextra_p.h> + +#include <dcopglobal.h> +#include <dcopclient.h> +#include <dcopobject.h> + +#if defined Q_WS_X11 && ! defined K_WS_QTONLY +#include <X11/Xmd.h> +#endif +extern "C" { +#include <KDE-ICE/ICElib.h> +#include <KDE-ICE/ICEutil.h> +#include <KDE-ICE/ICEmsg.h> +#include <KDE-ICE/ICEproto.h> +} + +// #define DCOPCLIENT_DEBUG 1 + +extern QMap<QCString, DCOPObject *> * kde_dcopObjMap; // defined in dcopobject.cpp + +/********************************************* + * Keep track of local clients + *********************************************/ +typedef QAsciiDict<DCOPClient> client_map_t; +static client_map_t *DCOPClient_CliMap = 0; + +static +client_map_t *cliMap() +{ + if (!DCOPClient_CliMap) + DCOPClient_CliMap = new client_map_t; + return DCOPClient_CliMap; +} + +DCOPClient *DCOPClient::findLocalClient( const QCString &_appId ) +{ + return cliMap()->find(_appId.data()); +} + +static +void registerLocalClient( const QCString &_appId, DCOPClient *client ) +{ + cliMap()->replace(_appId.data(), client); +} + +static +void unregisterLocalClient( const QCString &_appId ) +{ + client_map_t *map = cliMap(); + map->remove(_appId.data()); +} +///////////////////////////////////////////////////////// + +template class QPtrList<DCOPObjectProxy>; +template class QPtrList<DCOPClientTransaction>; +template class QPtrList<_IceConn>; + +struct DCOPClientMessage +{ + int opcode; + CARD32 key; + QByteArray data; +}; + +class DCOPClient::ReplyStruct +{ +public: + enum ReplyStatus { Pending, Ok, Failed }; + ReplyStruct() { + status = Pending; + replyType = 0; + replyData = 0; + replyId = -1; + transactionId = -1; + replyObject = 0; + } + ReplyStatus status; + QCString* replyType; + QByteArray* replyData; + int replyId; + Q_INT32 transactionId; + QCString calledApp; + QGuardedPtr<QObject> replyObject; + QCString replySlot; +}; + +class DCOPClientPrivate +{ +public: + DCOPClient *parent; + QCString appId; + IceConn iceConn; + int majorOpcode; // major opcode negotiated w/server and used to tag all comms. + + int majorVersion, minorVersion; // protocol versions negotiated w/server + + static const char* serverAddr; // location of server in ICE-friendly format. + QSocketNotifier *notifier; + bool non_blocking_call_lock; + bool registered; + bool foreign_server; + bool accept_calls; + bool accept_calls_override; // If true, user has specified policy. + bool qt_bridge_enabled; + + QCString senderId; + QCString objId; + QCString function; + + QCString defaultObject; + QPtrList<DCOPClientTransaction> *transactionList; + bool transaction; + Q_INT32 transactionId; + int opcode; + + // Special key values: + // 0 : Not specified + // 1 : DCOPSend + // 2 : Priority + // >= 42: Normal + CARD32 key; + CARD32 currentKey; + CARD32 currentKeySaved; + + QTimer postMessageTimer; + QPtrList<DCOPClientMessage> messages; + + QPtrList<DCOPClient::ReplyStruct> pendingReplies; + QPtrList<DCOPClient::ReplyStruct> asyncReplyQueue; + + struct LocalTransactionResult + { + QCString replyType; + QByteArray replyData; + }; + + QIntDict<LocalTransactionResult> localTransActionList; + + QTimer eventLoopTimer; +}; + +class DCOPClientTransaction +{ +public: + Q_INT32 id; + CARD32 key; + QCString senderId; +}; + +QCString DCOPClient::iceauthPath() +{ +#ifdef Q_OS_WIN32 + char szPath[512]; + char * pszFilePart; + int ret; + ret = SearchPathA(NULL,"iceauth.exe",NULL,sizeof(szPath)/sizeof(szPath[0]),szPath,&pszFilePart); + if(ret != 0) + return QCString(szPath); +#else + QCString path = ::getenv("PATH"); + if (path.isEmpty()) + path = "/bin:/usr/bin"; + path += ":/usr/bin/X11:/usr/X11/bin:/usr/X11R6/bin"; + QCString fPath = strtok(path.data(), ":\b"); + while (!fPath.isNull()) + { + fPath += "/iceauth"; + if (access(fPath.data(), X_OK) == 0) + { + return fPath; + } + + fPath = strtok(NULL, ":\b"); + } +#endif + return 0; +} + +static QCString dcopServerFile(const QCString &hostname, bool old) +{ + QCString fName = ::getenv("DCOPAUTHORITY"); + if (!old && !fName.isEmpty()) + return fName; + + fName = QFile::encodeName( QDir::homeDirPath() ); +// fName = ::getenv("HOME"); + if (fName.isEmpty()) + { + fprintf(stderr, "Aborting. $HOME is not set.\n"); + exit(1); + } +#ifdef Q_WS_X11 + QCString disp = getenv("DISPLAY"); +#elif defined(Q_WS_QWS) + QCString disp = getenv("QWS_DISPLAY"); +#else + QCString disp; +#endif + if (disp.isEmpty()) + disp = "NODISPLAY"; + + int i; + if((i = disp.findRev('.')) > disp.findRev(KPATH_SEPARATOR) && i >= 0) + disp.truncate(i); + + if (!old) + { + while( (i = disp.find(KPATH_SEPARATOR)) >= 0) + disp[i] = '_'; + } + + fName += "/.DCOPserver_"; + if (hostname.isEmpty()) + { + char hostName[256]; + hostName[0] = '\0'; + if (gethostname(hostName, sizeof(hostName))) + { + fName += "localhost"; + } + else + { + hostName[sizeof(hostName)-1] = '\0'; + fName += hostName; + } + } + else + { + fName += hostname; + } + fName += "_"+disp; + return fName; +} + + +// static +QCString DCOPClient::dcopServerFile(const QCString &hostname) +{ + return ::dcopServerFile(hostname, false); +} + + +// static +QCString DCOPClient::dcopServerFileOld(const QCString &hostname) +{ + return ::dcopServerFile(hostname, true); +} + + +const char* DCOPClientPrivate::serverAddr = 0; + +static void DCOPProcessInternal( DCOPClientPrivate *d, int opcode, CARD32 key, const QByteArray& dataReceived, bool canPost ); + +void DCOPClient::handleAsyncReply(ReplyStruct *replyStruct) +{ + if (replyStruct->replyObject) + { + QObject::connect(this, SIGNAL(callBack(int, const QCString&, const QByteArray &)), + replyStruct->replyObject, replyStruct->replySlot); + emit callBack(replyStruct->replyId, *(replyStruct->replyType), *(replyStruct->replyData)); + QObject::disconnect(this, SIGNAL(callBack(int, const QCString&, const QByteArray &)), + replyStruct->replyObject, replyStruct->replySlot); + } + delete replyStruct; +} + +/** + * Callback for ICE. + */ +static void DCOPProcessMessage(IceConn iceConn, IcePointer clientObject, + int opcode, unsigned long length, Bool /*swap*/, + IceReplyWaitInfo *replyWait, + Bool *replyWaitRet) +{ + DCOPMsg *pMsg = 0; + DCOPClientPrivate *d = static_cast<DCOPClientPrivate *>(clientObject); + DCOPClient::ReplyStruct *replyStruct = replyWait ? static_cast<DCOPClient::ReplyStruct*>(replyWait->reply) : 0; + + IceReadMessageHeader(iceConn, sizeof(DCOPMsg), DCOPMsg, pMsg); + CARD32 key = pMsg->key; + if ( d->key == 0 ) + d->key = key; // received a key from the server + + QByteArray dataReceived( length ); + IceReadData(iceConn, length, dataReceived.data() ); + + d->opcode = opcode; + switch (opcode ) { + + case DCOPReplyFailed: + if ( replyStruct ) { + replyStruct->status = DCOPClient::ReplyStruct::Failed; + replyStruct->transactionId = 0; + *replyWaitRet = True; + return; + } else { + qWarning("Very strange! got a DCOPReplyFailed opcode, but we were not waiting for a reply!"); + return; + } + case DCOPReply: + if ( replyStruct ) { + QByteArray* b = replyStruct->replyData; + QCString* t = replyStruct->replyType; + replyStruct->status = DCOPClient::ReplyStruct::Ok; + replyStruct->transactionId = 0; + + QCString calledApp, app; + QDataStream ds( dataReceived, IO_ReadOnly ); + ds >> calledApp >> app >> *t >> *b; + + *replyWaitRet = True; + return; + } else { + qWarning("Very strange! got a DCOPReply opcode, but we were not waiting for a reply!"); + return; + } + case DCOPReplyWait: + if ( replyStruct ) { + QCString calledApp, app; + Q_INT32 id; + QDataStream ds( dataReceived, IO_ReadOnly ); + ds >> calledApp >> app >> id; + replyStruct->transactionId = id; + replyStruct->calledApp = calledApp; + d->pendingReplies.append(replyStruct); + *replyWaitRet = True; + return; + } else { + qWarning("Very strange! got a DCOPReplyWait opcode, but we were not waiting for a reply!"); + return; + } + case DCOPReplyDelayed: + { + QDataStream ds( dataReceived, IO_ReadOnly ); + QCString calledApp, app; + Q_INT32 id; + + ds >> calledApp >> app >> id; + if (replyStruct && (id == replyStruct->transactionId) && (calledApp == replyStruct->calledApp)) + { + *replyWaitRet = True; + } + + for(DCOPClient::ReplyStruct *rs = d->pendingReplies.first(); rs; + rs = d->pendingReplies.next()) + { + if ((rs->transactionId == id) && (rs->calledApp == calledApp)) + { + d->pendingReplies.remove(); + QByteArray* b = rs->replyData; + QCString* t = rs->replyType; + ds >> *t >> *b; + + rs->status = DCOPClient::ReplyStruct::Ok; + rs->transactionId = 0; + if (!rs->replySlot.isEmpty()) + { + d->parent->handleAsyncReply(rs); + } + return; + } + } + } + qWarning("Very strange! got a DCOPReplyDelayed opcode, but we were not waiting for a reply!"); + return; + case DCOPCall: + case DCOPFind: + case DCOPSend: + DCOPProcessInternal( d, opcode, key, dataReceived, true ); + } +} + +void DCOPClient::processPostedMessagesInternal() +{ + if ( d->messages.isEmpty() ) + return; + QPtrListIterator<DCOPClientMessage> it (d->messages ); + DCOPClientMessage* msg ; + while ( ( msg = it.current() ) ) { + ++it; + if ( d->currentKey && msg->key != d->currentKey ) + continue; + d->messages.removeRef( msg ); + d->opcode = msg->opcode; + DCOPProcessInternal( d, msg->opcode, msg->key, msg->data, false ); + delete msg; + } + if ( !d->messages.isEmpty() ) + d->postMessageTimer.start( 100, true ); +} + +/** + Processes DCOPCall, DCOPFind and DCOPSend + */ +void DCOPProcessInternal( DCOPClientPrivate *d, int opcode, CARD32 key, const QByteArray& dataReceived, bool canPost ) +{ + if (!d->accept_calls && (opcode == DCOPSend)) + return; + + IceConn iceConn = d->iceConn; + DCOPMsg *pMsg = 0; + DCOPClient *c = d->parent; + QDataStream ds( dataReceived, IO_ReadOnly ); + + QCString fromApp; + ds >> fromApp; + if (fromApp.isEmpty()) + return; // Reserved for local calls + + if (!d->accept_calls) + { + QByteArray reply; + QDataStream replyStream( reply, IO_WriteOnly ); + // Call rejected. + replyStream << d->appId << fromApp; + IceGetHeader( iceConn, d->majorOpcode, DCOPReplyFailed, + sizeof(DCOPMsg), DCOPMsg, pMsg ); + int datalen = reply.size(); + pMsg->key = key; + pMsg->length += datalen; + IceSendData( iceConn, datalen, const_cast<char *>(reply.data())); + return; + } + + QCString app, objId, fun; + QByteArray data; + ds >> app >> objId >> fun >> data; + d->senderId = fromApp; + d->objId = objId; + d->function = fun; + +// qWarning("DCOP: %s got call: %s:%s:%s key = %d currentKey = %d", d->appId.data(), app.data(), objId.data(), fun.data(), key, d->currentKey); + + if ( canPost && d->currentKey && key != d->currentKey ) { + DCOPClientMessage* msg = new DCOPClientMessage; + msg->opcode = opcode; + msg->key = key; + msg->data = dataReceived; + d->messages.append( msg ); + d->postMessageTimer.start( 0, true ); + return; + } + + d->objId = objId; + d->function = fun; + + QCString replyType; + QByteArray replyData; + bool b; + CARD32 oldCurrentKey = d->currentKey; + if ( opcode != DCOPSend ) // DCOPSend doesn't change the current key + d->currentKey = key; + + if ( opcode == DCOPFind ) + b = c->find(app, objId, fun, data, replyType, replyData ); + else + b = c->receive( app, objId, fun, data, replyType, replyData ); + // set notifier back to previous state + + if ( opcode == DCOPSend ) + return; + + if ((d->currentKey == key) || (oldCurrentKey != 2)) + d->currentKey = oldCurrentKey; + + QByteArray reply; + QDataStream replyStream( reply, IO_WriteOnly ); + + Q_INT32 id = c->transactionId(); + if (id) { + // Call delayed. Send back the transaction ID. + replyStream << d->appId << fromApp << id; + + IceGetHeader( iceConn, d->majorOpcode, DCOPReplyWait, + sizeof(DCOPMsg), DCOPMsg, pMsg ); + pMsg->key = key; + pMsg->length += reply.size(); + IceSendData( iceConn, reply.size(), const_cast<char *>(reply.data())); + return; + } + + if ( !b ) { + // Call failed. No data send back. + + replyStream << d->appId << fromApp; + IceGetHeader( iceConn, d->majorOpcode, DCOPReplyFailed, + sizeof(DCOPMsg), DCOPMsg, pMsg ); + int datalen = reply.size(); + pMsg->key = key; + pMsg->length += datalen; + IceSendData( iceConn, datalen, const_cast<char *>(reply.data())); + return; + } + + // Call successful. Send back replyType and replyData. + replyStream << d->appId << fromApp << replyType << replyData.size(); + + + // we are calling, so we need to set up reply data + IceGetHeader( iceConn, d->majorOpcode, DCOPReply, + sizeof(DCOPMsg), DCOPMsg, pMsg ); + int datalen = reply.size() + replyData.size(); + pMsg->key = key; + pMsg->length += datalen; + // use IceSendData not IceWriteData to avoid a copy. Output buffer + // shouldn't need to be flushed. + IceSendData( iceConn, reply.size(), const_cast<char *>(reply.data())); + IceSendData( iceConn, replyData.size(), const_cast<char *>(replyData.data())); +} + + + +static IcePoVersionRec DCOPClientVersions[] = { + { DCOPVersionMajor, DCOPVersionMinor, DCOPProcessMessage } +}; + + +static DCOPClient* dcop_main_client = 0; + +DCOPClient* DCOPClient::mainClient() +{ + return dcop_main_client; +} + +void DCOPClient::setMainClient( DCOPClient* client ) +{ + dcop_main_client = client; +} + + +DCOPClient::DCOPClient() +{ + d = new DCOPClientPrivate; + d->parent = this; + d->iceConn = 0L; + d->key = 0; + d->currentKey = 0; + d->majorOpcode = 0; + d->appId = 0; + d->notifier = 0L; + d->non_blocking_call_lock = false; + d->registered = false; + d->foreign_server = true; + d->accept_calls = true; + d->accept_calls_override = false; + d->qt_bridge_enabled = true; + d->transactionList = 0L; + d->transactionId = 0; + QObject::connect( &d->postMessageTimer, SIGNAL( timeout() ), this, SLOT( processPostedMessagesInternal() ) ); + QObject::connect( &d->eventLoopTimer, SIGNAL( timeout() ), this, SLOT( eventLoopTimeout() ) ); + + if ( !mainClient() ) + setMainClient( this ); +} + +DCOPClient::~DCOPClient() +{ +#ifdef DCOPCLIENT_DEBUG + qWarning("d->messages.count() = %d", d->messages.count()); + QPtrListIterator<DCOPClientMessage> it (d->messages ); + DCOPClientMessage* msg ; + while ( ( msg = it.current() ) ) { + ++it; + d->messages.removeRef( msg ); + qWarning("DROPPING UNHANDLED DCOP MESSAGE:"); + qWarning(" opcode = %d key = %d", msg->opcode, msg->key); + QDataStream ds( msg->data, IO_ReadOnly ); + + QCString fromApp, app, objId, fun; + ds >> fromApp >> app >> objId >> fun; + qWarning(" from = %s", fromApp.data()); + qWarning(" to = %s / %s / %s", app.data(), objId.data(), fun.data()); + delete msg; + } +#endif + if (d->iceConn) + if (IceConnectionStatus(d->iceConn) == IceConnectAccepted) + detach(); + + if (d->registered) + unregisterLocalClient( d->appId ); + + delete d->notifier; + delete d->transactionList; + d->messages.setAutoDelete(true); + delete d; + + if ( mainClient() == this ) + setMainClient( 0 ); +} + +void DCOPClient::setServerAddress(const QCString &addr) +{ + QCString env = "DCOPSERVER=" + addr; + putenv(strdup(env.data())); + delete [] DCOPClientPrivate::serverAddr; + DCOPClientPrivate::serverAddr = qstrdup( addr.data() ); +} + +bool DCOPClient::attach() +{ + if (!attachInternal( true )) + if (!attachInternal( true )) + return false; // Try two times! + return true; +} + +void DCOPClient::bindToApp() +{ + // check if we have a qApp instantiated. If we do, + // we can create a QSocketNotifier and use it for receiving data. + if (qApp) { + if ( d->notifier ) + delete d->notifier; + d->notifier = new QSocketNotifier(socket(), + QSocketNotifier::Read, 0, 0); + QObject::connect(d->notifier, SIGNAL(activated(int)), + SLOT(processSocketData(int))); + } +} + +void DCOPClient::suspend() +{ +#ifdef Q_WS_WIN //TODO: remove (win32 ports sometimes do not create notifiers) + if (!d->notifier) + return; +#endif + assert(d->notifier); // Suspending makes no sense if we didn't had a qApp yet + d->notifier->setEnabled(false); +} + +void DCOPClient::resume() +{ +#ifdef Q_WS_WIN //TODO: remove + if (!d->notifier) + return; +#endif + assert(d->notifier); // Should never happen + d->notifier->setEnabled(true); +} + +bool DCOPClient::isSuspended() const +{ +#if defined(Q_WS_WIN) || defined(Q_WS_MAC) //TODO: REMOVE + if (!d->notifier) + return false; +#endif + return !d->notifier->isEnabled(); +} + +#ifdef SO_PEERCRED +// Check whether the remote end is owned by the same user. +static bool peerIsUs(int sockfd) +{ + struct ucred cred; + socklen_t siz = sizeof(cred); + if (getsockopt(sockfd, SOL_SOCKET, SO_PEERCRED, &cred, &siz) != 0) + return false; + return (cred.uid == getuid()); +} +#else +// Check whether the socket is owned by the same user. +static bool isServerSocketOwnedByUser(const char*server) +{ +#ifdef Q_OS_WIN + if (strncmp(server, "tcp/", 4) != 0) + return false; // Not a local socket -> foreign. + else + return true; +#else + if (strncmp(server, "local/", 6) != 0) + return false; // Not a local socket -> foreign. + const char *path = strchr(server, KPATH_SEPARATOR); + if (!path) + return false; + path++; + + struct stat stat_buf; + if (stat(path, &stat_buf) != 0) + return false; + + return (stat_buf.st_uid == getuid()); +#endif +} +#endif + + +bool DCOPClient::attachInternal( bool registerAsAnonymous ) +{ + char errBuf[1024]; + + if ( isAttached() ) + detach(); + + if ((d->majorOpcode = IceRegisterForProtocolSetup(const_cast<char *>("DCOP"), + const_cast<char *>(DCOPVendorString), + const_cast<char *>(DCOPReleaseString), + 1, DCOPClientVersions, + DCOPAuthCount, + const_cast<char **>(DCOPAuthNames), + DCOPClientAuthProcs, 0L)) < 0) { + emit attachFailed(QString::fromLatin1( "Communications could not be established." )); + return false; + } + + bool bClearServerAddr = false; + // first, check if serverAddr was ever set. + if (!d->serverAddr) { + // here, we obtain the list of possible DCOP connections, + // and attach to them. + QCString dcopSrv; + dcopSrv = ::getenv("DCOPSERVER"); + if (dcopSrv.isEmpty()) { + QCString fName = dcopServerFile(); + QFile f(QFile::decodeName(fName)); + if (!f.open(IO_ReadOnly)) { + emit attachFailed(QString::fromLatin1( "Could not read network connection list.\n" )+QFile::decodeName(fName)); + return false; + } + int size = QMIN( 1024, f.size() ); // protection against a huge file + QCString contents( size+1 ); + if ( f.readBlock( contents.data(), size ) != size ) + { + qDebug("Error reading from %s, didn't read the expected %d bytes", fName.data(), size); + // Should we abort ? + } + contents[size] = '\0'; + int pos = contents.find('\n'); + if ( pos == -1 ) // Shouldn't happen + { + qDebug("Only one line in dcopserver file !: %s", contents.data()); + dcopSrv = contents; + } + else + { + if(contents[pos - 1] == '\r') // check for windows end of line + pos--; + dcopSrv = contents.left( pos ); +//#ifndef NDEBUG +// qDebug("dcopserver address: %s", dcopSrv.data()); +//#endif + } + } + d->serverAddr = qstrdup( const_cast<char *>(dcopSrv.data()) ); + bClearServerAddr = true; + } + + if ((d->iceConn = IceOpenConnection(const_cast<char*>(d->serverAddr), + static_cast<IcePointer>(this), False, d->majorOpcode, + sizeof(errBuf), errBuf)) == 0L) { + qDebug("DCOPClient::attachInternal. Attach failed %s", errBuf); + d->iceConn = 0; + if (bClearServerAddr) { + delete [] d->serverAddr; + d->serverAddr = 0; + } + emit attachFailed(QString::fromLatin1( errBuf )); + return false; + } + fcntl(socket(), F_SETFL, FD_CLOEXEC); + + IceSetShutdownNegotiation(d->iceConn, False); + + int setupstat; + char* vendor = 0; + char* release = 0; + setupstat = IceProtocolSetup(d->iceConn, d->majorOpcode, + static_cast<IcePointer>(d), + False, /* must authenticate */ + &(d->majorVersion), &(d->minorVersion), + &(vendor), &(release), 1024, errBuf); + if (vendor) free(vendor); + if (release) free(release); + + if (setupstat == IceProtocolSetupFailure || + setupstat == IceProtocolSetupIOError) { + IceCloseConnection(d->iceConn); + d->iceConn = 0; + if (bClearServerAddr) { + delete [] d->serverAddr; + d->serverAddr = 0; + } + emit attachFailed(QString::fromLatin1( errBuf )); + return false; + } else if (setupstat == IceProtocolAlreadyActive) { + if (bClearServerAddr) { + delete [] d->serverAddr; + d->serverAddr = 0; + } + /* should not happen because 3rd arg to IceOpenConnection was 0. */ + emit attachFailed(QString::fromLatin1( "internal error in IceOpenConnection" )); + return false; + } + + + if (IceConnectionStatus(d->iceConn) != IceConnectAccepted) { + if (bClearServerAddr) { + delete [] d->serverAddr; + d->serverAddr = 0; + } + emit attachFailed(QString::fromLatin1( "DCOP server did not accept the connection." )); + return false; + } + +#ifdef SO_PEERCRED + d->foreign_server = !peerIsUs(socket()); +#else + d->foreign_server = !isServerSocketOwnedByUser(d->serverAddr); +#endif + if (!d->accept_calls_override) + d->accept_calls = !d->foreign_server; + + bindToApp(); + + if ( registerAsAnonymous ) + registerAs( "anonymous", true ); + + return true; +} + + +bool DCOPClient::detach() +{ + int status; + + if (d->iceConn) { + IceProtocolShutdown(d->iceConn, d->majorOpcode); + status = IceCloseConnection(d->iceConn); + if (status != IceClosedNow) + return false; + else + d->iceConn = 0L; + } + + if (d->registered) + unregisterLocalClient(d->appId); + + delete d->notifier; + d->notifier = 0L; + d->registered = false; + d->foreign_server = true; + return true; +} + +bool DCOPClient::isAttached() const +{ + if (!d->iceConn) + return false; + + return (IceConnectionStatus(d->iceConn) == IceConnectAccepted); +} + +bool DCOPClient::isAttachedToForeignServer() const +{ + return isAttached() && d->foreign_server; +} + +bool DCOPClient::acceptCalls() const +{ + return isAttached() && d->accept_calls; +} + +void DCOPClient::setAcceptCalls(bool b) +{ + d->accept_calls = b; + d->accept_calls_override = true; +} + +bool DCOPClient::qtBridgeEnabled() +{ + return d->qt_bridge_enabled; +} + +void DCOPClient::setQtBridgeEnabled(bool b) +{ + d->qt_bridge_enabled = b; +} + +QCString DCOPClient::registerAs( const QCString &appId, bool addPID ) +{ + QCString result; + + QCString _appId = appId; + + if (addPID) { + QCString pid; + pid.sprintf("-%d", getpid()); + _appId = _appId + pid; + } + + if( d->appId == _appId ) + return d->appId; + +#if 0 // no need to detach, dcopserver can handle renaming + // Detach before reregistering. + if ( isRegistered() ) { + detach(); + } +#endif + + if ( !isAttached() ) { + if (!attachInternal( false )) + if (!attachInternal( false )) + return result; // Try two times + } + + // register the application identifier with the server + QCString replyType; + QByteArray data, replyData; + QDataStream arg( data, IO_WriteOnly ); + arg << _appId; + if ( call( "DCOPServer", "", "registerAs(QCString)", data, replyType, replyData ) ) { + QDataStream reply( replyData, IO_ReadOnly ); + reply >> result; + } + + d->appId = result; + d->registered = !result.isNull(); + + if (d->registered) + registerLocalClient( d->appId, this ); + + return result; +} + +bool DCOPClient::isRegistered() const +{ + return d->registered; +} + + +QCString DCOPClient::appId() const +{ + return d->appId; +} + + +int DCOPClient::socket() const +{ + if (d->iceConn) + return IceConnectionNumber(d->iceConn); + return 0; +} + +static inline bool isIdentChar( char x ) +{ // Avoid bug in isalnum + return x == '_' || (x >= '0' && x <= '9') || + (x >= 'a' && x <= 'z') || (x >= 'A' && x <= 'Z'); +} + +QCString DCOPClient::normalizeFunctionSignature( const QCString& fun ) { + if ( fun.isEmpty() ) // nothing to do + return fun.copy(); + QCString result( fun.size() ); + char *from = fun.data(); + char *to = result.data(); + char *first = to; + char last = 0; + while ( true ) { + while ( *from && isspace(*from) ) + from++; + if ( last && isIdentChar( last ) && isIdentChar( *from ) ) + *to++ = 0x20; + while ( *from && !isspace(*from) ) { + last = *from++; + *to++ = last; + } + if ( !*from ) + break; + } + if ( to > first && *(to-1) == 0x20 ) + to--; + *to = '\0'; + result.resize( (int)((long)to - (long)result.data()) + 1 ); + return result; +} + + +QCString DCOPClient::senderId() const +{ + return d->senderId; +} + + +bool DCOPClient::send(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QByteArray &data) +{ + if (remApp.isEmpty()) + return false; + DCOPClient *localClient = findLocalClient( remApp ); + + if ( localClient ) { + bool saveTransaction = d->transaction; + Q_INT32 saveTransactionId = d->transactionId; + QCString saveSenderId = d->senderId; + + d->senderId = 0; // Local call + QCString replyType; + QByteArray replyData; + (void) localClient->receive( remApp, remObjId, remFun, data, replyType, replyData ); + + d->transaction = saveTransaction; + d->transactionId = saveTransactionId; + d->senderId = saveSenderId; + // send() returns true if the data could be send to the DCOPServer, + // regardles of receiving the data on the other application. + // So we assume the data is successfully send to the (virtual) server + // and return true in any case. + return true; + } + + if ( !isAttached() ) + return false; + + + DCOPMsg *pMsg; + + QByteArray ba; + QDataStream ds(ba, IO_WriteOnly); + ds << d->appId << remApp << remObjId << normalizeFunctionSignature(remFun) << data.size(); + + IceGetHeader(d->iceConn, d->majorOpcode, DCOPSend, + sizeof(DCOPMsg), DCOPMsg, pMsg); + + pMsg->key = 1; // DCOPSend always uses the magic key 1 + int datalen = ba.size() + data.size(); + pMsg->length += datalen; + + IceSendData( d->iceConn, ba.size(), const_cast<char *>(ba.data()) ); + IceSendData( d->iceConn, data.size(), const_cast<char *>(data.data()) ); + + //IceFlush(d->iceConn); + + if (IceConnectionStatus(d->iceConn) == IceConnectAccepted) + return true; + return false; +} + +bool DCOPClient::send(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QString &data) +{ + QByteArray ba; + QDataStream ds(ba, IO_WriteOnly); + ds << data; + return send(remApp, remObjId, remFun, ba); +} + +bool DCOPClient::findObject(const QCString &remApp, const QCString &remObj, + const QCString &remFun, const QByteArray &data, + QCString &foundApp, QCString &foundObj, + bool useEventLoop) +{ + return findObject( remApp, remObj, remFun, data, foundApp, foundObj, useEventLoop, -1 ); +} + +bool DCOPClient::findObject(const QCString &remApp, const QCString &remObj, + const QCString &remFun, const QByteArray &data, + QCString &foundApp, QCString &foundObj, + bool useEventLoop, int timeout) +{ + QCStringList appList; + QCString app = remApp; + if (app.isEmpty()) + app = "*"; + + foundApp = 0; + foundObj = 0; + + if (app[app.length()-1] == '*') + { + // Find all apps that match 'app'. + // NOTE: It would be more efficient to do the filtering in + // the dcopserver itself. + int len = app.length()-1; + QCStringList apps=registeredApplications(); + for( QCStringList::ConstIterator it = apps.begin(); + it != apps.end(); + ++it) + { + if ( strncmp( (*it).data(), app.data(), len) == 0) + appList.append(*it); + } + } + else + { + appList.append(app); + } + + // We do all the local clients in phase1 and the rest in phase2 + for(int phase=1; phase <= 2; phase++) + { + for( QCStringList::ConstIterator it = appList.begin(); + it != appList.end(); + ++it) + { + QCString remApp = *it; + QCString replyType; + QByteArray replyData; + bool result = false; + DCOPClient *localClient = findLocalClient( remApp ); + + if ( (phase == 1) && localClient ) { + // In phase 1 we do all local clients + bool saveTransaction = d->transaction; + Q_INT32 saveTransactionId = d->transactionId; + QCString saveSenderId = d->senderId; + + d->senderId = 0; // Local call + result = localClient->find( remApp, remObj, remFun, data, replyType, replyData ); + + Q_INT32 id = localClient->transactionId(); + if (id) { + // Call delayed. We have to wait till it has been processed. + do { + QApplication::eventLoop()->processEvents( QEventLoop::WaitForMore); + } while( !localClient->isLocalTransactionFinished(id, replyType, replyData)); + result = true; + } + d->transaction = saveTransaction; + d->transactionId = saveTransactionId; + d->senderId = saveSenderId; + } + else if ((phase == 2) && !localClient) + { + // In phase 2 we do the other clients + result = callInternal(remApp, remObj, remFun, data, + replyType, replyData, useEventLoop, timeout, DCOPFind); + } + + if (result) + { + if (replyType == "DCOPRef") + { + DCOPRef ref; + QDataStream reply( replyData, IO_ReadOnly ); + reply >> ref; + + if (ref.app() == remApp) // Consistency check + { + // replyType contains objId. + foundApp = ref.app(); + foundObj = ref.object(); + return true; + } + } + } + } + } + return false; +} + +bool DCOPClient::process(const QCString &, const QByteArray &, + QCString&, QByteArray &) +{ + return false; +} + +bool DCOPClient::isApplicationRegistered( const QCString& remApp) +{ + QCString replyType; + QByteArray data, replyData; + QDataStream arg( data, IO_WriteOnly ); + arg << remApp; + int result = false; + if ( call( "DCOPServer", "", "isApplicationRegistered(QCString)", data, replyType, replyData ) ) { + QDataStream reply( replyData, IO_ReadOnly ); + reply >> result; + } + return result; +} + +QCStringList DCOPClient::registeredApplications() +{ + QCString replyType; + QByteArray data, replyData; + QCStringList result; + if ( call( "DCOPServer", "", "registeredApplications()", data, replyType, replyData ) ) { + QDataStream reply( replyData, IO_ReadOnly ); + reply >> result; + } + return result; +} + +QCStringList DCOPClient::remoteObjects( const QCString& remApp, bool *ok ) +{ + QCString replyType; + QByteArray data, replyData; + QCStringList result; + if ( ok ) + *ok = false; + if ( call( remApp, "DCOPClient", "objects()", data, replyType, replyData ) ) { + QDataStream reply( replyData, IO_ReadOnly ); + reply >> result; + if ( ok ) + *ok = true; + } + return result; +} + +QCStringList DCOPClient::remoteInterfaces( const QCString& remApp, const QCString& remObj, bool *ok ) +{ + QCString replyType; + QByteArray data, replyData; + QCStringList result; + if ( ok ) + *ok = false; + if ( call( remApp, remObj, "interfaces()", data, replyType, replyData ) && replyType == "QCStringList") { + QDataStream reply( replyData, IO_ReadOnly ); + reply >> result; + if ( ok ) + *ok = true; + } + return result; +} + +QCStringList DCOPClient::remoteFunctions( const QCString& remApp, const QCString& remObj, bool *ok ) +{ + QCString replyType; + QByteArray data, replyData; + QCStringList result; + if ( ok ) + *ok = false; + if ( call( remApp, remObj, "functions()", data, replyType, replyData ) && replyType == "QCStringList") { + QDataStream reply( replyData, IO_ReadOnly ); + reply >> result; + if ( ok ) + *ok = true; + } + return result; +} + +void DCOPClient::setNotifications(bool enabled) +{ + QByteArray data; + QDataStream ds(data, IO_WriteOnly); + ds << static_cast<Q_INT8>(enabled); + + QCString replyType; + QByteArray reply; + if (!call("DCOPServer", "", "setNotifications( bool )", data, replyType, reply)) + qWarning("I couldn't enable notifications at the dcopserver!"); +} + +void DCOPClient::setDaemonMode( bool daemonMode ) +{ + QByteArray data; + QDataStream ds(data, IO_WriteOnly); + ds << static_cast<Q_INT8>( daemonMode ); + + QCString replyType; + QByteArray reply; + if (!call("DCOPServer", "", "setDaemonMode(bool)", data, replyType, reply)) + qWarning("I couldn't enable daemon mode at the dcopserver!"); +} + + + +/* + DCOP <-> Qt bridge + + ******************************************************************************** + */ +static void fillQtObjects( QCStringList& l, QObject* o, QCString path ) +{ + if ( !path.isEmpty() ) + path += '/'; + + int unnamed = 0; + const QObjectList *list = o ? o->children() : QObject::objectTrees(); + if ( list ) { + QObjectListIt it( *list ); + QObject *obj; + while ( (obj=it.current()) ) { + ++it; + QCString n = obj->name(); + if ( n == "unnamed" || n.isEmpty() ) + { + n.sprintf("%p", (void *) obj); + n = QString("unnamed%1(%2, %3)").arg(++unnamed).arg(obj->className()).arg(n).latin1(); + } + QCString fn = path + n; + l.append( fn ); + if ( obj->children() ) + fillQtObjects( l, obj, fn ); + } + } +} + +namespace +{ +struct O +{ + O(): o(0) {} + O ( const QCString& str, QObject* obj ):s(str), o(obj){} + QCString s; + QObject* o; +}; +} // namespace + +static void fillQtObjectsEx( QValueList<O>& l, QObject* o, QCString path ) +{ + if ( !path.isEmpty() ) + path += '/'; + + int unnamed = 0; + const QObjectList *list = o ? o->children() : QObject::objectTrees(); + if ( list ) { + QObjectListIt it( *list ); + QObject *obj; + while ( (obj=it.current()) ) { + ++it; + QCString n = obj->name(); + if ( n == "unnamed" || n.isEmpty() ) + { + n.sprintf("%p", (void *) obj); + n = QString("unnamed%1(%2, %3)").arg(++unnamed).arg(obj->className()).arg(n).latin1(); + } + QCString fn = path + n; + l.append( O( fn, obj ) ); + if ( obj->children() ) + fillQtObjectsEx( l, obj, fn ); + } + } +} + + +static QObject* findQtObject( QCString id ) +{ + QRegExp expr( id ); + QValueList<O> l; + fillQtObjectsEx( l, 0, "qt" ); + // Prefer an exact match, but fall-back on the first that contains the substring + QObject* firstContains = 0L; + for ( QValueList<O>::ConstIterator it = l.begin(); it != l.end(); ++it ) { + if ( (*it).s == id ) // exact match + return (*it).o; + if ( !firstContains && (*it).s.contains( expr ) ) { + firstContains = (*it).o; + } + } + return firstContains; +} + +static QCStringList findQtObjects( QCString id ) +{ + QRegExp expr( id ); + QValueList<O> l; + fillQtObjectsEx( l, 0, "qt" ); + QCStringList result; + for ( QValueList<O>::ConstIterator it = l.begin(); it != l.end(); ++it ) { + if ( (*it).s.contains( expr ) ) + result << (*it).s; + } + return result; +} + +static bool receiveQtObject( const QCString &objId, const QCString &fun, const QByteArray &data, + QCString& replyType, QByteArray &replyData) +{ + if ( objId == "qt" ) { + if ( fun == "interfaces()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + l << "DCOPObject"; + l << "Qt"; + reply << l; + return true; + } else if ( fun == "functions()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + l << "QCStringList functions()"; + l << "QCStringList interfaces()"; + l << "QCStringList objects()"; + l << "QCStringList find(QCString)"; + reply << l; + return true; + } else if ( fun == "objects()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + fillQtObjects( l, 0, "qt" ); + reply << l; + return true; + } else if ( fun == "find(QCString)" ) { + QDataStream ds( data, IO_ReadOnly ); + QCString id; + ds >> id ; + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + reply << findQtObjects( id ) ; + return true; + } + } else if ( objId.left(3) == "qt/" ) { + QObject* o = findQtObject( objId ); + if ( !o ) + return false; + if ( fun == "functions()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + l << "QCStringList functions()"; + l << "QCStringList interfaces()"; + l << "QCStringList properties()"; + l << "bool setProperty(QCString,QVariant)"; + l << "QVariant property(QCString)"; + QStrList lst = o->metaObject()->slotNames( true ); + int i = 0; + for ( QPtrListIterator<char> it( lst ); it.current(); ++it ) { + if ( o->metaObject()->slot( i++, true )->access != QMetaData::Public ) + continue; + QCString slot = it.current(); + if ( slot.contains( "()" ) ) { + slot.prepend("void "); + l << slot; + } + } + reply << l; + return true; + } else if ( fun == "interfaces()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + QMetaObject *meta = o->metaObject(); + while ( meta ) { + l.prepend( meta->className() ); + meta = meta->superClass(); + } + reply << l; + return true; + } else if ( fun == "properties()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + QStrList lst = o->metaObject()->propertyNames( true ); + for ( QPtrListIterator<char> it( lst ); it.current(); ++it ) { + QMetaObject *mo = o->metaObject(); + const QMetaProperty* p = mo->property( mo->findProperty( it.current(), true ), true ); + if ( !p ) + continue; + QCString prop = p->type(); + prop += ' '; + prop += p->name(); + if ( !p->writable() ) + prop += " readonly"; + l << prop; + } + reply << l; + return true; + } else if ( fun == "property(QCString)" ) { + replyType = "QVariant"; + QDataStream ds( data, IO_ReadOnly ); + QCString name; + ds >> name ; + QVariant result = o->property( name ); + QDataStream reply( replyData, IO_WriteOnly ); + reply << result; + return true; + } else if ( fun == "setProperty(QCString,QVariant)" ) { + QDataStream ds( data, IO_ReadOnly ); + QCString name; + QVariant value; + ds >> name >> value; + replyType = "bool"; + QDataStream reply( replyData, IO_WriteOnly ); + reply << (Q_INT8) o->setProperty( name, value ); + return true; + } else { + int slot = o->metaObject()->findSlot( fun, true ); + if ( slot != -1 ) { + replyType = "void"; + QUObject uo[ 1 ]; + o->qt_invoke( slot, uo ); + return true; + } + } + + + } + return false; +} + + +/* + ******************************************************************************** + End of DCOP <-> Qt bridge + */ + + +bool DCOPClient::receive(const QCString &/*app*/, const QCString &objId, + const QCString &fun, const QByteArray &data, + QCString& replyType, QByteArray &replyData) +{ + d->transaction = false; // Assume no transaction. + if ( objId == "DCOPClient" ) { + if ( fun == "objects()" ) { + replyType = "QCStringList"; + QDataStream reply( replyData, IO_WriteOnly ); + QCStringList l; + if (d->qt_bridge_enabled) + { + l << "qt"; // the Qt bridge object + } + if ( kde_dcopObjMap ) { + QMap<QCString, DCOPObject *>::ConstIterator it( kde_dcopObjMap->begin()); + for (; it != kde_dcopObjMap->end(); ++it) { + if ( !it.key().isEmpty() ) { + if ( it.key() == d->defaultObject ) + l << "default"; + l << it.key(); + } + } + } + reply << l; + return true; + } + } + + if ( objId.isEmpty() || objId == "DCOPClient" ) { + if ( fun == "applicationRegistered(QCString)" ) { + QDataStream ds( data, IO_ReadOnly ); + QCString r; + ds >> r; + emit applicationRegistered( r ); + return true; + } else if ( fun == "applicationRemoved(QCString)" ) { + QDataStream ds( data, IO_ReadOnly ); + QCString r; + ds >> r; + emit applicationRemoved( r ); + return true; + } + + if ( process( fun, data, replyType, replyData ) ) + return true; + // fall through and send to defaultObject if available + + } else if (d->qt_bridge_enabled && + (objId == "qt" || objId.left(3) == "qt/") ) { // dcop <-> qt bridge + return receiveQtObject( objId, fun, data, replyType, replyData ); + } + + if ( objId.isEmpty() || objId == "default" ) { + if ( !d->defaultObject.isEmpty() && DCOPObject::hasObject( d->defaultObject ) ) { + DCOPObject *objPtr = DCOPObject::find( d->defaultObject ); + objPtr->setCallingDcopClient(this); + if (objPtr->process(fun, data, replyType, replyData)) + return true; + } + + // fall through and send to object proxies + } + + if (!objId.isEmpty() && objId[objId.length()-1] == '*') { + // handle a multicast to several objects. + // doesn't handle proxies currently. should it? + QPtrList<DCOPObject> matchList = + DCOPObject::match(objId.left(objId.length()-1)); + for (DCOPObject *objPtr = matchList.first(); + objPtr != 0L; objPtr = matchList.next()) { + objPtr->setCallingDcopClient(this); + if (!objPtr->process(fun, data, replyType, replyData)) + return false; + } + return true; + } else if (!DCOPObject::hasObject(objId)) { + if ( DCOPObjectProxy::proxies ) { + for ( QPtrListIterator<DCOPObjectProxy> it( *DCOPObjectProxy::proxies ); it.current(); ++it ) { + // TODO: it.current()->setCallingDcopClient(this); + if ( it.current()->process( objId, fun, data, replyType, replyData ) ) + return true; + } + } + return false; + + } else { + DCOPObject *objPtr = DCOPObject::find(objId); + objPtr->setCallingDcopClient(this); + if (!objPtr->process(fun, data, replyType, replyData)) { + // obj doesn't understand function or some other error. + return false; + } + } + + return true; +} + +// Check if the function result is a bool with the value "true" +// If so set the function result to DCOPRef pointing to (app,objId) and +// return true. Return false otherwise. +static bool findResultOk(QCString &replyType, QByteArray &replyData) +{ + Q_INT8 success; // Tsk.. why is there no operator>>(bool)? + if (replyType != "bool") return false; + + QDataStream reply( replyData, IO_ReadOnly ); + reply >> success; + + if (!success) return false; + return true; +} + +// set the function result to DCOPRef pointing to (app,objId) and +// return true. +static bool findSuccess(const QCString &app, const QCString objId, QCString &replyType, QByteArray &replyData) +{ + DCOPRef ref(app, objId); + replyType = "DCOPRef"; + + replyData = QByteArray(); + QDataStream final_reply( replyData, IO_WriteOnly ); + final_reply << ref; + return true; +} + + +bool DCOPClient::find(const QCString &app, const QCString &objId, + const QCString &fun, const QByteArray &data, + QCString& replyType, QByteArray &replyData) +{ + d->transaction = false; // Transactions are not allowed. + if ( !app.isEmpty() && app != d->appId && app[app.length()-1] != '*') { + qWarning("WEIRD! we somehow received a DCOP message w/a different appId"); + return false; + } + + if (objId.isEmpty() || objId[objId.length()-1] != '*') + { + if (fun.isEmpty()) + { + if (objId.isEmpty() || DCOPObject::hasObject(objId)) + return findSuccess(app, objId, replyType, replyData); + return false; + } + // Message to application or single object... + if (receive(app, objId, fun, data, replyType, replyData)) + { + if (findResultOk(replyType, replyData)) + return findSuccess(app, objId, replyType, replyData); + } + } + else { + // handle a multicast to several objects. + // doesn't handle proxies currently. should it? + QPtrList<DCOPObject> matchList = + DCOPObject::match(objId.left(objId.length()-1)); + for (DCOPObject *objPtr = matchList.first(); + objPtr != 0L; objPtr = matchList.next()) + { + replyType = 0; + replyData = QByteArray(); + if (fun.isEmpty()) + return findSuccess(app, objPtr->objId(), replyType, replyData); + objPtr->setCallingDcopClient(this); + if (objPtr->process(fun, data, replyType, replyData)) + if (findResultOk(replyType, replyData)) + return findSuccess(app, objPtr->objId(), replyType, replyData); + } + } + return false; +} + + +bool DCOPClient::call(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QByteArray &data, + QCString& replyType, QByteArray &replyData, + bool useEventLoop) +{ + return call( remApp, remObjId, remFun, data, replyType, replyData, useEventLoop, -1 ); +} + +bool DCOPClient::call(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QByteArray &data, + QCString& replyType, QByteArray &replyData, + bool useEventLoop, int timeout) +{ + if (remApp.isEmpty()) + return false; + DCOPClient *localClient = findLocalClient( remApp ); + + if ( localClient ) { + bool saveTransaction = d->transaction; + Q_INT32 saveTransactionId = d->transactionId; + QCString saveSenderId = d->senderId; + + d->senderId = 0; // Local call + bool b = localClient->receive( remApp, remObjId, remFun, data, replyType, replyData ); + + Q_INT32 id = localClient->transactionId(); + if (id) { + // Call delayed. We have to wait till it has been processed. + do { + QApplication::eventLoop()->processEvents( QEventLoop::WaitForMore); + } while( !localClient->isLocalTransactionFinished(id, replyType, replyData)); + b = true; + } + d->transaction = saveTransaction; + d->transactionId = saveTransactionId; + d->senderId = saveSenderId; + return b; + } + + return callInternal(remApp, remObjId, remFun, data, + replyType, replyData, useEventLoop, timeout, DCOPCall); +} + +void DCOPClient::asyncReplyReady() +{ + while( d->asyncReplyQueue.count() ) + { + ReplyStruct *replyStruct = d->asyncReplyQueue.take(0); + handleAsyncReply(replyStruct); + } +} + +int DCOPClient::callAsync(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QByteArray &data, + QObject *callBackObj, const char *callBackSlot) +{ + QCString replyType; + QByteArray replyData; + + ReplyStruct *replyStruct = new ReplyStruct; + replyStruct->replyType = new QCString; + replyStruct->replyData = new QByteArray; + replyStruct->replyObject = callBackObj; + replyStruct->replySlot = callBackSlot; + replyStruct->replyId = ++d->transactionId; + if (d->transactionId < 0) // Ensure that ids > 0 + d->transactionId = 0; + + bool b = callInternal(remApp, remObjId, remFun, data, + replyStruct, false, -1, DCOPCall); + if (!b) + { + delete replyStruct->replyType; + delete replyStruct->replyData; + delete replyStruct; + return 0; + } + + if (replyStruct->transactionId == 0) + { + // Call is finished already + QTimer::singleShot(0, this, SLOT(asyncReplyReady())); + d->asyncReplyQueue.append(replyStruct); + } + + return replyStruct->replyId; +} + +bool DCOPClient::callInternal(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QByteArray &data, + QCString& replyType, QByteArray &replyData, + bool useEventLoop, int timeout, int minor_opcode) +{ + ReplyStruct replyStruct; + replyStruct.replyType = &replyType; + replyStruct.replyData = &replyData; + return callInternal(remApp, remObjId, remFun, data, &replyStruct, useEventLoop, timeout, minor_opcode); +} + +bool DCOPClient::callInternal(const QCString &remApp, const QCString &remObjId, + const QCString &remFun, const QByteArray &data, + ReplyStruct *replyStruct, + bool useEventLoop, int timeout, int minor_opcode) +{ + if ( !isAttached() ) + return false; + + DCOPMsg *pMsg; + + CARD32 oldCurrentKey = d->currentKey; + if ( !d->currentKey ) + d->currentKey = d->key; // no key yet, initiate new call + + QByteArray ba; + QDataStream ds(ba, IO_WriteOnly); + ds << d->appId << remApp << remObjId << normalizeFunctionSignature(remFun) << data.size(); + + IceGetHeader(d->iceConn, d->majorOpcode, minor_opcode, + sizeof(DCOPMsg), DCOPMsg, pMsg); + + pMsg->key = d->currentKey; + int datalen = ba.size() + data.size(); + pMsg->length += datalen; + +// qWarning("DCOP: %s made call %s:%s:%s key = %d", d->appId.data(), remApp.data(), remObjId.data(), remFun.data(), pMsg->key); + + IceSendData(d->iceConn, ba.size(), const_cast<char *>(ba.data())); + IceSendData(d->iceConn, data.size(), const_cast<char *>(data.data())); + + if (IceConnectionStatus(d->iceConn) != IceConnectAccepted) + return false; + + IceFlush (d->iceConn); + + IceReplyWaitInfo waitInfo; + waitInfo.sequence_of_request = IceLastSentSequenceNumber(d->iceConn); + waitInfo.major_opcode_of_request = d->majorOpcode; + waitInfo.minor_opcode_of_request = minor_opcode; + + replyStruct->transactionId = -1; + waitInfo.reply = static_cast<IcePointer>(replyStruct); + + Bool readyRet = False; + IceProcessMessagesStatus s; + + timeval time_start; + int time_left = -1; + if( timeout >= 0 ) + { + gettimeofday( &time_start, NULL ); + time_left = timeout; + } + for(;;) { + bool checkMessages = true; + if ( useEventLoop + ? d->notifier != NULL // useEventLoop needs a socket notifier and a qApp + : timeout >= 0 ) { // !useEventLoop doesn't block only for timeout >= 0 + const int guiTimeout = 100; + checkMessages = false; + + int msecs = useEventLoop + ? guiTimeout // timeout for the GUI refresh + : time_left; // time remaining for the whole call + fd_set fds; + struct timeval tv; + FD_ZERO( &fds ); + FD_SET( socket(), &fds ); + tv.tv_sec = msecs / 1000; + tv.tv_usec = (msecs % 1000) * 1000; + if ( select( socket() + 1, &fds, 0, 0, &tv ) <= 0 ) { + if( useEventLoop && (timeout < 0 || time_left > guiTimeout)) { + // nothing was available, we got a timeout. Reactivate + // the GUI in blocked state. + bool old_lock = d->non_blocking_call_lock; + if ( !old_lock ) { + d->non_blocking_call_lock = true; + emit blockUserInput( true ); + } + if( timeout >= 0 ) + d->eventLoopTimer.start(time_left - guiTimeout, true); + qApp->enter_loop(); + d->eventLoopTimer.stop(); + if ( !old_lock ) { + d->non_blocking_call_lock = false; + emit blockUserInput( false ); + } + } + } + else + { + checkMessages = true; + } + } + if (!d->iceConn) + return false; + + if( replyStruct->transactionId != -1 ) + { + if (replyStruct->transactionId == 0) + break; // Call complete + if (!replyStruct->replySlot.isEmpty()) + break; // Async call + } + + if( checkMessages ) { // something is available + s = IceProcessMessages(d->iceConn, &waitInfo, + &readyRet); + if (s == IceProcessMessagesIOError) { + detach(); + d->currentKey = oldCurrentKey; + return false; + } + } + + if( replyStruct->transactionId != -1 ) + { + if (replyStruct->transactionId == 0) + break; // Call complete + if (!replyStruct->replySlot.isEmpty()) + break; // Async call + } + + if( timeout < 0 ) + continue; + timeval time_now; + gettimeofday( &time_now, NULL ); + time_left = timeout - + ((time_now.tv_sec - time_start.tv_sec) * 1000) - + ((time_now.tv_usec - time_start.tv_usec) / 1000); + if( time_left <= 0) + { + if (useEventLoop) + { + // Before we fail, check one more time if something is available + time_left = 0; + useEventLoop = false; + continue; + } + *(replyStruct->replyType) = QCString(); + *(replyStruct->replyData) = QByteArray(); + replyStruct->status = ReplyStruct::Failed; + break; + } + } + + // Wake up parent call, maybe it's reply is available already. + if ( d->non_blocking_call_lock ) { + qApp->exit_loop(); + } + + d->currentKey = oldCurrentKey; + return replyStruct->status != ReplyStruct::Failed; +} + +void DCOPClient::eventLoopTimeout() +{ + qApp->exit_loop(); +} + +void DCOPClient::processSocketData(int fd) +{ + // Make sure there is data to read! + fd_set fds; + timeval timeout; + timeout.tv_sec = 0; + timeout.tv_usec = 0; + FD_ZERO(&fds); + FD_SET(fd, &fds); + int result = select(fd+1, &fds, 0, 0, &timeout); + if (result == 0) + return; + + if ( d->non_blocking_call_lock ) { + if( qApp ) + qApp->exit_loop(); + return; + } + + if (!d->iceConn) { + if( d->notifier ) + d->notifier->deleteLater(); + d->notifier = 0; + qWarning("received an error processing data from the DCOP server!"); + return; + } + + IceProcessMessagesStatus s = IceProcessMessages(d->iceConn, 0, 0); + + if (s == IceProcessMessagesIOError) { + detach(); + qWarning("received an error processing data from the DCOP server!"); + return; + } +} + +void DCOPClient::setDefaultObject( const QCString& objId ) +{ + d->defaultObject = objId; +} + + +QCString DCOPClient::defaultObject() const +{ + return d->defaultObject; +} + +bool +DCOPClient::isLocalTransactionFinished(Q_INT32 id, QCString &replyType, QByteArray &replyData) +{ + DCOPClientPrivate::LocalTransactionResult *result = d->localTransActionList.take(id); + if (!result) + return false; + + replyType = result->replyType; + replyData = result->replyData; + delete result; + + return true; +} + +DCOPClientTransaction * +DCOPClient::beginTransaction() +{ + if (d->opcode == DCOPSend) + return 0; + if (!d->transactionList) + d->transactionList = new QPtrList<DCOPClientTransaction>; + + d->transaction = true; + DCOPClientTransaction *trans = new DCOPClientTransaction(); + trans->senderId = d->senderId; + trans->id = ++d->transactionId; + if (d->transactionId < 0) // Ensure that ids > 0 + d->transactionId = 0; + trans->key = d->currentKey; + + d->transactionList->append( trans ); + + return trans; +} + +Q_INT32 +DCOPClient::transactionId() const +{ + if (d->transaction) + return d->transactionId; + else + return 0; +} + +void +DCOPClient::endTransaction( DCOPClientTransaction *trans, QCString& replyType, + QByteArray &replyData) +{ + if ( !trans ) + return; + + if ( !isAttached() ) + return; + + if ( !d->transactionList) { + qWarning("Transaction unknown: No pending transactions!"); + return; // No pending transactions! + } + + if ( !d->transactionList->removeRef( trans ) ) { + qWarning("Transaction unknown: Not on list of pending transactions!"); + return; // Transaction + } + + if (trans->senderId.isEmpty()) + { + // Local transaction + DCOPClientPrivate::LocalTransactionResult *result = new DCOPClientPrivate::LocalTransactionResult(); + result->replyType = replyType; + result->replyData = replyData; + + d->localTransActionList.insert(trans->id, result); + + delete trans; + + return; + } + + DCOPMsg *pMsg; + + QByteArray ba; + QDataStream ds(ba, IO_WriteOnly); + ds << d->appId << trans->senderId << trans->id << replyType << replyData; + + IceGetHeader(d->iceConn, d->majorOpcode, DCOPReplyDelayed, + sizeof(DCOPMsg), DCOPMsg, pMsg); + pMsg->key = trans->key; + pMsg->length += ba.size(); + + IceSendData( d->iceConn, ba.size(), const_cast<char *>(ba.data()) ); + + delete trans; +} + +void +DCOPClient::emitDCOPSignal( const QCString &object, const QCString &signal, const QByteArray &data) +{ + // We hack the sending object name into the signal name + send("DCOPServer", "emit", object+"#"+normalizeFunctionSignature(signal), data); +} + +void +DCOPClient::emitDCOPSignal( const QCString &signal, const QByteArray &data) +{ + emitDCOPSignal(0, signal, data); +} + +bool +DCOPClient::connectDCOPSignal( const QCString &sender, const QCString &senderObj, + const QCString &signal, + const QCString &receiverObj, const QCString &slot, bool Volatile) +{ + QCString replyType; + QByteArray data, replyData; + Q_INT8 iVolatile = Volatile ? 1 : 0; + + QDataStream args(data, IO_WriteOnly ); + args << sender << senderObj << normalizeFunctionSignature(signal) << receiverObj << normalizeFunctionSignature(slot) << iVolatile; + + if (!call("DCOPServer", 0, + "connectSignal(QCString,QCString,QCString,QCString,QCString,bool)", + data, replyType, replyData)) + { + return false; + } + + if (replyType != "bool") + return false; + + QDataStream reply(replyData, IO_ReadOnly ); + Q_INT8 result; + reply >> result; + return (result != 0); +} + +bool +DCOPClient::connectDCOPSignal( const QCString &sender, const QCString &signal, + const QCString &receiverObj, const QCString &slot, bool Volatile) +{ + return connectDCOPSignal( sender, 0, signal, receiverObj, slot, Volatile); +} + +bool +DCOPClient::disconnectDCOPSignal( const QCString &sender, const QCString &senderObj, + const QCString &signal, + const QCString &receiverObj, const QCString &slot) +{ + QCString replyType; + QByteArray data, replyData; + + QDataStream args(data, IO_WriteOnly ); + args << sender << senderObj << normalizeFunctionSignature(signal) << receiverObj << normalizeFunctionSignature(slot); + + if (!call("DCOPServer", 0, + "disconnectSignal(QCString,QCString,QCString,QCString,QCString)", + data, replyType, replyData)) + { + return false; + } + + if (replyType != "bool") + return false; + + QDataStream reply(replyData, IO_ReadOnly ); + Q_INT8 result; + reply >> result; + return (result != 0); +} + +bool +DCOPClient::disconnectDCOPSignal( const QCString &sender, const QCString &signal, + const QCString &receiverObj, const QCString &slot) +{ + return disconnectDCOPSignal( sender, 0, signal, receiverObj, slot); +} + +void +DCOPClient::setPriorityCall(bool b) +{ + if (b) + { + if (d->currentKey == 2) + return; + d->currentKeySaved = d->currentKey; + d->currentKey = 2; + } + else + { + if (d->currentKey != 2) + return; + d->currentKey = d->currentKeySaved; + if ( !d->messages.isEmpty() ) + d->postMessageTimer.start( 0, true ); // Process queued messages + } +} + + + +void +DCOPClient::emergencyClose() +{ + QPtrList<DCOPClient> list; + client_map_t *map = DCOPClient_CliMap; + if (!map) return; + QAsciiDictIterator<DCOPClient> it(*map); + while(it.current()) { + list.removeRef(it.current()); + list.append(it.current()); + ++it; + } + for(DCOPClient *cl = list.first(); cl; cl = list.next()) + { + if (cl->d->iceConn) { + IceProtocolShutdown(cl->d->iceConn, cl->d->majorOpcode); + IceCloseConnection(cl->d->iceConn); + cl->d->iceConn = 0L; + } + } +} + +const char * +DCOPClient::postMortemSender() +{ + if (!dcop_main_client) + return ""; + if (dcop_main_client->d->senderId.isEmpty()) + return ""; + return dcop_main_client->d->senderId.data(); +} + +const char * +DCOPClient::postMortemObject() +{ + if (!dcop_main_client) + return ""; + return dcop_main_client->d->objId.data(); +} +const char * +DCOPClient::postMortemFunction() +{ + if (!dcop_main_client) + return ""; + return dcop_main_client->d->function.data(); +} + +void DCOPClient::virtual_hook( int, void* ) +{ /*BASE::virtual_hook( id, data );*/ } + +#include <dcopclient.moc> + |