OSCAR-code/oscar/SleepLib/machine_loader.cpp

340 lines
7.9 KiB
C++

/* SleepLib Device Loader Class Implementation
*
* Copyright (c) 2019-2022 The OSCAR Team
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
*
* This file is subject to the terms and conditions of the GNU General Public
* License. See the file COPYING in the main directory of the source code
* for more details. */
#define TEST_MACROS_ENABLEDoff
#include <test_macros.h>
#include <QApplication>
#include <QFile>
#include <QDir>
#include <QThreadPool>
#include "machine_loader.h"
// GLOBALS:
bool genpixmapinit = false;
QList<MachineLoader *> m_loaders;
QPixmap * MachineLoader::genericCPAPPixmap;
MachineLoader::MachineLoader() :QObject(nullptr)
{
#ifndef UNITTEST_MODE // no QPixmap without a QGuiApplication
if (!genpixmapinit) {
genericCPAPPixmap = new QPixmap(genericPixmapPath);
genpixmapinit = true;
}
#endif
m_abort = false;
m_type = MT_UNKNOWN;
m_status = NEUTRAL;
m_ctx = nullptr;
}
MachineLoader::~MachineLoader()
{
}
void MachineLoader::addSession(Session * sess)
{
sessionMutex.lock();
new_sessions[sess->session()] = sess;
sessionMutex.unlock();
}
void MachineLoader::finishAddingSessions()
{
// Using a map specifically so they are inserted in order.
for (auto it=new_sessions.begin(), end=new_sessions.end(); it != end; ++it) {
Session * sess = it.value();
Machine * mach = sess->machine();
mach->AddSession(sess);
}
new_sessions.clear();
}
QPixmap & MachineLoader::getPixmap(QString series)
{
QHash<QString, QPixmap>::iterator it = m_pixmaps.find(series);
if (it != m_pixmaps.end()) {
return it.value();
}
return *genericCPAPPixmap;
}
QString MachineLoader::getPixmapPath(QString series)
{
QHash<QString, QString>::iterator it = m_pixmap_paths.find(series);
if (it != m_pixmap_paths.end()) {
return it.value();
}
return genericPixmapPath;
}
void MachineLoader::queTask(ImportTask * task)
{
m_MLtasklist.push_back(task);
}
void MachineLoader::runTasks(bool threaded)
{
m_totalMLtasks = m_MLtasklist.size();
qDebug() << "MachineLoader::runTasks MLtasklist size is" << m_totalMLtasks;
if (m_totalMLtasks == 0)
return;
emit setProgressMax(m_totalMLtasks);
m_currentMLtask=0;
threaded=AppSetting->multithreading();
if ( ! threaded) {
while (!m_MLtasklist.isEmpty() && !m_abort) {
ImportTask * task = m_MLtasklist.takeFirst();
task->run();
// update progress bar
m_currentMLtask++;
emit setProgressValue(m_currentMLtask);
QApplication::processEvents();
delete task;
}
} else {
ImportTask * task = m_MLtasklist[0];
QThreadPool * threadpool = QThreadPool::globalInstance();
while (!m_abort) {
if (threadpool->tryStart(task)) {
m_MLtasklist.pop_front();
if (!m_MLtasklist.isEmpty()) {
// next task to be run
task = m_MLtasklist[0];
// update progress bar
emit setProgressValue(++m_currentMLtask);
QApplication::processEvents();
} else {
// job list finished
break;
}
}
//QThread::sleep(100);
}
QThreadPool::globalInstance()->waitForDone(-1);
}
if (m_abort) {
// delete remaining tasks and clear task list
for (auto & task : m_MLtasklist) {
delete task;
}
m_MLtasklist.clear();
}
}
QList<MachineLoader *> GetLoaders(MachineType mt)
{
QList<MachineLoader *> list;
for (int i=0; i < m_loaders.size(); ++i) {
if (mt == MT_UNKNOWN) {
list.push_back(m_loaders.at(i));
} else {
MachineType mtype = m_loaders.at(i)->type();
if (mtype == mt) {
list.push_back(m_loaders.at(i));
}
}
}
return list;
}
MachineLoader * lookupLoader(Machine * m)
{
for (int i=0; i < m_loaders.size(); ++i) {
MachineLoader * loader = m_loaders.at(i);
if (loader->loaderName() == m->loaderName())
return loader;
}
return nullptr;
}
MachineLoader * lookupLoader(QString loaderName)
{
for (int i=0; i < m_loaders.size(); ++i) {
MachineLoader * loader = m_loaders.at(i);
if (loader->loaderName() == loaderName)
return loader;
}
return nullptr;
}
void RegisterLoader(MachineLoader *loader)
{
loader->initChannels();
m_loaders.push_back(loader);
}
void DestroyLoaders()
{
for (auto & loader : m_loaders) {
delete(loader);
}
m_loaders.clear();
}
/*
QList<ChannelID> CPAPLoader::eventFlags(Day * day)
{
Machine * mach = day->machine(MT_CPAP);
QList<ChannelID> list;
if (mach->loader() != this) {
qDebug() << "Trying to ask" << loaderName() << "for" << mach->loaderName() << "data";
return list;
}
for (int i = 0; i < ahiChannels.size(); i++)
list.push_back(ahiChannels.at(i));
// list.push_back(CPAP_ClearAirway);
// list.push_back(CPAP_AllApnea);
// list.push_back(CPAP_Obstructive);
// list.push_back(CPAP_Hypopnea);
// list.push_back(CPAP_Apnea);
return list;
}
*/
bool uncompressFile(QString infile, QString outfile)
{
if (!infile.endsWith(".gz",Qt::CaseInsensitive)) {
qDebug() << "uncompressFile()" << outfile << "missing .gz extension???";
return false;
}
if (QFile::exists(outfile)) {
qDebug() << "uncompressFile()" << outfile << "already exists";
return false;
}
// Get file length from inside gzip file
QFile fi(infile);
if (!fi.open(QFile::ReadOnly) || !fi.seek(fi.size() - 4)) {
return false;
}
unsigned char ch[4];
fi.read((char *)ch, 4);
quint32 datasize = ch[0] | (ch [1] << 8) | (ch[2] << 16) | (ch[3] << 24);
// Open gzip file for reading
gzFile f = gzopen(infile.toLatin1(), "rb");
if (!f) {
return false;
}
// Decompressed header and data block
char * buffer = new char [datasize];
gzread(f, buffer, datasize);
gzclose(f);
QFile out(outfile);
if (out.open(QFile::WriteOnly)) {
out.write(buffer, datasize);
out.close();
}
delete [] buffer;
return true;
}
bool compressFile(QString infile, QString outfile)
{
if (outfile.isEmpty()) {
outfile = infile + ".gz";
} else if (!outfile.endsWith(".gz")) {
outfile += ".gz";
}
if (QFile::exists(outfile)) {
qDebug() << "compressFile()" << outfile << "already exists";
return false;
}
QFile f(infile);
if (!f.exists(infile)) {
qDebug() << "compressFile()" << infile << "does not exist";
return false;
}
qint64 size = f.size();
if (!f.open(QFile::ReadOnly)) {
qDebug() << "compressFile() Couldn't open" << infile;
return false;
}
char *buf = new char [size];
if (!f.read(buf, size)) {
delete [] buf;
qDebug() << "compressFile() Couldn't read all of" << infile;
return false;
}
f.close();
gzFile gz = gzopen(outfile.toLatin1(), "wb");
//gzbuffer(gz,65536*2);
if (!gz) {
qDebug() << "compressFile() Couldn't open" << outfile << "for writing";
delete [] buf;
return false;
}
gzwrite(gz, buf, size);
gzclose(gz);
delete [] buf;
return true;
}
int MachineLoader::Open(const QStringList & paths)
{
int i, skipped = 0;
int size = paths.size();
for (i=0; i < size; i++) {
if (isAborted()) {
break;
}
QString filename = paths[i];
int res = OpenFile(filename);
if (res < 0) {
break;
}
if (res == 0) {
// Should we report on skipped count?
skipped++;
}
emit setProgressValue(i+1);
QCoreApplication::processEvents();
}
return i;
}