2018-04-22 12:06:48 +00:00
|
|
|
|
/* SleepLib (DeVilbiss) Intellipap Loader Implementation
|
2014-04-09 21:01:57 +00:00
|
|
|
|
*
|
2018-03-28 07:10:52 +00:00
|
|
|
|
* Notes: Intellipap DV54 requires the SmartLink attachment to access this data.
|
2014-04-09 21:01:57 +00:00
|
|
|
|
*
|
2018-03-28 07:10:52 +00:00
|
|
|
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
2014-04-09 21:01:57 +00:00
|
|
|
|
*
|
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
2018-06-04 20:48:38 +00:00
|
|
|
|
* License. See the file COPYING in the main directory of the source code
|
|
|
|
|
* for more details. */
|
2011-11-20 02:59:00 +00:00
|
|
|
|
|
2011-11-21 02:15:10 +00:00
|
|
|
|
#include <QDir>
|
|
|
|
|
|
2011-11-20 02:59:00 +00:00
|
|
|
|
#include "intellipap_loader.h"
|
|
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
|
ChannelID INTP_SmartFlexMode, INTP_SmartFlexLevel;
|
|
|
|
|
|
2018-06-04 23:26:46 +00:00
|
|
|
|
Intellipap::Intellipap(Profile *profile, MachineID id)
|
|
|
|
|
: CPAP(profile, id)
|
2011-11-20 02:59:00 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Intellipap::~Intellipap()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IntellipapLoader::IntellipapLoader()
|
|
|
|
|
{
|
2014-09-17 17:20:01 +00:00
|
|
|
|
const QString INTELLIPAP_ICON = ":/icons/intellipap.png";
|
2018-04-22 12:06:48 +00:00
|
|
|
|
const QString DV6_ICON = ":/icons/dv64.png";
|
|
|
|
|
|
2014-09-17 17:20:01 +00:00
|
|
|
|
QString s = newInfo().series;
|
|
|
|
|
m_pixmap_paths[s] = INTELLIPAP_ICON;
|
|
|
|
|
m_pixmaps[s] = QPixmap(INTELLIPAP_ICON);
|
2018-04-22 12:06:48 +00:00
|
|
|
|
m_pixmap_paths["DV6"] = DV6_ICON;
|
|
|
|
|
m_pixmaps["DV6"] = QPixmap(DV6_ICON);
|
2014-09-17 17:20:01 +00:00
|
|
|
|
|
2014-04-23 13:19:56 +00:00
|
|
|
|
m_buffer = nullptr;
|
2014-05-25 07:07:08 +00:00
|
|
|
|
m_type = MT_CPAP;
|
2011-11-20 02:59:00 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IntellipapLoader::~IntellipapLoader()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
const QString SET_BIN = "SET.BIN";
|
|
|
|
|
const QString SET1 = "SET1";
|
|
|
|
|
const QString DV6 = "DV6";
|
|
|
|
|
const QString SL = "SL";
|
|
|
|
|
|
|
|
|
|
const QString DV6_DIR = "/" + DV6;
|
|
|
|
|
const QString SL_DIR = "/" + SL;
|
|
|
|
|
|
2014-04-28 19:24:39 +00:00
|
|
|
|
bool IntellipapLoader::Detect(const QString & givenpath)
|
2014-04-26 09:54:08 +00:00
|
|
|
|
{
|
2016-03-06 02:50:22 +00:00
|
|
|
|
QString path = givenpath;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (path.endsWith(SL_DIR)) {
|
2016-03-06 02:50:22 +00:00
|
|
|
|
path.chop(3);
|
|
|
|
|
}
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (path.endsWith(DV6_DIR)) {
|
|
|
|
|
path.chop(4);
|
|
|
|
|
}
|
2016-03-06 02:50:22 +00:00
|
|
|
|
|
|
|
|
|
QDir dir(path);
|
2014-04-28 19:24:39 +00:00
|
|
|
|
|
|
|
|
|
if (!dir.exists()) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
// Intellipap DV54 has a folder called SL in the root directory, DV64 has DV6
|
|
|
|
|
if (dir.cd(SL)) {
|
|
|
|
|
// Test for presence of settings file
|
|
|
|
|
return dir.exists(SET1) ? true : false;
|
2014-04-28 19:24:39 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (dir.cd(DV6)) { // DV64
|
|
|
|
|
return dir.exists(SET_BIN) ? true : false;
|
2014-04-28 19:24:39 +00:00
|
|
|
|
}
|
2018-03-25 19:23:05 +00:00
|
|
|
|
return false;
|
2014-04-26 09:54:08 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
enum INTPAP_Type { INTPAP_Unknown, INTPAP_DV5, INTPAP_DV6 };
|
2013-09-14 23:32:14 +00:00
|
|
|
|
|
2011-11-21 02:15:10 +00:00
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
int IntellipapLoader::OpenDV5(const QString & path)
|
2018-03-25 19:23:05 +00:00
|
|
|
|
{
|
|
|
|
|
QString newpath = path + SL_DIR;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
QString filename;
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
//////////////////////////
|
|
|
|
|
// Parse the Settings File
|
|
|
|
|
//////////////////////////
|
2018-03-25 19:23:05 +00:00
|
|
|
|
filename = newpath + "/" + SET1;
|
2011-11-21 02:15:10 +00:00
|
|
|
|
QFile f(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2016-03-06 02:50:22 +00:00
|
|
|
|
if (!f.exists()) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
f.open(QFile::ReadOnly);
|
|
|
|
|
QTextStream tstream(&f);
|
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
const QString INT_PROP_Serial = "Serial";
|
|
|
|
|
const QString INT_PROP_Model = "Model";
|
|
|
|
|
const QString INT_PROP_Mode = "Mode";
|
|
|
|
|
const QString INT_PROP_MaxPressure = "Max Pressure";
|
|
|
|
|
const QString INT_PROP_MinPressure = "Min Pressure";
|
|
|
|
|
const QString INT_PROP_IPAP = "IPAP";
|
|
|
|
|
const QString INT_PROP_EPAP = "EPAP";
|
|
|
|
|
const QString INT_PROP_PS = "PS";
|
|
|
|
|
const QString INT_PROP_RampPressure = "Ramp Pressure";
|
|
|
|
|
const QString INT_PROP_RampTime = "Ramp Time";
|
|
|
|
|
|
|
|
|
|
const QString INT_PROP_HourMeter = "Usage Hours";
|
|
|
|
|
const QString INT_PROP_ComplianceMeter = "Compliance Hours";
|
|
|
|
|
const QString INT_PROP_ErrorCode = "Error";
|
|
|
|
|
const QString INT_PROP_LastErrorCode = "Long Error";
|
|
|
|
|
const QString INT_PROP_LowUseThreshold = "Low Usage";
|
|
|
|
|
const QString INT_PROP_SmartFlex = "SmartFlex";
|
|
|
|
|
const QString INT_PROP_SmartFlexMode = "SmartFlexMode";
|
|
|
|
|
|
|
|
|
|
|
2014-05-06 17:39:05 +00:00
|
|
|
|
QHash<QString, QString> lookup;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
lookup["Sn"] = INT_PROP_Serial;
|
|
|
|
|
lookup["Mn"] = INT_PROP_Model;
|
|
|
|
|
lookup["Mo"] = INT_PROP_Mode; // 0 cpap, 1 auto
|
2014-05-06 17:39:05 +00:00
|
|
|
|
//lookup["Pn"]="??";
|
2014-07-28 13:56:29 +00:00
|
|
|
|
lookup["Pu"] = INT_PROP_MaxPressure;
|
|
|
|
|
lookup["Pl"] = INT_PROP_MinPressure;
|
|
|
|
|
lookup["Pi"] = INT_PROP_IPAP;
|
|
|
|
|
lookup["Pe"] = INT_PROP_EPAP; // == WF on Auto models
|
|
|
|
|
lookup["Ps"] = INT_PROP_PS; // == WF on Auto models, Pressure support
|
2014-05-06 17:39:05 +00:00
|
|
|
|
//lookup["Ds"]="??";
|
|
|
|
|
//lookup["Pc"]="??";
|
2014-07-28 13:56:29 +00:00
|
|
|
|
lookup["Pd"] = INT_PROP_RampPressure;
|
|
|
|
|
lookup["Dt"] = INT_PROP_RampTime;
|
2014-05-06 17:39:05 +00:00
|
|
|
|
//lookup["Ld"]="??";
|
|
|
|
|
//lookup["Lh"]="??";
|
|
|
|
|
//lookup["FC"]="??";
|
|
|
|
|
//lookup["FE"]="??";
|
|
|
|
|
//lookup["FL"]="??";
|
|
|
|
|
lookup["A%"]="ApneaThreshold";
|
|
|
|
|
lookup["Ad"]="ApneaDuration";
|
|
|
|
|
lookup["H%"]="HypopneaThreshold";
|
|
|
|
|
lookup["Hd"]="HypopneaDuration";
|
|
|
|
|
//lookup["Pi"]="??";
|
|
|
|
|
//lookup["Pe"]="??";
|
|
|
|
|
lookup["Ri"]="SmartFlexIRnd"; // Inhale Rounding (0-5)
|
|
|
|
|
lookup["Re"]="SmartFlexERnd"; // Inhale Rounding (0-5)
|
|
|
|
|
//lookup["Bu"]="??"; // WF
|
|
|
|
|
//lookup["Ie"]="??"; // 20
|
2014-07-28 13:56:29 +00:00
|
|
|
|
//lookup["Se"]="??"; // 05 //Inspiratory trigger?
|
|
|
|
|
//lookup["Si"]="??"; // 05 // Expiratory Trigger?
|
2014-05-06 17:39:05 +00:00
|
|
|
|
//lookup["Mi"]="??"; // 0
|
|
|
|
|
lookup["Uh"]="HoursMeter"; // 0000.0
|
|
|
|
|
lookup["Up"]="ComplianceMeter"; // 0000.00
|
|
|
|
|
//lookup["Er"]="ErrorCode";, // E00
|
2014-07-28 13:56:29 +00:00
|
|
|
|
//lookup["El"]="LongErrorCode"; // E00 00/00/0000
|
2014-05-06 17:39:05 +00:00
|
|
|
|
//lookup["Hp"]="??";, // 1
|
|
|
|
|
//lookup["Hs"]="??";, // 02
|
|
|
|
|
//lookup["Lu"]="LowUseThreshold"; // defaults to 0 (4 hours)
|
2014-07-28 13:56:29 +00:00
|
|
|
|
lookup["Sf"] = INT_PROP_SmartFlex;
|
|
|
|
|
lookup["Sm"] = INT_PROP_SmartFlexMode;
|
2014-05-06 17:39:05 +00:00
|
|
|
|
lookup["Ks=s"]="Ks_s";
|
|
|
|
|
lookup["Ks=i"]="ks_i";
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
QHash<QString, QString> set1;
|
|
|
|
|
QHash<QString, QString>::iterator hi;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
Machine *mach = nullptr;
|
|
|
|
|
|
|
|
|
|
MachineInfo info = newInfo();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool ok;
|
|
|
|
|
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//EventDataType min_pressure = 0, max_pressure = 0, set_ipap = 0, set_ps = 0,
|
|
|
|
|
EventDataType ramp_pressure = 0, set_epap = 0, ramp_time = 0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
|
|
|
|
|
int papmode = 0, smartflex = 0, smartflexmode = 0;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
while (1) {
|
2014-04-17 05:58:57 +00:00
|
|
|
|
QString line = tstream.readLine();
|
|
|
|
|
|
|
|
|
|
if ((line.length() <= 2) ||
|
|
|
|
|
(line.isNull())) { break; }
|
|
|
|
|
|
|
|
|
|
QString key = line.section("\t", 0, 0).trimmed();
|
|
|
|
|
hi = lookup.find(key);
|
|
|
|
|
|
|
|
|
|
if (hi != lookup.end()) {
|
|
|
|
|
key = hi.value();
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
QString value = line.section("\t", 1).trimmed();
|
2014-07-28 13:56:29 +00:00
|
|
|
|
|
|
|
|
|
if (key == INT_PROP_Mode) {
|
|
|
|
|
papmode = value.toInt(&ok);
|
|
|
|
|
} else if (key == INT_PROP_Serial) {
|
|
|
|
|
info.serial = value;
|
|
|
|
|
} else if (key == INT_PROP_Model) {
|
|
|
|
|
info.model = value;
|
|
|
|
|
} else if (key == INT_PROP_MinPressure) {
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//min_pressure = value.toFloat() / 10.0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
} else if (key == INT_PROP_MaxPressure) {
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//max_pressure = value.toFloat() / 10.0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
} else if (key == INT_PROP_IPAP) {
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//set_ipap = value.toFloat() / 10.0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
} else if (key == INT_PROP_EPAP) {
|
|
|
|
|
set_epap = value.toFloat() / 10.0;
|
|
|
|
|
} else if (key == INT_PROP_PS) {
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//set_ps = value.toFloat() / 10.0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
} else if (key == INT_PROP_RampPressure) {
|
|
|
|
|
ramp_pressure = value.toFloat() / 10.0;
|
|
|
|
|
} else if (key == INT_PROP_RampTime) {
|
|
|
|
|
ramp_time = value.toFloat() / 10.0;
|
|
|
|
|
} else if (key == INT_PROP_SmartFlex) {
|
|
|
|
|
smartflex = value.toInt();
|
|
|
|
|
} else if (key == INT_PROP_SmartFlexMode) {
|
|
|
|
|
smartflexmode = value.toInt();
|
|
|
|
|
} else {
|
|
|
|
|
set1[key] = value;
|
|
|
|
|
}
|
2011-11-21 10:20:11 +00:00
|
|
|
|
qDebug() << key << "=" << value;
|
|
|
|
|
}
|
2011-11-21 02:15:10 +00:00
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
CPAPMode mode = MODE_UNKNOWN;
|
|
|
|
|
|
|
|
|
|
switch (papmode) {
|
|
|
|
|
case 0:
|
|
|
|
|
mode = MODE_CPAP;
|
|
|
|
|
break;
|
|
|
|
|
case 1:
|
|
|
|
|
mode = (set_epap > 0) ? MODE_BILEVEL_FIXED : MODE_APAP;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
qDebug() << "New machine mode";
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
if (!info.serial.isEmpty()) {
|
2018-04-22 12:06:48 +00:00
|
|
|
|
mach = p_profile->CreateMachine(info);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
if (!mach) {
|
|
|
|
|
qDebug() << "Couldn't get Intellipap machine record";
|
2014-07-29 14:38:59 +00:00
|
|
|
|
return -1;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2011-11-21 02:15:10 +00:00
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
QString backupPath = mach->getBackupPath();
|
|
|
|
|
QString copypath = path;
|
|
|
|
|
|
|
|
|
|
if (QDir::cleanPath(path).compare(QDir::cleanPath(backupPath)) != 0) {
|
|
|
|
|
copyPath(path, backupPath);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
// Refresh properties data..
|
2014-04-17 05:58:57 +00:00
|
|
|
|
for (QHash<QString, QString>::iterator i = set1.begin(); i != set1.end(); i++) {
|
|
|
|
|
mach->properties[i.key()] = i.value();
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
f.close();
|
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
///////////////////////////////////////////////
|
|
|
|
|
// Parse the Session Index (U File)
|
|
|
|
|
///////////////////////////////////////////////
|
2011-11-21 10:20:11 +00:00
|
|
|
|
unsigned char buf[27];
|
2014-04-17 05:58:57 +00:00
|
|
|
|
filename = newpath + "/U";
|
2011-11-21 10:20:11 +00:00
|
|
|
|
f.setFileName(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-07-29 14:38:59 +00:00
|
|
|
|
if (!f.exists()) { return -1; }
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
|
|
|
|
QVector<quint32> SessionStart;
|
|
|
|
|
QVector<quint32> SessionEnd;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
QHash<SessionID, Session *> Sessions;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
|
|
|
|
quint32 ts1, ts2;//, length;
|
|
|
|
|
//unsigned char cs;
|
2011-11-21 02:15:10 +00:00
|
|
|
|
f.open(QFile::ReadOnly);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
int cnt = 0;
|
|
|
|
|
QDateTime epoch(QDate(2002, 1, 1), QTime(0, 0, 0), Qt::UTC); // Intellipap Epoch
|
|
|
|
|
int ep = epoch.toTime_t();
|
|
|
|
|
|
2011-11-21 02:15:10 +00:00
|
|
|
|
do {
|
2014-04-17 05:58:57 +00:00
|
|
|
|
cnt = f.read((char *)buf, 9);
|
2014-07-28 13:56:29 +00:00
|
|
|
|
// big endian
|
2014-04-17 05:58:57 +00:00
|
|
|
|
ts1 = (buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3];
|
|
|
|
|
ts2 = (buf[4] << 24) | (buf[5] << 16) | (buf[6] << 8) | buf[7];
|
2014-07-28 13:56:29 +00:00
|
|
|
|
// buf[8] == ??? What is this byte? A Bit Field? A checksum?
|
2014-04-17 05:58:57 +00:00
|
|
|
|
ts1 += ep;
|
|
|
|
|
ts2 += ep;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
SessionStart.append(ts1);
|
|
|
|
|
SessionEnd.append(ts2);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
} while (cnt > 0);
|
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
qDebug() << "U file logs" << SessionStart.size() << "sessions.";
|
2011-11-21 02:15:10 +00:00
|
|
|
|
f.close();
|
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
///////////////////////////////////////////////
|
|
|
|
|
// Parse the Session Data (L File)
|
|
|
|
|
///////////////////////////////////////////////
|
2014-04-17 05:58:57 +00:00
|
|
|
|
filename = newpath + "/L";
|
2011-11-21 02:15:10 +00:00
|
|
|
|
f.setFileName(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-07-29 14:38:59 +00:00
|
|
|
|
if (!f.exists()) { return -1; }
|
2011-11-21 02:15:10 +00:00
|
|
|
|
|
|
|
|
|
f.open(QFile::ReadOnly);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
long size = f.size();
|
|
|
|
|
int recs = size / 26;
|
|
|
|
|
m_buffer = new unsigned char [size];
|
2011-11-21 02:15:10 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (size != f.read((char *)m_buffer, size)) {
|
|
|
|
|
qDebug() << "Couldn't read 'L' data" << filename;
|
2014-07-29 14:38:59 +00:00
|
|
|
|
return -1;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Session *sess;
|
|
|
|
|
SessionID sid;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
QHash<SessionID,qint64> rampstart;
|
2014-07-29 14:38:59 +00:00
|
|
|
|
QHash<SessionID,qint64> rampend;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
for (int i = 0; i < SessionStart.size(); i++) {
|
|
|
|
|
sid = SessionStart[i];
|
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
if (mach->SessionExists(sid)) {
|
|
|
|
|
// knock out the already imported sessions..
|
2014-04-17 05:58:57 +00:00
|
|
|
|
SessionStart[i] = 0;
|
|
|
|
|
SessionEnd[i] = 0;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
} else if (!Sessions.contains(sid)) {
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sess = Sessions[sid] = new Session(mach, sid);
|
2014-07-29 14:38:59 +00:00
|
|
|
|
|
|
|
|
|
sess->really_set_first(qint64(sid) * 1000L);
|
|
|
|
|
// sess->really_set_last(qint64(SessionEnd[i]) * 1000L);
|
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
rampstart[sid] = 0;
|
2014-07-29 14:38:59 +00:00
|
|
|
|
rampend[sid] = 0;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
sess->SetChanged(true);
|
2014-07-28 13:56:29 +00:00
|
|
|
|
if (mode >= MODE_BILEVEL_FIXED) {
|
|
|
|
|
sess->AddEventList(CPAP_IPAP, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_EPAP, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_PS, EVL_Event);
|
|
|
|
|
} else {
|
|
|
|
|
sess->AddEventList(CPAP_Pressure, EVL_Event);
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
sess->AddEventList(INTELLIPAP_Unknown1, EVL_Event);
|
|
|
|
|
sess->AddEventList(INTELLIPAP_Unknown2, EVL_Event);
|
|
|
|
|
|
|
|
|
|
sess->AddEventList(CPAP_LeakTotal, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_MaxLeak, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_TidalVolume, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_MinuteVent, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_RespRate, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_Snore, EVL_Event);
|
2014-07-29 17:29:54 +00:00
|
|
|
|
|
|
|
|
|
sess->AddEventList(CPAP_Obstructive, EVL_Event);
|
2014-08-03 13:00:13 +00:00
|
|
|
|
sess->AddEventList(CPAP_VSnore, EVL_Event);
|
2014-07-29 17:29:54 +00:00
|
|
|
|
sess->AddEventList(CPAP_Hypopnea, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_NRI, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_LeakFlag, EVL_Event);
|
|
|
|
|
sess->AddEventList(CPAP_ExP, EVL_Event);
|
|
|
|
|
|
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
} else {
|
|
|
|
|
// If there is a double up, null out the earlier session
|
|
|
|
|
// otherwise there will be a crash on shutdown.
|
2014-04-17 05:58:57 +00:00
|
|
|
|
for (int z = 0; z < SessionStart.size(); z++) {
|
|
|
|
|
if (SessionStart[z] == (quint32)sid) {
|
|
|
|
|
SessionStart[z] = 0;
|
|
|
|
|
SessionEnd[z] = 0;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
QDateTime d = QDateTime::fromTime_t(sid);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
qDebug() << sid << "has double ups" << d;
|
|
|
|
|
/*Session *sess=Sessions[sid];
|
|
|
|
|
Sessions.erase(Sessions.find(sid));
|
|
|
|
|
delete sess;
|
|
|
|
|
SessionStart[i]=0;
|
|
|
|
|
SessionEnd[i]=0; */
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
long pos = 0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
int rampval = 0;
|
2014-07-29 14:38:59 +00:00
|
|
|
|
sid = 0;
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//SessionID lastsid = 0;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-10-02 07:56:57 +00:00
|
|
|
|
//int last_minp=0, last_maxp=0, last_ps=0, last_pres = 0;
|
2014-08-03 13:00:13 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
for (int i = 0; i < recs; i++) {
|
2011-11-21 10:20:11 +00:00
|
|
|
|
// convert timestamp to real epoch
|
2014-07-09 03:49:20 +00:00
|
|
|
|
ts1 = ((m_buffer[pos] << 24) | (m_buffer[pos + 1] << 16) | (m_buffer[pos + 2] << 8) | m_buffer[pos + 3]) + ep;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
for (int j = 0; j < SessionStart.size(); j++) {
|
|
|
|
|
sid = SessionStart[j];
|
|
|
|
|
|
|
|
|
|
if (!sid) { continue; }
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if ((ts1 >= (quint32)sid) && (ts1 <= SessionEnd[j])) {
|
|
|
|
|
Session *sess = Sessions[sid];
|
2014-07-29 14:38:59 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
qint64 time = quint64(ts1) * 1000L;
|
2014-07-29 15:00:27 +00:00
|
|
|
|
sess->really_set_last(time);
|
2014-07-28 13:56:29 +00:00
|
|
|
|
sess->settings[CPAP_Mode] = mode;
|
|
|
|
|
|
|
|
|
|
int minp = m_buffer[pos + 0x13];
|
|
|
|
|
int maxp = m_buffer[pos + 0x14];
|
|
|
|
|
int ps = m_buffer[pos + 0x15];
|
|
|
|
|
int pres = m_buffer[pos + 0xd];
|
|
|
|
|
|
|
|
|
|
if (mode >= MODE_BILEVEL_FIXED) {
|
|
|
|
|
rampval = maxp;
|
|
|
|
|
} else {
|
|
|
|
|
rampval = minp;
|
|
|
|
|
}
|
2014-08-03 13:00:13 +00:00
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
qint64 rs = rampstart[sid];
|
|
|
|
|
|
|
|
|
|
if (pres < rampval) {
|
|
|
|
|
if (!rs) {
|
2014-08-03 13:00:13 +00:00
|
|
|
|
// ramp started
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// int rv = pres-rampval;
|
|
|
|
|
// double ramp =
|
|
|
|
|
|
2014-07-28 13:56:29 +00:00
|
|
|
|
rampstart[sid] = time;
|
|
|
|
|
}
|
2014-07-29 14:38:59 +00:00
|
|
|
|
rampend[sid] = time;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (rs > 0) {
|
|
|
|
|
if (!sess->eventlist.contains(CPAP_Ramp)) {
|
|
|
|
|
sess->AddEventList(CPAP_Ramp, EVL_Event);
|
|
|
|
|
}
|
|
|
|
|
int duration = (time - rs) / 1000L;
|
|
|
|
|
sess->eventlist[CPAP_Ramp][0]->AddEvent(time, duration);
|
|
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
|
rampstart.remove(sid);
|
|
|
|
|
rampend.remove(sid);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Do this after ramp, because ramp calcs might need to insert interpolated pressure samples
|
|
|
|
|
if (mode >= MODE_BILEVEL_FIXED) {
|
|
|
|
|
|
|
|
|
|
sess->settings[CPAP_EPAP] = float(minp) / 10.0;
|
|
|
|
|
sess->settings[CPAP_IPAP] = float(maxp) / 10.0;
|
|
|
|
|
|
|
|
|
|
sess->settings[CPAP_PS] = float(ps) / 10.0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sess->eventlist[CPAP_IPAP][0]->AddEvent(time, float(pres) / 10.0);
|
|
|
|
|
sess->eventlist[CPAP_EPAP][0]->AddEvent(time, float(pres-ps) / 10.0);
|
|
|
|
|
// rampval = maxp;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
sess->eventlist[CPAP_Pressure][0]->AddEvent(time, float(pres) / 10.0); // current pressure
|
|
|
|
|
// rampval = minp;
|
|
|
|
|
|
|
|
|
|
if (mode == MODE_APAP) {
|
|
|
|
|
sess->settings[CPAP_PressureMin] = float(minp) / 10.0;
|
|
|
|
|
sess->settings[CPAP_PressureMax] = float(maxp) / 10.0;
|
|
|
|
|
} else if (mode == MODE_CPAP) {
|
|
|
|
|
sess->settings[CPAP_Pressure] = float(maxp) / 10.0;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sess->eventlist[CPAP_LeakTotal][0]->AddEvent(time, m_buffer[pos + 0x7]); // "Average Leak"
|
|
|
|
|
sess->eventlist[CPAP_MaxLeak][0]->AddEvent(time, m_buffer[pos + 0x6]); // "Max Leak"
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
int rr = m_buffer[pos + 0xa];
|
|
|
|
|
sess->eventlist[CPAP_RespRate][0]->AddEvent(time, rr); // Respiratory Rate
|
2014-07-28 13:56:29 +00:00
|
|
|
|
// sess->eventlist[INTELLIPAP_Unknown1][0]->AddEvent(time, m_buffer[pos + 0xf]); //
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sess->eventlist[INTELLIPAP_Unknown1][0]->AddEvent(time, m_buffer[pos + 0xc]);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sess->eventlist[CPAP_Snore][0]->AddEvent(time, m_buffer[pos + 0x4]); //4/5??
|
2011-11-21 02:15:10 +00:00
|
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
|
if (m_buffer[pos+0x4] > 0) {
|
|
|
|
|
sess->eventlist[CPAP_VSnore][0]->AddEvent(time, m_buffer[pos + 0x5]);
|
|
|
|
|
}
|
|
|
|
|
|
2011-11-24 12:47:25 +00:00
|
|
|
|
// 0x0f == Leak Event
|
|
|
|
|
// 0x04 == Snore?
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (m_buffer[pos + 0xf] > 0) { // Leak Event
|
|
|
|
|
sess->eventlist[CPAP_LeakFlag][0]->AddEvent(time, m_buffer[pos + 0xf]);
|
2011-11-24 12:47:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (m_buffer[pos + 0x5] > 4) { // This matches Exhale Puff.. not sure why 4
|
2014-07-09 03:49:20 +00:00
|
|
|
|
//MW: Are the lower 2 bits something else?
|
2011-11-30 12:32:16 +00:00
|
|
|
|
|
2014-07-29 05:19:10 +00:00
|
|
|
|
sess->eventlist[CPAP_ExP][0]->AddEvent(time, m_buffer[pos + 0x5]);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (m_buffer[pos + 0x10] > 0) {
|
2014-07-29 05:19:10 +00:00
|
|
|
|
sess->eventlist[CPAP_Obstructive][0]->AddEvent(time, m_buffer[pos + 0x10]);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2011-11-30 12:32:16 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
if (m_buffer[pos + 0x11] > 0) {
|
2014-07-29 05:19:10 +00:00
|
|
|
|
sess->eventlist[CPAP_Hypopnea][0]->AddEvent(time, m_buffer[pos + 0x11]);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
if (m_buffer[pos + 0x12] > 0) { // NRI // is this == to RERA?? CA??
|
2014-07-29 05:19:10 +00:00
|
|
|
|
sess->eventlist[CPAP_NRI][0]->AddEvent(time, m_buffer[pos + 0x12]);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2011-11-30 12:32:16 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
quint16 tv = (m_buffer[pos + 0x8] << 8) | m_buffer[pos + 0x9]; // correct
|
2011-11-30 12:32:16 +00:00
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sess->eventlist[CPAP_TidalVolume][0]->AddEvent(time, tv);
|
|
|
|
|
|
|
|
|
|
EventDataType mv = tv * rr; // MinuteVent=TidalVolume * Respiratory Rate
|
|
|
|
|
sess->eventlist[CPAP_MinuteVent][0]->AddEvent(time, mv / 1000.0);
|
2011-11-21 10:20:11 +00:00
|
|
|
|
break;
|
2014-08-03 13:00:13 +00:00
|
|
|
|
} else {
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2014-10-08 16:51:09 +00:00
|
|
|
|
//lastsid = sid;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
|
|
|
|
pos += 26;
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-07-29 14:38:59 +00:00
|
|
|
|
// Close any open ramps and store the event.
|
|
|
|
|
QHash<SessionID,qint64>::iterator rit;
|
|
|
|
|
QHash<SessionID,qint64>::iterator rit_end = rampstart.end();
|
|
|
|
|
|
|
|
|
|
for (rit = rampstart.begin(); rit != rit_end; ++rit) {
|
|
|
|
|
qint64 rs = rit.value();
|
|
|
|
|
SessionID sid = rit.key();
|
|
|
|
|
if (rs > 0) {
|
|
|
|
|
qint64 re = rampend[rit.key()];
|
|
|
|
|
|
|
|
|
|
Session *sess = Sessions[sid];
|
|
|
|
|
if (!sess->eventlist.contains(CPAP_Ramp)) {
|
|
|
|
|
sess->AddEventList(CPAP_Ramp, EVL_Event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int duration = (re - rs) / 1000L;
|
|
|
|
|
sess->eventlist[CPAP_Ramp][0]->AddEvent(re, duration);
|
|
|
|
|
rit.value() = 0;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
|
for (int i = 0; i < SessionStart.size(); i++) {
|
|
|
|
|
SessionID sid = SessionStart[i];
|
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
if (sid) {
|
2014-04-17 05:58:57 +00:00
|
|
|
|
sess = Sessions[sid];
|
2014-07-30 17:14:28 +00:00
|
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
|
if (!sess) continue;
|
2014-07-09 03:49:20 +00:00
|
|
|
|
|
2014-07-30 17:14:28 +00:00
|
|
|
|
// quint64 first = qint64(sid) * 1000L;
|
2014-10-02 07:56:57 +00:00
|
|
|
|
//quint64 last = qint64(SessionEnd[i]) * 1000L;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
|
if (sess->last() > 0) {
|
2014-07-30 20:25:06 +00:00
|
|
|
|
// sess->really_set_last(last);
|
2014-07-30 17:14:28 +00:00
|
|
|
|
|
|
|
|
|
|
2014-08-03 13:00:13 +00:00
|
|
|
|
sess->settings[INTP_SmartFlexLevel] = smartflex;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
|
if (smartflexmode == 0) {
|
2014-08-03 13:00:13 +00:00
|
|
|
|
sess->settings[INTP_SmartFlexMode] = PM_FullTime;
|
2014-07-29 15:00:27 +00:00
|
|
|
|
} else {
|
2014-08-03 13:00:13 +00:00
|
|
|
|
sess->settings[INTP_SmartFlexMode] = PM_RampOnly;
|
2014-07-29 15:00:27 +00:00
|
|
|
|
}
|
2014-07-28 13:56:29 +00:00
|
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
|
sess->settings[CPAP_RampPressure] = ramp_pressure;
|
|
|
|
|
sess->settings[CPAP_RampTime] = ramp_time;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
|
sess->UpdateSummaries();
|
2014-09-17 06:12:38 +00:00
|
|
|
|
|
2014-07-30 20:33:24 +00:00
|
|
|
|
addSession(sess);
|
2014-07-29 15:00:27 +00:00
|
|
|
|
} else {
|
|
|
|
|
delete sess;
|
|
|
|
|
}
|
2014-07-11 12:09:38 +00:00
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
|
|
2014-07-30 20:33:24 +00:00
|
|
|
|
finishAddingSessions();
|
2011-11-21 10:20:11 +00:00
|
|
|
|
mach->Save();
|
|
|
|
|
|
2014-07-30 20:33:24 +00:00
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
|
delete [] m_buffer;
|
|
|
|
|
|
|
|
|
|
f.close();
|
2011-11-20 23:39:55 +00:00
|
|
|
|
|
2014-07-29 14:38:59 +00:00
|
|
|
|
int c = Sessions.size();
|
|
|
|
|
return c;
|
2011-11-20 02:59:00 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
struct DV6_S_Record
|
|
|
|
|
{
|
|
|
|
|
Session * sess;
|
|
|
|
|
unsigned char u1; //00 (position)
|
|
|
|
|
unsigned int start_time; //01
|
|
|
|
|
unsigned int stop_time; //05
|
|
|
|
|
unsigned int atpressure_time;//09
|
|
|
|
|
EventDataType hours; //13
|
|
|
|
|
EventDataType meh; //14
|
|
|
|
|
EventDataType pressureAvg; //15
|
|
|
|
|
EventDataType pressureMax; //16
|
|
|
|
|
EventDataType pressure50; //17 50th percentile
|
|
|
|
|
EventDataType pressure90; //18 90th percentile
|
|
|
|
|
EventDataType pressure95; //19 95th percentile
|
|
|
|
|
EventDataType pressureStdDev;//20 std deviation
|
|
|
|
|
EventDataType u2; //21
|
|
|
|
|
EventDataType leakAvg; //22
|
|
|
|
|
EventDataType leakMax; //23
|
|
|
|
|
EventDataType leak50; //24 50th percentile
|
|
|
|
|
EventDataType leak90; //25 90th percentile
|
|
|
|
|
EventDataType leak95; //26 95th percentile
|
|
|
|
|
EventDataType leakStdDev; //27 std deviation
|
|
|
|
|
EventDataType tidalVolume; //28 & 0x29
|
|
|
|
|
EventDataType avgBreathRate; //30
|
|
|
|
|
EventDataType u3;
|
|
|
|
|
EventDataType u4; //32 snores / hypopnea per minute
|
|
|
|
|
EventDataType timeInExPuf; //33 Time in Expiratory Puff
|
|
|
|
|
EventDataType timeInFL; //34 Time in Flow Limitation
|
|
|
|
|
EventDataType timeInPB; //35 Time in Periodic Breathing
|
|
|
|
|
EventDataType maskFit; //36 mask fit (or rather, not fit) percentage
|
|
|
|
|
EventDataType indexOA; //37 Obstructive
|
|
|
|
|
EventDataType indexCA; //38 Central index
|
|
|
|
|
EventDataType indexHyp; //39 Hypopnea Index
|
|
|
|
|
EventDataType r0; //40 Reserved?
|
|
|
|
|
EventDataType r1; //41 Reserved?
|
|
|
|
|
//42-48 unknown
|
|
|
|
|
EventDataType pressureSetMin; //49
|
|
|
|
|
EventDataType pressureSetMax; //50
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
int IntellipapLoader::OpenDV6(const QString & path)
|
2018-03-25 19:23:05 +00:00
|
|
|
|
{
|
|
|
|
|
QString newpath = path + DV6_DIR;
|
|
|
|
|
|
|
|
|
|
// Prime the machine database's info field with stuff relevant to this machine
|
|
|
|
|
MachineInfo info = newInfo();
|
|
|
|
|
info.series = "DV6";
|
2018-03-28 06:22:42 +00:00
|
|
|
|
info.serial = "Unknown";
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int vmin=0, vmaj=0;
|
2018-03-28 06:22:42 +00:00
|
|
|
|
EventDataType max_pressure=0, min_pressure=0; //, starting_pressure;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
|
|
|
|
QByteArray str, dataBA;
|
|
|
|
|
unsigned char *data = NULL;
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Parse SET.BIN settings file
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
QFile f(newpath+"/"+SET_BIN);
|
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
// Guessing settings is just a binary packed 0 terminated string list
|
|
|
|
|
// as in this is a continuation of the old string SET1 settings file, just the value fields.
|
|
|
|
|
// Each field is zero terminated
|
|
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
|
|
// Read and parse entire SET.BIN file
|
|
|
|
|
dataBA = f.readAll();
|
|
|
|
|
f.close();
|
|
|
|
|
|
|
|
|
|
// Parse it as we go...
|
|
|
|
|
for (int i=0; i< dataBA.size(); ++i) { // deliberately going one further to catch end condition
|
|
|
|
|
if ((dataBA.at(i) == 0) || (i >= dataBA.size()-1)) { // if null terminated or last byte
|
|
|
|
|
|
|
|
|
|
switch(cnt) {
|
|
|
|
|
case 1: // Serial Number
|
|
|
|
|
info.serial = QString(str);
|
|
|
|
|
break;
|
|
|
|
|
case 2: // Firmware version?
|
|
|
|
|
vmaj = (unsigned char)str.at(0);
|
|
|
|
|
vmin = (unsigned char)str.at(1);
|
|
|
|
|
break;
|
|
|
|
|
case 3: // ??? 0x64 100 // Starting Pressure?
|
|
|
|
|
//starting_pressure = (unsigned char)str.at(0);
|
|
|
|
|
// or is it 64, as in BCD coded model number?
|
|
|
|
|
break;
|
|
|
|
|
case 4: // Max Pressure
|
|
|
|
|
max_pressure = (unsigned char)str.at(0);
|
|
|
|
|
break;
|
|
|
|
|
case 5: // Min Pressure
|
|
|
|
|
min_pressure = (unsigned char)str.at(0);
|
|
|
|
|
break;
|
|
|
|
|
case 6: // The settings that were used to flag OA's and Hyp's...
|
|
|
|
|
//OA_min = (unsigned char)str.at(0); // minimum OA duration
|
|
|
|
|
//OA_thresh = (unsigned char)str.at(1); // OA flow restriction threshold
|
|
|
|
|
//HY_min = (unsigned char)str.at(2); // minimum Hyp duration
|
|
|
|
|
//HY_thresh = (unsigned char)str.at(3); // Hyp flow restriction threshold
|
|
|
|
|
break;
|
|
|
|
|
case 7:
|
|
|
|
|
//ramp_time = (unsigned char)str.at(0);
|
|
|
|
|
// ??? 250 = (unsigned char)str.at(1); // 25.0 (div 10) is maximum CPAP pressure
|
|
|
|
|
break;
|
|
|
|
|
case 8: // 0
|
|
|
|
|
break;
|
|
|
|
|
case 9: // 01
|
|
|
|
|
break;
|
|
|
|
|
case 10:
|
|
|
|
|
//SFFRI = (unsigned char)str.at(0); //Smartflex flow rounding inhalation setting
|
|
|
|
|
//SFFRE = (unsigned char)str.at(1); //Smartflex flow rounding exhalation setting
|
|
|
|
|
//??? = (unsigned char)str.at(2); // 0x04
|
|
|
|
|
break;
|
|
|
|
|
case 11: // 0
|
|
|
|
|
break;
|
|
|
|
|
case 12:
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
// Clear and start a new data record
|
|
|
|
|
str.clear();
|
|
|
|
|
cnt++;
|
|
|
|
|
} else {
|
|
|
|
|
// Add the character to the current string
|
|
|
|
|
str.append(dataBA[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else { // if f.open settings file
|
|
|
|
|
// Settings file open failed, return
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Parser VER.BIN for model number
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
f.setFileName(newpath+"/VER.BIN");
|
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
dataBA = f.readAll();
|
|
|
|
|
f.close();
|
|
|
|
|
|
|
|
|
|
int cnt = 0;
|
|
|
|
|
data = (unsigned char *)dataBA.data();
|
|
|
|
|
for (int i=0; i< dataBA.size(); ++i) { // deliberately going one further to catch end condition
|
|
|
|
|
if ((dataBA.at(i) == 0) || (i >= dataBA.size()-1)) { // if null terminated or last byte
|
|
|
|
|
|
|
|
|
|
switch(cnt) {
|
|
|
|
|
case 1: // serial
|
|
|
|
|
break;
|
|
|
|
|
case 2: // model
|
|
|
|
|
info.model = str;
|
|
|
|
|
break;
|
|
|
|
|
case 7: // ??? V025RN20170
|
|
|
|
|
break;
|
|
|
|
|
case 9: // ??? V014BL20150630
|
|
|
|
|
break;
|
|
|
|
|
case 11: // ??? 01 09
|
|
|
|
|
break;
|
|
|
|
|
case 12: // ??? 0C 0C
|
|
|
|
|
break;
|
|
|
|
|
case 14: // ??? BA 0C
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Clear and start a new data record
|
|
|
|
|
str.clear();
|
|
|
|
|
cnt++;
|
|
|
|
|
} else {
|
|
|
|
|
// Add the character to the current string
|
|
|
|
|
str.append(dataBA[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else { // if (f.open(...)
|
|
|
|
|
// VER.BIN open failed
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Creates Machine database record if it doesn't exist already
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
2018-04-22 12:06:48 +00:00
|
|
|
|
Machine *mach = p_profile->CreateMachine(info);
|
2018-03-28 06:22:42 +00:00
|
|
|
|
if (mach == nullptr) {
|
2018-03-25 19:23:05 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
qDebug() << "Opening DV6 (" << info.serial << ")" << "v" << vmaj << "." << vmin << "Min:" << min_pressure << "Max:" << max_pressure;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Open and parse session list and create a list of sessions to import
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
const int DV6_L_RecLength = 45;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
const int DV6_E_RecLength = 0x18;
|
|
|
|
|
const int DV6_S_RecLength = 55;
|
|
|
|
|
unsigned int ts1,ts2;
|
|
|
|
|
|
|
|
|
|
QMap<quint32, DV6_S_Record> summaryList; // QHash is faster, but QMap keeps order
|
|
|
|
|
|
|
|
|
|
QDateTime epoch(QDate(2002, 1, 1), QTime(0, 0, 0), Qt::UTC); // Intellipap Epoch
|
|
|
|
|
int ep = epoch.toTime_t();
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
f.setFileName(newpath+"/S.BIN");
|
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
// Settings is just a binary packed 0 terminated string list
|
|
|
|
|
dataBA = f.readAll();
|
|
|
|
|
f.close();
|
|
|
|
|
|
|
|
|
|
data = (unsigned char *)dataBA.data();
|
|
|
|
|
|
|
|
|
|
int records = dataBA.size() / DV6_S_RecLength;
|
|
|
|
|
|
2018-03-26 04:11:42 +00:00
|
|
|
|
//data[0x11]; // Start of data block
|
|
|
|
|
//data[0x12]; // Record count
|
2018-03-25 19:23:05 +00:00
|
|
|
|
// First record is block header
|
|
|
|
|
for (int r=1; r<records-1; r++) {
|
|
|
|
|
data += DV6_S_RecLength; // just so happen the headers the same length, though we probably should parse it to get it's version
|
|
|
|
|
DV6_S_Record R;
|
|
|
|
|
|
|
|
|
|
ts1 = ((data[4] << 24) | (data[3] << 16) | (data[2] << 8) | data[1])+ep; // session start time
|
|
|
|
|
ts2 = ((data[8] << 24) | (data[7] << 16) | (data[6] << 8) | data[5])+ep; // session end
|
|
|
|
|
|
|
|
|
|
if (!mach->sessionlist.contains(ts1)) { // Check if already imported
|
|
|
|
|
qDebug() << "Detected new Session" << ts1;
|
|
|
|
|
R.sess = new Session(mach, ts1);
|
|
|
|
|
R.sess->SetChanged(true);
|
|
|
|
|
|
|
|
|
|
R.sess->really_set_first(qint64(ts1) * 1000L);
|
|
|
|
|
R.sess->really_set_last(qint64(ts2) * 1000L);
|
|
|
|
|
|
|
|
|
|
R.start_time = ts1;
|
|
|
|
|
R.stop_time = ts2;
|
|
|
|
|
|
|
|
|
|
R.atpressure_time = ((data[12] << 24) | (data[11] << 16) | (data[10] << 8) | data[9])+ep;
|
|
|
|
|
R.hours = float(data[13]) / 10.0F;
|
|
|
|
|
R.pressureSetMin = float(data[49]) / 10.0F;
|
|
|
|
|
R.pressureSetMax = float(data[50]) / 10.0F;
|
|
|
|
|
|
|
|
|
|
// The following stuff is not necessary to decode, but can be used to verify we are on the right track
|
|
|
|
|
//data[14]... unknown
|
|
|
|
|
R.pressureAvg = float(data[15]) / 10.0F;
|
|
|
|
|
R.pressureMax = float(data[16]) / 10.0F;
|
|
|
|
|
R.pressure50 = float(data[17]) / 10.0F;
|
|
|
|
|
R.pressure90 = float(data[18]) / 10.0F;
|
|
|
|
|
R.pressure95 = float(data[19]) / 10.0F;
|
|
|
|
|
R.pressureStdDev = float(data[20]) / 10.0F;
|
|
|
|
|
//data[21]... unknown
|
|
|
|
|
R.leakAvg = float(data[22]) / 10.0F;
|
|
|
|
|
R.leakMax = float(data[23]) / 10.0F;
|
|
|
|
|
R.leak50= float(data[24]) / 10.0F;
|
|
|
|
|
R.leak90 = float(data[25]) / 10.0F;
|
|
|
|
|
R.leak95 = float(data[26]) / 10.0F;
|
|
|
|
|
R.leakStdDev = float(data[27]) / 10.0F;
|
|
|
|
|
|
|
|
|
|
R.tidalVolume = float(data[28] | data[29] << 8);
|
|
|
|
|
R.avgBreathRate = float(data[30] | data[31] << 8);
|
|
|
|
|
|
|
|
|
|
R.sess->settings[CPAP_PressureMin] = R.pressureSetMin;
|
|
|
|
|
R.sess->settings[CPAP_PressureMax] = R.pressureSetMax;
|
|
|
|
|
R.sess->settings[CPAP_Mode] = MODE_APAP;
|
|
|
|
|
|
|
|
|
|
summaryList[ts1] = R;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else { // if (f.open(...)
|
|
|
|
|
// S.BIN open failed
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
QMap<quint32, DV6_S_Record>::iterator SR;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
const int DV6_R_RecLength = 117;
|
|
|
|
|
const int DV6_R_HeaderSize = 55;
|
|
|
|
|
f.setFileName(newpath+"/R.BIN");
|
|
|
|
|
int numRrecs = (f.size()-DV6_R_HeaderSize) / DV6_R_RecLength;
|
2018-03-26 11:51:43 +00:00
|
|
|
|
Session *sess = NULL;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
// Let's not parse R all at once, it's huge
|
|
|
|
|
dataBA = f.read(DV6_R_HeaderSize);
|
|
|
|
|
if (dataBA.size() < DV6_R_HeaderSize) {
|
|
|
|
|
// bit mean aborting on corrupt R file... but oh well
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
sess = NULL;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
EventList * flow = NULL;
|
2018-03-26 05:08:44 +00:00
|
|
|
|
EventList * pressure = NULL;
|
2018-03-26 11:51:43 +00:00
|
|
|
|
// EventList * leak = NULL;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
EventList * OA = NULL;
|
|
|
|
|
EventList * HY = NULL;
|
|
|
|
|
EventList * NOA = NULL;
|
|
|
|
|
EventList * EXP = NULL;
|
|
|
|
|
EventList * FL = NULL;
|
|
|
|
|
EventList * PB = NULL;
|
|
|
|
|
EventList * VS = NULL;
|
|
|
|
|
EventList * LL = NULL;
|
|
|
|
|
EventList * RE = NULL;
|
2018-03-26 04:11:42 +00:00
|
|
|
|
bool inOA = false, inH = false, inCA = false, inExP = false, inVS = false, inFL = false, inPB = false, inRE = false, inLL = false;
|
|
|
|
|
qint64 OAstart = 0, OAend = 0;
|
|
|
|
|
qint64 Hstart = 0, Hend = 0;
|
|
|
|
|
qint64 CAstart = 0, CAend = 0;
|
|
|
|
|
qint64 ExPstart = 0, ExPend = 0;
|
|
|
|
|
qint64 VSstart = 0, VSend = 0;
|
|
|
|
|
qint64 FLstart = 0, FLend = 0;
|
|
|
|
|
qint64 PBstart = 0, PBend = 0;
|
|
|
|
|
qint64 REstart =0, REend = 0;
|
|
|
|
|
qint64 LLstart =0, LLend = 0;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
SR = summaryList.begin();
|
2018-03-25 19:23:05 +00:00
|
|
|
|
for (int r=0; r<numRrecs; ++r) {
|
|
|
|
|
dataBA=f.read(DV6_R_RecLength);
|
|
|
|
|
data = (unsigned char *)dataBA.data();
|
|
|
|
|
if (dataBA.size() < DV6_R_RecLength) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DV6_S_Record *R = &SR.value();
|
|
|
|
|
|
|
|
|
|
ts1 = ((data[4] << 24) | (data[3] << 16) | (data[2] << 8) | data[1]) + ep;
|
|
|
|
|
|
|
|
|
|
while (ts1 > R->stop_time) {
|
|
|
|
|
if (flow && sess) {
|
|
|
|
|
// update min and max
|
|
|
|
|
// then add to machine
|
|
|
|
|
EventDataType min = flow->Min();
|
|
|
|
|
EventDataType max = flow->Max();
|
|
|
|
|
|
|
|
|
|
sess->setMin(CPAP_FlowRate, min);
|
|
|
|
|
sess->setMax(CPAP_FlowRate, max);
|
|
|
|
|
|
|
|
|
|
sess->setPhysMax(CPAP_FlowRate, min);
|
|
|
|
|
sess->setPhysMin(CPAP_FlowRate, max);
|
|
|
|
|
|
|
|
|
|
sess = NULL;
|
|
|
|
|
flow = NULL;
|
|
|
|
|
}
|
|
|
|
|
SR++;
|
|
|
|
|
if (SR == summaryList.end()) break;
|
|
|
|
|
R = &SR.value();
|
|
|
|
|
}
|
|
|
|
|
if (SR == summaryList.end())
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (ts1 >= R->start_time) {
|
|
|
|
|
if (!flow && R->sess) {
|
2018-06-07 00:09:06 +00:00
|
|
|
|
flow = R->sess->AddEventList(CPAP_FlowRate, EVL_Waveform, 1.0f/60.0f, 0.0f, 0.0f, 0.0f, double(2000) / double(50));
|
|
|
|
|
pressure = R->sess->AddEventList(CPAP_Pressure, EVL_Waveform, 0.1f, 0.0f, 0.0f, 0.0f, double(2000) / double(2));
|
2018-03-26 11:51:43 +00:00
|
|
|
|
//leak = R->sess->AddEventList(CPAP_Leak, EVL_Waveform, 1.0, 0.0, 0.0, 0.0, double(2000) / double(1));
|
2018-03-25 19:23:05 +00:00
|
|
|
|
OA = R->sess->AddEventList(CPAP_Obstructive, EVL_Event);
|
|
|
|
|
NOA = R->sess->AddEventList(CPAP_NRI, EVL_Event);
|
|
|
|
|
RE = R->sess->AddEventList(CPAP_RERA, EVL_Event);
|
|
|
|
|
VS = R->sess->AddEventList(CPAP_VSnore, EVL_Event);
|
|
|
|
|
HY = R->sess->AddEventList(CPAP_Hypopnea, EVL_Event);
|
|
|
|
|
EXP = R->sess->AddEventList(CPAP_ExP, EVL_Event);
|
|
|
|
|
FL = R->sess->AddEventList(CPAP_FlowLimit, EVL_Event);
|
|
|
|
|
PB = R->sess->AddEventList(CPAP_PB, EVL_Event);
|
|
|
|
|
LL = R->sess->AddEventList(CPAP_LargeLeak, EVL_Event);
|
|
|
|
|
}
|
|
|
|
|
if (flow) {
|
|
|
|
|
sess = R->sess;
|
2018-03-26 11:51:43 +00:00
|
|
|
|
// starting at position 5 is 100 bytes, 16bit LE 25hz samples
|
2018-03-25 19:23:05 +00:00
|
|
|
|
qint16 *wavedata = (qint16 *)(&data[5]);
|
2018-03-26 11:51:43 +00:00
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
qint64 ti = qint64(ts1) * 1000;
|
|
|
|
|
|
2018-03-26 05:08:44 +00:00
|
|
|
|
unsigned char d[2];
|
|
|
|
|
d[0] = data[105];
|
|
|
|
|
d[1] = data[106];
|
2018-03-26 04:11:42 +00:00
|
|
|
|
flow->AddWaveform(ti+40000,wavedata,50,2000);
|
2018-03-26 05:08:44 +00:00
|
|
|
|
pressure->AddWaveform(ti+40000, d, 2, 2000);
|
|
|
|
|
// Fields data[107] && data[108] are bitfields default is 0x90, occasionally 0x98
|
|
|
|
|
|
|
|
|
|
d[0] = data[107];
|
|
|
|
|
d[1] = data[108];
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
//leak->AddWaveform(ti+40000, d, 2, 2000);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 04:11:42 +00:00
|
|
|
|
|
|
|
|
|
// Needs to track state to pull events out cleanly..
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// High Leak
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
if (data[110] & 3) { // LL state 1st second
|
|
|
|
|
if (!inLL) {
|
|
|
|
|
LLstart = ti;
|
|
|
|
|
inLL = true;
|
|
|
|
|
}
|
|
|
|
|
LLend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inLL) {
|
|
|
|
|
inLL = false;
|
|
|
|
|
LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
|
|
|
|
|
LLstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[114] & 3) {
|
|
|
|
|
if (!inLL) {
|
|
|
|
|
LLstart = ti+1000L;
|
|
|
|
|
inLL = true;
|
|
|
|
|
}
|
|
|
|
|
LLend = ti+2000L;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
if (inLL) {
|
|
|
|
|
inLL = false;
|
|
|
|
|
LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
|
|
|
|
|
LLstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Obstructive Apnea
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
if (data[110] & 12) { // OA state 1st second
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (!inOA) {
|
|
|
|
|
OAstart = ti;
|
|
|
|
|
inOA = true;
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
OAend = ti+1000L;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (inOA) {
|
|
|
|
|
inOA = false;
|
2018-03-26 04:11:42 +00:00
|
|
|
|
OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
OAstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
if (data[114] & 12) {
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (!inOA) {
|
|
|
|
|
OAstart = ti+1000L;
|
|
|
|
|
inOA = true;
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
OAend = ti+2000L;
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (inOA) {
|
|
|
|
|
inOA = false;
|
2018-03-26 04:11:42 +00:00
|
|
|
|
OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
OAstart = 0;
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Hypopnea
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
if (data[110] & 192) {
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (!inH) {
|
|
|
|
|
Hstart = ti;
|
|
|
|
|
inH = true;
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
Hend = ti + 1000L;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (inH) {
|
|
|
|
|
inH = false;
|
2018-03-26 04:11:42 +00:00
|
|
|
|
HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
Hstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-26 04:11:42 +00:00
|
|
|
|
if (data[114] & 192) {
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (!inH) {
|
|
|
|
|
Hstart = ti+1000L;
|
|
|
|
|
inH = true;
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
Hend = ti + 2000L;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (inH) {
|
|
|
|
|
inH = false;
|
2018-03-26 04:11:42 +00:00
|
|
|
|
HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
Hstart = 0;
|
|
|
|
|
}
|
2018-03-26 04:11:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Non Responding Apnea Event (Are these CA's???)
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
if (data[110] & 48) { // OA state 1st second
|
|
|
|
|
if (!inCA) {
|
|
|
|
|
CAstart = ti;
|
|
|
|
|
inCA = true;
|
|
|
|
|
}
|
|
|
|
|
CAend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inCA) {
|
|
|
|
|
inCA = false;
|
|
|
|
|
NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
|
|
|
|
|
CAstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[114] & 48) {
|
|
|
|
|
if (!inCA) {
|
|
|
|
|
CAstart = ti+1000L;
|
|
|
|
|
inCA = true;
|
|
|
|
|
}
|
|
|
|
|
CAend = ti+2000L;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
if (inCA) {
|
|
|
|
|
inCA = false;
|
|
|
|
|
NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
|
|
|
|
|
CAstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 04:11:42 +00:00
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// VSnore Event
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
if (data[109] & 3) { // OA state 1st second
|
|
|
|
|
if (!inVS) {
|
|
|
|
|
VSstart = ti;
|
|
|
|
|
inVS = true;
|
|
|
|
|
}
|
|
|
|
|
VSend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inVS) {
|
|
|
|
|
inVS = false;
|
|
|
|
|
VS->AddEvent(VSstart,(VSend-VSstart) / 1000L);
|
|
|
|
|
VSstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[113] & 3) {
|
|
|
|
|
if (!inVS) {
|
|
|
|
|
VSstart = ti+1000L;
|
|
|
|
|
inVS = true;
|
|
|
|
|
}
|
|
|
|
|
VSend = ti+2000L;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 04:11:42 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (inVS) {
|
|
|
|
|
inVS = false;
|
|
|
|
|
VS->AddEvent(VSstart,(VSend-VSstart) / 1000L);
|
|
|
|
|
VSstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 04:11:42 +00:00
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Expiratory puff Event
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
if (data[109] & 12) { // OA state 1st second
|
|
|
|
|
if (!inExP) {
|
|
|
|
|
ExPstart = ti;
|
|
|
|
|
inExP = true;
|
|
|
|
|
}
|
|
|
|
|
ExPend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inExP) {
|
|
|
|
|
inExP = false;
|
|
|
|
|
EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
|
|
|
|
|
ExPstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[113] & 12) {
|
|
|
|
|
if (!inExP) {
|
|
|
|
|
ExPstart = ti+1000L;
|
|
|
|
|
inExP = true;
|
|
|
|
|
}
|
|
|
|
|
ExPend = ti+2000L;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
if (inExP) {
|
|
|
|
|
inExP = false;
|
|
|
|
|
EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
|
|
|
|
|
ExPstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Flow Limitation Event
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
if (data[109] & 48) { // OA state 1st second
|
|
|
|
|
if (!inFL) {
|
|
|
|
|
FLstart = ti;
|
|
|
|
|
inFL = true;
|
|
|
|
|
}
|
|
|
|
|
FLend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inFL) {
|
|
|
|
|
inFL = false;
|
|
|
|
|
FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
|
|
|
|
|
FLstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[113] & 48) {
|
|
|
|
|
if (!inFL) {
|
|
|
|
|
FLstart = ti+1000L;
|
|
|
|
|
inFL = true;
|
|
|
|
|
}
|
|
|
|
|
FLend = ti+2000L;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
if (inFL) {
|
|
|
|
|
inFL = false;
|
|
|
|
|
FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
|
|
|
|
|
FLstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Periodic Breathing Event
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
if (data[109] & 192) { // OA state 1st second
|
|
|
|
|
if (!inPB) {
|
|
|
|
|
PBstart = ti;
|
|
|
|
|
inPB = true;
|
|
|
|
|
}
|
|
|
|
|
PBend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inPB) {
|
|
|
|
|
inPB = false;
|
|
|
|
|
PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
|
|
|
|
|
PBstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[113] & 192) {
|
|
|
|
|
if (!inPB) {
|
|
|
|
|
PBstart = ti+1000L;
|
|
|
|
|
inPB = true;
|
|
|
|
|
}
|
|
|
|
|
PBend = ti+2000L;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
if (inPB) {
|
|
|
|
|
inPB = false;
|
|
|
|
|
PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
|
|
|
|
|
PBstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
// Respiratory Effort Related Arousal Event
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
if (data[111] & 48) { // OA state 1st second
|
|
|
|
|
if (!inRE) {
|
|
|
|
|
REstart = ti;
|
|
|
|
|
inRE = true;
|
|
|
|
|
}
|
|
|
|
|
REend = ti+1000L;
|
|
|
|
|
} else {
|
|
|
|
|
if (inRE) {
|
|
|
|
|
inRE = false;
|
|
|
|
|
RE->AddEvent(REstart,(REend-REstart) / 1000L);
|
|
|
|
|
REstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (data[115] & 48) {
|
|
|
|
|
if (!inRE) {
|
|
|
|
|
REstart = ti+1000L;
|
|
|
|
|
inRE = true;
|
|
|
|
|
}
|
|
|
|
|
REend = ti+2000L;
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
if (inRE) {
|
|
|
|
|
inRE = false;
|
|
|
|
|
RE->AddEvent(REstart,(REend-REstart) / 1000L);
|
|
|
|
|
REstart = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-03-25 19:23:05 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data += DV6_R_RecLength;
|
|
|
|
|
}
|
|
|
|
|
if (flow && sess) {
|
2018-03-26 04:11:42 +00:00
|
|
|
|
// Close event states if they are still open, and write event.
|
|
|
|
|
if (inH) HY->AddEvent(Hstart,(Hend-Hstart) / 1000L);
|
|
|
|
|
if (inOA) OA->AddEvent(OAstart,(OAend-OAstart) / 1000L);
|
|
|
|
|
if (inCA) NOA->AddEvent(CAstart,(CAend-CAstart) / 1000L);
|
|
|
|
|
if (inLL) LL->AddEvent(LLstart,(LLend-LLstart) / 1000L);
|
|
|
|
|
if (inVS) HY->AddEvent(VSstart,(VSend-VSstart) / 1000L);
|
|
|
|
|
if (inExP) EXP->AddEvent(ExPstart,(ExPend-ExPstart) / 1000L);
|
|
|
|
|
if (inFL) FL->AddEvent(FLstart,(FLend-FLstart) / 1000L);
|
|
|
|
|
if (inPB) PB->AddEvent(PBstart,(PBend-PBstart) / 1000L);
|
|
|
|
|
if (inPB) RE->AddEvent(REstart,(REend-REstart) / 1000L);
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
// update min and max
|
|
|
|
|
// then add to machine
|
|
|
|
|
EventDataType min = flow->Min();
|
|
|
|
|
EventDataType max = flow->Max();
|
|
|
|
|
sess->setMin(CPAP_FlowRate, min);
|
|
|
|
|
sess->setMax(CPAP_FlowRate, max);
|
|
|
|
|
|
|
|
|
|
sess->setPhysMax(CPAP_FlowRate, min); // not sure :/
|
|
|
|
|
sess->setPhysMin(CPAP_FlowRate, max);
|
|
|
|
|
sess->really_set_last(flow->last());
|
|
|
|
|
|
|
|
|
|
sess = NULL;
|
|
|
|
|
flow = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
f.close();
|
|
|
|
|
data = (unsigned char *)dataBA.data();
|
|
|
|
|
} else { // if (f.open(...)
|
|
|
|
|
// L.BIN open failed
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
/// Parse L.BIN and extract per-minute data.
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
EventList *leak = NULL;
|
|
|
|
|
EventList *maxleak = NULL;
|
|
|
|
|
sess = NULL;
|
|
|
|
|
const int DV6_L_HeaderSize = 55;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
// Need to parse L.bin minute table to get graphs
|
|
|
|
|
f.setFileName(newpath+"/L.BIN");
|
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
dataBA = f.readAll();
|
2018-03-26 11:51:43 +00:00
|
|
|
|
if (dataBA.size() <= DV6_L_HeaderSize) {
|
2018-03-25 19:23:05 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
2018-03-26 11:51:43 +00:00
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
f.close();
|
2018-03-26 11:51:43 +00:00
|
|
|
|
data = (unsigned char *)dataBA.data()+DV6_L_HeaderSize;
|
|
|
|
|
int numLrecs = (dataBA.size()-DV6_L_HeaderSize) / DV6_L_RecLength;
|
|
|
|
|
|
|
|
|
|
SR = summaryList.begin();
|
|
|
|
|
DV6_S_Record *R = &SR.value();
|
|
|
|
|
|
|
|
|
|
for (int r=0; r<numLrecs; ++r) {
|
|
|
|
|
ts1 = ((data[4] << 24) | (data[3] << 16) | (data[2] << 8) | data[1])+ep; // session start time
|
|
|
|
|
|
|
|
|
|
while (ts1 > R->stop_time) {
|
|
|
|
|
if (leak && sess) {
|
|
|
|
|
// Close the open session and update the min and max
|
|
|
|
|
|
|
|
|
|
EventDataType min = leak->Min();
|
|
|
|
|
EventDataType max = leak->Max();
|
|
|
|
|
sess->setMin(CPAP_Leak, min);
|
|
|
|
|
sess->setMax(CPAP_Leak, max);
|
|
|
|
|
|
|
|
|
|
sess->setPhysMax(CPAP_Leak, min);
|
|
|
|
|
sess->setPhysMin(CPAP_Leak, max);
|
|
|
|
|
|
|
|
|
|
min = maxleak->Min();
|
|
|
|
|
max = maxleak->Max();
|
|
|
|
|
sess->setMin(CPAP_MaxLeak, min);
|
|
|
|
|
sess->setMax(CPAP_MaxLeak, max);
|
|
|
|
|
|
|
|
|
|
sess->setPhysMax(CPAP_MaxLeak, min);
|
|
|
|
|
sess->setPhysMin(CPAP_MaxLeak, max);
|
|
|
|
|
sess->set_last(maxleak->last());
|
|
|
|
|
|
|
|
|
|
sess = NULL;
|
|
|
|
|
leak = NULL;
|
|
|
|
|
}
|
|
|
|
|
SR++;
|
|
|
|
|
if (SR == summaryList.end()) break;
|
|
|
|
|
R = &SR.value();
|
|
|
|
|
}
|
|
|
|
|
if (SR == summaryList.end())
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (ts1 >= R->start_time) {
|
|
|
|
|
if (!leak && R->sess) {
|
|
|
|
|
qDebug() << "Adding Leak data for session" << R->sess->session() << "starting at" << ts1;
|
|
|
|
|
leak = R->sess->AddEventList(CPAP_Leak, EVL_Waveform, 1.0, 0.0, 0.0, 0.0, double(60000) / double(1));
|
|
|
|
|
maxleak = R->sess->AddEventList(CPAP_MaxLeak, EVL_Waveform, 1.0, 0.0, 0.0, 0.0, double(60000) / double(1));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
//SR
|
|
|
|
|
}
|
|
|
|
|
if (leak) {
|
|
|
|
|
sess = R->sess;
|
|
|
|
|
|
|
|
|
|
qint64 ti = qint64(ts1) * 1000;
|
|
|
|
|
|
|
|
|
|
maxleak->AddEvent(ti, data[5]);
|
|
|
|
|
leak->AddEvent(ti, data[6]);
|
|
|
|
|
|
|
|
|
|
if (!sess->channelExists(CPAP_FlowRate)) {
|
|
|
|
|
// No flow rate, so lets grab this data...
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data += DV6_L_RecLength;
|
|
|
|
|
} // for
|
|
|
|
|
if (sess && leak) {
|
|
|
|
|
EventDataType min = leak->Min();
|
|
|
|
|
EventDataType max = leak->Max();
|
|
|
|
|
sess->setMin(CPAP_Leak, min);
|
|
|
|
|
sess->setMax(CPAP_Leak, max);
|
|
|
|
|
|
|
|
|
|
sess->setPhysMax(CPAP_Leak, min);
|
|
|
|
|
sess->setPhysMin(CPAP_Leak, max);
|
|
|
|
|
min = maxleak->Min();
|
|
|
|
|
max = maxleak->Max();
|
|
|
|
|
sess->setMin(CPAP_MaxLeak, min);
|
|
|
|
|
sess->setMax(CPAP_MaxLeak, max);
|
|
|
|
|
|
|
|
|
|
sess->setPhysMax(CPAP_MaxLeak, min);
|
|
|
|
|
sess->setPhysMin(CPAP_MaxLeak, max);
|
|
|
|
|
sess->set_last(maxleak->last());
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-25 19:23:05 +00:00
|
|
|
|
} else { // if (f.open(...)
|
|
|
|
|
// L.BIN open failed
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Now sessionList is populated with summary data, lets parse the Events list in E.BIN
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
f.setFileName(newpath+"/E.BIN");
|
|
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
|
dataBA = f.readAll();
|
|
|
|
|
if (dataBA.size() == 0) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
f.close();
|
|
|
|
|
data = (unsigned char *)dataBA.data();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} else { // if (f.open(...)
|
|
|
|
|
// E.BIN open failed
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QMap<quint32, DV6_S_Record>::iterator it;
|
|
|
|
|
|
|
|
|
|
for (it=summaryList.begin(); it!= summaryList.end(); ++it) {
|
|
|
|
|
Session * sess = it.value().sess;
|
|
|
|
|
|
|
|
|
|
mach->AddSession(sess);
|
|
|
|
|
|
|
|
|
|
// Update indexes, process waveform and perform flagging
|
|
|
|
|
sess->UpdateSummaries();
|
|
|
|
|
|
|
|
|
|
// Save is not threadsafe
|
|
|
|
|
sess->Store(mach->getDataPath());
|
|
|
|
|
|
|
|
|
|
// Unload them from memory
|
|
|
|
|
sess->TrashEvents();
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
return summaryList.size();
|
2018-03-25 19:23:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-04-27 04:29:03 +00:00
|
|
|
|
int IntellipapLoader::Open(const QString & dirpath)
|
2018-03-25 19:23:05 +00:00
|
|
|
|
{
|
|
|
|
|
// Check for SL directory
|
|
|
|
|
// Check for DV5MFirm.bin?
|
2018-04-27 04:29:03 +00:00
|
|
|
|
QString path(dirpath);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
path = path.replace("\\", "/");
|
|
|
|
|
|
|
|
|
|
if (path.endsWith(SL_DIR)) {
|
|
|
|
|
path.chop(3);
|
|
|
|
|
} else if (path.endsWith(DV6_DIR)) {
|
|
|
|
|
path.chop(4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QDir dir;
|
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
int r = -1;
|
|
|
|
|
// Sometimes there can be an SL folder because SmartLink dumps an old DV5 firmware in it, so check it first
|
2018-03-25 19:23:05 +00:00
|
|
|
|
if (dir.exists(path + SL_DIR))
|
2018-03-26 11:51:43 +00:00
|
|
|
|
r = OpenDV5(path);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
if ((r<0) && dir.exists(path + DV6_DIR))
|
|
|
|
|
r = OpenDV6(path);
|
2018-03-25 19:23:05 +00:00
|
|
|
|
|
2018-03-26 11:51:43 +00:00
|
|
|
|
return r;
|
2018-03-25 19:23:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2014-08-06 14:06:44 +00:00
|
|
|
|
void IntellipapLoader::initChannels()
|
2011-11-20 02:59:00 +00:00
|
|
|
|
{
|
2014-08-03 13:00:13 +00:00
|
|
|
|
using namespace schema;
|
|
|
|
|
Channel * chan = nullptr;
|
2014-08-23 06:21:50 +00:00
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(INTP_SmartFlexMode = 0x1165, SETTING, MT_CPAP, SESSION,
|
2014-08-03 13:00:13 +00:00
|
|
|
|
"INTPSmartFlexMode", QObject::tr("SmartFlex Mode"),
|
|
|
|
|
QObject::tr("Intellipap pressure relief mode."),
|
|
|
|
|
QObject::tr("SmartFlex Mode"),
|
|
|
|
|
"", DEFAULT, Qt::green));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
chan->addOption(0, STR_TR_Off);
|
|
|
|
|
chan->addOption(1, QObject::tr("Ramp Only"));
|
|
|
|
|
chan->addOption(2, QObject::tr("Full Time"));
|
|
|
|
|
|
2014-08-23 06:21:50 +00:00
|
|
|
|
channel.add(GRP_CPAP, chan = new Channel(INTP_SmartFlexLevel = 0x1169, SETTING, MT_CPAP, SESSION,
|
2014-08-03 13:00:13 +00:00
|
|
|
|
"INTPSmartFlexLevel", QObject::tr("SmartFlex Level"),
|
|
|
|
|
QObject::tr("Intellipap pressure relief level."),
|
|
|
|
|
QObject::tr("SmartFlex Level"),
|
|
|
|
|
"", DEFAULT, Qt::green));
|
2011-11-20 02:59:00 +00:00
|
|
|
|
}
|
2014-08-06 14:06:44 +00:00
|
|
|
|
|
|
|
|
|
bool intellipap_initialized = false;
|
|
|
|
|
void IntellipapLoader::Register()
|
|
|
|
|
{
|
|
|
|
|
if (intellipap_initialized) { return; }
|
|
|
|
|
|
|
|
|
|
qDebug() << "Registering IntellipapLoader";
|
|
|
|
|
RegisterLoader(new IntellipapLoader());
|
|
|
|
|
//InitModelMap();
|
|
|
|
|
intellipap_initialized = true;
|
|
|
|
|
|
|
|
|
|
}
|