2020-02-10 21:04:03 +00:00
|
|
|
/* SleepLib RESMED Loader Header
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
2020-02-10 21:04:03 +00:00
|
|
|
* Copyright (c) 2019-2020 The OSCAR Team
|
2018-03-28 07:10:52 +00:00
|
|
|
* Copyright (C) 2011-2018 Mark Watkins <mark@jedimark.net>
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
2018-06-04 20:48:38 +00:00
|
|
|
* License. See the file COPYING in the main directory of the source code
|
|
|
|
* for more details. */
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
#ifndef RESMED_LOADER_H
|
|
|
|
#define RESMED_LOADER_H
|
2018-04-25 17:00:09 +00:00
|
|
|
|
2011-07-31 20:24:43 +00:00
|
|
|
#include <QVector>
|
2011-06-28 02:21:38 +00:00
|
|
|
#include "SleepLib/machine.h" // Base class: MachineLoader
|
|
|
|
#include "SleepLib/machine_loader.h"
|
|
|
|
#include "SleepLib/profiles.h"
|
2018-04-25 15:22:29 +00:00
|
|
|
#include "SleepLib/loader_plugins/edfparser.h"
|
2019-11-17 21:41:47 +00:00
|
|
|
#include "SleepLib/loader_plugins/resmed_EDFinfo.h"
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
//********************************************************************************************
|
|
|
|
/// IMPORTANT!!!
|
|
|
|
//********************************************************************************************
|
|
|
|
// Please INCREMENT the following value when making changes to this loaders implementation.
|
|
|
|
//
|
2020-04-29 18:11:39 +00:00
|
|
|
const int resmed_data_version = 14;
|
2011-06-28 02:21:38 +00:00
|
|
|
//
|
|
|
|
//********************************************************************************************
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
|
|
|
|
class ResmedLoader;
|
|
|
|
|
2019-09-25 11:25:44 +00:00
|
|
|
class ResMedDay {
|
|
|
|
public:
|
|
|
|
ResMedDay( QDate d) : date(d) {}
|
|
|
|
|
2018-04-28 05:33:26 +00:00
|
|
|
QDate date;
|
|
|
|
STRRecord str;
|
2019-09-25 11:25:44 +00:00
|
|
|
QHash<QString, QString> files; // key is filename, value is fullpath
|
2018-04-28 05:33:26 +00:00
|
|
|
};
|
|
|
|
|
2020-02-10 21:04:03 +00:00
|
|
|
typedef void (*ResDaySaveCallback)(ResmedLoader* loader, Session* session);
|
|
|
|
|
2018-04-28 05:33:26 +00:00
|
|
|
class ResDayTask:public ImportTask
|
|
|
|
{
|
|
|
|
public:
|
2020-02-10 21:04:03 +00:00
|
|
|
ResDayTask(ResmedLoader * l, Machine * m, ResMedDay * d, ResDaySaveCallback s): reimporting(false), loader(l), mach(m), resday(d), save(s) {}
|
2018-04-28 05:33:26 +00:00
|
|
|
virtual ~ResDayTask() {}
|
|
|
|
virtual void run();
|
2019-09-25 11:25:44 +00:00
|
|
|
|
2018-05-03 05:44:19 +00:00
|
|
|
bool reimporting;
|
2018-04-28 05:33:26 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
ResmedLoader * loader;
|
|
|
|
Machine * mach;
|
|
|
|
ResMedDay * resday;
|
2020-02-10 21:04:03 +00:00
|
|
|
ResDaySaveCallback save;
|
2018-04-28 05:33:26 +00:00
|
|
|
};
|
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
/*! \class ResmedLoader
|
|
|
|
\brief Importer for ResMed S9 Data
|
|
|
|
*/
|
2014-08-03 13:00:13 +00:00
|
|
|
class ResmedLoader : public CPAPLoader
|
2011-06-28 02:21:38 +00:00
|
|
|
{
|
2018-05-07 00:37:22 +00:00
|
|
|
Q_OBJECT
|
2014-05-20 11:51:47 +00:00
|
|
|
friend class ResmedImport;
|
2014-06-30 11:20:12 +00:00
|
|
|
friend class ResmedImportStage2;
|
2014-04-17 05:58:57 +00:00
|
|
|
public:
|
2011-06-28 02:21:38 +00:00
|
|
|
ResmedLoader();
|
|
|
|
virtual ~ResmedLoader();
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2019-12-24 02:44:10 +00:00
|
|
|
//! \brief Register the ResmedLoader with the list of other machine loaders
|
|
|
|
static void Register();
|
|
|
|
|
2014-04-26 09:54:08 +00:00
|
|
|
//! \brief Detect if the given path contains a valid Folder structure
|
|
|
|
virtual bool Detect(const QString & path);
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
//! \brief Look up machine model information of ResMed file structure stored at path
|
|
|
|
virtual MachineInfo PeekInfo(const QString & path);
|
|
|
|
|
2019-08-16 23:50:09 +00:00
|
|
|
virtual void checkSummaryDay( ResMedDay & resday, QDate date, Machine * mach );
|
|
|
|
|
2018-04-25 18:04:05 +00:00
|
|
|
//! \brief Scans for ResMed SD folder structure signature, and loads any new data if found
|
2018-04-27 04:29:03 +00:00
|
|
|
virtual int Open(const QString &);
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Returns the version number of this ResMed loader
|
2011-07-27 09:21:53 +00:00
|
|
|
virtual int Version() { return resmed_data_version; }
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Returns the Machine class name of this loader. ("ResMed")
|
2014-07-28 13:56:29 +00:00
|
|
|
virtual const QString &loaderName() { return resmed_class_name; }
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2019-12-24 02:44:10 +00:00
|
|
|
virtual MachineInfo newInfo() {
|
|
|
|
return MachineInfo(MT_CPAP, 0, resmed_class_name, QObject::tr("ResMed"), QString(),
|
|
|
|
QString(), QString(), QObject::tr("S9"), QDateTime::currentDateTime(), resmed_data_version);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void initChannels();
|
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Converts EDFSignal data to time delta packed EventList, and adds to Session
|
2019-07-31 18:36:40 +00:00
|
|
|
void ToTimeDelta(Session *sess, ResMedEDFInfo &edf, EDFSignal &es, ChannelID code, long recs,
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 duration, EventDataType min = 0, EventDataType max = 0, bool square = false);
|
2011-06-29 20:30:23 +00:00
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Parse the EVE Event annotation data, and save to Session * sess
|
|
|
|
//! This contains all Hypopnea, Obstructive Apnea, Central and Apnea codes
|
2014-05-20 11:51:47 +00:00
|
|
|
bool LoadEVE(Session *sess, const QString & path);
|
2014-09-22 04:32:15 +00:00
|
|
|
|
|
|
|
//! \brief Parse the CSL Event annotation data, and save to Session * sess
|
|
|
|
//! This contains Cheyne Stokes Respiration flagging on the AirSense 10
|
|
|
|
bool LoadCSL(Session *sess, const QString & path);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Parse the BRP High Resolution data, and save to Session * sess
|
|
|
|
//! This contains Flow Rate, Mask Pressure, and Resp. Event data
|
2014-05-20 11:51:47 +00:00
|
|
|
bool LoadBRP(Session *sess, const QString & path);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
|
|
|
//! \brief Parse the SAD Pulse oximetry attachment data, and save to Session * sess
|
|
|
|
//! This contains Pulse Rate and SpO2 Oxygen saturation data
|
2014-05-20 11:51:47 +00:00
|
|
|
bool LoadSAD(Session *sess, const QString & path);
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2011-12-19 05:35:05 +00:00
|
|
|
//! \brief Parse the PRD low resolution data, and save to Session * sess
|
|
|
|
//! This contains the Pressure, Leak, Respiratory Rate, Minute Ventilation, Tidal Volume, etc..
|
2014-05-20 11:51:47 +00:00
|
|
|
bool LoadPLD(Session *sess, const QString & path);
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Now for some CPAPLoader overrides
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
virtual QString PresReliefLabel() { return QObject::tr("EPR: "); }
|
|
|
|
|
2019-12-24 02:44:10 +00:00
|
|
|
virtual ChannelID PresReliefMode() ;
|
|
|
|
virtual ChannelID PresReliefLevel() ;
|
|
|
|
virtual ChannelID CPAPModeChannel() ;
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2018-04-28 05:33:26 +00:00
|
|
|
volatile int sessionCount;
|
2020-02-10 21:04:03 +00:00
|
|
|
static void SaveSession(ResmedLoader* loader, Session* session);
|
|
|
|
ResDaySaveCallback saveCallback;
|
2021-04-27 00:10:37 +00:00
|
|
|
int OpenWithCallback(const QString & dirpath, ResDaySaveCallback s);
|
2014-08-03 13:00:13 +00:00
|
|
|
|
2021-08-01 03:30:24 +00:00
|
|
|
void LogUnexpectedMessage(const QString & message);
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
protected:
|
2019-07-31 18:36:40 +00:00
|
|
|
//! \brief The STR.edf file is a unique edf file with many signals
|
2020-02-14 20:18:25 +00:00
|
|
|
bool ProcessSTRfiles(Machine *, QMap<QDate, STRFile> &, QDate);
|
2014-05-18 17:06:58 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
//! \brief Scan for new files to import, group into sessions and add to task que
|
2019-12-24 02:44:10 +00:00
|
|
|
int ScanFiles(Machine * mach, const QString & datalog_path, QDate firstImport);
|
2014-05-18 17:06:58 +00:00
|
|
|
|
2019-07-31 18:36:40 +00:00
|
|
|
//! \brief Write a backup copy to the backup path
|
2019-12-24 02:44:10 +00:00
|
|
|
QString Backup(const QString & file, const QString & backup_path);
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2019-07-31 18:36:40 +00:00
|
|
|
// The data members
|
|
|
|
// QMap<SessionID, QStringList> sessfiles;
|
|
|
|
// QMap<quint32, STRRecord> strsess;
|
|
|
|
// QMap<QDate, QList<STRRecord *> > strdate;
|
|
|
|
|
2018-05-03 05:08:45 +00:00
|
|
|
QMap<QDate, ResMedDay> resdayList;
|
2014-05-18 17:06:58 +00:00
|
|
|
|
2012-01-05 06:54:07 +00:00
|
|
|
#ifdef DEBUG_EFFICIENCY
|
2014-04-17 05:58:57 +00:00
|
|
|
QHash<ChannelID, qint64> channel_efficiency;
|
|
|
|
QHash<ChannelID, qint64> channel_time;
|
2018-05-06 16:59:50 +00:00
|
|
|
volatile qint64 timeInLoadBRP;
|
|
|
|
volatile qint64 timeInLoadPLD;
|
|
|
|
volatile qint64 timeInLoadEVE;
|
|
|
|
volatile qint64 timeInLoadCSL;
|
|
|
|
volatile qint64 timeInLoadSAD;
|
|
|
|
volatile qint64 timeInEDFOpen;
|
2019-07-31 18:36:40 +00:00
|
|
|
volatile qint64 timeInEDFInfo;
|
2018-05-06 16:59:50 +00:00
|
|
|
volatile qint64 timeInAddWaveform;
|
|
|
|
volatile qint64 timeInTimeDelta;
|
|
|
|
QMutex timeMutex;
|
2021-08-01 03:30:24 +00:00
|
|
|
#endif
|
2018-05-06 16:59:50 +00:00
|
|
|
|
2021-08-01 03:30:24 +00:00
|
|
|
// TODO: This really belongs in a generic location that all loaders can use.
|
|
|
|
// But that will require retooling the overall call structure so that there's
|
|
|
|
// a top-level import job that's managing a specific import. Right now it's
|
|
|
|
// essentially managed by the importCPAP method rather than an object instance
|
|
|
|
// with state.
|
|
|
|
QMutex m_importMutex;
|
|
|
|
QSet<QString> m_unexpectedMessages;
|
2018-05-06 16:59:50 +00:00
|
|
|
|
2011-06-28 02:21:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // RESMED_LOADER_H
|