Proxmark3GUI/common/pm3process.cpp

163 lines
4.6 KiB
C++
Raw Normal View History

2020-04-06 23:48:08 +08:00
#include "pm3process.h"
PM3Process::PM3Process(QThread* thread, QObject* parent): QProcess(parent)
2020-04-06 23:48:08 +08:00
{
moveToThread(thread);
2020-04-07 18:16:00 +08:00
setProcessChannelMode(PM3Process::MergedChannels);
2020-04-22 23:13:00 +08:00
isRequiringOutput = false;
requiredOutput = new QString();
serialListener = new QTimer(); // if using new QTimer(this), the debug output will show "Cannot create children for a parent that is in a different thread."
serialListener->moveToThread(this->thread());// I've tried many ways to creat a QTimer instance, but all of the instances are in the main thread(UI thread), so I have to move it manually
serialListener->setInterval(1000);
serialListener->setTimerType(Qt::VeryCoarseTimer);
2020-04-22 23:13:00 +08:00
connect(serialListener, &QTimer::timeout, this, &PM3Process::onTimeout);
connect(this, &PM3Process::readyRead, this, &PM3Process::onReadyRead);
portInfo = nullptr;
2020-04-07 18:16:00 +08:00
}
2020-04-06 23:48:08 +08:00
void PM3Process::connectPM3(const QString& path, const QStringList args)
2020-04-07 18:16:00 +08:00
{
2020-08-05 19:08:02 +08:00
QString result;
Util::ClientType clientType;
setRequiringOutput(true);
2020-04-06 23:48:08 +08:00
// stash for reconnect
currPath = path;
currArgs = args;
2020-04-07 18:16:00 +08:00
// using "-f" option to make the client output flushed after every print.
start(path, args, QProcess::Unbuffered | QProcess::ReadWrite | QProcess::Text);
if(waitForStarted(10000))
{
waitForReadyRead(10000);
setRequiringOutput(false);
2020-08-05 19:08:02 +08:00
result = *requiredOutput;
if(result.indexOf("[=]") != -1)
{
clientType = Util::CLIENTTYPE_ICEMAN;
2020-08-05 19:08:02 +08:00
setRequiringOutput(true);
write("hw version\n");
for(int i = 0; i < 50; i++)
{
waitForReadyRead(200);
result += *requiredOutput;
if(result.indexOf("os: ") != -1)
break;
}
2020-08-05 19:08:02 +08:00
setRequiringOutput(false);
}
else
{
clientType = Util::CLIENTTYPE_OFFICIAL;
}
2020-04-22 23:13:00 +08:00
if(result.indexOf("os: ") != -1) // make sure the PM3 is connected
{
emit changeClientType(clientType);
result = result.mid(result.indexOf("os: "));
result = result.left(result.indexOf("\n"));
result = result.mid(4, result.indexOf(" ", 4) - 4);
2020-04-22 23:13:00 +08:00
emit PM3StatedChanged(true, result);
}
else
kill();
}
2020-04-06 23:48:08 +08:00
}
void PM3Process::reconnectPM3()
{
connectPM3(currPath, currArgs);
}
void PM3Process::setRequiringOutput(bool st)
{
2020-04-22 23:13:00 +08:00
isRequiringOutput = st;
if(isRequiringOutput)
requiredOutput->clear();
}
2020-04-15 00:38:43 +08:00
bool PM3Process::waitForReadyRead(int msecs)
{
return QProcess::waitForReadyRead(msecs);
}
2020-04-16 00:39:10 +08:00
2020-04-22 23:13:00 +08:00
void PM3Process::setSerialListener(const QString& name, bool state)
{
if(state)
{
currPort = name;
2020-04-22 23:13:00 +08:00
portInfo = new QSerialPortInfo(name);
serialListener->start();
2020-04-22 23:13:00 +08:00
qDebug() << serialListener->thread();
}
else
{
serialListener->stop();
if(portInfo != nullptr)
{
delete portInfo;
portInfo = nullptr;
}
}
}
void PM3Process::setSerialListener(bool state)
{
setSerialListener(currPort, state);
}
2020-04-22 21:14:33 +08:00
void PM3Process::onTimeout() //when the proxmark3 client is unexpectedly terminated or the PM3 hardware is removed, the isBusy() will return false(only tested on Windows);
{
// isBusy() is a deprecated function because it will block the serial port when the port is not in use.
2021-09-19 23:59:55 +08:00
// However, the PM3 client is supposed to use the target serial port exclusively, so it should be fine
// isBusy() will always return false on Raspbian, in this case, check "Keep the client active" in the Settings panel.
//
2020-04-22 23:13:00 +08:00
// qDebug()<<portInfo->isBusy();
if(!portInfo->isBusy())
{
killPM3();
}
}
void PM3Process::testThread()
{
2020-04-22 23:13:00 +08:00
qDebug() << "PM3:" << QThread::currentThread();
}
2020-04-22 16:00:56 +08:00
qint64 PM3Process::write(QString data)
{
return QProcess::write(data.toLatin1());
}
2020-04-22 16:42:58 +08:00
void PM3Process::onReadyRead()
{
2020-04-22 21:14:33 +08:00
QString out = readAll();
if(isRequiringOutput)
requiredOutput->append(out);
if(out != "")
{
2020-04-23 17:50:20 +08:00
// qDebug() << "PM3Process::onReadyRead:" << out;
2020-04-22 21:14:33 +08:00
emit newOutput(out);
}
2020-04-22 16:42:58 +08:00
}
void PM3Process::setProcEnv(const QStringList* env)
{
// qDebug() << "passed Env List" << *env;
this->setEnvironment(*env);
// qDebug() << "final Env List" << processEnvironment().toStringList();
}
void PM3Process::setWorkingDir(const QString& dir)
{
// the working directory cannot be the default, or the client will failed to load the dll
this->setWorkingDirectory(dir);
}
2021-02-22 12:47:55 +08:00
void PM3Process::killPM3()
{
kill();
emit PM3StatedChanged(false);
setSerialListener(false);
}