2014-04-09 21:01:57 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
* vim: set ts=8 sts=4 et sw=4 tw=99:
|
|
|
|
*
|
|
|
|
* SleepLib (DeVilbiss) Intellipap Loader Implementation
|
|
|
|
* Notes: Intellipap requires the SmartLink attachment to access this data.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2011-2014 Mark Watkins <jedimark@users.sourceforge.net>
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
|
|
* License. See the file COPYING in the main directory of the Linux
|
|
|
|
* distribution for more details. */
|
2011-11-20 02:59:00 +00:00
|
|
|
|
2011-11-21 02:15:10 +00:00
|
|
|
#include <QDir>
|
2011-11-21 10:20:11 +00:00
|
|
|
#include <QProgressBar>
|
2011-11-21 02:15:10 +00:00
|
|
|
|
2011-11-20 02:59:00 +00:00
|
|
|
#include "intellipap_loader.h"
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
extern QProgressBar *qprogress;
|
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
Intellipap::Intellipap(MachineID id)
|
|
|
|
: CPAP(id)
|
2011-11-20 02:59:00 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Intellipap::~Intellipap()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
IntellipapLoader::IntellipapLoader()
|
|
|
|
{
|
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()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-04-28 19:24:39 +00:00
|
|
|
bool IntellipapLoader::Detect(const QString & givenpath)
|
2014-04-26 09:54:08 +00:00
|
|
|
{
|
2014-04-28 19:24:39 +00:00
|
|
|
QDir dir(givenpath);
|
|
|
|
|
|
|
|
if (!dir.exists()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Intellipap has a folder called SL in the root directory
|
|
|
|
if (!dir.cd("SL")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for the settings file inside the SL folder
|
|
|
|
if (!dir.exists("SET1")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2014-04-26 09:54:08 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 12:09:38 +00:00
|
|
|
int IntellipapLoader::Open(QString path)
|
2011-11-20 02:59:00 +00:00
|
|
|
{
|
2011-11-21 02:15:10 +00:00
|
|
|
// Check for SL directory
|
2011-11-21 10:20:11 +00:00
|
|
|
// Check for DV5MFirm.bin?
|
2011-11-21 02:15:10 +00:00
|
|
|
QString newpath;
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
path = path.replace("\\", "/");
|
2013-09-14 23:32:14 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
QString dirtag = "SL";
|
|
|
|
|
|
|
|
if (path.endsWith("/" + dirtag)) {
|
2014-07-29 14:38:59 +00:00
|
|
|
return -1;
|
2011-11-21 02:15:10 +00:00
|
|
|
//newpath=path;
|
|
|
|
} else {
|
2014-04-17 05:58:57 +00:00
|
|
|
newpath = path + "/" + dirtag;
|
2011-11-21 02:15:10 +00:00
|
|
|
}
|
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
QString filename;
|
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
// Parse the Settings File
|
|
|
|
//////////////////////////
|
2014-04-17 05:58:57 +00:00
|
|
|
filename = newpath + "/SET1";
|
2011-11-21 02:15:10 +00:00
|
|
|
QFile f(filename);
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-07-29 14:38:59 +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;
|
|
|
|
|
|
|
|
EventDataType min_pressure = 0, max_pressure = 0, ramp_pressure = 0, set_epap = 0, set_ipap = 0, set_ps = 0, ramp_time = 0;
|
|
|
|
|
|
|
|
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) {
|
|
|
|
min_pressure = value.toFloat() / 10.0;
|
|
|
|
} else if (key == INT_PROP_MaxPressure) {
|
|
|
|
max_pressure = value.toFloat() / 10.0;
|
|
|
|
} else if (key == INT_PROP_IPAP) {
|
|
|
|
set_ipap = value.toFloat() / 10.0;
|
|
|
|
} else if (key == INT_PROP_EPAP) {
|
|
|
|
set_epap = value.toFloat() / 10.0;
|
|
|
|
} else if (key == INT_PROP_PS) {
|
|
|
|
set_ps = value.toFloat() / 10.0;
|
|
|
|
} 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()) {
|
|
|
|
mach = 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);
|
|
|
|
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;
|
|
|
|
SessionID lastsid = 0;
|
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) {
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
qint64 rs = rampstart[sid];
|
|
|
|
|
|
|
|
if (pres < rampval) {
|
|
|
|
if (!rs) {
|
|
|
|
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);
|
|
|
|
|
|
|
|
rampstart[sid] = 0;
|
2014-07-29 14:38:59 +00:00
|
|
|
//rampend[sid] = 0; // don't need to
|
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
|
|
|
|
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-07-29 14:38:59 +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-29 15:00:27 +00:00
|
|
|
if (!sess) continue;
|
2014-07-09 03:49:20 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
quint64 first = qint64(sid) * 1000L;
|
|
|
|
quint64 last = qint64(SessionEnd[i]) * 1000L;
|
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
if (sess->last() > 0) {
|
|
|
|
sess->settings[CPAP_PresReliefType] = (PRTypes)PR_SMARTFLEX;
|
2014-04-17 05:58:57 +00:00
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
sess->settings[CPAP_PresReliefSet] = smartflex;
|
2014-07-28 13:56:29 +00:00
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
if (smartflexmode == 0) {
|
|
|
|
sess->settings[CPAP_PresReliefMode] = PM_FullTime;
|
|
|
|
} else {
|
|
|
|
sess->settings[CPAP_PresReliefMode] = PM_RampOnly;
|
|
|
|
}
|
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
|
|
|
|
2011-12-27 13:21:10 +00:00
|
|
|
|
2014-07-29 15:00:27 +00:00
|
|
|
sess->UpdateSummaries();
|
|
|
|
mach->AddSession(sess);
|
|
|
|
} 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
|
|
|
|
2011-11-21 10:20:11 +00:00
|
|
|
mach->Save();
|
|
|
|
|
|
|
|
delete [] m_buffer;
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
if (qprogress) { qprogress->setValue(100); }
|
2011-11-21 10:20:11 +00:00
|
|
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
bool intellipap_initialized = false;
|
2011-11-20 02:59:00 +00:00
|
|
|
void IntellipapLoader::Register()
|
|
|
|
{
|
2014-04-17 05:58:57 +00:00
|
|
|
if (intellipap_initialized) { return; }
|
|
|
|
|
2011-11-20 02:59:00 +00:00
|
|
|
qDebug() << "Registering IntellipapLoader";
|
|
|
|
RegisterLoader(new IntellipapLoader());
|
|
|
|
//InitModelMap();
|
2014-04-17 05:58:57 +00:00
|
|
|
intellipap_initialized = true;
|
2011-11-20 02:59:00 +00:00
|
|
|
}
|