mumble-voip_mumble/src/tests/Benchmark.cpp

415 lines
10 KiB
C++

// Copyright 2007-2023 The Mumble Developers. All rights reserved.
// Use of this source code is governed by a BSD-style license
// that can be found in the LICENSE file at the root of the
// Mumble source tree or at <https://www.mumble.info/LICENSE>.
/**
* Provided a target address spawns a specified number of senders/speakers,
* UDP-listeners and TCP-listeners.
*/
#include <QtCore>
#include <QtNetwork>
#ifndef Q_OS_WIN
# include <unistd.h>
#endif
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#ifndef Q_OS_WIN
# include <netinet/in.h>
# include <netinet/ip.h>
# include <sys/socket.h>
# include <sys/utsname.h>
#endif
#include <errno.h>
#include "Message.h"
#include "Mumble.pb.h"
#include "PacketDataStream.h"
#include "Timer.h"
#include "crypto/CryptState.h"
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
# include <QRandomGenerator>
#endif
class Client : public QThread {
Q_OBJECT
public:
bool udp;
bool sender;
struct sockaddr_in srv;
unsigned int uiSession;
CryptStateOCB2 crypt;
int rcvd;
int socket;
int seq;
void run();
void ping();
void sendVoice();
int numbytes;
int ptype;
QSslSocket *ssl;
Client(QObject *parent, QHostAddress srvaddr, unsigned short prt, bool send, bool tcponly);
void doUdp(const unsigned char *buffer, int size);
void sendMessage(const ::google::protobuf::Message &msg, unsigned int msgType);
~Client();
public slots:
void readyRead();
void disconnected();
};
Client::Client(QObject *p, QHostAddress qha, unsigned short prt, bool send, bool tcponly) : QThread(p) {
srv.sin_family = AF_INET;
srv.sin_addr.s_addr = htonl(qha.toIPv4Address());
srv.sin_port = htons(prt);
udp = !tcponly;
sender = send;
ssl = new QSslSocket(this);
connect(ssl, SIGNAL(readyRead()), this, SLOT(readyRead()));
connect(ssl, SIGNAL(disconnected()), this, SLOT(disconnected()));
ssl->setProtocol(QSsl::TlsV1);
ssl->connectToHostEncrypted(qha.toString(), prt);
ssl->ignoreSslErrors();
if (!ssl->waitForEncrypted())
qFatal("Connection failure");
static int ctr = 1;
#ifdef Q_OS_WIN
DWORD pid = GetCurrentProcessId();
wchar_t buf[64];
DWORD bufsize = 64;
GetComputerName(buf, &bufsize);
QString name = QString("%1.%2").arg(QString::fromWCharArray(buf)).arg(pid * 1000 + ctr);
#else
struct utsname uts;
uname(&uts);
QString name = QString("%1.%2").arg(uts.nodename).arg(getpid() * 1000 + ctr);
#endif
ctr++;
MumbleProto::Version mpv;
mpv.set_release(u8(QLatin1String("1.2.1 Benchmark")));
mpv.set_version_v1(Version::toLegacyVersion(Version::fromComponents(1, 2, 3)));
sendMessage(mpv, MessageHandler::Version);
MumbleProto::Authenticate mpa;
mpa.set_username(u8(name));
sendMessage(mpa, MessageHandler::Authenticate);
if (udp)
socket = ::socket(PF_INET, SOCK_DGRAM, 0);
seq = 0;
rcvd = 0;
numbytes = -1;
}
Client::~Client() {
terminate();
wait();
}
void Client::sendMessage(const ::google::protobuf::Message &msg, unsigned int msgType) {
unsigned char uc[4096];
int len = msg.ByteSize();
Q_ASSERT(len < 4090);
*reinterpret_cast< quint16 * >(&uc[0]) = qToBigEndian(static_cast< quint16 >(msgType));
*reinterpret_cast< quint32 * >(&uc[2]) = qToBigEndian(static_cast< quint32 >(len));
msg.SerializeToArray(uc + 6, len);
ssl->write(reinterpret_cast< const char * >(uc), len + 6);
}
void Client::ping() {
unsigned char buffer[256];
buffer[0] = MessageHandler::UDPPing << 5;
PacketDataStream pds(buffer + 1, 255);
pds << 123;
doUdp(buffer, pds.size() + 1);
MumbleProto::Ping mpp;
mpp.set_timestamp(123);
sendMessage(mpp, MessageHandler::Ping);
}
void Client::sendVoice() {
unsigned char buffer[1024];
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
int len = 32 + (QRandomGenerator::global()->generate() & 0x3f);
#else
// Qt 5.10 introduces the QRandomGenerator class and in Qt 5.15 qrand got deprecated in its favor
int len = 32 + (qrand() & 0x3f);
#endif
// Regular voice, nothing special
buffer[0] = 0;
PacketDataStream ods(buffer + 1, 1024);
ods << 1;
ods.append(len);
ods.skip(len);
doUdp(buffer, ods.size() + 1);
}
void Client::doUdp(const unsigned char *buffer, int size) {
if (!udp || !crypt.isValid())
return;
unsigned char crypted[2048];
crypt.encrypt(reinterpret_cast< const unsigned char * >(buffer), crypted, size);
::sendto(socket, reinterpret_cast< const char * >(crypted), size + 4, 0,
reinterpret_cast< struct sockaddr * >(&srv), sizeof(srv));
}
void Client::run() {
unsigned char buffer[1024];
struct sockaddr_in addr;
socklen_t sz;
int len;
if (!udp)
return;
forever {
sz = sizeof(addr);
len = ::recvfrom(socket, reinterpret_cast< char * >(buffer), 1024, 0,
reinterpret_cast< struct sockaddr * >(&addr), &sz);
if (len <= 0)
break;
if (len >= 32)
rcvd++;
}
}
void Client::readyRead() {
forever {
int avail = ssl->bytesAvailable();
if (numbytes == -1) {
if (avail < 6)
break;
unsigned char b[6];
ssl->read(reinterpret_cast< char * >(b), 6);
ptype = qFromBigEndian(*reinterpret_cast< quint16 * >(&b[0]));
numbytes = qFromBigEndian(*reinterpret_cast< quint32 * >(&b[2]));
avail = ssl->bytesAvailable();
}
if ((numbytes >= 0) && (avail >= numbytes)) {
int want = numbytes;
numbytes = -1;
unsigned char buff[65536];
Q_ASSERT(want < 65536);
ssl->read(reinterpret_cast< char * >(buff), want);
avail = ssl->bytesAvailable();
switch (ptype) {
case MessageHandler::CryptSetup: {
MumbleProto::CryptSetup msg;
if (!msg.ParseFromArray(buff, want))
qFatal("Failed parse crypt");
if (msg.has_key() && msg.has_client_nonce() && msg.has_server_nonce()) {
const std::string &key = msg.key();
const std::string &client_nonce = msg.client_nonce();
const std::string &server_nonce = msg.server_nonce();
if (key.size() == AES_BLOCK_SIZE && client_nonce.size() == AES_BLOCK_SIZE
&& server_nonce.size() == AES_BLOCK_SIZE)
crypt.setKey(key, client_nonce, server_nonce);
} else if (msg.has_server_nonce()) {
const std::string &server_nonce = msg.server_nonce();
if (server_nonce.size() == AES_BLOCK_SIZE) {
crypt.uiResync++;
crypt.setDecryptIV(server_nonce);
}
} else {
MumbleProto::CryptSetup mpcs;
mpcs.set_client_nonce(crypt.getEncryptIV());
sendMessage(mpcs, MessageHandler::CryptSetup);
}
break;
}
case MessageHandler::ServerSync: {
MumbleProto::ServerSync msg;
if (!msg.ParseFromArray(buff, want))
qFatal("Failed parse sync");
uiSession = msg.session();
break;
}
case MessageHandler::UDPTunnel: {
unsigned int msgUDPType = (buff[0] >> 5) & 0x7;
if (msgUDPType == MessageHandler::UDPVoiceCELTAlpha)
rcvd++;
break;
}
}
} else {
break;
}
}
}
void Client::disconnected() {
qWarning("SSL Socket disconnected");
QCoreApplication::instance()->quit();
}
class Container : public QObject {
Q_OBJECT
public:
int sent;
int numsender, numudplistener, numtcplistener;
int isender, iudplistener, itcplistener;
bool live, forceping;
QHostAddress qha;
unsigned short port;
QTimer qtTick;
Timer tickPing, tickVoice, tickGo, tickSpawn;
QList< Client * > speakers;
QList< Client * > clients;
Container(QHostAddress srvaddr, unsigned short port, int nsend, int nudp, int ntcp);
public slots:
void tick();
void go();
};
Container::Container(QHostAddress qha, unsigned short port, int numsend, int numudp, int numtcp) {
isender = iudplistener = itcplistener = 0;
live = false;
forceping = false;
sent = 0;
Timer t;
qWarning("Spawning %d speakers and %d listeners (%d UDP, %d TCP)", numsend, numudp + numtcp, numudp, numtcp);
this->qha = qha;
this->port = port;
numsender = numsend;
numudplistener = numudp;
numtcplistener = numtcp;
connect(&qtTick, SIGNAL(timeout()), this, SLOT(tick()));
qtTick.start(0);
tickSpawn.restart();
}
void Container::tick() {
if (forceping || tickPing.isElapsed(5000000ULL)) {
forceping = false;
foreach (Client *c, clients)
c->ping();
if (live) {
int lost = 0;
quint64 totrcv = 0;
int nrcv = 0;
foreach (Client *c, clients) {
if (!c->sender) {
totrcv += c->rcvd;
lost += sent - c->rcvd;
nrcv++;
}
}
qWarning("Sent: %8d Rcvd: %8lld Lost: %8d BW: %6.1fMbit/s", sent, totrcv / nrcv,
(lost + nrcv - 1) / nrcv, (totrcv * 8.0 * 123.0) / (tickGo.elapsed() * 1.0));
} else {
qWarning("Spawned %3d/%3d", isender + iudplistener + itcplistener,
numsender + numudplistener + numtcplistener);
}
}
if (live && tickVoice.isElapsed(10000ULL)) {
foreach (Client *c, speakers) {
sent++;
c->sendVoice();
}
}
if (!live) {
if (isender < numsender) {
// Spawn a sender
Client *c = new Client(this, qha, port, true, false);
speakers << c;
c->start();
clients << c;
isender++;
} else if (iudplistener < numudplistener) {
// Spawn a listener which uses UDP
Client *c = new Client(this, qha, port, false, false);
c->start();
clients << c;
iudplistener++;
} else if (itcplistener < numtcplistener) {
// Spawn a listener which uses TCP-only
Client *c = new Client(this, qha, port, false, true);
c->start();
clients << c;
itcplistener++;
} else {
live = true;
quint64 elapsed = tickSpawn.elapsed();
qWarning("Spawning took %lld ms (%lld us per client)", elapsed / 1000ULL,
elapsed / (numsender + numudplistener + numtcplistener));
foreach (Client *c, clients)
c->rcvd = 0;
sent = 0;
forceping = true;
qtTick.start(10);
}
}
}
void Container::go() {
foreach (Client *c, clients)
c->start();
qtTick.start(10);
tickGo.restart();
}
int main(int argc, char **argv) {
QCoreApplication a(argc, argv);
qWarning("Maximum # sockets is %d", FD_SETSIZE);
if (argc != 6)
qFatal(
"Invalid number of arguments. These need to be passed: <host address> <port> <numsend> <numudp> <numtcp>");
QHostAddress qha = QHostAddress(argv[1]);
int port = atoi(argv[2]);
int numsender = atoi(argv[3]);
int numudplistener = atoi(argv[4]);
int numtcplistener = atoi(argv[5]);
Container c(qha, port, numsender, numudplistener, numtcplistener);
c.go();
a.exec();
}
#include "Benchmark.moc"