2012-04-09 17:53:46 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 Felix Geyer <debfx@fobos.de>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 2 or (at your option)
|
|
|
|
* version 3 of the License.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Tools.h"
|
|
|
|
|
2015-07-22 18:11:05 -04:00
|
|
|
#include <QCoreApplication>
|
2013-10-03 09:18:16 -04:00
|
|
|
#include <QImageReader>
|
|
|
|
#include <QIODevice>
|
|
|
|
#include <QLocale>
|
|
|
|
#include <QStringList>
|
2012-04-09 17:53:46 -04:00
|
|
|
|
2013-10-03 09:18:16 -04:00
|
|
|
#include <QElapsedTimer>
|
2012-08-05 13:19:01 -04:00
|
|
|
|
2012-07-12 07:51:50 -04:00
|
|
|
#ifdef Q_OS_WIN
|
2015-12-06 08:32:06 -05:00
|
|
|
#include <windows.h> // for Sleep(), SetDllDirectoryA() and SetSearchPathMode()
|
2012-07-12 07:51:50 -04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef Q_OS_UNIX
|
|
|
|
#include <time.h> // for nanosleep()
|
|
|
|
#endif
|
|
|
|
|
2012-10-21 13:24:17 -04:00
|
|
|
#include "config-keepassx.h"
|
|
|
|
|
2015-05-14 06:58:00 -04:00
|
|
|
#if defined(HAVE_RLIMIT_CORE)
|
|
|
|
#include <sys/resource.h>
|
|
|
|
#endif
|
|
|
|
|
2012-10-13 05:05:50 -04:00
|
|
|
#if defined(HAVE_PR_SET_DUMPABLE)
|
|
|
|
#include <sys/prctl.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_PT_DENY_ATTACH
|
2012-10-23 11:06:50 -04:00
|
|
|
#include <sys/types.h>
|
2012-10-13 05:05:50 -04:00
|
|
|
#include <sys/ptrace.h>
|
|
|
|
#endif
|
|
|
|
|
2012-04-18 16:08:22 -04:00
|
|
|
namespace Tools {
|
|
|
|
|
|
|
|
QString humanReadableFileSize(qint64 bytes)
|
2012-04-09 17:53:46 -04:00
|
|
|
{
|
|
|
|
double size = bytes;
|
|
|
|
|
|
|
|
QStringList units = QStringList() << "B" << "KiB" << "MiB" << "GiB";
|
|
|
|
int i = 0;
|
|
|
|
int maxI = units.size() - 1;
|
|
|
|
|
|
|
|
while ((size >= 1024) && (i < maxI)) {
|
|
|
|
size /= 1024;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return QString("%1 %2").arg(QLocale().toString(size, 'f', 2), units.at(i));
|
|
|
|
}
|
2012-04-18 16:08:22 -04:00
|
|
|
|
2012-04-18 18:04:52 -04:00
|
|
|
bool hasChild(const QObject* parent, const QObject* child)
|
|
|
|
{
|
|
|
|
if (!parent || !child) {
|
|
|
|
return false;
|
|
|
|
}
|
2016-09-02 13:51:51 -04:00
|
|
|
|
|
|
|
const QObjectList children = parent->children();
|
|
|
|
for (QObject* c : children) {
|
2012-04-18 18:04:52 -04:00
|
|
|
if (child == c || hasChild(c, child)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-05-10 13:00:36 -04:00
|
|
|
bool readFromDevice(QIODevice* device, QByteArray& data, int size)
|
|
|
|
{
|
|
|
|
QByteArray buffer;
|
|
|
|
buffer.resize(size);
|
|
|
|
|
|
|
|
qint64 readResult = device->read(buffer.data(), size);
|
|
|
|
if (readResult == -1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
buffer.resize(readResult);
|
|
|
|
data = buffer;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-02 05:06:24 -04:00
|
|
|
bool readAllFromDevice(QIODevice* device, QByteArray& data)
|
|
|
|
{
|
|
|
|
QByteArray result;
|
|
|
|
qint64 readBytes = 0;
|
|
|
|
qint64 readResult;
|
|
|
|
do {
|
|
|
|
result.resize(result.size() + 16384);
|
|
|
|
readResult = device->read(result.data() + readBytes, result.size() - readBytes);
|
|
|
|
if (readResult > 0) {
|
|
|
|
readBytes += readResult;
|
|
|
|
}
|
|
|
|
} while (readResult > 0);
|
|
|
|
|
|
|
|
if (readResult == -1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result.resize(static_cast<int>(readBytes));
|
|
|
|
data = result;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-10 12:09:42 -04:00
|
|
|
QString imageReaderFilter()
|
|
|
|
{
|
2016-09-02 13:51:51 -04:00
|
|
|
const QList<QByteArray> formats = QImageReader::supportedImageFormats();
|
2012-05-10 12:09:42 -04:00
|
|
|
QStringList formatsStringList;
|
|
|
|
|
2016-09-02 13:51:51 -04:00
|
|
|
for (const QByteArray& format : formats) {
|
2012-05-10 12:09:42 -04:00
|
|
|
for (int i = 0; i < format.size(); i++) {
|
|
|
|
if (!QChar(format.at(i)).isLetterOrNumber()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-24 15:19:20 -05:00
|
|
|
formatsStringList.append("*." + QString::fromLatin1(format).toLower());
|
2012-05-10 12:09:42 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return formatsStringList.join(" ");
|
|
|
|
}
|
|
|
|
|
2012-05-10 12:34:51 -04:00
|
|
|
bool isHex(const QByteArray& ba)
|
|
|
|
{
|
2016-09-02 13:51:51 -04:00
|
|
|
for (char c : ba) {
|
2012-05-10 12:34:51 -04:00
|
|
|
if ( !( (c >= '0' && c <= '9') || (c >= 'a' && c <= 'f') || (c >= 'A' && c <= 'F') ) ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-11-01 12:30:50 -05:00
|
|
|
bool isBase64(const QByteArray& ba)
|
|
|
|
{
|
|
|
|
QRegExp regexp("^(?:[a-z0-9+/]{4})*(?:[a-z0-9+/]{3}=|[a-z0-9+/]{2}==)?$",
|
|
|
|
Qt::CaseInsensitive, QRegExp::RegExp2);
|
|
|
|
|
|
|
|
QString base64 = QString::fromLatin1(ba.constData(), ba.size());
|
|
|
|
|
|
|
|
return regexp.exactMatch(base64);
|
|
|
|
}
|
|
|
|
|
2012-07-12 07:51:50 -04:00
|
|
|
void sleep(int ms)
|
|
|
|
{
|
|
|
|
Q_ASSERT(ms >= 0);
|
|
|
|
|
|
|
|
if (ms == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
Sleep(uint(ms));
|
|
|
|
#else
|
|
|
|
timespec ts;
|
|
|
|
ts.tv_sec = ms / 1000;
|
|
|
|
ts.tv_nsec = (ms % 1000) * 1000 * 1000;
|
2015-07-24 12:28:12 -04:00
|
|
|
nanosleep(&ts, nullptr);
|
2012-07-12 07:51:50 -04:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void wait(int ms)
|
|
|
|
{
|
2012-11-02 10:33:39 -04:00
|
|
|
Q_ASSERT(ms >= 0);
|
|
|
|
|
|
|
|
if (ms == 0) {
|
|
|
|
return;
|
|
|
|
}
|
2012-07-12 07:51:50 -04:00
|
|
|
|
|
|
|
QElapsedTimer timer;
|
|
|
|
timer.start();
|
|
|
|
|
|
|
|
if (ms <= 50) {
|
|
|
|
QCoreApplication::processEvents(QEventLoop::AllEvents, ms);
|
|
|
|
sleep(qMax(ms - static_cast<int>(timer.elapsed()), 0));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
int timeLeft;
|
|
|
|
do {
|
|
|
|
timeLeft = ms - timer.elapsed();
|
|
|
|
if (timeLeft > 0) {
|
|
|
|
QCoreApplication::processEvents(QEventLoop::AllEvents, timeLeft);
|
|
|
|
sleep(10);
|
|
|
|
}
|
2015-07-22 17:54:39 -04:00
|
|
|
} while (!timer.hasExpired(ms));
|
2012-07-12 07:51:50 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-13 05:05:50 -04:00
|
|
|
void disableCoreDumps()
|
|
|
|
{
|
2015-05-14 06:58:00 -04:00
|
|
|
// default to true
|
|
|
|
// there is no point in printing a warning if this is not implemented on the platform
|
|
|
|
bool success = true;
|
2012-10-13 05:05:50 -04:00
|
|
|
|
2015-05-14 06:58:00 -04:00
|
|
|
#if defined(HAVE_RLIMIT_CORE)
|
2012-10-13 05:05:50 -04:00
|
|
|
struct rlimit limit;
|
|
|
|
limit.rlim_cur = 0;
|
|
|
|
limit.rlim_max = 0;
|
2015-05-14 06:58:00 -04:00
|
|
|
success = success && (setrlimit(RLIMIT_CORE, &limit) == 0);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(HAVE_PR_SET_DUMPABLE)
|
|
|
|
success = success && (prctl(PR_SET_DUMPABLE, 0) == 0);
|
2012-10-13 05:05:50 -04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Mac OS X
|
|
|
|
#ifdef HAVE_PT_DENY_ATTACH
|
|
|
|
success = success && (ptrace(PT_DENY_ATTACH, 0, 0, 0) == 0);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!success) {
|
|
|
|
qWarning("Unable to disable core dumps.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-06 08:32:06 -05:00
|
|
|
void setupSearchPaths()
|
|
|
|
{
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
// Make sure Windows doesn't load DLLs from the current working directory
|
|
|
|
SetDllDirectoryA("");
|
|
|
|
SetSearchPathMode(BASE_SEARCH_PATH_ENABLE_SAFE_SEARCHMODE);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2012-04-18 16:08:22 -04:00
|
|
|
} // namespace Tools
|