2014-08-17 12:56:05 +00:00
|
|
|
/* SleepLib ResMed Loader Implementation
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
2016-03-01 11:59:04 +00:00
|
|
|
* Copyright (c) 2011-2016 Mark Watkins <jedimark@users.sourceforge.net>
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
|
|
|
* 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 Linux
|
|
|
|
* distribution for more details. */
|
2011-07-30 00:36:31 +00:00
|
|
|
|
|
|
|
#include <QApplication>
|
2011-06-28 02:21:38 +00:00
|
|
|
#include <QString>
|
|
|
|
#include <QDateTime>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFile>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QProgressBar>
|
2014-07-27 16:35:49 +00:00
|
|
|
#include <QTextStream>
|
2011-07-01 10:10:44 +00:00
|
|
|
#include <QDebug>
|
2015-08-02 05:53:09 +00:00
|
|
|
#include <QStringList>
|
2011-07-27 09:21:53 +00:00
|
|
|
#include <cmath>
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
#include "resmed_loader.h"
|
|
|
|
#include "SleepLib/session.h"
|
2011-11-28 01:39:28 +00:00
|
|
|
#include "SleepLib/calcs.h"
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2012-01-05 06:54:07 +00:00
|
|
|
#ifdef DEBUG_EFFICIENCY
|
|
|
|
#include <QElapsedTimer> // only available in 4.8
|
|
|
|
#endif
|
|
|
|
|
2011-06-30 04:55:20 +00:00
|
|
|
extern QProgressBar *qprogress;
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
QHash<QString, QList<quint16> > Resmed_Model_Map;
|
|
|
|
|
|
|
|
const QString STR_UnknownModel = "Resmed S9 ???";
|
|
|
|
|
2016-03-08 13:55:29 +00:00
|
|
|
ChannelID RMS9_EPR, RMS9_EPRLevel, RMS9_Mode, RMS9_SmartStart, RMS9_HumidStatus, RMS9_HumidLevel,
|
|
|
|
RMS9_PtAccess, RMS9_Mask, RMS9_ABFilter, RMS9_ClimateControl, RMS9_TubeType,
|
|
|
|
RMS9_Temp, RMS9_TempEnable, RMS9_RampEnable;
|
2014-08-03 13:00:13 +00:00
|
|
|
|
2014-09-17 17:20:01 +00:00
|
|
|
const QString STR_ResMed_AirSense10 = "AirSense 10";
|
2016-03-02 02:09:32 +00:00
|
|
|
const QString STR_ResMed_AirCurve10= "AirCurve 10";
|
2014-09-17 17:20:01 +00:00
|
|
|
const QString STR_ResMed_S9 = "S9";
|
|
|
|
|
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Return the model name matching the supplied model number.
|
|
|
|
const QString & lookupModel(quint16 model)
|
|
|
|
{
|
|
|
|
|
2014-05-06 18:03:13 +00:00
|
|
|
QHash<QString, QList<quint16> >::iterator end = Resmed_Model_Map.end();
|
|
|
|
for (QHash<QString, QList<quint16> >::iterator it = Resmed_Model_Map.begin(); it != end; ++it) {
|
2014-05-04 18:02:41 +00:00
|
|
|
QList<quint16> & list = it.value();
|
|
|
|
for (int i=0; i < list.size(); ++i) {
|
|
|
|
if (list.at(i) == model) {
|
|
|
|
return it.key();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return STR_UnknownModel;
|
|
|
|
}
|
|
|
|
|
|
|
|
QHash<ChannelID, QStringList> resmed_codes;
|
2011-06-30 04:55:20 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
const QString STR_ext_TGT = "tgt";
|
|
|
|
const QString STR_ext_CRC = "crc";
|
|
|
|
const QString STR_ext_EDF = "edf";
|
|
|
|
const QString STR_ext_gz = ".gz";
|
2013-09-14 23:32:14 +00:00
|
|
|
|
|
|
|
|
2011-12-26 18:26:06 +00:00
|
|
|
// Looks up foreign language Signal names that match this channelID
|
2014-04-17 05:58:57 +00:00
|
|
|
EDFSignal *EDFParser::lookupSignal(ChannelID ch)
|
2011-11-25 12:13:35 +00:00
|
|
|
{
|
2014-05-04 18:02:41 +00:00
|
|
|
// Get list of all known foreign language names for this channel
|
2014-05-06 18:03:13 +00:00
|
|
|
QHash<ChannelID, QStringList>::iterator channames = resmed_codes.find(ch);
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
if (channames == resmed_codes.end()) {
|
|
|
|
// no alternatives strings found for this channel
|
2014-04-23 13:19:56 +00:00
|
|
|
return nullptr;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
// This is bad, because ResMed thinks it was a cool idea to use two channels with the same name.
|
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Scan through EDF's list of signals to see if any match
|
|
|
|
for (int i = 0; i < channames.value().size(); i++) {
|
2014-05-18 17:06:58 +00:00
|
|
|
EDFSignal *sig = lookupLabel(channames.value()[i]);
|
|
|
|
if (sig) return sig;
|
2011-11-25 12:13:35 +00:00
|
|
|
}
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Failed
|
2014-04-23 13:19:56 +00:00
|
|
|
return nullptr;
|
2011-11-25 12:13:35 +00:00
|
|
|
}
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Check if given string matches any alternative signal names for this channel
|
|
|
|
bool matchSignal(ChannelID ch, const QString & name)
|
|
|
|
{
|
2014-05-06 18:03:13 +00:00
|
|
|
QHash<ChannelID, QStringList>::iterator channames = resmed_codes.find(ch);
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
if (channames == resmed_codes.end()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList & strings = channames.value();
|
|
|
|
int size = strings.size();
|
|
|
|
|
|
|
|
for (int i=0; i < size; ++i) {
|
|
|
|
// Using starts with, because ResMed is very lazy about consistency
|
|
|
|
if (name.startsWith(strings.at(i), Qt::CaseInsensitive)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
EDFSignal *EDFParser::lookupLabel(QString name, int index)
|
2012-01-03 07:15:02 +00:00
|
|
|
{
|
2014-08-26 07:55:01 +00:00
|
|
|
QHash<QString, QList<EDFSignal *> >::iterator it = signalList.find(name);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
if (it == signalList.end()) return nullptr;
|
|
|
|
|
|
|
|
if (index >= it.value().size()) return nullptr;
|
|
|
|
|
|
|
|
return it.value()[index];
|
2012-01-03 07:15:02 +00:00
|
|
|
}
|
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
EDFParser::EDFParser(QString name)
|
|
|
|
{
|
2014-04-23 13:19:56 +00:00
|
|
|
buffer = nullptr;
|
2011-06-29 14:19:38 +00:00
|
|
|
Open(name);
|
|
|
|
}
|
|
|
|
EDFParser::~EDFParser()
|
|
|
|
{
|
2014-05-06 18:03:13 +00:00
|
|
|
for (QVector<EDFSignal>::iterator s = edfsignals.begin(); s != edfsignals.end(); s++) {
|
2014-04-17 05:58:57 +00:00
|
|
|
if ((*s).data) { delete [](*s).data; }
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (buffer) { delete [] buffer; }
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2014-05-05 06:14:29 +00:00
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
void ResmedLoader::ParseSTR(Machine *mach, QStringList strfiles)
|
|
|
|
{
|
|
|
|
QStringList::iterator strend = strfiles.end();
|
|
|
|
for (QStringList::iterator it = strfiles.begin(); it != strend; ++it) {
|
|
|
|
EDFParser str(*it);
|
|
|
|
if (!str.Parse()) continue;
|
2014-07-28 13:56:29 +00:00
|
|
|
if (mach->serial() != str.serialnumber) {
|
|
|
|
qDebug() << "Trying to import a STR.edf from another machine, skipping" << mach->serial() << str.serialnumber;
|
2014-05-18 17:06:58 +00:00
|
|
|
qDebug() << (*it);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-08-11 20:01:24 +00:00
|
|
|
QDateTime start = QDateTime::fromMSecsSinceEpoch(str.startdate, Qt::UTC);
|
2014-05-18 17:06:58 +00:00
|
|
|
QDate date = start.date();
|
|
|
|
|
|
|
|
qDebug() << "Parsing" << *it << date << str.GetNumDataRecords() << str.GetNumSignals();
|
|
|
|
|
|
|
|
EDFSignal *maskon = str.lookupLabel("Mask On");
|
2014-08-24 05:00:51 +00:00
|
|
|
if (!maskon) {
|
|
|
|
maskon = str.lookupLabel("MaskOn");
|
|
|
|
}
|
2014-05-18 17:06:58 +00:00
|
|
|
EDFSignal *maskoff = str.lookupLabel("Mask Off");
|
2014-08-24 05:00:51 +00:00
|
|
|
if (!maskoff) {
|
|
|
|
maskoff = str.lookupLabel("MaskOff");
|
|
|
|
}
|
2014-05-18 17:06:58 +00:00
|
|
|
EDFSignal *sig = nullptr;
|
|
|
|
quint32 laston = 0;
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//bool skipday;
|
2014-05-18 17:06:58 +00:00
|
|
|
|
|
|
|
int size = str.GetNumDataRecords();
|
|
|
|
int cnt=0;
|
2014-07-10 14:54:05 +00:00
|
|
|
QDateTime dt = start;
|
2014-05-18 17:06:58 +00:00
|
|
|
|
|
|
|
// For each data record, representing 1 day each
|
|
|
|
for (int rec = 0; rec < str.GetNumDataRecords(); ++rec) {
|
2014-07-10 14:54:05 +00:00
|
|
|
uint timestamp = dt.toTime_t();
|
2014-05-18 17:06:58 +00:00
|
|
|
int recstart = rec * maskon->nr;
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
// skipday = false;
|
2014-05-18 17:06:58 +00:00
|
|
|
if ((++cnt % 10) == 0) {
|
|
|
|
// TODO: Change me to emit once MachineLoader is QObjectified...
|
|
|
|
if (qprogress) { qprogress->setValue(10.0 + (float(cnt) / float(size) * 90.0)); }
|
|
|
|
|
|
|
|
QApplication::processEvents();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan the mask on/off events
|
|
|
|
for (int s = 0; s < maskon->nr; ++s) {
|
|
|
|
qint32 on = maskon->data[recstart+s];
|
|
|
|
qint32 off = maskoff->data[recstart+s];
|
|
|
|
quint32 ontime = timestamp + on * 60;
|
|
|
|
quint32 offtime = timestamp + off * 60;
|
|
|
|
|
|
|
|
// -1 marks empty record, but can start with mask off, if sleep crosses noon
|
|
|
|
if (on < 0) {
|
|
|
|
if (off < 0) continue; // Both are -1, skip the rest of this day
|
|
|
|
// laston stops on this record
|
|
|
|
|
|
|
|
QMap<quint32, STRRecord>::iterator si = strsess.find(laston);
|
|
|
|
if (si != strsess.end()) {
|
|
|
|
if (si.value().maskoff == 0) {
|
2014-05-31 21:25:07 +00:00
|
|
|
if (offtime > laston) {
|
|
|
|
si.value().maskoff = offtime;
|
|
|
|
}
|
2014-05-18 17:06:58 +00:00
|
|
|
} else {
|
|
|
|
if (si.value().maskoff != offtime) {
|
|
|
|
// not sure why this happens.
|
2014-05-31 21:25:07 +00:00
|
|
|
qDebug() << "WTF?? mask off's don't match"
|
|
|
|
<< QDateTime::fromTime_t(laston).toString()
|
|
|
|
<< QDateTime::fromTime_t(si.value().maskoff).toString()
|
|
|
|
<< "!=" << QDateTime::fromTime_t(offtime).toString();
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
|
|
|
//Q_ASSERT(si.value().maskoff == offtime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
QMap<quint32, STRRecord>::iterator sid = strsess.find(ontime);
|
2014-05-20 11:51:47 +00:00
|
|
|
// Record already exists?
|
2014-05-18 17:06:58 +00:00
|
|
|
if (sid != strsess.end()) {
|
2014-05-20 11:51:47 +00:00
|
|
|
// then skip
|
|
|
|
laston = ontime;
|
|
|
|
continue;
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// For every mask on, there will be a session within 1 minute either way
|
|
|
|
// We can use that for data matching
|
|
|
|
STRRecord R;
|
|
|
|
|
|
|
|
R.maskon = ontime;
|
|
|
|
if (offtime > 0) {
|
|
|
|
R.maskoff = offtime;
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
CPAPMode mode = MODE_UNKNOWN;
|
|
|
|
|
|
|
|
if ((sig = str.lookupSignal(CPAP_Mode))) {
|
|
|
|
int mod = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
2014-09-01 04:49:05 +00:00
|
|
|
R.rms9_mode = mod;
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
if (mod == 11) {
|
2016-03-04 00:17:19 +00:00
|
|
|
mode = MODE_APAP; // For her
|
2014-09-01 04:49:05 +00:00
|
|
|
} else if (mod >= 8) { // mod 8 == vpap adapt variable epap
|
2014-08-26 07:55:01 +00:00
|
|
|
mode = MODE_ASV_VARIABLE_EPAP;
|
|
|
|
} else if (mod >= 7) { // mod 7 == vpap adapt
|
|
|
|
mode = MODE_ASV;
|
|
|
|
} else if (mod >= 6) { // mod 6 == vpap auto (Min EPAP, Max IPAP, PS)
|
|
|
|
mode = MODE_BILEVEL_AUTO_FIXED_PS;
|
|
|
|
} else if (mod >= 3) {// mod 3 == vpap s fixed pressure (EPAP, IPAP, No PS)
|
|
|
|
mode = MODE_BILEVEL_FIXED;
|
|
|
|
// 4,5 are S/T types...
|
|
|
|
|
|
|
|
} else if (mod >= 1) {
|
|
|
|
mode = MODE_APAP; // mod 1 == apap
|
|
|
|
// not sure what mode 2 is ?? split ?
|
|
|
|
} else {
|
|
|
|
mode = MODE_CPAP; // mod 0 == cpap
|
|
|
|
}
|
|
|
|
R.mode = mode;
|
|
|
|
|
2015-09-21 01:30:23 +00:00
|
|
|
// Settings.CPAP.Starting Pressure
|
2014-09-01 04:49:05 +00:00
|
|
|
if ((mod == 0) && (sig = str.lookupLabel("S.C.StartPress"))) {
|
|
|
|
R.ramp_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2015-09-21 01:30:23 +00:00
|
|
|
// Settings.Adaptive Starting Pressure? // mode 11 = APAP for her?
|
2014-09-01 04:49:05 +00:00
|
|
|
if (((mod == 1) || (mod == 11)) && (sig = str.lookupLabel("S.AS.StartPress"))) {
|
|
|
|
R.ramp_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2015-09-21 01:30:23 +00:00
|
|
|
|
|
|
|
if ((R.mode == MODE_BILEVEL_FIXED) && (sig = str.lookupLabel("S.BL.StartPress"))) {
|
|
|
|
// Bilevel Starting Pressure
|
|
|
|
R.ramp_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if (((R.mode == MODE_ASV) || (R.mode == MODE_ASV_VARIABLE_EPAP)) && (sig = str.lookupLabel("S.VA.StartPress"))) {
|
|
|
|
// Bilevel Starting Pressure
|
|
|
|
R.ramp_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
}
|
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
if ((sig = str.lookupLabel("Mask Dur"))) {
|
2014-05-18 17:06:58 +00:00
|
|
|
R.maskdur = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
if ((sig = str.lookupLabel("Leak Med"))) {
|
|
|
|
float gain = sig->gain * 60.0;
|
|
|
|
R.leakgain = gain;
|
|
|
|
R.leakmed = EventDataType(sig->data[rec]) * gain + sig->offset;
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
if ((sig = str.lookupLabel("Leak Max"))) {
|
|
|
|
float gain = sig->gain * 60.0;
|
|
|
|
R.leakgain = gain;
|
|
|
|
R.leakmax = EventDataType(sig->data[rec]) * gain + sig->offset;
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
if ((sig = str.lookupLabel("Leak 95"))) {
|
|
|
|
float gain = sig->gain * 60.0;
|
|
|
|
R.leakgain = gain;
|
|
|
|
R.leak95 = EventDataType(sig->data[rec]) * gain + sig->offset;
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
|
|
|
|
2014-07-25 07:53:48 +00:00
|
|
|
bool haveipap = false;
|
2015-09-21 01:30:23 +00:00
|
|
|
// if (R.mode == MODE_BILEVEL_FIXED) {
|
|
|
|
if ((sig = str.lookupSignal(CPAP_IPAP))) {
|
|
|
|
R.ipap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
haveipap = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sig = str.lookupSignal(CPAP_EPAP))) {
|
|
|
|
R.epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
2016-03-04 00:17:19 +00:00
|
|
|
|
|
|
|
if (R.mode == MODE_ASV) {
|
|
|
|
if ((sig = str.lookupLabel("S.AV.StartPress"))) {
|
2016-03-08 13:55:29 +00:00
|
|
|
EventDataType sp = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
R.ramp_pressure = sp;
|
2016-03-04 00:17:19 +00:00
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AV.EPAP"))) {
|
|
|
|
R.min_epap = R.max_epap = R.epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AV.MinPS"))) {
|
|
|
|
R.min_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AV.MaxPS"))) {
|
|
|
|
R.max_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
R.max_ipap = R.epap + R.max_ps;
|
|
|
|
R.min_ipap = R.epap + R.min_ps;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (R.mode == MODE_ASV_VARIABLE_EPAP) {
|
|
|
|
if ((sig = str.lookupLabel("S.AA.StartPress"))) {
|
2016-03-08 13:55:29 +00:00
|
|
|
EventDataType sp = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
R.ramp_pressure = sp;
|
2016-03-04 00:17:19 +00:00
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AA.MinEPAP"))) {
|
|
|
|
R.min_epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AA.MaxEPAP"))) {
|
|
|
|
R.max_epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AA.MinPS"))) {
|
|
|
|
R.min_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.AA.MaxPS"))) {
|
|
|
|
R.max_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
R.max_ipap = R.max_epap + R.max_ps;
|
|
|
|
R.min_ipap = R.min_epap + R.min_ps;
|
|
|
|
}
|
|
|
|
}
|
2015-09-21 01:30:23 +00:00
|
|
|
|
|
|
|
if ((sig = str.lookupSignal(CPAP_PressureMax))) {
|
|
|
|
R.max_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupSignal(CPAP_PressureMin))) {
|
|
|
|
R.min_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupSignal(RMS9_SetPressure))) {
|
|
|
|
R.set_pressure = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sig = str.lookupSignal(CPAP_EPAPHi))) {
|
|
|
|
R.max_epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupSignal(CPAP_EPAPLo))) {
|
|
|
|
R.min_epap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sig = str.lookupSignal(CPAP_IPAPHi))) {
|
|
|
|
R.max_ipap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
haveipap = true;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupSignal(CPAP_IPAPLo))) {
|
|
|
|
R.min_ipap = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
haveipap = true;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupSignal(CPAP_PS))) {
|
|
|
|
R.ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
// }
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Okay, problem here: THere are TWO PSMin & MAX values on the 36037 with the same string
|
|
|
|
// One is for ASV mode, and one is for ASVAuto
|
|
|
|
int psvar = (mode == MODE_ASV_VARIABLE_EPAP) ? 1 : 0;
|
2015-09-21 01:30:23 +00:00
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
if ((sig = str.lookupLabel("Max PS", psvar))) {
|
2014-05-18 17:06:58 +00:00
|
|
|
R.max_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
if ((sig = str.lookupLabel("Min PS", psvar))) {
|
2014-05-18 17:06:58 +00:00
|
|
|
R.min_ps = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2014-08-03 13:00:13 +00:00
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2015-09-21 01:30:23 +00:00
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
if (!haveipap) {
|
|
|
|
}
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
if (mode == MODE_ASV_VARIABLE_EPAP) {
|
|
|
|
R.min_ipap = R.min_epap + R.min_ps;
|
|
|
|
R.max_ipap = R.max_epap + R.max_ps;
|
|
|
|
} else if (mode == MODE_ASV) {
|
|
|
|
R.min_ipap = R.epap + R.min_ps;
|
|
|
|
R.max_ipap = R.epap + R.max_ps;
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
EventDataType epr = -1, epr_level = -1;
|
2014-09-30 16:24:43 +00:00
|
|
|
if ((mode == MODE_CPAP) || (mode == MODE_APAP)) {
|
|
|
|
if ((sig = str.lookupSignal(RMS9_EPR))) {
|
|
|
|
epr= EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupSignal(RMS9_EPRLevel))) {
|
|
|
|
epr_level= EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
2014-08-03 13:00:13 +00:00
|
|
|
|
2014-09-30 16:24:43 +00:00
|
|
|
if ((sig = str.lookupLabel("S.EPR.EPRType"))) {
|
|
|
|
epr = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
epr += 1;
|
|
|
|
}
|
2016-03-03 08:28:45 +00:00
|
|
|
int epr_on=0, clin_epr_on=0;
|
|
|
|
if ((sig = str.lookupLabel("S.EPR.EPREnable"))) { // first check machines opinion
|
|
|
|
epr_on = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if (epr_on && (sig = str.lookupLabel("S.EPR.ClinEnable"))) {
|
|
|
|
clin_epr_on = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if (!(epr_on && clin_epr_on)) {
|
|
|
|
epr = 0;
|
|
|
|
epr_level = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-17 17:20:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
if ((epr >= 0) && (epr_level >= 0)) {
|
|
|
|
R.epr_level = epr_level;
|
|
|
|
R.epr = epr;
|
|
|
|
} else {
|
|
|
|
if (epr >= 0) {
|
|
|
|
static bool warn=false;
|
|
|
|
if (!warn) { // just nag once
|
|
|
|
qDebug() << "If you can read this, please tell Jedimark you found a ResMed with EPR but no EPR_LEVEL so he can remove this warning";
|
|
|
|
warn = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
R.epr = (epr > 0) ? 1 : 0;
|
|
|
|
R.epr_level = epr;
|
|
|
|
} else if (epr_level >= 0) {
|
|
|
|
R.epr_level = epr_level;
|
|
|
|
R.epr = (epr_level > 0) ? 1 : 0;
|
|
|
|
}
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
2014-07-25 07:53:48 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
if ((sig = str.lookupLabel("AHI"))) {
|
|
|
|
R.ahi = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("AI"))) {
|
|
|
|
R.ai = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("HI"))) {
|
|
|
|
R.hi = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("UAI"))) {
|
|
|
|
R.uai = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("CAI"))) {
|
|
|
|
R.cai = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
if ((sig = str.lookupLabel("S.RampTime"))) {
|
|
|
|
R.s_RampTime = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.RampEnable"))) {
|
|
|
|
R.s_RampEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.EPR.ClinEnable"))) {
|
|
|
|
R.s_EPR_ClinEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.EPR.EPREnable"))) {
|
|
|
|
R.s_EPREnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sig = str.lookupLabel("S.ABFilter"))) {
|
|
|
|
R.s_ABFilter = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sig = str.lookupLabel("S.ClimateControl"))) {
|
|
|
|
R.s_ClimateControl = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sig = str.lookupLabel("S.Mask"))) {
|
|
|
|
R.s_Mask = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.PtAccess"))) {
|
|
|
|
R.s_PtAccess = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.SmartStart"))) {
|
|
|
|
R.s_SmartStart = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.HumEnable"))) {
|
|
|
|
R.s_HumEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.HumLevel"))) {
|
|
|
|
R.s_HumLevel = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.TempEnable"))) {
|
|
|
|
R.s_TempEnable = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.Temp"))) {
|
|
|
|
R.s_Temp = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
if ((sig = str.lookupLabel("S.Tube"))) {
|
|
|
|
R.s_Tube = EventDataType(sig->data[rec]) * sig->gain + sig->offset;
|
|
|
|
}
|
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
laston = ontime;
|
|
|
|
|
2015-08-11 20:01:24 +00:00
|
|
|
// TODO UTC
|
2014-05-18 17:06:58 +00:00
|
|
|
QDateTime dontime = QDateTime::fromTime_t(ontime);
|
|
|
|
date = dontime.date();
|
|
|
|
R.date = date;
|
2014-06-20 02:06:57 +00:00
|
|
|
|
|
|
|
//CHECKME: Should I be taking noon day split time into account here?
|
2014-05-20 11:51:47 +00:00
|
|
|
strdate[date].push_back(&strsess.insert(ontime, R).value());
|
|
|
|
|
2014-07-13 10:03:00 +00:00
|
|
|
//QDateTime dofftime = QDateTime::fromTime_t(offtime);
|
|
|
|
//qDebug() << "Mask on" << dontime << "Mask off" << dofftime;
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
// Wait... ResMed has a DST bug here...should I be replicating it by using multiples of 86400 seconds?
|
2014-07-10 15:51:14 +00:00
|
|
|
dt = dt.addDays(1);
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-05 06:14:29 +00:00
|
|
|
// Read a 16 bits integer
|
2011-06-29 16:19:57 +00:00
|
|
|
qint16 EDFParser::Read16()
|
|
|
|
{
|
2014-05-05 06:14:29 +00:00
|
|
|
if ((pos + 2) > filesize) {
|
2014-04-09 22:47:23 +00:00
|
|
|
return 0;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2014-04-09 22:47:23 +00:00
|
|
|
|
2014-05-13 21:30:06 +00:00
|
|
|
#ifdef Q_LITTLE_ENDIAN
|
2014-05-05 06:14:29 +00:00
|
|
|
// Intel, etc...
|
2014-04-09 22:47:23 +00:00
|
|
|
qint16 res = *(qint16 *)&buffer[pos];
|
2014-05-05 06:14:29 +00:00
|
|
|
#else
|
|
|
|
// ARM, PPC, etc..
|
2014-05-13 21:30:06 +00:00
|
|
|
qint16 res = quint8(buffer[pos]) | (qint8(buffer[pos+1]) << 8);
|
2014-05-05 06:14:29 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
pos += 2;
|
2011-06-29 16:19:57 +00:00
|
|
|
return res;
|
|
|
|
}
|
2014-05-05 06:14:29 +00:00
|
|
|
|
2014-04-09 22:47:23 +00:00
|
|
|
QString EDFParser::Read(unsigned n)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2014-05-08 04:46:23 +00:00
|
|
|
if ((pos + long(n)) > filesize) {
|
2014-04-09 22:47:23 +00:00
|
|
|
return "";
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2014-04-09 22:47:23 +00:00
|
|
|
|
2014-07-06 12:44:59 +00:00
|
|
|
QByteArray buf(&buffer[pos], n);
|
|
|
|
pos+=n;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-07-06 12:44:59 +00:00
|
|
|
return buf.trimmed();
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
|
|
|
bool EDFParser::Parse()
|
|
|
|
{
|
|
|
|
bool ok;
|
2014-04-17 05:58:57 +00:00
|
|
|
QString temp, temp2;
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
version = QString::fromLatin1(header.version, 8).toLong(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
2011-06-29 14:19:38 +00:00
|
|
|
return false;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2013-01-17 18:26:11 +00:00
|
|
|
//patientident=QString::fromLatin1(header.patientident,80);
|
2014-04-17 05:58:57 +00:00
|
|
|
recordingident = QString::fromLatin1(header.recordingident, 80); // Serial number is in here..
|
|
|
|
int snp = recordingident.indexOf("SRN=");
|
2011-07-02 14:35:50 +00:00
|
|
|
serialnumber.clear();
|
2011-07-05 05:37:11 +00:00
|
|
|
/*char * idx=index(header.recordingident,'=');
|
|
|
|
idx++;
|
2011-07-02 14:35:50 +00:00
|
|
|
for (int i=0;i<16;++i) {
|
|
|
|
if (*idx==0x20) break;
|
|
|
|
serialnumber+=*idx;
|
|
|
|
++idx;
|
2011-07-05 05:37:11 +00:00
|
|
|
} */
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int i = snp + 4; i < recordingident.length(); i++) {
|
|
|
|
if (recordingident[i] == ' ') {
|
2011-07-05 05:37:11 +00:00
|
|
|
break;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
serialnumber += recordingident[i];
|
2011-07-05 05:37:11 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-09-30 11:02:35 +00:00
|
|
|
QDateTime startDate = QDateTime::fromString(QString::fromLatin1(header.datetime, 16), "dd.MM.yyHH.mm.ss");
|
2011-12-07 12:23:19 +00:00
|
|
|
//startDate.toTimeSpec(Qt::UTC);
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QDate d2 = startDate.date();
|
|
|
|
|
|
|
|
if (d2.year() < 2000) {
|
|
|
|
d2.setDate(d2.year() + 100, d2.month(), d2.day());
|
2011-07-02 14:35:50 +00:00
|
|
|
startDate.setDate(d2);
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-07-02 14:35:50 +00:00
|
|
|
if (!startDate.isValid()) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Invalid date time retreieved parsing EDF File " << filename;
|
2011-06-29 16:19:57 +00:00
|
|
|
return false;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
startdate = qint64(startDate.toTime_t()) * 1000L;
|
2011-12-07 12:23:19 +00:00
|
|
|
//startdate-=timezoneOffset();
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-07-21 03:38:17 +00:00
|
|
|
//qDebug() << startDate.toString("yyyy-MM-dd HH:mm:ss");
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
num_header_bytes = QString::fromLatin1(header.num_header_bytes, 8).toLong(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
2011-06-29 14:19:38 +00:00
|
|
|
return false;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
2013-01-17 18:26:11 +00:00
|
|
|
//reserved44=QString::fromLatin1(header.reserved,44);
|
2014-04-17 05:58:57 +00:00
|
|
|
num_data_records = QString::fromLatin1(header.num_data_records, 8).toLong(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
2011-06-29 14:19:38 +00:00
|
|
|
return false;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
2014-06-20 02:06:57 +00:00
|
|
|
dur_data_record = (QString::fromLatin1(header.dur_data_records, 8).toDouble(&ok) * 1000.0L);
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (!ok) {
|
2011-06-29 14:19:38 +00:00
|
|
|
return false;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
num_signals = QString::fromLatin1(header.num_signals, 4).toLong(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
2011-06-29 14:19:38 +00:00
|
|
|
return false;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
enddate = startdate + dur_data_record * qint64(num_data_records);
|
|
|
|
// if (dur_data_record==0)
|
|
|
|
// return false;
|
2011-07-02 14:35:50 +00:00
|
|
|
|
|
|
|
// this could be loaded quicker by transducer_type[signal] etc..
|
|
|
|
|
2014-04-09 19:39:03 +00:00
|
|
|
// Initialize fixed-size signal list.
|
|
|
|
edfsignals.resize(num_signals);
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) {
|
2014-05-18 17:06:58 +00:00
|
|
|
EDFSignal &sig = edfsignals[i];
|
|
|
|
sig.data = nullptr;
|
|
|
|
sig.label = Read(16);
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
signal_labels.push_back(sig.label);
|
2014-08-26 07:55:01 +00:00
|
|
|
signalList[sig.label].push_back(&sig);
|
2014-05-18 17:06:58 +00:00
|
|
|
signal.push_back(&sig);
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].transducer_type = Read(80); }
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].physical_dimension = Read(8); }
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].physical_minimum = Read(8).toDouble(&ok); }
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].physical_maximum = Read(8).toDouble(&ok); }
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].digital_minimum = Read(8).toDouble(&ok); }
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) {
|
2014-04-09 19:39:03 +00:00
|
|
|
EDFSignal &e = edfsignals[i];
|
2014-04-17 05:58:57 +00:00
|
|
|
e.digital_maximum = Read(8).toDouble(&ok);
|
|
|
|
e.gain = (e.physical_maximum - e.physical_minimum) / (e.digital_maximum - e.digital_minimum);
|
|
|
|
e.offset = 0;
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].prefiltering = Read(80); }
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].nr = Read(8).toLong(&ok); }
|
|
|
|
|
|
|
|
for (int i = 0; i < num_signals; i++) { edfsignals[i].reserved = Read(32); }
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
// allocate the buffers
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int i = 0; i < num_signals; i++) {
|
2011-06-29 14:19:38 +00:00
|
|
|
//qDebug//cout << "Reading signal " << signals[i]->label << endl;
|
2014-04-09 19:39:03 +00:00
|
|
|
EDFSignal &sig = edfsignals[i];
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
long recs = sig.nr * num_data_records;
|
|
|
|
|
|
|
|
if (num_data_records < 0) {
|
2014-07-26 12:46:11 +00:00
|
|
|
sig.data = nullptr;
|
|
|
|
continue;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sig.data = new qint16 [recs];
|
|
|
|
sig.pos = 0;
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int x = 0; x < num_data_records; x++) {
|
|
|
|
for (int i = 0; i < num_signals; i++) {
|
2014-04-09 19:39:03 +00:00
|
|
|
EDFSignal &sig = edfsignals[i];
|
2014-05-13 21:30:06 +00:00
|
|
|
#ifdef Q_LITTLE_ENDIAN
|
2014-05-05 06:14:29 +00:00
|
|
|
// Intel x86, etc..
|
2014-04-17 05:58:57 +00:00
|
|
|
memcpy((char *)&sig.data[sig.pos], (char *)&buffer[pos], sig.nr * 2);
|
|
|
|
sig.pos += sig.nr;
|
|
|
|
pos += sig.nr * 2;
|
2014-05-05 06:14:29 +00:00
|
|
|
#else
|
2014-05-14 05:15:06 +00:00
|
|
|
// Big endian safe
|
2014-05-05 06:14:29 +00:00
|
|
|
for (int j=0;j<sig.nr;j++) {
|
2011-06-29 16:19:57 +00:00
|
|
|
qint16 t=Read16();
|
|
|
|
sig.data[sig.pos++]=t;
|
2014-05-05 06:14:29 +00:00
|
|
|
}
|
|
|
|
#endif
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool EDFParser::Open(QString name)
|
|
|
|
{
|
2014-05-05 06:14:29 +00:00
|
|
|
Q_ASSERT(buffer == nullptr);
|
2011-12-19 05:35:05 +00:00
|
|
|
|
2013-09-14 23:32:14 +00:00
|
|
|
if (name.endsWith(STR_ext_gz)) {
|
2014-05-05 06:14:29 +00:00
|
|
|
// Open and decempress file to buffer
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
filename = name.mid(0, -3);
|
2014-05-05 06:14:29 +00:00
|
|
|
|
|
|
|
// Get file length from inside gzip file
|
2012-01-05 04:37:22 +00:00
|
|
|
QFile fi(name);
|
2014-05-05 06:14:29 +00:00
|
|
|
|
|
|
|
if (!fi.open(QFile::ReadOnly) || !fi.seek(fi.size() - 4)) {
|
|
|
|
goto badfile;
|
|
|
|
}
|
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
unsigned char ch[4];
|
2014-04-17 05:58:57 +00:00
|
|
|
fi.read((char *)ch, 4);
|
|
|
|
filesize = ch[0] | (ch [1] << 8) | (ch[2] << 16) | (ch[3] << 24);
|
|
|
|
|
2014-05-05 06:14:29 +00:00
|
|
|
datasize = filesize - EDFHeaderSize;
|
|
|
|
if (datasize < 0) {
|
|
|
|
goto badfile;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-05-05 06:14:29 +00:00
|
|
|
// Open gzip file for reading
|
2014-04-17 05:58:57 +00:00
|
|
|
gzFile f = gzopen(name.toLatin1(), "rb");
|
2012-01-05 04:37:22 +00:00
|
|
|
if (!f) {
|
2014-05-05 06:14:29 +00:00
|
|
|
goto badfile;
|
2012-01-05 04:37:22 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-05-05 06:14:29 +00:00
|
|
|
// Decompressed header and data block
|
2014-04-17 05:58:57 +00:00
|
|
|
gzread(f, (char *)&header, EDFHeaderSize);
|
|
|
|
buffer = new char [datasize];
|
|
|
|
gzread(f, buffer, datasize);
|
2012-01-05 04:37:22 +00:00
|
|
|
gzclose(f);
|
|
|
|
} else {
|
2014-05-05 06:14:29 +00:00
|
|
|
|
|
|
|
// Open and read uncompressed file
|
2012-01-05 04:37:22 +00:00
|
|
|
QFile f(name);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (!f.open(QIODevice::ReadOnly)) {
|
2014-05-05 06:14:29 +00:00
|
|
|
goto badfile;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
filename = name;
|
|
|
|
filesize = f.size();
|
|
|
|
datasize = filesize - EDFHeaderSize;
|
2012-01-05 04:37:22 +00:00
|
|
|
|
2014-05-05 06:14:29 +00:00
|
|
|
if (datasize < 0) {
|
|
|
|
goto badfile;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
f.read((char *)&header, EDFHeaderSize);
|
2014-05-05 06:14:29 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
buffer = new char [datasize];
|
|
|
|
f.read(buffer, datasize);
|
2012-01-05 04:37:22 +00:00
|
|
|
f.close();
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
pos = 0;
|
2011-06-29 16:19:57 +00:00
|
|
|
return true;
|
2014-05-05 06:14:29 +00:00
|
|
|
|
|
|
|
badfile:
|
|
|
|
qDebug() << "EDFParser::Open() Couldn't open file" << name;
|
|
|
|
return false;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
|
|
|
|
void ResmedImport::run()
|
|
|
|
{
|
2014-07-25 07:53:48 +00:00
|
|
|
loader->saveMutex.lock();
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
Session * sess = mach->SessionExists(sessionid);
|
|
|
|
if (sess) {
|
2014-07-02 11:58:36 +00:00
|
|
|
if (sess->summaryOnly()) {
|
2014-05-20 11:51:47 +00:00
|
|
|
// Reuse this session
|
|
|
|
sess->wipeSummary();
|
|
|
|
} else {
|
|
|
|
// Already imported
|
2014-07-25 07:53:48 +00:00
|
|
|
loader->saveMutex.unlock();
|
2014-05-20 11:51:47 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Could be importing from an older backup.. if so, destroy the summary only records
|
|
|
|
quint32 key = int(sessionid / 60) * 60;
|
2014-07-25 07:53:48 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
sess = mach->SessionExists(key);
|
|
|
|
if (sess) {
|
2014-07-02 11:58:36 +00:00
|
|
|
if (sess->summaryOnly()) {
|
2014-07-25 07:53:48 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
sess->Destroy();
|
|
|
|
delete sess;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the session
|
|
|
|
sess = new Session(mach, sessionid);
|
|
|
|
}
|
2014-07-25 07:53:48 +00:00
|
|
|
loader->saveMutex.unlock();
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2015-08-02 05:53:09 +00:00
|
|
|
if (files.contains(EDF_PLD)) {
|
|
|
|
QStringList & sl = files[EDF_PLD];
|
|
|
|
for (int i=0; i<sl.size(); ++i) {
|
|
|
|
QString file = sl.at(i);
|
|
|
|
loader->LoadPLD(sess, file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
2015-08-02 05:53:09 +00:00
|
|
|
sess->session_files.append(file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#endif
|
2015-08-02 05:53:09 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
}
|
2015-08-02 05:53:09 +00:00
|
|
|
if (files.contains(EDF_BRP)) {
|
|
|
|
QStringList & sl = files[EDF_BRP];
|
|
|
|
for (int i=0; i<sl.size(); ++i) {
|
|
|
|
QString file = sl.at(i);
|
|
|
|
|
|
|
|
loader->LoadBRP(sess, file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
2015-08-02 05:53:09 +00:00
|
|
|
sess->session_files.append(file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#endif
|
2015-08-02 05:53:09 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
}
|
2015-08-02 05:53:09 +00:00
|
|
|
if (files.contains(EDF_SAD)) {
|
|
|
|
QStringList & sl = files[EDF_SAD];
|
|
|
|
for (int i=0; i<sl.size(); ++i) {
|
|
|
|
QString file = sl.at(i);
|
|
|
|
loader->LoadSAD(sess, file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
2015-08-02 05:53:09 +00:00
|
|
|
sess->session_files.append(file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#endif
|
2015-08-02 05:53:09 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
}
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2015-08-02 05:53:09 +00:00
|
|
|
if (files.contains(EDF_CSL)) {
|
|
|
|
QStringList & sl = files[EDF_CSL];
|
|
|
|
for (int i=0; i<sl.size(); ++i) {
|
|
|
|
QString file = sl.at(i);
|
|
|
|
loader->LoadCSL(sess, file);
|
2014-09-22 04:32:15 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
2015-08-02 05:53:09 +00:00
|
|
|
sess->session_files.append(file);
|
2014-09-22 04:32:15 +00:00
|
|
|
#endif
|
2015-08-02 05:53:09 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
}
|
2014-09-17 06:12:38 +00:00
|
|
|
|
|
|
|
bool haveeve = false;
|
2015-08-02 05:53:09 +00:00
|
|
|
if (files.contains(EDF_EVE)) {
|
|
|
|
QStringList & sl = files[EDF_EVE];
|
|
|
|
for (int i=0; i<sl.size(); ++i) {
|
|
|
|
QString file = sl.at(i);
|
|
|
|
loader->LoadEVE(sess, file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
2015-08-02 05:53:09 +00:00
|
|
|
sess->session_files.append(file);
|
2014-09-01 04:49:05 +00:00
|
|
|
#endif
|
2015-08-02 05:53:09 +00:00
|
|
|
|
|
|
|
haveeve = true;
|
|
|
|
}
|
2014-09-17 06:12:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!haveeve) {
|
|
|
|
sess->AddEventList(CPAP_Obstructive, EVL_Event);
|
|
|
|
sess->AddEventList(CPAP_ClearAirway, EVL_Event);
|
|
|
|
sess->AddEventList(CPAP_Apnea, EVL_Event);
|
|
|
|
sess->AddEventList(CPAP_Hypopnea, EVL_Event);
|
2014-09-01 04:49:05 +00:00
|
|
|
}
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2014-07-25 07:53:48 +00:00
|
|
|
if (sess->first() == 0) {
|
2014-07-26 12:46:11 +00:00
|
|
|
// loader->saveMutex.lock();
|
|
|
|
|
2014-07-25 07:53:48 +00:00
|
|
|
//if (mach->sessionlist.contains(sess->session())) {
|
2014-07-26 12:46:11 +00:00
|
|
|
// sess->Destroy();
|
2014-07-25 07:53:48 +00:00
|
|
|
//mach->sessionlist.remove(sess->session());
|
|
|
|
//}
|
2014-05-20 11:51:47 +00:00
|
|
|
delete sess;
|
2014-07-26 12:46:11 +00:00
|
|
|
//loader->saveMutex.unlock();
|
2014-05-20 11:51:47 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-02 11:58:36 +00:00
|
|
|
sess->setSummaryOnly(false);
|
2014-05-20 11:51:47 +00:00
|
|
|
sess->SetChanged(true);
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Process STR.edf now all valid Session data is imported
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
quint32 key = quint32(sessionid / 60) * 60; // round to 1 minute
|
|
|
|
|
|
|
|
QMap<quint32, STRRecord>::iterator strsess_end = loader->strsess.end();
|
|
|
|
QMap<quint32, STRRecord>::iterator it = loader->strsess.find(key);
|
|
|
|
|
|
|
|
if (it == strsess_end) {
|
|
|
|
// ResMed merges mask on/off groups that are less than a minute apart
|
|
|
|
// this means have to jump back to the last session closest.
|
|
|
|
|
|
|
|
it = loader->strsess.lowerBound(key);
|
|
|
|
if (it != loader->strsess.begin()) it--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (it != strsess_end) {
|
|
|
|
STRRecord & R = it.value();
|
|
|
|
|
|
|
|
// calculate the time between session record and mask-on record.
|
|
|
|
int gap = sessionid - R.maskon;
|
|
|
|
|
|
|
|
if (gap > 3600*6) {
|
|
|
|
QDateTime dt = QDateTime::fromTime_t(sessionid);
|
|
|
|
QDateTime rt = QDateTime::fromTime_t(R.maskon);
|
|
|
|
|
2014-06-20 06:17:27 +00:00
|
|
|
QString msg = QString("Warning: Closest matching STR record for %1 is %2 by %3 seconds").
|
|
|
|
arg(dt.toString(Qt::ISODate)).
|
|
|
|
arg(sess->length() / 1000.0L,0,'f',1).
|
|
|
|
arg(gap);
|
|
|
|
qDebug() << msg;
|
2014-05-20 11:51:47 +00:00
|
|
|
}
|
|
|
|
|
2014-06-20 06:17:27 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
// Claim this session
|
|
|
|
R.sessionid = sessionid;
|
|
|
|
|
2014-07-09 03:49:20 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
// Save maskon time in session setting so we can use it later to avoid doubleups.
|
|
|
|
sess->settings[RMS9_MaskOnTime] = R.maskon;
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
|
|
|
sess->session_files.append("STR.edf");
|
|
|
|
#endif
|
|
|
|
|
2014-08-04 19:57:48 +00:00
|
|
|
if (R.mode >= 0) {
|
|
|
|
sess->settings[CPAP_Mode] = R.mode;
|
2014-09-01 04:49:05 +00:00
|
|
|
sess->settings[RMS9_Mode] = R.rms9_mode;
|
2014-08-04 19:57:48 +00:00
|
|
|
if (R.mode == MODE_CPAP) {
|
|
|
|
if (R.set_pressure >= 0) {
|
|
|
|
sess->settings[CPAP_Pressure] = R.set_pressure;
|
|
|
|
}
|
|
|
|
} else if (R.mode == MODE_APAP) {
|
|
|
|
if (R.min_pressure >= 0) sess->settings[CPAP_PressureMin] = R.min_pressure;
|
|
|
|
if (R.max_pressure >= 0) sess->settings[CPAP_PressureMax] = R.max_pressure;
|
|
|
|
} else if (R.mode == MODE_BILEVEL_FIXED) {
|
|
|
|
if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap;
|
|
|
|
if (R.ipap >= 0) sess->settings[CPAP_IPAP] = R.ipap;
|
|
|
|
if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps;
|
|
|
|
} else if (R.mode == MODE_BILEVEL_AUTO_FIXED_PS) {
|
|
|
|
if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps;
|
|
|
|
} else if (R.mode == MODE_ASV) {
|
|
|
|
if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap;
|
|
|
|
if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps;
|
|
|
|
if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
} else if (R.mode == MODE_ASV_VARIABLE_EPAP) {
|
|
|
|
if (R.max_epap >= 0) sess->settings[CPAP_EPAPHi] = R.max_epap;
|
|
|
|
if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
if (R.min_ipap >= 0) sess->settings[CPAP_IPAPLo] = R.min_ipap;
|
|
|
|
if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps;
|
|
|
|
if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (R.set_pressure >= 0) sess->settings[CPAP_Pressure] = R.set_pressure;
|
|
|
|
if (R.min_pressure >= 0) sess->settings[CPAP_PressureMin] = R.min_pressure;
|
|
|
|
if (R.max_pressure >= 0) sess->settings[CPAP_PressureMax] = R.max_pressure;
|
|
|
|
if (R.max_epap >= 0) sess->settings[CPAP_EPAPHi] = R.max_epap;
|
|
|
|
if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
if (R.min_ipap >= 0) sess->settings[CPAP_IPAPLo] = R.min_ipap;
|
|
|
|
if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps;
|
|
|
|
if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps;
|
|
|
|
if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps;
|
|
|
|
if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap;
|
|
|
|
if (R.ipap >= 0) sess->settings[CPAP_IPAP] = R.ipap;
|
|
|
|
}
|
2014-06-20 02:53:12 +00:00
|
|
|
|
|
|
|
if (R.epr >= 0) {
|
2014-07-02 13:45:53 +00:00
|
|
|
sess->settings[RMS9_EPR] = (int)R.epr;
|
2014-08-04 19:57:48 +00:00
|
|
|
if (R.epr > 0) {
|
|
|
|
if (R.epr_level >= 0) {
|
|
|
|
sess->settings[RMS9_EPRLevel] = (int)R.epr_level;
|
|
|
|
}
|
|
|
|
}
|
2016-03-08 13:55:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (R.s_RampEnable >= 0) {
|
|
|
|
sess->settings[RMS9_RampEnable] = R.s_RampEnable;
|
|
|
|
|
|
|
|
if (R.s_RampEnable >= 1) {
|
|
|
|
if (R.s_RampTime >= 0) {
|
|
|
|
sess->settings[CPAP_RampTime] = R.s_RampTime;
|
|
|
|
}
|
|
|
|
if (R.ramp_pressure >= 0) {
|
|
|
|
sess->settings[CPAP_RampPressure] = R.ramp_pressure;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (R.s_SmartStart >= 0) {
|
|
|
|
sess->settings[RMS9_SmartStart] = R.s_SmartStart;
|
|
|
|
}
|
|
|
|
if (R.s_ABFilter >= 0) {
|
|
|
|
sess->settings[RMS9_ABFilter] = R.s_ABFilter;
|
|
|
|
}
|
|
|
|
if (R.s_ClimateControl >= 0) {
|
|
|
|
sess->settings[RMS9_ClimateControl] = R.s_ClimateControl;
|
|
|
|
}
|
|
|
|
if (R.s_Mask >= 0) {
|
|
|
|
sess->settings[RMS9_Mask] = R.s_Mask;
|
|
|
|
}
|
|
|
|
if (R.s_PtAccess >= 0) {
|
|
|
|
sess->settings[RMS9_PtAccess] = R.s_PtAccess;
|
|
|
|
}
|
2014-08-04 19:57:48 +00:00
|
|
|
|
2016-03-08 13:55:29 +00:00
|
|
|
if (R.s_HumEnable >= 0) {
|
|
|
|
sess->settings[RMS9_HumidStatus] = (short)R.s_HumEnable;
|
|
|
|
if ((R.s_HumEnable >= 1) && (R.s_HumLevel >= 0)) {
|
|
|
|
sess->settings[RMS9_HumidLevel] = (short)R.s_HumLevel;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (R.s_TempEnable >= 0) {
|
|
|
|
sess->settings[RMS9_TempEnable] = (short)R.s_TempEnable;
|
2016-03-08 14:39:56 +00:00
|
|
|
if ((R.s_TempEnable >= 1) && (R.s_Temp >= 0)){
|
2016-03-08 13:55:29 +00:00
|
|
|
sess->settings[RMS9_Temp] = (short)R.s_Temp;
|
|
|
|
}
|
2014-06-20 02:53:12 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
|
|
|
|
// Ignore all the rest of the sumary data, because there is enough available to calculate it with higher accuracy.
|
|
|
|
|
|
|
|
if (sess->length() > 0) {
|
2014-07-30 20:25:06 +00:00
|
|
|
loader->addSession(sess);
|
2014-05-20 11:51:47 +00:00
|
|
|
} else {
|
|
|
|
delete sess;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2016-03-04 03:34:05 +00:00
|
|
|
CPAPMode mode = (CPAPMode)sess->settings[CPAP_Mode].toInt();
|
|
|
|
|
|
|
|
if ((mode >= MODE_BILEVEL_FIXED) && (!sess->eventlist.contains(CPAP_IPAP))) {
|
|
|
|
QVector<EventList *> & evl = sess->eventlist[CPAP_Pressure];
|
|
|
|
|
|
|
|
for (int i=0; i<evl.size(); ++i) {
|
|
|
|
EventList * el = evl[i];
|
|
|
|
sess->eventlist[CPAP_IPAP].push_back(el);
|
|
|
|
}
|
|
|
|
|
|
|
|
sess->eventlist.remove(CPAP_Pressure);
|
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
|
|
|
|
// Update indexes, process waveform and perform flagging
|
|
|
|
sess->UpdateSummaries();
|
|
|
|
|
|
|
|
// Save is not threadsafe
|
|
|
|
loader->saveMutex.lock();
|
2014-07-28 13:56:29 +00:00
|
|
|
sess->Store(mach->getDataPath());
|
2014-05-20 11:51:47 +00:00
|
|
|
loader->saveMutex.unlock();
|
|
|
|
|
|
|
|
// Free the memory used by this session
|
|
|
|
sess->TrashEvents();
|
|
|
|
}
|
2014-05-17 05:04:40 +00:00
|
|
|
|
2011-06-28 02:21:38 +00:00
|
|
|
ResmedLoader::ResmedLoader()
|
|
|
|
{
|
2014-09-17 17:20:01 +00:00
|
|
|
const QString RMS9_ICON = ":/icons/rms9.png";
|
2016-02-28 03:36:58 +00:00
|
|
|
const QString RM10_ICON = ":/icons/airsense10.png";
|
2016-03-03 04:48:08 +00:00
|
|
|
const QString RM10C_ICON = ":/icons/aircurve.png";
|
2014-09-17 17:20:01 +00:00
|
|
|
|
|
|
|
m_pixmaps[STR_ResMed_S9] = QPixmap(RMS9_ICON);
|
|
|
|
m_pixmap_paths[STR_ResMed_S9] = RMS9_ICON;
|
2016-03-02 02:09:32 +00:00
|
|
|
m_pixmaps[STR_ResMed_AirSense10] = QPixmap(RM10_ICON);
|
2014-09-17 17:20:01 +00:00
|
|
|
m_pixmap_paths[STR_ResMed_AirSense10] = RM10_ICON;
|
2016-03-03 04:48:08 +00:00
|
|
|
m_pixmaps[STR_ResMed_AirCurve10] = QPixmap(RM10C_ICON);
|
2016-03-03 08:32:43 +00:00
|
|
|
m_pixmap_paths[STR_ResMed_AirCurve10] = RM10C_ICON;
|
2014-05-25 07:07:08 +00:00
|
|
|
m_type = MT_CPAP;
|
2011-06-28 02:21:38 +00:00
|
|
|
}
|
|
|
|
ResmedLoader::~ResmedLoader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-06-30 11:20:12 +00:00
|
|
|
void ResmedImportStage2::run()
|
|
|
|
{
|
2014-09-01 04:49:05 +00:00
|
|
|
if (R.maskon == R.maskoff) return;
|
2014-06-30 11:20:12 +00:00
|
|
|
Session * sess = new Session(mach, R.maskon);
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2014-06-30 11:20:12 +00:00
|
|
|
sess->really_set_first(qint64(R.maskon) * 1000L);
|
|
|
|
sess->really_set_last(qint64(R.maskoff) * 1000L);
|
|
|
|
|
|
|
|
// Claim this record for future imports
|
|
|
|
sess->settings[RMS9_MaskOnTime] = R.maskon;
|
2014-07-02 11:58:36 +00:00
|
|
|
sess->setSummaryOnly(true);
|
2014-09-17 06:12:38 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
#ifdef SESSION_DEBUG
|
|
|
|
sess->session_files.append("STR.edf");
|
|
|
|
#endif
|
2014-06-30 11:20:12 +00:00
|
|
|
sess->SetChanged(true);
|
|
|
|
|
|
|
|
// First take the settings
|
2014-08-04 19:57:48 +00:00
|
|
|
|
|
|
|
if (R.mode >= 0) {
|
|
|
|
sess->settings[CPAP_Mode] = R.mode;
|
2014-09-01 04:49:05 +00:00
|
|
|
sess->settings[RMS9_Mode] = R.rms9_mode;
|
|
|
|
|
2014-08-04 19:57:48 +00:00
|
|
|
if (R.mode == MODE_CPAP) {
|
|
|
|
if (R.set_pressure >= 0) {
|
|
|
|
sess->settings[CPAP_Pressure] = R.set_pressure;
|
|
|
|
}
|
|
|
|
} else if (R.mode == MODE_APAP) {
|
|
|
|
if (R.min_pressure >= 0) sess->settings[CPAP_PressureMin] = R.min_pressure;
|
|
|
|
if (R.max_pressure >= 0) sess->settings[CPAP_PressureMax] = R.max_pressure;
|
|
|
|
} else if (R.mode == MODE_BILEVEL_FIXED) {
|
|
|
|
if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap;
|
|
|
|
if (R.ipap >= 0) sess->settings[CPAP_IPAP] = R.ipap;
|
|
|
|
if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps;
|
|
|
|
} else if (R.mode == MODE_BILEVEL_AUTO_FIXED_PS) {
|
|
|
|
if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps;
|
|
|
|
} else if (R.mode == MODE_ASV) {
|
|
|
|
if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap;
|
|
|
|
if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps;
|
|
|
|
if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
} else if (R.mode == MODE_ASV_VARIABLE_EPAP) {
|
|
|
|
if (R.max_epap >= 0) sess->settings[CPAP_EPAPHi] = R.max_epap;
|
|
|
|
if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
if (R.min_ipap >= 0) sess->settings[CPAP_IPAPLo] = R.min_ipap;
|
|
|
|
if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps;
|
|
|
|
if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (R.set_pressure >= 0) sess->settings[CPAP_Pressure] = R.set_pressure;
|
|
|
|
if (R.min_pressure >= 0) sess->settings[CPAP_PressureMin] = R.min_pressure;
|
|
|
|
if (R.max_pressure >= 0) sess->settings[CPAP_PressureMax] = R.max_pressure;
|
|
|
|
if (R.max_epap >= 0) sess->settings[CPAP_EPAPHi] = R.max_epap;
|
|
|
|
if (R.min_epap >= 0) sess->settings[CPAP_EPAPLo] = R.min_epap;
|
|
|
|
if (R.max_ipap >= 0) sess->settings[CPAP_IPAPHi] = R.max_ipap;
|
|
|
|
if (R.min_ipap >= 0) sess->settings[CPAP_IPAPLo] = R.min_ipap;
|
|
|
|
if (R.min_ps >= 0) sess->settings[CPAP_PSMin] = R.min_ps;
|
|
|
|
if (R.max_ps >= 0) sess->settings[CPAP_PSMax] = R.max_ps;
|
|
|
|
if (R.ps >= 0) sess->settings[CPAP_PS] = R.ps;
|
|
|
|
if (R.epap >= 0) sess->settings[CPAP_EPAP] = R.epap;
|
|
|
|
if (R.ipap >= 0) sess->settings[CPAP_IPAP] = R.ipap;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-30 11:20:12 +00:00
|
|
|
if (R.epr >= 0) {
|
2014-07-02 13:45:53 +00:00
|
|
|
sess->settings[RMS9_EPR] = (int)R.epr;
|
2014-08-04 19:57:48 +00:00
|
|
|
if (R.epr > 0) {
|
|
|
|
if (R.epr_level >= 0) {
|
|
|
|
sess->settings[RMS9_EPRLevel] = (int)R.epr_level;
|
|
|
|
}
|
|
|
|
}
|
2014-06-30 11:20:12 +00:00
|
|
|
}
|
|
|
|
if (R.leakmax >= 0) sess->setMax(CPAP_Leak, R.leakmax);
|
|
|
|
if (R.leakmax >= 0) sess->setMin(CPAP_Leak, 0);
|
|
|
|
if ((R.leakmed >= 0) && (R.leak95 >= 0) && (R.leakmax >= 0)) {
|
|
|
|
sess->m_timesummary[CPAP_Leak][short(R.leakmax / R.leakgain)]=1;
|
|
|
|
sess->m_timesummary[CPAP_Leak][short(R.leak95 / R.leakgain)]=9;
|
|
|
|
sess->m_timesummary[CPAP_Leak][short(R.leakmed / R.leakgain)]=65;
|
|
|
|
sess->m_timesummary[CPAP_Leak][0]=25;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Find the matching date group for this record
|
|
|
|
QMap<QDate, QList<STRRecord *> >::iterator dtit = loader->strdate.find(R.date);
|
|
|
|
|
|
|
|
// should not be possible, but my brain hurts...
|
|
|
|
Q_ASSERT(dtit != loader->strdate.end());
|
|
|
|
|
|
|
|
if (dtit != loader->strdate.end()) {
|
|
|
|
QList<STRRecord *> & dayrecs = dtit.value();
|
|
|
|
bool hasdatasess=false;
|
|
|
|
EventDataType time=0, totaltime=0;
|
|
|
|
|
|
|
|
for (int c=0; c < dayrecs.size(); ++c) {
|
|
|
|
STRRecord *r = dayrecs[c];
|
|
|
|
if (r->sessionid > 0) {
|
|
|
|
// get complicated.. calculate all the counts for valid sessions, and use the summary to make up the rest
|
|
|
|
hasdatasess = true;
|
|
|
|
}
|
|
|
|
totaltime += r->maskoff - r->maskon;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!hasdatasess) {
|
|
|
|
for (int c=0; c < dayrecs.size(); ++c) {
|
|
|
|
STRRecord *r = dayrecs[c];
|
|
|
|
time = r->maskoff - r->maskon;
|
|
|
|
float ratio = time / totaltime;
|
|
|
|
|
|
|
|
// Add the time weighted proportion of the events counts
|
|
|
|
if (r->ai >= 0) {
|
|
|
|
sess->setCount(CPAP_Obstructive, r->ai * ratio);
|
|
|
|
sess->setCph(CPAP_Obstructive, (r->ai * ratio) / (time / 3600.0));
|
|
|
|
}
|
|
|
|
if (r->uai >= 0) {
|
|
|
|
sess->setCount(CPAP_Apnea, r->uai * ratio);
|
|
|
|
sess->setCph(CPAP_Apnea, (r->uai * ratio) / (time / 3600.0));
|
|
|
|
}
|
|
|
|
if (r->hi >= 0) {
|
|
|
|
sess->setCount(CPAP_Hypopnea, r->hi * ratio);
|
|
|
|
sess->setCph(CPAP_Hypopnea, (r->hi * ratio) / (time / 3600.0));
|
|
|
|
}
|
|
|
|
if (r->cai >= 0) {
|
|
|
|
sess->setCount(CPAP_ClearAirway, r->cai * ratio);
|
|
|
|
sess->setCph(CPAP_ClearAirway, (r->ai * ratio) / (time / 3600.0));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-30 20:25:06 +00:00
|
|
|
loader->addSession(sess);
|
2014-06-30 11:20:12 +00:00
|
|
|
loader->saveMutex.lock();
|
2014-07-28 13:56:29 +00:00
|
|
|
sess->Store(mach->getDataPath());
|
2014-06-30 11:20:12 +00:00
|
|
|
loader->saveMutex.unlock();
|
|
|
|
}
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
long event_cnt = 0;
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
const QString RMS9_STR_datalog = "DATALOG";
|
|
|
|
const QString RMS9_STR_idfile = "Identification.";
|
|
|
|
const QString RMS9_STR_strfile = "STR.";
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-04-28 04:01:55 +00:00
|
|
|
bool ResmedLoader::Detect(const QString & givenpath)
|
|
|
|
{
|
2014-04-28 18:45:33 +00:00
|
|
|
QDir dir(givenpath);
|
2014-04-28 04:01:55 +00:00
|
|
|
|
2014-04-28 18:45:33 +00:00
|
|
|
if (!dir.exists()) {
|
|
|
|
return false;
|
2014-04-28 04:01:55 +00:00
|
|
|
}
|
|
|
|
|
2014-04-28 18:45:33 +00:00
|
|
|
// ResMed drives contain a folder named "DATALOG".
|
|
|
|
if (!dir.exists(RMS9_STR_datalog)) {
|
2014-04-28 04:01:55 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-04-28 18:45:33 +00:00
|
|
|
// They also contain a file named "STR.edf".
|
|
|
|
if (!dir.exists("STR.edf")) {
|
2014-04-28 04:01:55 +00:00
|
|
|
return false;
|
2014-04-28 18:45:33 +00:00
|
|
|
}
|
2014-04-28 04:01:55 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
MachineInfo ResmedLoader::PeekInfo(const QString & path)
|
|
|
|
{
|
|
|
|
if (!Detect(path)) return MachineInfo();
|
|
|
|
|
|
|
|
QFile f(path+"/"+RMS9_STR_idfile+"tgt");
|
|
|
|
|
|
|
|
// Abort if this file is dodgy..
|
|
|
|
if (!f.exists() || !f.open(QIODevice::ReadOnly)) {
|
|
|
|
return MachineInfo();
|
|
|
|
}
|
|
|
|
MachineInfo info = newInfo();
|
|
|
|
|
|
|
|
// Parse # entries into idmap.
|
|
|
|
while (!f.atEnd()) {
|
|
|
|
QString line = f.readLine().trimmed();
|
|
|
|
|
|
|
|
if (!line.isEmpty()) {
|
|
|
|
QString key = line.section(" ", 0, 0).section("#", 1);
|
|
|
|
QString value = line.section(" ", 1);
|
|
|
|
|
|
|
|
if (key == "SRN") { // Serial Number
|
|
|
|
info.serial = value;
|
|
|
|
|
|
|
|
} else if (key == "PNA") { // Product Name
|
|
|
|
value.replace("_"," ");
|
2014-08-26 08:59:35 +00:00
|
|
|
|
2014-09-17 17:20:01 +00:00
|
|
|
if (value.contains(STR_ResMed_S9)) {
|
|
|
|
value.replace(STR_ResMed_S9, "");
|
|
|
|
info.series = STR_ResMed_S9;
|
|
|
|
} else if (value.contains(STR_ResMed_AirSense10)) {
|
|
|
|
value.replace(STR_ResMed_AirSense10, "");
|
|
|
|
info.series = STR_ResMed_AirSense10;
|
2016-03-02 02:09:32 +00:00
|
|
|
} else if (value.contains(STR_ResMed_AirCurve10)) {
|
|
|
|
value.replace(STR_ResMed_AirCurve10, "");
|
|
|
|
info.series = STR_ResMed_AirCurve10;
|
2014-08-26 08:59:35 +00:00
|
|
|
}
|
2014-07-30 20:49:44 +00:00
|
|
|
value.replace("(","");
|
|
|
|
value.replace(")","");
|
|
|
|
if (value.contains("Adapt", Qt::CaseInsensitive)) {
|
|
|
|
if (!value.contains("VPAP")) {
|
|
|
|
value.replace("Adapt", QObject::tr("VPAP Adapt"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
info.model = value.trimmed();
|
2014-07-28 13:56:29 +00:00
|
|
|
} else if (key == "PCD") { // Product Code
|
|
|
|
info.modelnumber = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
struct EDFduration {
|
2014-08-26 07:55:01 +00:00
|
|
|
EDFduration() { start = end = 0; type = EDF_UNKNOWN; }
|
2014-07-27 16:35:49 +00:00
|
|
|
EDFduration(const EDFduration & copy) {
|
|
|
|
path = copy.path;
|
|
|
|
start = copy.start;
|
|
|
|
end = copy.end;
|
2014-08-26 07:55:01 +00:00
|
|
|
type = copy.type;
|
|
|
|
filename = copy.filename;
|
2014-07-27 16:35:49 +00:00
|
|
|
}
|
|
|
|
EDFduration(quint32 start, quint32 end, QString path) :
|
|
|
|
start(start), end(end), path(path) {}
|
|
|
|
quint32 start;
|
|
|
|
quint32 end;
|
|
|
|
QString path;
|
2014-08-26 07:55:01 +00:00
|
|
|
QString filename;
|
|
|
|
EDFType type;
|
2014-07-27 16:35:49 +00:00
|
|
|
};
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
EDFType lookupEDFType(QString text)
|
|
|
|
{
|
|
|
|
if (text == "EVE") {
|
|
|
|
return EDF_EVE;
|
|
|
|
} else if (text =="BRP") {
|
|
|
|
return EDF_BRP;
|
|
|
|
} else if (text == "PLD") {
|
|
|
|
return EDF_PLD;
|
|
|
|
} else if (text == "SAD") {
|
|
|
|
return EDF_SAD;
|
|
|
|
} else if (text == "CSL") {
|
|
|
|
return EDF_CSL;
|
|
|
|
} else return EDF_UNKNOWN;
|
|
|
|
}
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
// Pretend to parse the EVE file to get the duration out of it.
|
2014-09-22 04:32:15 +00:00
|
|
|
int PeekAnnotations(const QString & path, quint32 &start, quint32 &end)
|
2014-09-01 04:49:05 +00:00
|
|
|
{
|
|
|
|
EDFParser edf(path);
|
|
|
|
if (!edf.Parse())
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
QString t;
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//double duration;
|
2014-09-01 04:49:05 +00:00
|
|
|
char *data;
|
|
|
|
char c;
|
|
|
|
long pos;
|
|
|
|
bool sign, ok;
|
|
|
|
double d;
|
|
|
|
double tt;
|
|
|
|
|
|
|
|
int recs = 0;
|
|
|
|
int goodrecs = 0;
|
|
|
|
|
|
|
|
// Notes: Event records have useless duration record.
|
|
|
|
|
|
|
|
start = edf.startdate / 1000L;
|
|
|
|
// Process event annotation records
|
|
|
|
for (int s = 0; s < edf.GetNumSignals(); s++) {
|
|
|
|
recs = edf.edfsignals[s].nr * edf.GetNumDataRecords() * 2;
|
|
|
|
|
|
|
|
data = (char *)edf.edfsignals[s].data;
|
|
|
|
pos = 0;
|
|
|
|
tt = edf.startdate;
|
2014-10-08 16:51:09 +00:00
|
|
|
//duration = 0;
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
while (pos < recs) {
|
|
|
|
c = data[pos];
|
|
|
|
|
|
|
|
if ((c != '+') && (c != '-')) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data[pos++] == '+') { sign = true; }
|
|
|
|
else { sign = false; }
|
|
|
|
|
|
|
|
t = "";
|
|
|
|
c = data[pos];
|
|
|
|
|
|
|
|
do {
|
|
|
|
t += c;
|
|
|
|
pos++;
|
|
|
|
c = data[pos];
|
|
|
|
} while ((c != 20) && (c != 21)); // start code
|
|
|
|
|
|
|
|
d = t.toDouble(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
2014-09-22 04:32:15 +00:00
|
|
|
qDebug() << "Faulty EDF Annotations file " << edf.filename;
|
2014-09-01 04:49:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!sign) { d = -d; }
|
|
|
|
|
|
|
|
tt = edf.startdate + qint64(d * 1000.0);
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//duration = 0;
|
2014-09-01 04:49:05 +00:00
|
|
|
// First entry
|
|
|
|
|
|
|
|
if (data[pos] == 21) {
|
|
|
|
pos++;
|
|
|
|
// get duration.
|
|
|
|
t = "";
|
|
|
|
|
|
|
|
do {
|
|
|
|
t += data[pos];
|
|
|
|
pos++;
|
|
|
|
} while ((data[pos] != 20) && (pos < recs)); // start code
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//duration = t.toDouble(&ok);
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
if (!ok) {
|
2014-09-22 04:32:15 +00:00
|
|
|
qDebug() << "Faulty EDF Annotations file (at %" << pos << ") " << edf.filename;
|
2014-09-01 04:49:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end = (tt / 1000.0);
|
|
|
|
|
|
|
|
while ((data[pos] == 20) && (pos < recs)) {
|
|
|
|
t = "";
|
|
|
|
pos++;
|
|
|
|
|
|
|
|
if (data[pos] == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data[pos] == 20) {
|
|
|
|
pos++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
t += tolower(data[pos++]);
|
|
|
|
} while ((data[pos] != 20) && (pos < recs)); // start code
|
|
|
|
|
|
|
|
if (!t.isEmpty() && (t!="recording starts")) {
|
|
|
|
goodrecs++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pos >= recs) {
|
2014-09-22 04:32:15 +00:00
|
|
|
qDebug() << "Short EDF Annotations file" << edf.filename;
|
2014-09-01 04:49:05 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((data[pos] == 0) && (pos < recs)) { pos++; }
|
|
|
|
|
|
|
|
if (pos >= recs) { break; }
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
return goodrecs;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
// Looks inside an EDF or EDF.gz and grabs the start and duration
|
|
|
|
EDFduration getEDFDuration(QString filename)
|
|
|
|
{
|
2014-09-01 04:49:05 +00:00
|
|
|
QString ext = filename.section("_", -1).section(".",0,0).toUpper();
|
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
bool ok1, ok2;
|
|
|
|
|
|
|
|
int num_records;
|
|
|
|
double rec_duration;
|
|
|
|
|
|
|
|
QDateTime startDate;
|
|
|
|
|
|
|
|
if (!filename.endsWith(".gz", Qt::CaseInsensitive)) {
|
|
|
|
QFile file(filename);
|
|
|
|
if (!file.open(QFile::ReadOnly)) {
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!file.seek(0xa8)) {
|
|
|
|
file.close();
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
QByteArray bytes = file.read(16).trimmed();
|
|
|
|
|
|
|
|
startDate = QDateTime::fromString(QString::fromLatin1(bytes, 16), "dd.MM.yyHH.mm.ss");
|
|
|
|
|
|
|
|
if (!file.seek(0xec)) {
|
|
|
|
file.close();
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes = file.read(8).trimmed();
|
|
|
|
num_records = bytes.toInt(&ok1);
|
|
|
|
bytes = file.read(8).trimmed();
|
|
|
|
rec_duration = bytes.toDouble(&ok2);
|
2014-08-11 06:13:25 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
file.close();
|
|
|
|
} else {
|
|
|
|
gzFile f = gzopen(filename.toLatin1(), "rb");
|
|
|
|
if (!f) {
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!gzseek(f, 0xa8, SEEK_SET)) {
|
|
|
|
gzclose(f);
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
char datebytes[17] = {0};
|
|
|
|
gzread(f, (char *)&datebytes, 16);
|
|
|
|
QString str = QString(QString::fromLatin1(datebytes,16)).trimmed();
|
|
|
|
|
|
|
|
startDate = QDateTime::fromString(str, "dd.MM.yyHH.mm.ss");
|
|
|
|
|
|
|
|
if (!gzseek(f, 0xec-0xa8-16, SEEK_CUR)) { // 0xec
|
|
|
|
gzclose(f);
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decompressed header and data block
|
|
|
|
char cbytes[9] = {0};
|
|
|
|
gzread(f, (char *)&cbytes, 8);
|
|
|
|
str = QString(cbytes).trimmed();
|
|
|
|
num_records = str.toInt(&ok1);
|
|
|
|
|
|
|
|
gzread(f, (char *)&cbytes, 8);
|
|
|
|
str = QString(cbytes).trimmed();
|
|
|
|
rec_duration = str.toDouble(&ok2);
|
|
|
|
|
|
|
|
gzclose(f);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
QDate d2 = startDate.date();
|
|
|
|
|
|
|
|
if (d2.year() < 2000) {
|
|
|
|
d2.setDate(d2.year() + 100, d2.month(), d2.day());
|
|
|
|
startDate.setDate(d2);
|
|
|
|
}
|
|
|
|
if (!startDate.isValid()) {
|
|
|
|
qDebug() << "Invalid date time retreieved parsing EDF duration for" << filename;
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(ok1 && ok2)) {
|
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
quint32 start = startDate.toTime_t();
|
|
|
|
quint32 end = start + rec_duration * num_records;
|
|
|
|
|
|
|
|
QString filedate = filename.section("/",-1).section("_",0,1);
|
|
|
|
|
|
|
|
QDateTime dt2 = QDateTime::fromString(filedate, "yyyyMMdd_hhmmss");
|
|
|
|
quint32 st2 = dt2.toTime_t();
|
|
|
|
|
2014-08-20 03:03:01 +00:00
|
|
|
start = qMin(st2, start);
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
if (end < start) end = qMax(st2, start);
|
2014-08-11 06:13:25 +00:00
|
|
|
|
2014-09-22 04:32:15 +00:00
|
|
|
if ((ext == "EVE") || (ext == "CSL")) {
|
2014-09-01 04:49:05 +00:00
|
|
|
// S10 Forces us to parse EVE files to find their real durations
|
|
|
|
quint32 en2;
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
// Have to get the actual duration of the EVE file by parsing the annotations. :(
|
2014-09-22 04:32:15 +00:00
|
|
|
int recs = PeekAnnotations(filename, st2, en2);
|
2014-09-01 04:49:05 +00:00
|
|
|
if (recs > 0) {
|
|
|
|
start = qMin(st2, start);
|
|
|
|
end = qMax(en2, end);
|
|
|
|
EDFduration dur(start, end, filename);
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
dur.type = lookupEDFType(ext.toUpper());
|
2014-08-20 03:03:01 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
return dur;
|
|
|
|
} else {
|
2014-09-22 04:32:15 +00:00
|
|
|
// empty annotations file, don't give a crap about it...
|
2014-09-01 04:49:05 +00:00
|
|
|
return EDFduration(0, 0, filename);
|
|
|
|
}
|
|
|
|
// A Firmware bug causes (perhaps with failing SD card) sessions to sometimes take a long time to write
|
|
|
|
}
|
2014-07-27 16:35:49 +00:00
|
|
|
|
|
|
|
EDFduration dur(start, end, filename);
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
dur.type = lookupEDFType(ext.toUpper());
|
2014-08-20 03:03:01 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
return dur;
|
|
|
|
}
|
|
|
|
|
2014-07-29 14:38:59 +00:00
|
|
|
int ResmedLoader::scanFiles(Machine * mach, QString datalog_path)
|
2014-07-27 16:35:49 +00:00
|
|
|
{
|
2014-07-28 13:56:29 +00:00
|
|
|
QHash<QString, SessionID> skipfiles;
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
bool create_backups = true; //p_profile->session->backupCardData();
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
QString backup_path = mach->getBackupPath();
|
2014-07-27 16:35:49 +00:00
|
|
|
|
|
|
|
QString dlog = datalog_path;
|
|
|
|
|
|
|
|
if (datalog_path == backup_path + RMS9_STR_datalog + "/") {
|
|
|
|
// Don't create backups if importing from backup folder
|
|
|
|
create_backups = false;
|
|
|
|
}
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
// Read the "already imported" file list
|
2014-07-27 16:35:49 +00:00
|
|
|
QFile impfile(mach->getDataPath()+"/imported_files.csv");
|
|
|
|
if (impfile.open(QFile::ReadOnly)) {
|
|
|
|
QTextStream impstream(&impfile);
|
|
|
|
QString serial;
|
|
|
|
impstream >> serial;
|
2014-07-28 13:56:29 +00:00
|
|
|
if (mach->serial() == serial) {
|
2014-07-27 16:35:49 +00:00
|
|
|
QString line, file, str;
|
|
|
|
SessionID sid;
|
|
|
|
bool ok;
|
|
|
|
do {
|
|
|
|
line = impstream.readLine();
|
|
|
|
file = line.section(',',0,0);
|
|
|
|
str = line.section(',',1);
|
|
|
|
sid = str.toInt(&ok);
|
|
|
|
|
|
|
|
skipfiles[file] = sid;
|
|
|
|
} while (!impstream.atEnd());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impfile.close();
|
|
|
|
|
|
|
|
QStringList dirs;
|
|
|
|
dirs.push_back(datalog_path);
|
|
|
|
|
|
|
|
QDir dir(datalog_path);
|
|
|
|
|
|
|
|
dir.setFilter(QDir::Dirs | QDir::Hidden | QDir::NoDotAndDotDot);
|
|
|
|
QFileInfoList flist = dir.entryInfoList();
|
|
|
|
QString filename;
|
|
|
|
bool ok, gz;
|
|
|
|
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
// Scan for any sub folders
|
2014-07-27 16:35:49 +00:00
|
|
|
for (int i = 0; i < flist.size(); i++) {
|
|
|
|
QFileInfo fi = flist.at(i);
|
|
|
|
filename = fi.fileName();
|
|
|
|
|
|
|
|
if (filename.length() == 4) {
|
2014-08-26 07:55:01 +00:00
|
|
|
// year folder (used in backups)
|
2014-07-27 16:35:49 +00:00
|
|
|
filename.toInt(&ok);
|
|
|
|
|
2014-08-24 05:00:51 +00:00
|
|
|
if (ok) {
|
|
|
|
dirs.push_back(fi.canonicalFilePath());
|
|
|
|
}
|
|
|
|
} else if (filename.length() == 8) {
|
|
|
|
// S10 stores sessions per day folders
|
|
|
|
filename.toInt(&ok);
|
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
if (ok) {
|
|
|
|
dirs.push_back(fi.canonicalFilePath());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QStringList newSkipFiles;
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
QMap<QString, EDFduration> newfiles; // used for duplicate checking, and session overlap testing to group sessions
|
|
|
|
QHash<EDFType, QList<EDFduration *> > filesbytype;
|
2014-07-27 16:35:49 +00:00
|
|
|
|
2015-08-02 05:53:09 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
// Scan through all folders looking for EDF files, skip any already imported and peek inside to get durations
|
|
|
|
for (int d=0; d < dirs.size(); ++d) {
|
|
|
|
dir.setPath(dirs.at(d));
|
|
|
|
dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks);
|
|
|
|
dir.setSorting(QDir::Name);
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
flist = dir.entryInfoList();
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
// get number of files in current directory being processed
|
2014-07-27 16:35:49 +00:00
|
|
|
int size = flist.size();
|
|
|
|
|
|
|
|
// For each file in flist...
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
QFileInfo fi = flist.at(i);
|
|
|
|
filename = fi.fileName();
|
|
|
|
|
|
|
|
// Forget about it if it can't be read.
|
|
|
|
if (!fi.isReadable()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Chop off the .gz component if it exists
|
|
|
|
if (filename.endsWith(STR_ext_gz)) {
|
|
|
|
filename.chop(3);
|
|
|
|
gz = true;
|
|
|
|
} else { gz = false; }
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
Q_UNUSED(gz)
|
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
// Skip if this file is in the already imported list
|
|
|
|
if (skipfiles.contains(filename)) continue;
|
|
|
|
|
|
|
|
if (newfiles.contains(filename)) {
|
|
|
|
// Not sure what to do with it.. delete it? check compress status and delete the other one?
|
|
|
|
qDebug() << "Duplicate EDF file detected" << filename;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Peek inside file and get duration in seconds..
|
|
|
|
|
|
|
|
// Accept only .edf and .edf.gz files
|
|
|
|
if (filename.right(4).toLower() != "." + STR_ext_EDF) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
QString fullname = fi.canonicalFilePath();
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
// Peek inside the EDF file and get the EDFDuration record for the session matching that follows
|
2014-09-01 04:49:05 +00:00
|
|
|
EDFduration dur = getEDFDuration(fullname);
|
|
|
|
dur.filename = filename;
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
if (dur.start != dur.end) { // make sure empty EVE's are skipped
|
|
|
|
QMap<QString, EDFduration>::iterator it = newfiles.insert(filename, getEDFDuration(fullname));
|
|
|
|
filesbytype[dur.type].append(&it.value());
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<EDFType> EDForder;
|
|
|
|
EDForder.push_back(EDF_PLD);
|
|
|
|
EDForder.push_back(EDF_BRP);
|
|
|
|
EDForder.push_back(EDF_SAD);
|
2015-08-02 05:53:09 +00:00
|
|
|
QHash<EDFType, QStringList>::iterator gi;
|
2014-08-26 07:55:01 +00:00
|
|
|
|
2014-09-22 04:32:15 +00:00
|
|
|
for (int i=0; i<3; i++) {
|
2014-08-26 07:55:01 +00:00
|
|
|
EDFType basetype = EDForder.takeFirst();
|
|
|
|
|
|
|
|
// Process PLD files
|
|
|
|
QList<EDFduration *> & LIST = filesbytype[basetype];
|
2014-09-01 04:49:05 +00:00
|
|
|
int base_size = LIST.size();
|
|
|
|
for (int f=0; f < base_size; ++f) {
|
2014-08-26 07:55:01 +00:00
|
|
|
const EDFduration * dur = LIST.at(f);
|
|
|
|
|
|
|
|
quint32 start = dur->start;
|
|
|
|
if (start == 0) continue;
|
|
|
|
|
|
|
|
quint32 end = dur->end;
|
2014-09-01 04:49:05 +00:00
|
|
|
QHash<EDFType, QStringList> grp;
|
2015-08-02 05:53:09 +00:00
|
|
|
|
|
|
|
// grp[EDF_PLD] = QStringList();
|
|
|
|
// grp[EDF_SAD] = QStringList();
|
|
|
|
// grp[EDF_BRP] = QStringList();
|
|
|
|
// grp[EDF_EVE] = QStringList();
|
|
|
|
// grp[EDF_CSL] = QStringList();
|
|
|
|
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
grp[basetype].append(create_backups ? backup(dur->path, backup_path) : dur->path);
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
QStringList files;
|
|
|
|
files.append(dur->filename);
|
|
|
|
|
|
|
|
for (int o=0; o<EDForder.size(); ++o) {
|
|
|
|
EDFType type = EDForder.at(o);
|
|
|
|
|
|
|
|
QList<EDFduration *> & EDF_list = filesbytype[type];
|
|
|
|
QList<EDFduration *>::iterator item;
|
|
|
|
QList<EDFduration *>::iterator list_end = EDF_list.end();
|
|
|
|
for (item = EDF_list.begin(); item != list_end; ++item) {
|
|
|
|
const EDFduration * dur2 = *item;
|
2014-09-01 04:49:05 +00:00
|
|
|
if (dur2->start == 0) continue;
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
// Do the sessions Overlap?
|
|
|
|
if ((start < dur2->end) && ( dur2->start < end)) {
|
|
|
|
start = qMin(start, dur2->start);
|
|
|
|
end = qMax(end, dur2->end);
|
|
|
|
|
|
|
|
files.append(dur2->filename);
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
grp[type].append(create_backups ? backup(dur2->path, backup_path) : dur2->path);
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
filesbytype[type].erase(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
// EVE annotation files can cover multiple sessions
|
|
|
|
QList<EDFduration *> & EDF_list = filesbytype[EDF_EVE];
|
|
|
|
QList<EDFduration *>::iterator item;
|
|
|
|
QList<EDFduration *>::iterator list_end = EDF_list.end();
|
|
|
|
for (item = EDF_list.begin(); item != list_end; ++item) {
|
|
|
|
const EDFduration * dur2 = *item;
|
|
|
|
if (dur2->start == 0) continue;
|
|
|
|
|
|
|
|
// Do the sessions Overlap?
|
|
|
|
if ((start < dur2->end) && ( dur2->start < end)) {
|
|
|
|
// start = qMin(start, dur2->start);
|
|
|
|
// end = qMax(end, dur2->end);
|
|
|
|
|
|
|
|
files.append(dur2->filename);
|
|
|
|
|
|
|
|
grp[EDF_EVE].append(create_backups ? backup(dur2->path, backup_path) : dur2->path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-22 04:32:15 +00:00
|
|
|
// CSL files contain CSR flags
|
|
|
|
QList<EDFduration *> & CSL_list = filesbytype[EDF_CSL];
|
|
|
|
list_end = CSL_list.end();
|
|
|
|
for (item = CSL_list.begin(); item != list_end; ++item) {
|
|
|
|
const EDFduration * dur2 = *item;
|
|
|
|
if (dur2->start == 0) continue;
|
|
|
|
|
|
|
|
// Do the sessions Overlap?
|
|
|
|
if ((start < dur2->end) && ( dur2->start < end)) {
|
|
|
|
// start = qMin(start, dur2->start);
|
|
|
|
// end = qMax(end, dur2->end);
|
|
|
|
|
|
|
|
files.append(dur2->filename);
|
|
|
|
|
|
|
|
grp[EDF_CSL].append(create_backups ? backup(dur2->path, backup_path) : dur2->path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
if (mach->SessionExists(start) == nullptr) {
|
2014-09-01 04:49:05 +00:00
|
|
|
//EDFGroup group(grp[EDF_BRP], grp[EDF_EVE], grp[EDF_PLD], grp[EDF_SAD], grp[EDF_CSL]);
|
|
|
|
if (grp.size() > 0) {
|
|
|
|
queTask(new ResmedImport(this, start, grp, mach));
|
|
|
|
for (int i=0; i<files.size(); i++) skipfiles[files.at(i)] = start;
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
}
|
2014-07-27 16:35:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
|
|
|
|
// No PLD files
|
|
|
|
|
|
|
|
/* QMap<QString, EDFduration>::iterator it;
|
2014-07-27 16:35:49 +00:00
|
|
|
QMap<QString, EDFduration>::iterator itn;
|
|
|
|
QMap<QString, EDFduration>::iterator it_end = newfiles.end();
|
|
|
|
|
|
|
|
// Now scan through all new files, and group together into sessions
|
|
|
|
for (it = newfiles.begin(); it != it_end; ++it) {
|
|
|
|
quint32 start = it.value().start;
|
|
|
|
|
|
|
|
if (start == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const QString & file = it.key();
|
|
|
|
|
|
|
|
quint32 end = it.value().end;
|
|
|
|
|
|
|
|
|
2014-08-26 07:55:01 +00:00
|
|
|
QString type = file.section("_", -1).section(".", 0, 0).toUpper();
|
2014-07-27 16:35:49 +00:00
|
|
|
|
|
|
|
QString newpath = create_backups ? backup(it.value().path, backup_path) : it.value().path;
|
|
|
|
|
|
|
|
EDFGroup group;
|
|
|
|
|
|
|
|
if (type == "BRP") group.BRP = newpath;
|
2014-08-20 03:03:01 +00:00
|
|
|
else if (type == "EVE") {
|
|
|
|
if (group.BRP.isEmpty()) {
|
|
|
|
qDebug() << "Jedimark's Order theory was wrong.. EVE's need to be parsed seperately!";
|
|
|
|
}
|
|
|
|
group.EVE = newpath;
|
|
|
|
}
|
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
else if (type == "PLD") group.PLD = newpath;
|
|
|
|
else if (type == "SAD") group.SAD = newpath;
|
|
|
|
else continue;
|
|
|
|
|
|
|
|
QStringList sessfiles;
|
|
|
|
sessfiles.push_back(file);
|
|
|
|
|
|
|
|
for (itn = it+1; itn != it_end; ++itn) {
|
|
|
|
if (itn.value().start == 0) continue; // already processed
|
|
|
|
const EDFduration & dur2 = itn.value();
|
|
|
|
|
|
|
|
// Do the sessions Overlap?
|
|
|
|
if ((start < dur2.end) && ( dur2.start < end)) {
|
|
|
|
|
|
|
|
start = qMin(start, dur2.start);
|
|
|
|
end = qMax(end, dur2.end);
|
|
|
|
|
|
|
|
type = itn.key().section("_",-1).section(".",0,0).toUpper();
|
|
|
|
|
|
|
|
newpath = create_backups ? backup(dur2.path, backup_path) : dur2.path;
|
|
|
|
|
|
|
|
if (type == "BRP") {
|
|
|
|
if (!group.BRP.isEmpty()) {
|
|
|
|
itn.value().start = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
group.BRP = newpath;
|
|
|
|
} else if (type == "EVE") {
|
|
|
|
if (!group.EVE.isEmpty()) {
|
|
|
|
itn.value().start = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
group.EVE = newpath;
|
|
|
|
} else if (type == "PLD") {
|
|
|
|
if (!group.PLD.isEmpty()) {
|
|
|
|
itn.value().start = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
group.PLD = newpath;
|
|
|
|
} else if (type == "SAD") {
|
|
|
|
if (!group.SAD.isEmpty()) {
|
|
|
|
itn.value().start = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
group.SAD = newpath;
|
|
|
|
} else {
|
|
|
|
itn.value().start = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
sessfiles.push_back(itn.key());
|
|
|
|
|
|
|
|
itn.value().start = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mach->SessionExists(start) == nullptr) {
|
|
|
|
queTask(new ResmedImport(this, start, group, mach));
|
|
|
|
for (int i=0; i < sessfiles.size(); ++i) {
|
|
|
|
skipfiles[sessfiles.at(i)] = start;
|
|
|
|
}
|
|
|
|
}
|
2014-08-26 07:55:01 +00:00
|
|
|
} */
|
2014-07-27 16:35:49 +00:00
|
|
|
|
|
|
|
// Run the tasks...
|
2014-07-29 14:38:59 +00:00
|
|
|
int c = countTasks();
|
2014-07-27 16:35:49 +00:00
|
|
|
runTasks(p_profile->session->multithreading());
|
|
|
|
|
|
|
|
newSkipFiles.append(skipfiles.keys());
|
|
|
|
impfile.remove();
|
|
|
|
|
|
|
|
if (impfile.open(QFile::WriteOnly)) {
|
|
|
|
QTextStream out(&impfile);
|
2014-07-28 13:56:29 +00:00
|
|
|
out << mach->serial();
|
2014-07-27 16:35:49 +00:00
|
|
|
QHash<QString, SessionID>::iterator skit;
|
|
|
|
QHash<QString, SessionID>::iterator skit_end = skipfiles.end();
|
|
|
|
for (skit = skipfiles.begin(); skit != skit_end; ++skit) {
|
|
|
|
QString a = QString("%1,%2\n").arg(skit.key()).arg(skit.value());;
|
|
|
|
out << a;
|
|
|
|
}
|
|
|
|
out.flush();
|
|
|
|
}
|
|
|
|
impfile.close();
|
|
|
|
|
2014-07-29 14:38:59 +00:00
|
|
|
return c;
|
2014-07-27 16:35:49 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
int ResmedLoader::Open(QString path)
|
2011-06-28 02:21:38 +00:00
|
|
|
{
|
2011-12-30 23:02:45 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QString key, value;
|
2011-12-30 23:02:45 +00:00
|
|
|
QString line;
|
2011-06-28 02:21:38 +00:00
|
|
|
QString newpath;
|
2011-12-30 23:02:45 +00:00
|
|
|
QString filename;
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QHash<QString, QString> idmap; // Temporary properties hash
|
2011-12-30 23:02:45 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
path = path.replace("\\", "/");
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
// Strip off end "/" if any
|
2014-04-17 05:58:57 +00:00
|
|
|
if (path.endsWith("/")) {
|
|
|
|
path = path.section("/", 0, -2);
|
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
// Strip off DATALOG from path, and set newpath to the path contianing DATALOG
|
2013-09-14 23:32:14 +00:00
|
|
|
if (path.endsWith(RMS9_STR_datalog)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
newpath = path + "/";
|
|
|
|
path = path.section("/", 0, -2);
|
2011-06-28 02:21:38 +00:00
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
newpath = path + "/" + RMS9_STR_datalog + "/";
|
2011-06-28 02:21:38 +00:00
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
// Add separator back
|
2014-04-17 05:58:57 +00:00
|
|
|
path += "/";
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
// Check DATALOG folder exists and is readable
|
2014-04-17 05:58:57 +00:00
|
|
|
if (!QDir().exists(newpath)) {
|
2014-07-29 14:38:59 +00:00
|
|
|
return -1;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Parse Identification.tgt file (containing serial number and machine information)
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2014-04-17 05:58:57 +00:00
|
|
|
filename = path + RMS9_STR_idfile + STR_ext_TGT;
|
2011-12-30 23:02:45 +00:00
|
|
|
QFile f(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
// Abort if this file is dodgy..
|
2014-04-17 05:58:57 +00:00
|
|
|
if (!f.exists() || !f.open(QIODevice::ReadOnly)) {
|
2014-07-29 14:38:59 +00:00
|
|
|
return -1;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2014-07-28 13:56:29 +00:00
|
|
|
MachineInfo info = newInfo();
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
// Parse # entries into idmap.
|
|
|
|
while (!f.atEnd()) {
|
2014-04-17 05:58:57 +00:00
|
|
|
line = f.readLine().trimmed();
|
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
if (!line.isEmpty()) {
|
2014-05-04 18:02:41 +00:00
|
|
|
key = line.section(" ", 0, 0).section("#", 1);
|
2014-04-17 05:58:57 +00:00
|
|
|
value = line.section(" ", 1);
|
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
if (key == "SRN") { // Serial Number
|
2014-07-28 13:56:29 +00:00
|
|
|
info.serial = value;
|
|
|
|
continue;
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
} else if (key == "PNA") { // Product Name
|
|
|
|
value.replace("_"," ");
|
2016-03-02 02:09:32 +00:00
|
|
|
if (value.contains(STR_ResMed_S9)) {
|
|
|
|
value.replace(STR_ResMed_S9, "");
|
|
|
|
info.series = STR_ResMed_S9;
|
|
|
|
} else if (value.contains(STR_ResMed_AirSense10)) {
|
|
|
|
value.replace(STR_ResMed_AirSense10, "");
|
|
|
|
info.series = STR_ResMed_AirSense10;
|
|
|
|
} else if (value.contains(STR_ResMed_AirCurve10)) {
|
|
|
|
value.replace(STR_ResMed_AirCurve10, "");
|
|
|
|
info.series = STR_ResMed_AirCurve10;
|
2014-08-26 08:59:35 +00:00
|
|
|
}
|
2014-07-30 20:25:06 +00:00
|
|
|
value.replace("(","");
|
|
|
|
value.replace(")","");
|
2014-08-26 08:59:35 +00:00
|
|
|
|
2014-07-30 20:25:06 +00:00
|
|
|
if (value.contains("Adapt", Qt::CaseInsensitive)) {
|
|
|
|
if (!value.contains("VPAP")) {
|
|
|
|
value.replace("Adapt", QObject::tr("VPAP Adapt"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
info.model = value.trimmed();
|
2014-07-28 13:56:29 +00:00
|
|
|
continue;
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
} else if (key == "PCD") { // Product Code
|
2014-07-28 13:56:29 +00:00
|
|
|
info.modelnumber = value;
|
|
|
|
continue;
|
2011-06-29 19:06:49 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
idmap[key] = value;
|
2011-12-30 23:02:45 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
f.close();
|
|
|
|
|
|
|
|
// Abort if no serial number
|
2014-07-28 13:56:29 +00:00
|
|
|
if (info.serial.isEmpty()) {
|
2011-12-30 23:02:45 +00:00
|
|
|
qDebug() << "S9 Data card has no valid serial number in Indentification.tgt";
|
2014-07-29 14:38:59 +00:00
|
|
|
return -1;
|
2011-12-30 23:02:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Early check for STR.edf file, so we can early exit before creating faulty machine record.
|
2014-04-17 05:58:57 +00:00
|
|
|
QString strpath = path + RMS9_STR_strfile + STR_ext_EDF; // STR.edf file
|
2011-12-30 23:02:45 +00:00
|
|
|
f.setFileName(strpath);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
if (!f.exists()) { // No STR.edf.. Do we have a STR.edf.gz?
|
2014-04-17 05:58:57 +00:00
|
|
|
strpath += STR_ext_gz;
|
2012-01-05 04:37:22 +00:00
|
|
|
f.setFileName(strpath);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
if (!f.exists()) {
|
|
|
|
qDebug() << "Missing STR.edf file";
|
2014-07-29 14:38:59 +00:00
|
|
|
return -1;
|
2012-01-05 04:37:22 +00:00
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Create machine object (unless it's already registered)
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2014-07-28 13:56:29 +00:00
|
|
|
Machine *m = CreateMachine(info);
|
2011-12-30 23:02:45 +00:00
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
bool create_backups = p_profile->session->backupCardData();
|
|
|
|
bool compress_backups = p_profile->session->compressBackupData();
|
2012-01-05 04:37:22 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
QString backup_path = m->getBackupPath();
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (path == backup_path) {
|
2014-05-04 18:02:41 +00:00
|
|
|
// Don't create backups if importing from backup folder
|
2014-04-17 05:58:57 +00:00
|
|
|
create_backups = false;
|
2012-01-05 04:37:22 +00:00
|
|
|
}
|
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Parse the idmap into machine objects properties, (overwriting any old values)
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2014-05-06 18:03:13 +00:00
|
|
|
for (QHash<QString, QString>::iterator i = idmap.begin(); i != idmap.end(); i++) {
|
2014-04-17 05:58:57 +00:00
|
|
|
m->properties[i.key()] = i.value();
|
2011-06-29 19:06:49 +00:00
|
|
|
}
|
2011-09-17 12:39:00 +00:00
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Open and Parse STR.edf file
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2014-05-18 17:06:58 +00:00
|
|
|
QStringList strfiles;
|
|
|
|
strfiles.push_back(strpath);
|
|
|
|
QDir dir(path + "STR_Backup");
|
|
|
|
dir.setFilter(QDir::Files | QDir::Hidden | QDir::Readable);
|
|
|
|
QFileInfoList flist = dir.entryInfoList();
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
{
|
|
|
|
int size = flist.size();
|
|
|
|
for (int i = 0; i < size; i++) {
|
2014-05-18 17:06:58 +00:00
|
|
|
QFileInfo fi = flist.at(i);
|
|
|
|
filename = fi.fileName();
|
|
|
|
if (filename.startsWith("STR", Qt::CaseInsensitive)) {
|
|
|
|
strfiles.push_back(fi.filePath());
|
|
|
|
}
|
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
}
|
2014-05-18 17:06:58 +00:00
|
|
|
|
|
|
|
strsess.clear();
|
|
|
|
ParseSTR(m, strfiles);
|
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
EDFParser stredf(strpath);
|
2011-09-17 12:39:00 +00:00
|
|
|
if (!stredf.Parse()) {
|
2013-09-14 23:32:14 +00:00
|
|
|
qDebug() << "Faulty file" << RMS9_STR_strfile;
|
2011-09-17 12:39:00 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
if (stredf.serialnumber != info.serial) {
|
2011-12-30 23:02:45 +00:00
|
|
|
qDebug() << "Identification.tgt Serial number doesn't match STR.edf!";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Creating early as we need the object
|
2014-05-18 17:06:58 +00:00
|
|
|
dir.setPath(newpath);
|
2011-12-30 23:02:45 +00:00
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Create the backup folder for storing a copy of everything in..
|
2012-01-05 04:37:22 +00:00
|
|
|
// (Unless we are importing from this backup folder)
|
2011-12-30 23:02:45 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2012-01-05 04:37:22 +00:00
|
|
|
if (create_backups) {
|
|
|
|
if (!dir.exists(backup_path)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (!dir.mkpath(backup_path + RMS9_STR_datalog)) {
|
2012-01-05 04:37:22 +00:00
|
|
|
qDebug() << "Could not create S9 backup directory :-/";
|
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
}
|
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
// Copy Identification files to backup folder
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(path + RMS9_STR_idfile + STR_ext_TGT, backup_path + RMS9_STR_idfile + STR_ext_TGT);
|
|
|
|
QFile::copy(path + RMS9_STR_idfile + STR_ext_CRC, backup_path + RMS9_STR_idfile + STR_ext_CRC);
|
2012-01-05 04:37:22 +00:00
|
|
|
|
2015-08-11 20:01:24 +00:00
|
|
|
QDateTime dts = QDateTime::fromMSecsSinceEpoch(stredf.startdate, Qt::UTC);
|
2014-04-17 05:58:57 +00:00
|
|
|
dir.mkpath(backup_path + "STR_Backup");
|
|
|
|
QString strmonthly = backup_path + "STR_Backup/STR-" + dts.toString("yyyyMM") + "." + STR_ext_EDF;
|
2013-10-20 10:20:54 +00:00
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
//copy STR files to backup folder
|
2013-10-20 10:20:54 +00:00
|
|
|
if (strpath.endsWith(STR_ext_gz)) { // Already compressed. Don't bother decompressing..
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(strpath, backup_path + RMS9_STR_strfile + STR_ext_EDF + STR_ext_gz);
|
2013-10-20 10:20:54 +00:00
|
|
|
} else { // Compress STR file to backup folder
|
2014-04-17 05:58:57 +00:00
|
|
|
QString strf = backup_path + RMS9_STR_strfile + STR_ext_EDF;
|
2013-10-20 10:20:54 +00:00
|
|
|
|
|
|
|
// Copy most recent to STR.edf
|
2014-04-17 05:58:57 +00:00
|
|
|
if (QFile::exists(strf)) {
|
2012-01-06 16:07:54 +00:00
|
|
|
QFile::remove(strf);
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2012-01-06 16:07:54 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (QFile::exists(strf + STR_ext_gz)) {
|
|
|
|
QFile::remove(strf + STR_ext_gz);
|
|
|
|
}
|
2013-10-20 10:20:54 +00:00
|
|
|
|
2012-01-05 04:37:22 +00:00
|
|
|
compress_backups ?
|
2014-04-17 05:58:57 +00:00
|
|
|
compressFile(strpath, strf)
|
2012-01-05 04:37:22 +00:00
|
|
|
:
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(strpath, strf);
|
2013-10-20 10:20:54 +00:00
|
|
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2013-10-20 10:20:54 +00:00
|
|
|
// Keep one STR.edf backup every month
|
2014-04-17 05:58:57 +00:00
|
|
|
if (!QFile::exists(strmonthly) && !QFile::exists(strmonthly + ".gz")) {
|
2013-10-20 10:20:54 +00:00
|
|
|
compress_backups ?
|
2014-04-17 05:58:57 +00:00
|
|
|
compressFile(strpath, strmonthly)
|
2013-10-20 10:20:54 +00:00
|
|
|
:
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(strpath, strmonthly);
|
2012-01-05 04:37:22 +00:00
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
|
2013-10-22 11:42:57 +00:00
|
|
|
// Meh.. these can be calculated if ever needed for ResScan SDcard export
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(path + "STR.crc", backup_path + "STR.crc");
|
2012-01-05 04:37:22 +00:00
|
|
|
}
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Process the actual STR.edf data
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2011-09-17 12:39:00 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 numrecs = stredf.GetNumDataRecords();
|
|
|
|
qint64 duration = numrecs * stredf.GetDuration();
|
2014-07-10 14:54:05 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
int days = duration / 86400000L; // GetNumDataRecords = this.. Duh!
|
2011-09-17 12:39:00 +00:00
|
|
|
|
2015-07-30 22:36:52 +00:00
|
|
|
if (days<0) {
|
|
|
|
qDebug() << "Error: Negative number of days in STR.edf, aborting import";
|
|
|
|
days=0;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-12-30 23:02:45 +00:00
|
|
|
// Process STR.edf and find first and last time for each day
|
|
|
|
|
|
|
|
QVector<qint8> dayused;
|
|
|
|
dayused.resize(days);
|
2014-05-18 17:06:58 +00:00
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//time_t time = stredf.startdate / 1000L; // == 12pm on first day
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
// reset time to first day
|
2014-10-08 16:51:09 +00:00
|
|
|
//time = stredf.startdate / 1000;
|
2011-12-30 23:02:45 +00:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2014-07-28 13:56:29 +00:00
|
|
|
// Scan DATALOG files, sort, and import any new sessions
|
2011-12-30 23:02:45 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
int num_new_sessions = scanFiles(m, newpath);
|
2014-07-28 13:56:29 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////
|
2014-05-18 17:06:58 +00:00
|
|
|
// Now look for any new summary data that can be extracted from STR.edf records
|
2014-07-28 13:56:29 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////
|
2013-10-16 02:52:25 +00:00
|
|
|
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//int size = m->sessionlist.size();
|
|
|
|
//int cnt=0;
|
2014-07-27 16:35:49 +00:00
|
|
|
Session * sess;
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
// Scan through all sessions, and remove any strsess records that have a matching session already
|
2014-09-01 04:49:05 +00:00
|
|
|
// for (sessit = m->sessionlist.begin(); sessit != sessend; ++sessit) {
|
|
|
|
// sess = *sessit;
|
|
|
|
// quint32 key = sess->settings[RMS9_MaskOnTime].toUInt();
|
|
|
|
|
|
|
|
// // Ugly.. need to check sessions overlaps..
|
|
|
|
|
|
|
|
// QMap<quint32, STRRecord>::iterator e = strsess.find(key);
|
|
|
|
// if (e != end) {
|
|
|
|
// strsess.erase(e);
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
///
|
|
|
|
|
|
|
|
QHash<SessionID, Session *>::iterator sessit;
|
|
|
|
QHash<SessionID, Session *>::iterator sessend = m->sessionlist.end();;
|
|
|
|
|
|
|
|
QMap<SessionID, Session *>::iterator sit;
|
|
|
|
QMap<SessionID, Session *>::iterator ns_end = new_sessions.end();
|
|
|
|
|
|
|
|
|
|
|
|
QMap<quint32, STRRecord>::iterator it;
|
|
|
|
QMap<quint32, STRRecord>::iterator end = strsess.end();
|
|
|
|
|
|
|
|
QList<quint32> strlist;
|
|
|
|
for (it = strsess.begin(); it != end; ++it) {
|
|
|
|
STRRecord & R = it.value();
|
|
|
|
quint32 s1 = R.maskon;
|
|
|
|
quint32 e1 = R.maskoff;
|
|
|
|
bool fnd = false;
|
|
|
|
for (sessit = m->sessionlist.begin(); sessit != sessend; ++sessit) {
|
|
|
|
sess = sessit.value();
|
|
|
|
quint32 s2 = sess->session();
|
|
|
|
quint32 e2 = s2 + (sess->length() / 1000L);
|
|
|
|
|
|
|
|
if ((s1 < e2) && (s2 < e1)) {
|
|
|
|
strlist.push_back(it.key());
|
|
|
|
fnd = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!fnd) for (sit = new_sessions.begin(); sit != ns_end; ++sit) {
|
|
|
|
sess = sit.value();
|
|
|
|
quint32 s2 = sess->session();
|
|
|
|
quint32 e2 = s2 + (sess->length() / 1000L);
|
|
|
|
|
|
|
|
if ((s1 < e2) && (s2 < e1)) {
|
|
|
|
strlist.push_back(it.key());
|
|
|
|
fnd = true;
|
|
|
|
break;
|
|
|
|
}
|
2013-10-16 02:52:25 +00:00
|
|
|
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
|
|
|
}
|
2013-10-16 02:52:25 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
for (int i=0; i<strlist.size(); i++) {
|
|
|
|
int k = strlist.at(i);
|
|
|
|
strsess.remove(k);
|
|
|
|
}
|
|
|
|
|
|
|
|
///
|
2013-10-16 02:52:25 +00:00
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
//size = strsess.size();
|
|
|
|
//cnt=0;
|
2014-07-11 12:09:38 +00:00
|
|
|
quint32 ignoreolder = p_profile->session->ignoreOlderSessionsDate().toTime_t();
|
2014-05-19 07:30:10 +00:00
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
bool ignoreold = p_profile->session->ignoreOlderSessions();
|
2014-05-20 11:51:47 +00:00
|
|
|
// strsess end can change above.
|
|
|
|
end = strsess.end();
|
|
|
|
|
2014-06-20 02:06:57 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Scan through unmatched strsess records, and attempt to get at summary data
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
2014-05-18 17:06:58 +00:00
|
|
|
for (it = strsess.begin(); it != end; ++it) {
|
2014-05-20 11:51:47 +00:00
|
|
|
STRRecord & R = it.value();
|
2014-05-19 07:30:10 +00:00
|
|
|
|
2014-06-20 02:06:57 +00:00
|
|
|
if (ignoreold && (R.maskon < ignoreolder)) {
|
2014-05-20 11:51:47 +00:00
|
|
|
m->skipSaveTask();
|
|
|
|
continue;
|
|
|
|
}
|
2014-05-19 07:30:10 +00:00
|
|
|
|
|
|
|
//Q_ASSERT(R.sessionid == 0);
|
2014-06-30 11:20:12 +00:00
|
|
|
|
2014-06-20 02:06:57 +00:00
|
|
|
// the following should not happen
|
2014-05-20 11:51:47 +00:00
|
|
|
if (R.sessionid > 0) {
|
|
|
|
m->skipSaveTask();
|
|
|
|
continue;
|
|
|
|
}
|
2013-10-16 02:52:25 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2014-06-30 11:20:12 +00:00
|
|
|
queTask(new ResmedImportStage2(this, R, m));
|
2014-05-18 17:06:58 +00:00
|
|
|
}
|
2014-07-29 14:38:59 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
num_new_sessions += countTasks();
|
2014-06-30 11:20:12 +00:00
|
|
|
runTasks();
|
2013-10-16 02:52:25 +00:00
|
|
|
|
2014-07-30 20:25:06 +00:00
|
|
|
finishAddingSessions();
|
|
|
|
|
2012-01-05 06:54:07 +00:00
|
|
|
#ifdef DEBUG_EFFICIENCY
|
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 totalbytes = 0;
|
|
|
|
qint64 totalns = 0;
|
|
|
|
qDebug() << "Time Delta Efficiency Information";
|
|
|
|
|
|
|
|
for (QHash<ChannelID, qint64>::iterator it = channel_efficiency.begin();
|
|
|
|
it != channel_efficiency.end(); it++) {
|
|
|
|
ChannelID code = it.key();
|
|
|
|
qint64 value = it.value();
|
|
|
|
qint64 ns = channel_time[code];
|
|
|
|
totalbytes += value;
|
|
|
|
totalns += ns;
|
|
|
|
double secs = double(ns) / 1000000000.0L;
|
|
|
|
QString s = value < 0 ? "saved" : "cost";
|
|
|
|
qDebug() << "Time-Delta conversion for " + schema::channel[code].label() + " " + s + " " +
|
|
|
|
QString::number(qAbs(value)) + " bytes and took " + QString::number(secs, 'f', 4) + "s";
|
|
|
|
}
|
|
|
|
|
|
|
|
qDebug() << "Total toTimeDelta function usage:" << totalbytes << "in" << double(
|
|
|
|
totalns) / 1000000000.0 << "seconds";
|
2012-01-05 06:54:07 +00:00
|
|
|
}
|
|
|
|
#endif
|
2011-12-21 04:25:01 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (qprogress) { qprogress->setValue(100); }
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
sessfiles.clear();
|
|
|
|
strsess.clear();
|
|
|
|
|
|
|
|
strdate.clear();
|
|
|
|
channel_efficiency.clear();
|
|
|
|
channel_time.clear();
|
|
|
|
|
2011-07-02 14:35:50 +00:00
|
|
|
qDebug() << "Total Events " << event_cnt;
|
2014-09-01 04:49:05 +00:00
|
|
|
return num_new_sessions;
|
2011-06-28 02:21:38 +00:00
|
|
|
}
|
2011-06-29 17:58:28 +00:00
|
|
|
|
2014-07-27 16:35:49 +00:00
|
|
|
|
|
|
|
QString ResmedLoader::backup(QString fullname, QString backup_path)
|
2013-10-22 11:42:57 +00:00
|
|
|
{
|
2014-07-27 16:35:49 +00:00
|
|
|
bool compress = p_profile->session->compressBackupData();
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QString filename, yearstr, newname, oldname;
|
|
|
|
bool ok, gz = (fullname.right(3).toLower() == STR_ext_gz);
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
filename = fullname.section("/", -1);
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (gz) {
|
2013-10-22 11:42:57 +00:00
|
|
|
filename.chop(3);
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
yearstr = filename.left(4);
|
|
|
|
yearstr.toInt(&ok, 10);
|
2013-10-22 11:42:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
if (!ok) {
|
|
|
|
qDebug() << "Invalid EDF filename given to ResMedLoader::backup()";
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
newname = backup_path + RMS9_STR_datalog + "/" + yearstr;
|
2013-10-22 11:42:57 +00:00
|
|
|
QDir dir;
|
|
|
|
dir.mkpath(newname);
|
2014-04-17 05:58:57 +00:00
|
|
|
newname += "/" + filename;
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QString tmpname = newname;
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (compress) {
|
|
|
|
newname += STR_ext_gz;
|
|
|
|
}
|
2013-10-22 11:42:57 +00:00
|
|
|
|
|
|
|
// First make sure the correct backup exists.
|
|
|
|
if (!QFile::exists(newname)) {
|
|
|
|
if (compress) {
|
|
|
|
gz ?
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(fullname, newname) // Already compressed.. copy it to the right location
|
2013-10-22 11:42:57 +00:00
|
|
|
:
|
2014-04-17 05:58:57 +00:00
|
|
|
compressFile(fullname, newname);
|
2013-10-22 11:42:57 +00:00
|
|
|
} else {
|
|
|
|
// dont really care if it's compressed and not meant to be, leave it that way
|
2014-04-17 05:58:57 +00:00
|
|
|
QFile::copy(fullname, newname);
|
2013-10-22 11:42:57 +00:00
|
|
|
}
|
|
|
|
} // else backup already exists...
|
|
|
|
|
|
|
|
// Now the correct backup is in place, we can trash any
|
|
|
|
if (compress) {
|
|
|
|
// Remove any uncompressed duplicate
|
2014-04-17 05:58:57 +00:00
|
|
|
if (QFile::exists(tmpname)) {
|
2013-10-22 11:42:57 +00:00
|
|
|
QFile::remove(tmpname);
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2013-10-22 11:42:57 +00:00
|
|
|
} else {
|
|
|
|
// Delete the non compressed copy and choose it instead.
|
2014-04-17 05:58:57 +00:00
|
|
|
if (QFile::exists(tmpname + STR_ext_gz)) {
|
2013-10-22 11:42:57 +00:00
|
|
|
QFile::remove(tmpname);
|
2014-04-17 05:58:57 +00:00
|
|
|
newname = tmpname + STR_ext_gz;
|
2013-10-22 11:42:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove any traces from old backup directory structure
|
2014-04-17 05:58:57 +00:00
|
|
|
oldname = backup_path + RMS9_STR_datalog + "/" + filename;
|
|
|
|
|
|
|
|
if (QFile::exists(oldname)) {
|
2013-10-22 11:42:57 +00:00
|
|
|
QFile::remove(oldname);
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (QFile::exists(oldname + STR_ext_gz)) {
|
|
|
|
QFile::remove(oldname + STR_ext_gz);
|
|
|
|
}
|
2013-10-22 11:42:57 +00:00
|
|
|
|
|
|
|
return newname;
|
|
|
|
}
|
|
|
|
|
2014-09-22 04:32:15 +00:00
|
|
|
bool ResmedLoader::LoadCSL(Session *sess, const QString & path)
|
|
|
|
{
|
|
|
|
EDFParser edf(path);
|
|
|
|
if (!edf.Parse())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
QString t;
|
|
|
|
|
|
|
|
long recs;
|
|
|
|
double duration;
|
|
|
|
char *data;
|
|
|
|
char c;
|
|
|
|
long pos;
|
|
|
|
bool sign, ok;
|
|
|
|
double d;
|
|
|
|
double tt;
|
|
|
|
|
|
|
|
// Notes: Event records have useless duration record.
|
|
|
|
// sess->updateFirst(edf.startdate);
|
|
|
|
|
|
|
|
EventList *CSR = nullptr;
|
|
|
|
|
|
|
|
// Allow for empty sessions..
|
|
|
|
qint64 csr_starts = 0;
|
|
|
|
|
|
|
|
|
|
|
|
// Process event annotation records
|
|
|
|
for (int s = 0; s < edf.GetNumSignals(); s++) {
|
|
|
|
recs = edf.edfsignals[s].nr * edf.GetNumDataRecords() * 2;
|
|
|
|
|
|
|
|
data = (char *)edf.edfsignals[s].data;
|
|
|
|
pos = 0;
|
|
|
|
tt = edf.startdate;
|
|
|
|
// sess->updateFirst(tt);
|
|
|
|
duration = 0;
|
|
|
|
|
|
|
|
while (pos < recs) {
|
|
|
|
c = data[pos];
|
|
|
|
|
|
|
|
if ((c != '+') && (c != '-')) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data[pos++] == '+') { sign = true; }
|
|
|
|
else { sign = false; }
|
|
|
|
|
|
|
|
t = "";
|
|
|
|
c = data[pos];
|
|
|
|
|
|
|
|
do {
|
|
|
|
t += c;
|
|
|
|
pos++;
|
|
|
|
c = data[pos];
|
|
|
|
} while ((c != 20) && (c != 21)); // start code
|
|
|
|
|
|
|
|
d = t.toDouble(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
|
|
|
qDebug() << "Faulty EDF CSL file " << edf.filename;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!sign) { d = -d; }
|
|
|
|
|
|
|
|
tt = edf.startdate + qint64(d * 1000.0);
|
|
|
|
duration = 0;
|
|
|
|
// First entry
|
|
|
|
|
|
|
|
if (data[pos] == 21) {
|
|
|
|
pos++;
|
|
|
|
// get duration.
|
|
|
|
t = "";
|
|
|
|
|
|
|
|
do {
|
|
|
|
t += data[pos];
|
|
|
|
pos++;
|
|
|
|
} while ((data[pos] != 20) && (pos < recs)); // start code
|
|
|
|
|
|
|
|
duration = t.toDouble(&ok);
|
|
|
|
|
|
|
|
if (!ok) {
|
|
|
|
qDebug() << "Faulty EDF CSL file (at %" << pos << ") " << edf.filename;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((data[pos] == 20) && (pos < recs)) {
|
|
|
|
t = "";
|
|
|
|
pos++;
|
|
|
|
|
|
|
|
if (data[pos] == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data[pos] == 20) {
|
|
|
|
pos++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
t += tolower(data[pos++]);
|
|
|
|
} while ((data[pos] != 20) && (pos < recs)); // start code
|
|
|
|
|
|
|
|
if (!t.isEmpty()) {
|
|
|
|
if (t == "csr start") {
|
|
|
|
csr_starts = tt;
|
|
|
|
} else if (t == "csr end") {
|
|
|
|
if (!CSR) {
|
|
|
|
CSR = sess->AddEventList(CPAP_CSR, EVL_Event);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (csr_starts > 0) {
|
|
|
|
if (sess->checkInside(csr_starts))
|
|
|
|
CSR->AddEvent(tt, double(tt - csr_starts) / 1000.0);
|
|
|
|
csr_starts = 0;
|
|
|
|
} else {
|
|
|
|
qDebug() << "If you can read this, ResMed sucks and split CSR flagging!";
|
|
|
|
}
|
|
|
|
} else if (t != "recording starts") {
|
|
|
|
qDebug() << "Unobserved ResMed CSL annotation field: " << t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pos >= recs) {
|
|
|
|
qDebug() << "Short EDF CSL file" << edf.filename;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// pos++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((data[pos] == 0) && (pos < recs)) { pos++; }
|
|
|
|
|
|
|
|
if (pos >= recs) { break; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// sess->updateLast(tt);
|
|
|
|
}
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
Q_UNUSED(duration)
|
|
|
|
|
2014-09-22 04:32:15 +00:00
|
|
|
return true;
|
|
|
|
}
|
2013-10-22 11:42:57 +00:00
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
bool ResmedLoader::LoadEVE(Session *sess, const QString & path)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2014-05-20 11:51:47 +00:00
|
|
|
EDFParser edf(path);
|
|
|
|
if (!edf.Parse())
|
|
|
|
return false;
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
QString t;
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2011-06-30 10:56:22 +00:00
|
|
|
long recs;
|
2011-07-27 09:21:53 +00:00
|
|
|
double duration;
|
2014-04-17 05:58:57 +00:00
|
|
|
char *data;
|
2011-06-30 10:56:22 +00:00
|
|
|
char c;
|
|
|
|
long pos;
|
2014-04-17 05:58:57 +00:00
|
|
|
bool sign, ok;
|
2011-06-30 10:56:22 +00:00
|
|
|
double d;
|
2011-07-02 15:48:55 +00:00
|
|
|
double tt;
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Notes: Event records have useless duration record.
|
2014-08-24 05:00:51 +00:00
|
|
|
// sess->updateFirst(edf.startdate);
|
2012-01-12 05:25:51 +00:00
|
|
|
|
2014-09-17 17:20:01 +00:00
|
|
|
EventList *OA = nullptr, *HY = nullptr, *CA = nullptr, *UA = nullptr, *RE = nullptr;
|
2012-01-12 05:25:51 +00:00
|
|
|
|
|
|
|
// Allow for empty sessions..
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
// Create EventLists
|
2014-04-17 05:58:57 +00:00
|
|
|
OA = sess->AddEventList(CPAP_Obstructive, EVL_Event);
|
|
|
|
HY = sess->AddEventList(CPAP_Hypopnea, EVL_Event);
|
|
|
|
UA = sess->AddEventList(CPAP_Apnea, EVL_Event);
|
2012-01-12 05:25:51 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Process event annotation records
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int s = 0; s < edf.GetNumSignals(); s++) {
|
|
|
|
recs = edf.edfsignals[s].nr * edf.GetNumDataRecords() * 2;
|
2011-07-02 09:49:53 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
data = (char *)edf.edfsignals[s].data;
|
|
|
|
pos = 0;
|
|
|
|
tt = edf.startdate;
|
2014-09-01 04:49:05 +00:00
|
|
|
// sess->updateFirst(tt);
|
2014-04-17 05:58:57 +00:00
|
|
|
duration = 0;
|
|
|
|
|
|
|
|
while (pos < recs) {
|
|
|
|
c = data[pos];
|
|
|
|
|
|
|
|
if ((c != '+') && (c != '-')) {
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (data[pos++] == '+') { sign = true; }
|
|
|
|
else { sign = false; }
|
|
|
|
|
|
|
|
t = "";
|
|
|
|
c = data[pos];
|
|
|
|
|
2011-06-29 17:58:28 +00:00
|
|
|
do {
|
2014-04-17 05:58:57 +00:00
|
|
|
t += c;
|
2011-06-29 17:58:28 +00:00
|
|
|
pos++;
|
2014-04-17 05:58:57 +00:00
|
|
|
c = data[pos];
|
|
|
|
} while ((c != 20) && (c != 21)); // start code
|
|
|
|
|
|
|
|
d = t.toDouble(&ok);
|
|
|
|
|
2011-06-29 17:58:28 +00:00
|
|
|
if (!ok) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Faulty EDF EVE file " << edf.filename;
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (!sign) { d = -d; }
|
|
|
|
|
|
|
|
tt = edf.startdate + qint64(d * 1000.0);
|
|
|
|
duration = 0;
|
2011-06-29 17:58:28 +00:00
|
|
|
// First entry
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (data[pos] == 21) {
|
2011-06-29 17:58:28 +00:00
|
|
|
pos++;
|
|
|
|
// get duration.
|
2014-04-17 05:58:57 +00:00
|
|
|
t = "";
|
|
|
|
|
2011-06-29 17:58:28 +00:00
|
|
|
do {
|
2014-04-17 05:58:57 +00:00
|
|
|
t += data[pos];
|
2011-06-29 17:58:28 +00:00
|
|
|
pos++;
|
2014-04-17 05:58:57 +00:00
|
|
|
} while ((data[pos] != 20) && (pos < recs)); // start code
|
|
|
|
|
|
|
|
duration = t.toDouble(&ok);
|
|
|
|
|
2011-06-29 17:58:28 +00:00
|
|
|
if (!ok) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Faulty EDF EVE file (at %" << pos << ") " << edf.filename;
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
while ((data[pos] == 20) && (pos < recs)) {
|
|
|
|
t = "";
|
2011-06-29 17:58:28 +00:00
|
|
|
pos++;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (data[pos] == 0) {
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (data[pos] == 20) {
|
2011-06-29 17:58:28 +00:00
|
|
|
pos++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
2014-04-17 05:58:57 +00:00
|
|
|
t += tolower(data[pos++]);
|
|
|
|
} while ((data[pos] != 20) && (pos < recs)); // start code
|
|
|
|
|
2011-06-29 17:58:28 +00:00
|
|
|
if (!t.isEmpty()) {
|
2014-05-04 18:02:41 +00:00
|
|
|
if (matchSignal(CPAP_Obstructive, t)) {
|
2014-09-01 04:49:05 +00:00
|
|
|
|
|
|
|
if (sess->checkInside(tt)) OA->AddEvent(tt, duration);
|
2014-05-04 18:02:41 +00:00
|
|
|
} else if (matchSignal(CPAP_Hypopnea, t)) {
|
2014-09-01 04:49:05 +00:00
|
|
|
if (sess->checkInside(tt)) HY->AddEvent(tt, duration + 10); // Only Hyponea's Need the extra duration???
|
2014-05-04 18:02:41 +00:00
|
|
|
} else if (matchSignal(CPAP_Apnea, t)) {
|
2014-09-01 04:49:05 +00:00
|
|
|
if (sess->checkInside(tt)) UA->AddEvent(tt, duration);
|
2014-09-17 17:20:01 +00:00
|
|
|
} else if (matchSignal(CPAP_RERA, t)) {
|
|
|
|
// Not all machines have it, so only create it when necessary..
|
|
|
|
if (!RE) {
|
|
|
|
if (!(RE = sess->AddEventList(CPAP_RERA, EVL_Event))) { return false; }
|
|
|
|
}
|
|
|
|
if (sess->checkInside(tt)) RE->AddEvent(tt, duration);
|
2014-05-04 18:02:41 +00:00
|
|
|
} else if (matchSignal(CPAP_ClearAirway, t)) {
|
2012-01-12 05:25:51 +00:00
|
|
|
// Not all machines have it, so only create it when necessary..
|
|
|
|
if (!CA) {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (!(CA = sess->AddEventList(CPAP_ClearAirway, EVL_Event))) { return false; }
|
2011-07-27 09:21:53 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
if (sess->checkInside(tt)) CA->AddEvent(tt, duration);
|
2011-06-29 17:58:28 +00:00
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (t != "recording starts") {
|
2011-08-09 23:44:36 +00:00
|
|
|
qDebug() << "Unobserved ResMed annotation field: " << t;
|
2011-06-29 17:58:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (pos >= recs) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Short EDF EVE file" << edf.filename;
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
// pos++;
|
2011-06-29 17:58:28 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
while ((data[pos] == 0) && (pos < recs)) { pos++; }
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (pos >= recs) { break; }
|
2011-06-29 17:58:28 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
// sess->updateLast(tt);
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
|
|
|
|
bool ResmedLoader::LoadBRP(Session *sess, const QString & path)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2014-05-20 11:51:47 +00:00
|
|
|
EDFParser edf(path);
|
|
|
|
if (!edf.Parse())
|
|
|
|
return false;
|
|
|
|
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateFirst(edf.startdate);
|
2014-05-04 18:02:41 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 duration = edf.GetNumDataRecords() * edf.GetDuration();
|
|
|
|
sess->updateLast(edf.startdate + duration);
|
2011-07-03 02:43:50 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int s = 0; s < edf.GetNumSignals(); s++) {
|
2014-04-09 19:39:03 +00:00
|
|
|
EDFSignal &es = edf.edfsignals[s];
|
2014-05-04 18:02:41 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
long recs = es.nr * edf.GetNumDataRecords();
|
2014-07-26 12:46:11 +00:00
|
|
|
if (recs < 0)
|
|
|
|
continue;
|
2011-07-31 20:24:43 +00:00
|
|
|
ChannelID code;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
if (matchSignal(CPAP_FlowRate, es.label)) {
|
|
|
|
code = CPAP_FlowRate;
|
2014-04-17 05:58:57 +00:00
|
|
|
es.gain *= 60.0;
|
|
|
|
es.physical_minimum *= 60.0;
|
|
|
|
es.physical_maximum *= 60.0;
|
|
|
|
es.physical_dimension = "L/M";
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
} else if (matchSignal(CPAP_MaskPressureHi, es.label)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
code = CPAP_MaskPressureHi;
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
} else if (matchSignal(CPAP_RespEvent, es.label)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
code = CPAP_RespEvent;
|
2014-05-04 18:02:41 +00:00
|
|
|
|
2014-09-01 11:39:38 +00:00
|
|
|
} else if (es.label != "Crc16") {
|
2014-04-09 19:39:03 +00:00
|
|
|
qDebug() << "Unobserved ResMed BRP Signal " << es.label;
|
2011-06-29 17:58:28 +00:00
|
|
|
continue;
|
2014-09-01 11:39:38 +00:00
|
|
|
} else continue;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
if (code) {
|
|
|
|
double rate = double(duration) / double(recs);
|
|
|
|
EventList *a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
|
|
|
|
a->setDimension(es.physical_dimension);
|
|
|
|
a->AddWaveform(edf.startdate, es.data, recs, duration);
|
2015-09-21 00:02:15 +00:00
|
|
|
EventDataType min = a->Min();
|
|
|
|
EventDataType max = a->Max();
|
|
|
|
|
|
|
|
// Cap to physical dimensions, because there can be ram glitches/whatever that throw really big outliers.
|
|
|
|
if (min < es.physical_minimum) min = es.physical_minimum;
|
|
|
|
if (max > es.physical_maximum) max = es.physical_maximum;
|
|
|
|
|
|
|
|
sess->setMin(code, min);
|
|
|
|
sess->setMax(code, max);
|
2014-05-04 18:02:41 +00:00
|
|
|
sess->setPhysMin(code, es.physical_minimum);
|
|
|
|
sess->setPhysMax(code, es.physical_maximum);
|
|
|
|
}
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2014-05-04 18:02:41 +00:00
|
|
|
|
2014-05-06 02:19:26 +00:00
|
|
|
// Convert EDFSignal data to sleepyheads Time-Delta Event format
|
2014-04-17 05:58:57 +00:00
|
|
|
void ResmedLoader::ToTimeDelta(Session *sess, EDFParser &edf, EDFSignal &es, ChannelID code,
|
|
|
|
long recs, qint64 duration, EventDataType t_min, EventDataType t_max, bool square)
|
2011-06-29 20:30:23 +00:00
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
if (t_min == t_max) {
|
|
|
|
t_min = es.physical_minimum;
|
|
|
|
t_max = es.physical_maximum;
|
2013-10-25 10:39:30 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2012-01-05 06:54:07 +00:00
|
|
|
#ifdef DEBUG_EFFICIENCY
|
|
|
|
QElapsedTimer time;
|
|
|
|
time.start();
|
|
|
|
#endif
|
2012-01-09 15:38:41 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
double rate = (duration / recs); // milliseconds per record
|
|
|
|
double tt = edf.startdate;
|
2014-05-04 18:02:41 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
EventStoreType c, last;
|
2011-09-17 12:39:00 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
int startpos = 0;
|
2011-09-13 08:12:07 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if ((code == CPAP_Pressure) || (code == CPAP_IPAP) || (code == CPAP_EPAP)) {
|
|
|
|
startpos = 20; // Shave the first 20 seconds of pressure data
|
|
|
|
tt += rate * startpos;
|
2011-09-17 12:39:00 +00:00
|
|
|
}
|
2012-01-09 15:38:41 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
qint16 *sptr = es.data;
|
|
|
|
qint16 *eptr = sptr + recs;
|
|
|
|
sptr += startpos;
|
|
|
|
|
|
|
|
EventDataType min = t_max, max = t_min, tmp;
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-23 13:19:56 +00:00
|
|
|
EventList *el = nullptr;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (recs > startpos + 1) {
|
2013-10-25 10:39:30 +00:00
|
|
|
|
|
|
|
// Prime last with a good starting value
|
|
|
|
do {
|
2014-04-17 05:58:57 +00:00
|
|
|
last = *sptr++;
|
|
|
|
tmp = EventDataType(last) * es.gain;
|
2013-10-25 10:39:30 +00:00
|
|
|
|
|
|
|
if ((tmp >= t_min) && (tmp <= t_max)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
min = tmp;
|
|
|
|
max = tmp;
|
|
|
|
el = sess->AddEventList(code, EVL_Event, es.gain, es.offset, 0, 0);
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
el->AddEvent(tt, last);
|
|
|
|
tt += rate;
|
2013-10-25 10:39:30 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
tt += rate;
|
2013-10-25 10:39:30 +00:00
|
|
|
|
|
|
|
} while (sptr < eptr);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
if (!el) {
|
2013-10-25 10:39:30 +00:00
|
|
|
return;
|
2014-04-17 05:58:57 +00:00
|
|
|
}
|
2012-01-09 15:38:41 +00:00
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
for (; sptr < eptr; sptr++) {
|
|
|
|
c = *sptr;
|
2011-06-29 20:30:23 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (last != c) {
|
2013-10-25 10:39:30 +00:00
|
|
|
if (square) {
|
2014-04-17 05:58:57 +00:00
|
|
|
tmp = EventDataType(last) * es.gain;
|
|
|
|
|
2013-10-25 10:39:30 +00:00
|
|
|
if ((tmp >= t_min) && (tmp <= t_max)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (tmp < min) {
|
|
|
|
min = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tmp > max) {
|
|
|
|
max = tmp;
|
|
|
|
}
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
el->AddEvent(tt, last);
|
2013-10-25 10:39:30 +00:00
|
|
|
} else {
|
|
|
|
// Out of bounds value, start a new eventlist
|
2014-04-17 05:58:57 +00:00
|
|
|
if (el->count() > 1) {
|
2013-10-25 10:39:30 +00:00
|
|
|
// that should be in session, not the eventlist.. handy for debugging though
|
|
|
|
el->setDimension(es.physical_dimension);
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
el = sess->AddEventList(code, EVL_Event, es.gain, es.offset, 0, 0);
|
2013-10-25 10:39:30 +00:00
|
|
|
} else {
|
|
|
|
el->clear(); // reuse the object
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
tmp = EventDataType(c) * es.gain;
|
|
|
|
|
2013-10-25 10:39:30 +00:00
|
|
|
if ((tmp >= t_min) && (tmp <= t_max)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (tmp < min) {
|
|
|
|
min = tmp;
|
|
|
|
}
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (tmp > max) {
|
|
|
|
max = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
el->AddEvent(tt, c);
|
2013-10-25 10:39:30 +00:00
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (el->count() > 1) {
|
2013-10-25 10:39:30 +00:00
|
|
|
el->setDimension(es.physical_dimension);
|
|
|
|
|
|
|
|
// Create and attach new EventList
|
2014-04-17 05:58:57 +00:00
|
|
|
el = sess->AddEventList(code, EVL_Event, es.gain, es.offset, 0, 0);
|
|
|
|
} else { el->clear(); }
|
2013-10-25 10:39:30 +00:00
|
|
|
}
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
tt += rate;
|
|
|
|
|
|
|
|
last = c;
|
2012-01-09 15:38:41 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
tmp = EventDataType(c) * es.gain;
|
|
|
|
|
2013-10-25 10:39:30 +00:00
|
|
|
if ((tmp >= t_min) && (tmp <= t_max)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
el->AddEvent(tt, c);
|
2013-10-25 10:39:30 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
sess->setMin(code, min);
|
|
|
|
sess->setMax(code, max);
|
|
|
|
sess->setPhysMin(code, es.physical_minimum);
|
|
|
|
sess->setPhysMax(code, es.physical_maximum);
|
2012-01-09 15:38:41 +00:00
|
|
|
sess->updateLast(tt);
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
2012-01-05 06:54:07 +00:00
|
|
|
|
|
|
|
#ifdef DEBUG_EFFICIENCY
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 t = time.nsecsElapsed();
|
|
|
|
int cnt = el->count();
|
|
|
|
int bytes = cnt * (sizeof(EventStoreType) + sizeof(quint32));
|
|
|
|
int wvbytes = recs * (sizeof(EventStoreType));
|
|
|
|
QHash<ChannelID, qint64>::iterator it = channel_efficiency.find(code);
|
|
|
|
|
|
|
|
if (it == channel_efficiency.end()) {
|
|
|
|
channel_efficiency[code] = wvbytes - bytes;
|
|
|
|
channel_time[code] = t;
|
2012-01-05 06:54:07 +00:00
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
it.value() += wvbytes - bytes;
|
|
|
|
channel_time[code] += t;
|
2012-01-05 06:54:07 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2012-01-05 06:54:07 +00:00
|
|
|
#endif
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
2014-05-04 18:02:41 +00:00
|
|
|
|
|
|
|
// Load SAD Oximetry Signals
|
2014-05-20 11:51:47 +00:00
|
|
|
bool ResmedLoader::LoadSAD(Session *sess, const QString & path)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2014-05-20 11:51:47 +00:00
|
|
|
EDFParser edf(path);
|
|
|
|
if (!edf.Parse())
|
|
|
|
return false;
|
|
|
|
|
2011-08-06 13:37:06 +00:00
|
|
|
sess->updateFirst(edf.startdate);
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 duration = edf.GetNumDataRecords() * edf.GetDuration();
|
|
|
|
sess->updateLast(edf.startdate + duration);
|
2011-08-06 13:37:06 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
for (int s = 0; s < edf.GetNumSignals(); s++) {
|
2014-04-09 19:39:03 +00:00
|
|
|
EDFSignal &es = edf.edfsignals[s];
|
2011-08-07 01:26:28 +00:00
|
|
|
//qDebug() << "SAD:" << es.label << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum;
|
2014-04-17 05:58:57 +00:00
|
|
|
long recs = es.nr * edf.GetNumDataRecords();
|
2011-08-06 13:37:06 +00:00
|
|
|
ChannelID code;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
bool hasdata = false;
|
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
for (int i = 0; i < recs; ++i) {
|
2014-04-17 05:58:57 +00:00
|
|
|
if (es.data[i] != -1) {
|
|
|
|
hasdata = true;
|
2011-08-06 13:37:06 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
if (!hasdata) {
|
|
|
|
continue;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-05-13 02:56:51 +00:00
|
|
|
if (matchSignal(OXI_Pulse, es.label)) {
|
|
|
|
code = OXI_Pulse;
|
2014-05-04 18:02:41 +00:00
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration);
|
|
|
|
sess->setPhysMax(code, 180);
|
|
|
|
sess->setPhysMin(code, 18);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(OXI_SPO2, es.label)) {
|
|
|
|
code = OXI_SPO2;
|
2014-05-04 18:02:41 +00:00
|
|
|
es.physical_minimum = 60;
|
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration);
|
|
|
|
sess->setPhysMax(code, 100);
|
|
|
|
sess->setPhysMin(code, 60);
|
2014-09-01 11:39:38 +00:00
|
|
|
} else if (es.label != "Crc16") {
|
2014-05-04 18:02:41 +00:00
|
|
|
qDebug() << "Unobserved ResMed SAD Signal " << es.label;
|
|
|
|
}
|
2011-08-06 13:37:06 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-06-29 20:30:23 +00:00
|
|
|
|
|
|
|
|
2014-05-20 11:51:47 +00:00
|
|
|
bool ResmedLoader::LoadPLD(Session *sess, const QString & path)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2014-05-20 11:51:47 +00:00
|
|
|
EDFParser edf(path);
|
|
|
|
if (!edf.Parse())
|
|
|
|
return false;
|
|
|
|
|
2011-07-01 02:52:02 +00:00
|
|
|
// Is it save to assume the order does not change here?
|
2014-04-17 05:58:57 +00:00
|
|
|
enum PLDType { MaskPres = 0, TherapyPres, ExpPress, Leak, RR, Vt, Mv, SnoreIndex, FFLIndex, U1, U2 };
|
2011-07-01 02:52:02 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
qint64 duration = edf.GetNumDataRecords() * edf.GetDuration();
|
2011-09-17 12:39:00 +00:00
|
|
|
sess->updateFirst(edf.startdate);
|
2014-04-17 05:58:57 +00:00
|
|
|
sess->updateLast(edf.startdate + duration);
|
2011-06-29 17:58:28 +00:00
|
|
|
QString t;
|
2014-04-17 05:58:57 +00:00
|
|
|
int emptycnt = 0;
|
2014-04-23 13:19:56 +00:00
|
|
|
EventList *a = nullptr;
|
2011-08-01 08:53:26 +00:00
|
|
|
double rate;
|
|
|
|
long recs;
|
|
|
|
ChannelID code;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
for (int s = 0; s < edf.GetNumSignals(); s++) {
|
2014-04-23 13:19:56 +00:00
|
|
|
a = nullptr;
|
2014-04-09 19:39:03 +00:00
|
|
|
EDFSignal &es = edf.edfsignals[s];
|
2014-04-17 05:58:57 +00:00
|
|
|
recs = es.nr * edf.GetNumDataRecords();
|
|
|
|
|
|
|
|
if (recs <= 0) { continue; }
|
|
|
|
|
|
|
|
rate = double(duration) / double(recs);
|
|
|
|
|
2011-07-27 09:21:53 +00:00
|
|
|
//qDebug() << "EVE:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum << es.gain;
|
2014-05-13 02:56:51 +00:00
|
|
|
if (matchSignal(CPAP_Snore, es.label)) {
|
|
|
|
code = CPAP_Snore;
|
2014-05-04 18:02:41 +00:00
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_Pressure, es.label)) {
|
|
|
|
code = CPAP_Pressure;
|
2014-04-17 05:58:57 +00:00
|
|
|
es.physical_maximum = 25;
|
|
|
|
es.physical_minimum = 4;
|
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_IPAP, es.label)) {
|
|
|
|
code = CPAP_IPAP;
|
2014-04-17 05:58:57 +00:00
|
|
|
es.physical_maximum = 25;
|
|
|
|
es.physical_minimum = 4;
|
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-09-01 11:39:38 +00:00
|
|
|
} else if (matchSignal(CPAP_EPAP, es.label)) { // Expiratory Pressure
|
|
|
|
code = CPAP_EPAP;
|
|
|
|
es.physical_maximum = 25;
|
|
|
|
es.physical_minimum = 4;
|
|
|
|
|
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
|
|
|
} else if (matchSignal(CPAP_MinuteVent,es.label)) {
|
2014-05-13 02:56:51 +00:00
|
|
|
code = CPAP_MinuteVent;
|
2014-04-17 05:58:57 +00:00
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_RespRate, es.label)) {
|
|
|
|
code = CPAP_RespRate;
|
2014-04-17 05:58:57 +00:00
|
|
|
a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
|
|
|
|
a->AddWaveform(edf.startdate, es.data, recs, duration);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_TidalVolume, es.label)) {
|
|
|
|
code = CPAP_TidalVolume;
|
2014-04-17 05:58:57 +00:00
|
|
|
es.gain *= 1000.0;
|
|
|
|
es.physical_maximum *= 1000.0;
|
|
|
|
es.physical_minimum *= 1000.0;
|
|
|
|
// es.digital_maximum*=1000.0;
|
|
|
|
// es.digital_minimum*=1000.0;
|
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_Leak, es.label)) {
|
|
|
|
code = CPAP_Leak;
|
2014-05-06 02:19:26 +00:00
|
|
|
es.gain *= 60.0;
|
|
|
|
es.physical_maximum *= 60.0;
|
|
|
|
es.physical_minimum *= 60.0;
|
2014-04-17 05:58:57 +00:00
|
|
|
// es.digital_maximum*=60.0;
|
|
|
|
// es.digital_minimum*=60.0;
|
|
|
|
es.physical_dimension = "L/M";
|
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0, true);
|
|
|
|
sess->setPhysMax(code, 120.0);
|
|
|
|
sess->setPhysMin(code, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_FLG, es.label)) {
|
|
|
|
code = CPAP_FLG;
|
2014-04-17 05:58:57 +00:00
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_MaskPressure, es.label)) {
|
|
|
|
code = CPAP_MaskPressure;
|
2014-04-17 05:58:57 +00:00
|
|
|
es.physical_maximum = 25;
|
|
|
|
es.physical_minimum = 4;
|
2013-10-25 10:39:30 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
ToTimeDelta(sess, edf, es, code, recs, duration, 0, 0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_IE, es.label)) { //I:E ratio
|
|
|
|
code = CPAP_IE;
|
2014-04-17 05:58:57 +00:00
|
|
|
a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
|
|
|
|
a->AddWaveform(edf.startdate, es.data, recs, duration);
|
2011-08-09 23:44:36 +00:00
|
|
|
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_Ti, es.label)) {
|
|
|
|
code = CPAP_Ti;
|
2014-08-22 11:54:17 +00:00
|
|
|
// There are TWO of these with the same label on my VPAP Adapt 36037
|
|
|
|
|
|
|
|
if (sess->eventlist.contains(code)) {
|
|
|
|
continue;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
|
|
|
|
a->AddWaveform(edf.startdate, es.data, recs, duration);
|
2011-08-09 23:44:36 +00:00
|
|
|
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_Te, es.label)) {
|
|
|
|
code = CPAP_Te;
|
2014-08-22 11:54:17 +00:00
|
|
|
// There are TWO of these with the same label on my VPAP Adapt 36037
|
|
|
|
if (sess->eventlist.contains(code)) {
|
|
|
|
continue;
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
|
|
|
|
a->AddWaveform(edf.startdate, es.data, recs, duration);
|
2011-11-14 09:26:58 +00:00
|
|
|
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2014-05-13 02:56:51 +00:00
|
|
|
} else if (matchSignal(CPAP_TgMV, es.label)) {
|
|
|
|
code = CPAP_TgMV;
|
2014-04-17 05:58:57 +00:00
|
|
|
a = sess->AddEventList(code, EVL_Waveform, es.gain, es.offset, 0, 0, rate);
|
|
|
|
a->AddWaveform(edf.startdate, es.data, recs, duration);
|
2011-08-09 23:44:36 +00:00
|
|
|
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2014-08-21 16:13:22 +00:00
|
|
|
} else if (es.label == "") { // What the hell resmed??
|
2014-04-17 05:58:57 +00:00
|
|
|
if (emptycnt == 0) {
|
|
|
|
code = RMS9_E01;
|
2014-09-01 11:39:38 +00:00
|
|
|
// ToTimeDelta(sess, edf, es, code, recs, duration);
|
2014-04-17 05:58:57 +00:00
|
|
|
} else if (emptycnt == 1) {
|
|
|
|
code = RMS9_E02;
|
2014-09-01 11:39:38 +00:00
|
|
|
// ToTimeDelta(sess, edf, es, code, recs, duration);
|
2011-07-21 03:35:59 +00:00
|
|
|
} else {
|
2011-07-27 09:21:53 +00:00
|
|
|
qDebug() << "Unobserved Empty Signal " << es.label;
|
2011-07-21 03:35:59 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-07-21 03:35:59 +00:00
|
|
|
emptycnt++;
|
2014-09-01 11:39:38 +00:00
|
|
|
} else if (es.label != "Crc16") {
|
2011-08-09 23:44:36 +00:00
|
|
|
qDebug() << "Unobserved ResMed PLD Signal " << es.label;
|
2014-04-23 13:19:56 +00:00
|
|
|
a = nullptr;
|
2011-08-01 08:53:26 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-08-01 08:53:26 +00:00
|
|
|
if (a) {
|
2014-04-17 05:58:57 +00:00
|
|
|
sess->setMin(code, a->Min());
|
|
|
|
sess->setMax(code, a->Max());
|
|
|
|
sess->setPhysMin(code, es.physical_minimum);
|
|
|
|
sess->setPhysMax(code, es.physical_maximum);
|
2011-08-09 23:44:36 +00:00
|
|
|
a->setDimension(es.physical_dimension);
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
2014-05-20 11:51:47 +00:00
|
|
|
|
2011-06-29 17:58:28 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
void ResInitModelMap()
|
|
|
|
{
|
2014-05-04 18:02:41 +00:00
|
|
|
// don't really need this anymore
|
2014-05-06 17:39:05 +00:00
|
|
|
// Resmed_Model_Map = {
|
|
|
|
// { "S9 Escape", { 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301, 36361 } },
|
|
|
|
// { "S9 Escape Auto", { 36002, 36012, 36022, 36302, 36362 } },
|
|
|
|
// { "S9 Elite", { 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203, 36223, 36243, 36263, 36303, 36343, 36363 } },
|
|
|
|
// { "S9 Autoset", { 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205, 36225, 36245, 36265, 36305, 36325, 36345, 36365 } },
|
|
|
|
// { "S9 AutoSet CS", { 36100, 36110, 36120, 36140, 36200, 36220, 36360 } },
|
|
|
|
// { "S9 AutoSet 25", { 36106, 36116, 36126, 36146, 36206, 36226, 36366 } },
|
|
|
|
// { "S9 AutoSet for Her", { 36065 } },
|
|
|
|
// { "S9 VPAP S", { 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224, 36284, 36304 } },
|
|
|
|
// { "S9 VPAP Auto", { 36006, 36016, 36026 } },
|
|
|
|
// { "S9 VPAP Adapt", { 36037, 36007, 36017, 36027, 36367 } },
|
|
|
|
// { "S9 VPAP ST", { 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 } },
|
|
|
|
// { "S9 VPAP ST 22", { 36118, 36128 } },
|
|
|
|
// { "S9 VPAP ST-A", { 36039, 36159, 36169, 36379 } },
|
|
|
|
// //S8 Series
|
|
|
|
// { "S8 Escape", { 33007 } },
|
|
|
|
// { "S8 Elite II", { 33039 } },
|
|
|
|
// { "S8 Escape II", { 33051 } },
|
|
|
|
// { "S8 Escape II AutoSet", { 33064 } },
|
|
|
|
// { "S8 AutoSet II", { 33129 } },
|
|
|
|
// };
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-04-26 09:54:08 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////
|
2013-09-14 23:32:14 +00:00
|
|
|
// Translation lookup table for non-english machines
|
2014-04-26 09:54:08 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
// Only put the first part, enough to be identifiable, because ResMed likes
|
|
|
|
// to signal names crop short
|
2014-05-06 17:39:05 +00:00
|
|
|
// Read this from a table?
|
|
|
|
|
2014-08-06 14:06:44 +00:00
|
|
|
resmed_codes.clear();
|
2014-05-06 17:39:05 +00:00
|
|
|
|
2014-09-01 11:39:38 +00:00
|
|
|
// BRP file
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_FlowRate].push_back("Flow");
|
2014-09-01 11:39:38 +00:00
|
|
|
resmed_codes[CPAP_FlowRate].push_back("Flow.40ms");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_MaskPressureHi].push_back("Mask Pres");
|
2014-09-01 11:39:38 +00:00
|
|
|
resmed_codes[CPAP_MaskPressureHi].push_back("Press.40ms");
|
|
|
|
|
|
|
|
|
|
|
|
// PLD File
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_MaskPressure].push_back("Mask Pres");
|
2014-09-01 11:39:38 +00:00
|
|
|
|
|
|
|
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_RespEvent].push_back("Resp Event");
|
|
|
|
resmed_codes[CPAP_Pressure].push_back("Therapy Pres");
|
|
|
|
resmed_codes[CPAP_IPAP].push_back("Insp Pres");
|
2014-05-18 17:06:58 +00:00
|
|
|
resmed_codes[CPAP_IPAP].push_back("IPAP");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_IPAP].push_back("S.BL.IPAP");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_EPAP].push_back("Exp Pres");
|
2016-03-04 03:34:05 +00:00
|
|
|
resmed_codes[CPAP_EPAP].push_back("EprPress.2s");
|
2014-05-18 17:06:58 +00:00
|
|
|
resmed_codes[CPAP_EPAP].push_back("EPAP");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_EPAP].push_back("S.BL.EPAP");
|
2014-05-18 17:06:58 +00:00
|
|
|
resmed_codes[CPAP_EPAPHi].push_back("Max EPAP");
|
|
|
|
resmed_codes[CPAP_EPAPLo].push_back("Min EPAP");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_EPAPLo].push_back("S.VA.MinEPAP");
|
2014-05-18 17:06:58 +00:00
|
|
|
resmed_codes[CPAP_IPAPHi].push_back("Max IPAP");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_IPAPHi].push_back("S.VA.MaxIPAP");
|
2014-05-18 17:06:58 +00:00
|
|
|
resmed_codes[CPAP_IPAPLo].push_back("Min IPAP");
|
|
|
|
|
|
|
|
resmed_codes[CPAP_PS].push_back("PS");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_PS].push_back("S.VA.PS");
|
2014-05-18 17:06:58 +00:00
|
|
|
resmed_codes[CPAP_PSMin].push_back("Min PS");
|
|
|
|
resmed_codes[CPAP_PSMax].push_back("Max PS");
|
|
|
|
|
2014-09-11 14:23:08 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Leak"); // Leak Leck Lekk Läck Fuites
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Leck");
|
2014-09-11 14:23:08 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Fuites");
|
2015-08-02 05:53:09 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Fuite");
|
2015-08-02 06:05:56 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Fuga");
|
2014-06-26 14:05:59 +00:00
|
|
|
|
|
|
|
resmed_codes[CPAP_Leak].push_back("\xE6\xBC\x8F\xE6\xB0\x94");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Lekk");
|
|
|
|
resmed_codes[CPAP_Leak].push_back("Läck");
|
|
|
|
resmed_codes[CPAP_Leak].push_back("Läck");
|
|
|
|
resmed_codes[CPAP_RespRate].push_back("RR");
|
|
|
|
resmed_codes[CPAP_RespRate].push_back("AF");
|
|
|
|
resmed_codes[CPAP_RespRate].push_back("FR");
|
|
|
|
resmed_codes[CPAP_MinuteVent].push_back("MV");
|
|
|
|
resmed_codes[CPAP_MinuteVent].push_back("VM");
|
|
|
|
resmed_codes[CPAP_TidalVolume].push_back("Vt");
|
|
|
|
resmed_codes[CPAP_TidalVolume].push_back("VC");
|
|
|
|
resmed_codes[CPAP_IE].push_back("I:E");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_IE].push_back("IERatio.2s");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_Snore].push_back("Snore");
|
|
|
|
resmed_codes[CPAP_FLG].push_back("FFL Index");
|
|
|
|
resmed_codes[CPAP_Ti].push_back("Ti");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_Ti].push_back("B5ITime.2s");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_Te].push_back("Te");
|
2015-09-21 01:30:23 +00:00
|
|
|
resmed_codes[CPAP_Te].push_back("B5ETime.2s");
|
|
|
|
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_TgMV].push_back("TgMV");
|
2016-03-04 00:17:19 +00:00
|
|
|
resmed_codes[CPAP_TgMV].push_back("TgtVent.2s");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[OXI_Pulse].push_back("Pulse");
|
|
|
|
resmed_codes[OXI_Pulse].push_back("Puls");
|
2014-09-11 14:23:08 +00:00
|
|
|
resmed_codes[OXI_Pulse].push_back("Pouls");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[OXI_Pulse].push_back("Pols");
|
|
|
|
resmed_codes[OXI_SPO2].push_back("SpO2");
|
|
|
|
resmed_codes[CPAP_Obstructive].push_back("Obstructive apnea");
|
|
|
|
resmed_codes[CPAP_Hypopnea].push_back("Hypopnea");
|
|
|
|
resmed_codes[CPAP_Apnea].push_back("Apnea");
|
2014-09-17 17:20:01 +00:00
|
|
|
resmed_codes[CPAP_RERA].push_back("Arousal");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_ClearAirway].push_back("Central apnea");
|
|
|
|
resmed_codes[CPAP_Mode].push_back("Mode");
|
|
|
|
resmed_codes[CPAP_Mode].push_back("Modus");
|
|
|
|
resmed_codes[CPAP_Mode].push_back("Funktion");
|
2014-06-26 14:05:59 +00:00
|
|
|
resmed_codes[CPAP_Mode].push_back("\xE6\xA8\xA1\xE5\xBC\x8F"); // Chinese
|
|
|
|
|
2014-05-28 16:23:40 +00:00
|
|
|
resmed_codes[RMS9_SetPressure].push_back("Set Pressure");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[RMS9_SetPressure].push_back("Eingest. Druck");
|
|
|
|
resmed_codes[RMS9_SetPressure].push_back("Ingestelde druk");
|
2014-06-26 14:05:59 +00:00
|
|
|
resmed_codes[RMS9_SetPressure].push_back("\xE8\xAE\xBE\xE5\xAE\x9A\xE5\x8E\x8B\xE5\x8A\x9B"); // Chinese
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[RMS9_SetPressure].push_back("Pres. prescrite");
|
|
|
|
resmed_codes[RMS9_SetPressure].push_back("Inställt tryck");
|
|
|
|
resmed_codes[RMS9_SetPressure].push_back("Inställt tryck");
|
2014-09-01 11:39:38 +00:00
|
|
|
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[RMS9_EPR].push_back("EPR");
|
2014-09-17 17:20:01 +00:00
|
|
|
//resmed_codes[RMS9_EPR].push_back("S.EPR.EPRType");
|
2014-09-01 04:49:05 +00:00
|
|
|
|
2014-06-26 14:05:59 +00:00
|
|
|
resmed_codes[RMS9_EPR].push_back("\xE5\x91\xBC\xE6\xB0\x94\xE9\x87\x8A\xE5\x8E\x8B\x28\x45\x50"); // Chinese
|
2014-07-06 00:22:50 +00:00
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("EPR Level");
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("EPR-Stufe");
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("EPR-niveau");
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("\x45\x50\x52\x20\xE6\xB0\xB4\xE5\xB9\xB3"); // Chinese
|
|
|
|
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("Niveau EPR");
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("EPR-nivå");
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("EPR-nivå");
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_PressureMax].push_back("Max Pressure");
|
|
|
|
resmed_codes[CPAP_PressureMax].push_back("Max. Druck");
|
|
|
|
resmed_codes[CPAP_PressureMax].push_back("Max druk");
|
2014-06-26 14:05:59 +00:00
|
|
|
|
|
|
|
resmed_codes[CPAP_PressureMax].push_back("\xE6\x9C\x80\xE5\xA4\xA7\xE5\x8E\x8B\xE5\x8A\x9B"); // Chinese
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_PressureMax].push_back("Pression max.");
|
|
|
|
resmed_codes[CPAP_PressureMax].push_back("Max tryck");
|
|
|
|
resmed_codes[CPAP_PressureMin].push_back("Min Pressure");
|
|
|
|
resmed_codes[CPAP_PressureMin].push_back("Min. Druck");
|
|
|
|
resmed_codes[CPAP_PressureMin].push_back("Min druk");
|
2014-06-26 14:05:59 +00:00
|
|
|
resmed_codes[CPAP_PressureMin].push_back("\xE6\x9C\x80\xE5\xB0\x8F\xE5\x8E\x8B\xE5\x8A\x9B"); // Chinese
|
2014-05-06 17:39:05 +00:00
|
|
|
resmed_codes[CPAP_PressureMin].push_back("Pression min.");
|
|
|
|
resmed_codes[CPAP_PressureMin].push_back("Min tryck");
|
2014-08-24 02:26:53 +00:00
|
|
|
|
2014-09-01 11:39:38 +00:00
|
|
|
|
2014-08-24 02:26:53 +00:00
|
|
|
|
|
|
|
// SAD file
|
|
|
|
resmed_codes[OXI_Pulse].push_back("Pulse.1s");
|
|
|
|
resmed_codes[OXI_SPO2].push_back("SpO2.1s");
|
|
|
|
|
|
|
|
// PLD file
|
|
|
|
resmed_codes[CPAP_MaskPressure].push_back("MaskPress.2s");
|
2016-03-04 03:34:05 +00:00
|
|
|
resmed_codes[CPAP_Pressure].push_back("Press.2s"); // Un problemo... IPAP also uses this.. :/
|
2015-08-02 05:53:09 +00:00
|
|
|
resmed_codes[CPAP_EPAP].push_back("EPRPress.2s");
|
2014-08-24 02:26:53 +00:00
|
|
|
resmed_codes[CPAP_Leak].push_back("Leak.2s");
|
|
|
|
resmed_codes[CPAP_RespRate].push_back("RespRate.2s");
|
|
|
|
resmed_codes[CPAP_TidalVolume].push_back("TidVol.2s");
|
|
|
|
resmed_codes[CPAP_MinuteVent].push_back("MinVent.2s");
|
|
|
|
resmed_codes[CPAP_Snore].push_back("Snore.2s");
|
|
|
|
resmed_codes[CPAP_FLG].push_back("FlowLim.2s");
|
|
|
|
|
2014-08-24 05:00:51 +00:00
|
|
|
//S.AS.StartPress
|
|
|
|
resmed_codes[CPAP_PressureMin].push_back("S.AS.MinPress");
|
|
|
|
resmed_codes[CPAP_PressureMax].push_back("S.AS.MaxPress");
|
|
|
|
|
|
|
|
resmed_codes[RMS9_SetPressure].push_back("S.C.Press");
|
|
|
|
|
|
|
|
resmed_codes[RMS9_EPRLevel].push_back("S.EPR.Level");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-08-24 02:26:53 +00:00
|
|
|
|
2011-09-23 03:54:48 +00:00
|
|
|
}
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2014-09-01 04:49:05 +00:00
|
|
|
ChannelID ResmedLoader::CPAPModeChannel() { return RMS9_Mode; }
|
2014-08-03 13:00:13 +00:00
|
|
|
ChannelID ResmedLoader::PresReliefMode() { return RMS9_EPR; }
|
|
|
|
ChannelID ResmedLoader::PresReliefLevel() { return RMS9_EPRLevel; }
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2014-08-06 14:06:44 +00:00
|
|
|
void ResmedLoader::initChannels()
|
2011-06-28 02:21:38 +00:00
|
|
|
{
|
2014-08-03 13:00:13 +00:00
|
|
|
using namespace schema;
|
|
|
|
Channel * chan = nullptr;
|
2014-09-01 04:49:05 +00:00
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_Mode = 0xe203, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_Mode",
|
|
|
|
QObject::tr("Mode"),
|
|
|
|
QObject::tr("CPAP Mode"),
|
|
|
|
QObject::tr("Mode"),
|
|
|
|
"", LOOKUP, Qt::green));
|
|
|
|
|
|
|
|
chan->addOption(0, QObject::tr("CPAP"));
|
|
|
|
chan->addOption(1, QObject::tr("APAP"));
|
|
|
|
chan->addOption(2, QObject::tr("VPAP-T"));
|
|
|
|
chan->addOption(3, QObject::tr("VPAP-S"));
|
|
|
|
chan->addOption(4, QObject::tr("VPAP-S/T"));
|
|
|
|
chan->addOption(5, QObject::tr("??"));
|
|
|
|
chan->addOption(6, QObject::tr("VPAPauto"));
|
|
|
|
chan->addOption(7, QObject::tr("ASV"));
|
|
|
|
chan->addOption(8, QObject::tr("ASVAuto"));
|
|
|
|
chan->addOption(9, QObject::tr("???"));
|
|
|
|
chan->addOption(10, QObject::tr("???"));
|
|
|
|
chan->addOption(11, QObject::tr("Auto for Her"));
|
|
|
|
|
2014-08-23 06:21:50 +00:00
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_EPR = 0xe201, SETTING, MT_CPAP, SESSION,
|
2014-08-04 19:57:48 +00:00
|
|
|
"EPR", QObject::tr("EPR"),
|
|
|
|
QObject::tr("ResMed Exhale Pressure Relief"),
|
|
|
|
QObject::tr("EPR"),
|
|
|
|
"", LOOKUP, Qt::green));
|
2014-08-03 13:00:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
chan->addOption(1, QObject::tr("Ramp Only"));
|
|
|
|
chan->addOption(2, QObject::tr("Full Time"));
|
2014-08-06 14:06:44 +00:00
|
|
|
chan->addOption(3, QObject::tr("Patient???"));
|
2014-08-03 13:00:13 +00:00
|
|
|
|
2014-08-23 06:21:50 +00:00
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_EPRLevel = 0xe202, SETTING, MT_CPAP, SESSION,
|
2014-08-03 13:00:13 +00:00
|
|
|
"EPRLevel", QObject::tr("EPR Level"),
|
|
|
|
QObject::tr("Exhale Pressure Relief Level"),
|
|
|
|
QObject::tr("EPR Level"),
|
2014-08-04 19:57:48 +00:00
|
|
|
"", LOOKUP, Qt::blue));
|
2014-08-03 13:00:13 +00:00
|
|
|
|
|
|
|
chan->addOption(0, QObject::tr("0cmH2O"));
|
|
|
|
chan->addOption(1, QObject::tr("1cmH2O"));
|
|
|
|
chan->addOption(2, QObject::tr("2cmH2O"));
|
|
|
|
chan->addOption(3, QObject::tr("3cmH2O"));
|
2014-08-04 19:57:48 +00:00
|
|
|
|
2016-03-08 13:55:29 +00:00
|
|
|
// RMS9_SmartStart, RMS9_HumidStatus, RMS9_HumidLevel,
|
|
|
|
// RMS9_PtAccess, RMS9_Mask, RMS9_ABFilter, RMS9_ClimateControl, RMS9_TubeType,
|
|
|
|
// RMS9_Temp, RMS9_TempEnable;
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_SmartStart = 0xe204, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_SmartStart", QObject::tr("SmartStart"),
|
|
|
|
QObject::tr("Machine auto starts by breathing"),
|
|
|
|
QObject::tr("Smart Start"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
chan->addOption(1, STR_TR_On);
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_HumidStatus = 0xe205, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_HumidStat", QObject::tr("Humid. Status"),
|
|
|
|
QObject::tr("Humidifier Enabled Status"),
|
|
|
|
QObject::tr("Humidifier Status"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
chan->addOption(1, STR_TR_On);
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_HumidLevel = 0xe206, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_HumidLevel", QObject::tr("Humid. Level"),
|
|
|
|
QObject::tr("Humidity Level"),
|
|
|
|
QObject::tr("Humidity Level"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
chan->addOption(1, "1");
|
|
|
|
chan->addOption(2, "2");
|
|
|
|
chan->addOption(3, "3");
|
|
|
|
chan->addOption(4, "4");
|
|
|
|
chan->addOption(5, "5");
|
|
|
|
chan->addOption(6, "6");
|
|
|
|
chan->addOption(7, "7");
|
|
|
|
chan->addOption(8, "8");
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_Temp = 0xe207, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_Temp", QObject::tr("Temperature"),
|
|
|
|
QObject::tr("ClimateLine Temperature"),
|
|
|
|
QObject::tr("Temperature"),
|
|
|
|
"ºC", INTEGER, Qt::black));
|
|
|
|
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_TempEnable = 0xe208, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_TempEnable", QObject::tr("Temp. Enable"),
|
|
|
|
QObject::tr("ClimateLine Temperature Enable"),
|
|
|
|
QObject::tr("Temperature Enable"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
chan->addOption(1, "1");
|
|
|
|
chan->addOption(2, "2");
|
|
|
|
chan->addOption(3, "3");
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_ABFilter= 0xe209, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_ABFilter", QObject::tr("AB Filter"),
|
|
|
|
QObject::tr("Antibacterial Filter"),
|
|
|
|
QObject::tr("Antibacterial Filter"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_No);
|
|
|
|
chan->addOption(1, STR_TR_Yes);
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_PtAccess= 0xe20A, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_PTAccess", QObject::tr("Pt. Access"),
|
|
|
|
QObject::tr("Patient Access"),
|
|
|
|
QObject::tr("Patient Access"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_No);
|
|
|
|
chan->addOption(1, STR_TR_Yes);
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_ClimateControl= 0xe20B, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_ClimateControl", QObject::tr("Climate Control"),
|
|
|
|
QObject::tr("Climate Control"),
|
|
|
|
QObject::tr("Climate Control"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, QObject::tr("Manual"));
|
|
|
|
chan->addOption(1, QObject::tr("Auto"));
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_Mask= 0xe20C, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_Mask", QObject::tr("Mask"),
|
|
|
|
QObject::tr("ResMed Mask Setting"),
|
|
|
|
QObject::tr("Mask"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, QObject::tr("Pillows"));
|
|
|
|
chan->addOption(1, QObject::tr("Full Face"));
|
|
|
|
chan->addOption(2, QObject::tr("Nasal"));
|
|
|
|
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(RMS9_RampEnable = 0xe20D, SETTING, MT_CPAP, SESSION,
|
|
|
|
"RMS9_RampEnable", QObject::tr("Ramp"),
|
|
|
|
QObject::tr("Ramp Enable"),
|
|
|
|
QObject::tr("Ramp"),
|
|
|
|
"", LOOKUP, Qt::black));
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
chan->addOption(1, STR_TR_On);
|
|
|
|
|
|
|
|
|
2014-08-06 14:06:44 +00:00
|
|
|
// Modelmap needs channels initalized above!!!
|
2014-08-04 19:57:48 +00:00
|
|
|
ResInitModelMap();
|
2014-08-06 14:06:44 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool resmed_initialized = false;
|
|
|
|
void ResmedLoader::Register()
|
|
|
|
{
|
|
|
|
if (resmed_initialized) { return; }
|
|
|
|
|
|
|
|
qDebug() << "Registering ResmedLoader";
|
|
|
|
RegisterLoader(new ResmedLoader());
|
|
|
|
|
2014-08-04 19:57:48 +00:00
|
|
|
resmed_initialized = true;
|
2011-06-28 02:21:38 +00:00
|
|
|
}
|
|
|
|
|
2014-05-04 18:02:41 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Model number information
|
|
|
|
// 36003, 36013, 36023, 36103, 36113, 36123, 36143, 36203,
|
|
|
|
// 36223, 36243, 36263, 36303, 36343, 36363 S9 Elite Series
|
|
|
|
// 36005, 36015, 36025, 36105, 36115, 36125, 36145, 36205,
|
|
|
|
// 36225, 36245, 36265, 36305, 36325, 36345, 36365 S9 AutoSet Series
|
|
|
|
// 36065 S9 AutoSet for Her
|
|
|
|
// 36001, 36011, 36021, 36141, 36201, 36221, 36261, 36301,
|
|
|
|
// 36361 S9 Escape
|
|
|
|
// 36002, 36012, 36022, 36302, 36362 S9 Escape Auto
|
|
|
|
// 36004, 36014, 36024, 36114, 36124, 36144, 36204, 36224,
|
|
|
|
// 36284, 36304 S9 VPAP S (+ H5i, + Climate Control)
|
|
|
|
// 36006, 36016, 36026 S9 VPAP AUTO (+ H5i, + Climate Control)
|
|
|
|
|
|
|
|
// 36007, 36017, 36027, 36367
|
|
|
|
// S9 VPAP ADAPT (+ H5i, + Climate
|
|
|
|
// Control)
|
|
|
|
// 36008, 36018, 36028, 36108, 36148, 36208, 36228, 36368 S9 VPAP ST (+ H5i, + Climate Control)
|
|
|
|
// 36100, 36110, 36120, 36140, 36200, 36220, 36360 S9 AUTOSET CS
|
|
|
|
// 36106, 36116, 36126, 36146, 36206, 36226, 36366 S9 AUTOSET 25
|
|
|
|
// 36118, 36128 S9 VPAP ST 22
|
|
|
|
// 36039, 36159, 36169, 36379 S9 VPAP ST-A
|
|
|
|
// 24921, 24923, 24925, 24926, 24927 ResMed Power Station II (RPSII)
|
|
|
|
// 33030 S8 Compact
|
|
|
|
// 33001, 33007, 33013, 33036, 33060 S8 Escape
|
|
|
|
// 33032 S8 Lightweight
|
|
|
|
// 33033 S8 AutoScore
|
|
|
|
// 33048, 33051, 33052, 33053, 33054, 33061 S8 Escape II
|
|
|
|
// 33055 S8 Lightweight II
|
|
|
|
// 33021 S8 Elite
|
|
|
|
// 33039, 33045, 33062, 33072, 33073, 33074, 33075 S8 Elite II
|
|
|
|
// 33044 S8 AutoScore II
|
|
|
|
// 33105, 33112, 33126 S8 AutoSet (including Spirit & Vantage)
|
|
|
|
// 33128, 33137 S8 Respond
|
|
|
|
// 33129, 33141, 33150 S8 AutoSet II
|
|
|
|
// 33136, 33143, 33144, 33145, 33146, 33147, 33148 S8 AutoSet Spirit II
|
|
|
|
// 33138 S8 AutoSet C
|
|
|
|
// 26101, 26121 VPAP Auto 25
|
|
|
|
// 26119, 26120 VPAP S
|
|
|
|
// 26110, 26122 VPAP ST
|
|
|
|
// 26104, 26105, 26125, 26126 S8 Auto 25
|
|
|
|
// 26102, 26103, 26106, 26107, 26108, 26109, 26123, 26127 VPAP IV
|
|
|
|
// 26112, 26113, 26114, 26115, 26116, 26117, 26118, 26124 VPAP IV ST
|
|
|
|
|