gpt4 book ai didi

linux - Qt 5.6 使用 QLocalServer 和 QLocalSocket 传递文件描述符

转载 作者:太空宇宙 更新时间:2023-11-04 10:35:43 26 4
gpt4 key购买 nike

我正在学习如何在 Ubuntu 14.04 上使用带有 Qt 5.6 的 Unix 域套接字传递文件描述符。从文档中可以看出,执行此操作的方法是使用 QLocalServer 和 QLocalSocket 类。

我创建了 2 个简单的应用程序,1 个服务器和 1 个客户端。他们每个人只有 1 个类(class),我在下面粘贴了整个类(class)。

我可以接收整数形式的“文件描述符”,但是当我尝试以追加模式打开它以便在客户端进程中写入时,出现以下错误:

Cannot open existing file handle:  "Unknown error"
QIODevice::write: device not open
FD Received: 20

我做错了什么?

客户:

#include <QtWidgets>
#include <QtNetwork>

#include "localclient.h"

Client::Client(QObject *parent)
{

socket = new QLocalSocket(this);
connect(socket, SIGNAL(readyRead()), this, SLOT(readFd()));
connect(socket, SIGNAL(error(QLocalSocket::LocalSocketError)), this, SLOT(displayError(QLocalSocket::LocalSocketError)));

requestFd();

}

void Client::requestFd()
{

socket->abort();
socket->connectToServer("mysocket");

}

void Client::readFd()
{

QDataStream in(socket);
in.setVersion(QDataStream::Qt_4_0);

if (socket->bytesAvailable() < (int)sizeof(quint16))
return;

if (in.atEnd())
return;

int nextFd;
in >> nextFd;

QFile rxFile;
if ( !rxFile.open(nextFd,QIODevice::Append) ) {
qDebug() << "Cannot open existing file handle: " << rxFile.errorString();
}
rxFile.write("hello");

qDebug() << "FD Received: " << nextFd;

}

void Client::displayError(QLocalSocket::LocalSocketError socketError)
{
switch (socketError) {
case QLocalSocket::ServerNotFoundError:
tr("The host was not found. Please check the "
"host name and port settings.");
break;
case QLocalSocket::ConnectionRefusedError:
tr("The connection was refused by the peer. "
"Make sure the fortune server is running, "
"and check that the host name and port "
"settings are correct.");
break;
case QLocalSocket::PeerClosedError:
break;
default:
tr("The following error occurred: %1.").arg(socket->errorString());
}

}

服务器:

#include <QtWidgets>
#include <QtNetwork>

#include <stdlib.h>

#include "localserver.h"
#include <qlocalserver.h>
#include <qlocalsocket.h>

Server::Server(QObject *parent)
{

server = new QLocalServer(this);
if (!server->listen("mysocket")) {
qDebug() << QString("Unable to start the server: %1.").arg(server->errorString());
return;
}

qDebug() << tr("The server is running");

connect(server, SIGNAL(newConnection()), this, SLOT(sendFd()));

fileToSend = new QFile("/home/me/Desktop/test.bin");
if ( !fileToSend->open(QIODevice::WriteOnly) ) {
qDebug() << "Unable to open file to send";
}

}

void Server::sendFd()
{

QByteArray block;
QDataStream out(&block, QIODevice::WriteOnly);
out.setVersion(QDataStream::Qt_4_0);
out.device()->seek(0);
qDebug() << "Sending this file handle: " << fileToSend->handle();
out << fileToSend->handle();

QLocalSocket *clientConnection = server->nextPendingConnection();
connect(clientConnection, SIGNAL(disconnected()),
clientConnection, SLOT(deleteLater()));

clientConnection->write(block);
clientConnection->flush();
clientConnection->disconnectFromServer();

}

最佳答案

在 Linux 上,使用 Unix 域套接字发送文件描述符是使用 SCM_RIGHTS 类型的辅助数据完成的。 Qt 似乎不支持使用 QLocalSocketQLocalServer 的此类数据,有一个旧的 QTBUG谈论问题。而且好像还没有解决。

您可以使用 Qt D-Bus您可以在其中使用 QDBusUnixFileDescriptor 发送文件描述符类。或者恐怕您可能必须使用 Unix 域套接字来自己实现,有一个很好的例子 here .

但是您确定您真的需要在应用程序之间传递文件描述符吗?对于学习以外的目的,我认为总有办法克服这个限制。

附言您正在使用 QDataStream 序列化套接字文件描述符(就好像它是一个普通的 int),然后将序列化数据发送给其他进程以将其作为普通 读取>整数。这将允许您读取文件描述符的整数值,但您不会允许您与其进行任何交互,因为文件描述符值在它们所属的进程之外没有意义。

关于linux - Qt 5.6 使用 QLocalServer 和 QLocalSocket 传递文件描述符,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37648466/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com