mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 10:40:42 +00:00
Revise CPAP/APAP identification in SleepStyle loader
o While the Sleepstyle summary record has a data item that appears to identify the mode as CPAP or APAP, it is not reliable o Determine APAP or CPAP by comparing Max and 95 percentile pressures seen with the CPAP pressure setting o This could result in an incorrect identification if user sets CPAP pressure to APAP Max pressure. o Add DEBUGSS define to enable SleepStyle loader debugging.
This commit is contained in:
parent
3808943f65
commit
36c2968701
@ -276,7 +276,7 @@ int SleepStyleLoader::OpenMachine(Machine *mach, const QString & path, const QSt
|
|||||||
|
|
||||||
SessionID sid;//,st;
|
SessionID sid;//,st;
|
||||||
float hours, mins;
|
float hours, mins;
|
||||||
|
Q_UNUSED(mins) // Used only in debug mode
|
||||||
// For diagnostics, print summary of last 20 session or one week
|
// For diagnostics, print summary of last 20 session or one week
|
||||||
qDebug() << "SS Loader - last 20 Sessions:";
|
qDebug() << "SS Loader - last 20 Sessions:";
|
||||||
|
|
||||||
@ -300,9 +300,9 @@ int SleepStyleLoader::OpenMachine(Machine *mach, const QString & path, const QSt
|
|||||||
hours = sess->hours();
|
hours = sess->hours();
|
||||||
mins = hours * 60;
|
mins = hours * 60;
|
||||||
dt = QDateTime::fromTime_t(sid);
|
dt = QDateTime::fromTime_t(sid);
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << cnt << ":" << dt << "session" << sid << "," << mins << "minutes" << a;
|
qDebug() << cnt << ":" << dt << "session" << sid << "," << mins << "minutes" << a;
|
||||||
|
#endif
|
||||||
if (dt.date() < date) {
|
if (dt.date() < date) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -382,7 +382,9 @@ quint32 ssconvertDate(quint32 timestamp)
|
|||||||
|
|
||||||
QDateTime dt = QDateTime(QDate(year, month, day), QTime(hour, minute, second), Qt::UTC);
|
QDateTime dt = QDateTime(QDate(year, month, day), QTime(hour, minute, second), Qt::UTC);
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug().noquote() << "SS timestamp" << timestamp << year << month << day << dt << hour << minute << second;
|
qDebug().noquote() << "SS timestamp" << timestamp << year << month << day << dt << hour << minute << second;
|
||||||
|
#endif
|
||||||
|
|
||||||
// Q NO!!! _ASSERT(dt.isValid());
|
// Q NO!!! _ASSERT(dt.isValid());
|
||||||
// if ((year == 2013) && (month == 9) && (day == 18)) {
|
// if ((year == 2013) && (month == 9) && (day == 18)) {
|
||||||
@ -431,7 +433,9 @@ bool SleepStyleLoader::OpenRealTime(Machine *mach, const QString & fname, const
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug().noquote() << "SS ORT timestamp" << edf.startdate / 1000L << QDateTime::fromSecsSinceEpoch(edf.startdate / 1000L).toString("MM/dd/yyyy hh:mm:ss");
|
qDebug().noquote() << "SS ORT timestamp" << edf.startdate / 1000L << QDateTime::fromSecsSinceEpoch(edf.startdate / 1000L).toString("MM/dd/yyyy hh:mm:ss");
|
||||||
|
#endif
|
||||||
SessionID sessKey = findSession(edf.startdate / 1000L);
|
SessionID sessKey = findSession(edf.startdate / 1000L);
|
||||||
if (sessKey == 0) {
|
if (sessKey == 0) {
|
||||||
qWarning() << "SS ORT session not found";
|
qWarning() << "SS ORT session not found";
|
||||||
@ -526,13 +530,13 @@ bool SleepStyleLoader::OpenRealTime(Machine *mach, const QString & fname, const
|
|||||||
////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
||||||
{
|
{
|
||||||
qDebug() << filename;
|
qDebug() << "SS SUM File" << filename;
|
||||||
QByteArray header;
|
QByteArray header;
|
||||||
QFile file(filename);
|
QFile file(filename);
|
||||||
QString typex;
|
QString typex;
|
||||||
|
|
||||||
if (!file.open(QFile::ReadOnly)) {
|
if (!file.open(QFile::ReadOnly)) {
|
||||||
qDebug() << "SS SUM Couldn't open" << filename;
|
qWarning() << "SS SUM Couldn't open" << filename;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -540,7 +544,7 @@ bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
|||||||
header = file.read(0x200);
|
header = file.read(0x200);
|
||||||
|
|
||||||
if (header.size() != 0x200) {
|
if (header.size() != 0x200) {
|
||||||
qDebug() << "SS SUM Short file" << filename;
|
qWarning() << "SS SUM Short file" << filename;
|
||||||
file.close();
|
file.close();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -562,8 +566,9 @@ bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
|||||||
htxt >> type; // SPSAAN etc with 4th character being A (Auto) or C (CPAP)
|
htxt >> type; // SPSAAN etc with 4th character being A (Auto) or C (CPAP)
|
||||||
htxt >> unknownident; // Constant, but has different value when version number is different.
|
htxt >> unknownident; // Constant, but has different value when version number is different.
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "SS SUM header" << h1 << version << fname << serial << model << type << unknownident;
|
qDebug() << "SS SUM header" << h1 << version << fname << serial << model << type << unknownident;
|
||||||
|
#endif
|
||||||
if (type.length() > 4)
|
if (type.length() > 4)
|
||||||
typex = (type.at(3) == 'C' ? "CPAP" : "Auto");
|
typex = (type.at(3) == 'C' ? "CPAP" : "Auto");
|
||||||
mach->setModel(model + " " + typex);
|
mach->setModel(model + " " + typex);
|
||||||
@ -601,18 +606,23 @@ bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
|||||||
|
|
||||||
in >> ts;
|
in >> ts;
|
||||||
if (ts == 0xffffffff) {
|
if (ts == 0xffffffff) {
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "SS SUM 0xffffffff terminator found at block" << nblock;
|
qDebug() << "SS SUM 0xffffffff terminator found at block" << nblock;
|
||||||
|
#endif
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if ((ts & 0xffff) == 0xfafe) {
|
if ((ts & 0xffff) == 0xfafe) {
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "SS SUM 0xfafa terminator found at block" << nblock;
|
qDebug() << "SS SUM 0xfafa terminator found at block" << nblock;
|
||||||
|
#endif
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
ts = ssconvertDate(ts);
|
ts = ssconvertDate(ts);
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "\nSS SUM Session" << nblock << "with timestamp" << ts << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss");
|
qDebug() << "\nSS SUM Session" << nblock << "with timestamp" << ts << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss");
|
||||||
|
#endif
|
||||||
// the following two quite often match in value
|
// the following two quite often match in value
|
||||||
in >> runTime; // 0x04
|
in >> runTime; // 0x04
|
||||||
in >> useTime; // 0x05
|
in >> useTime; // 0x05
|
||||||
@ -654,20 +664,20 @@ bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
|||||||
unsigned char s [5];
|
unsigned char s [5];
|
||||||
for (unsigned int i=0; i < sizeof(s); i++)
|
for (unsigned int i=0; i < sizeof(s); i++)
|
||||||
in >> s[i];
|
in >> s[i];
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "SS SUM block" << nblock
|
qDebug() << "SS SUM block" << nblock
|
||||||
<< "a:" <<"Pressure Min"<<minPressSeen<<"95%"<<pct95PressSeen<<"Max"<<maxPressSeen
|
<< "a:" <<"Pressure Min"<<minPressSeen<<"95%"<<pct95PressSeen<<"Max"<<maxPressSeen
|
||||||
<< "\nd:" <<d1<<d2<<d3<<d4<<d5<<d6
|
<< "\nd:" <<d1<<d2<<d3<<d4<<d5<<d6
|
||||||
<< "\nj:" <<j1 << " c:" << c1 << c2 << c3 << c4
|
<< "\nj:" <<j1 << " c:" << c1 << c2 << c3 << c4
|
||||||
<< "\np:" <<p3<<"Ramp"<<(ramp?"on":"off")
|
<< "\np:" <<p3<<"Ramp"<<(ramp?"on":"off")
|
||||||
<< "\nx:" <<x1<<"CPAP Pressure" << CPAPpressSet << "Mode" << (mode?"CPAP":"APAP")
|
<< "\nx:" <<x1<<"CPAP Pressure" << CPAPpressSet << "Mode" << mode << (mode==0?"APAP":"CPAP")
|
||||||
<< "\ns:" <<"Min set" <<minPressSet<<"Max set"<<maxPressSet<<"SenseAwake"<<senseAwakeLevel<<"Humid"<<humidityLevel<<"EPR"<<EPRLevel<<"flags"<<flags
|
<< "\ns:" <<"Min set" <<minPressSet<<"Max set"<<maxPressSet<<"SenseAwake"<<senseAwakeLevel<<"Humid"<<humidityLevel<<"EPR"<<EPRLevel<<"flags"<<flags
|
||||||
<< "\ns:" <<s[0]<<s[1]<<s[2]<<s[3]<<s[4];
|
<< "\ns:" <<s[0]<<s[1]<<s[2]<<s[3]<<s[4];
|
||||||
|
|
||||||
if (runTime != useTime) {
|
if (runTime != useTime) {
|
||||||
qDebug() << "SS SUM run time" << runTime << "!= use time" << useTime << "-" << nblock << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss");
|
qDebug() << "SS SUM run time" << runTime << "!= use time" << useTime << "-" << nblock << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss");
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
if (!mach->SessionExists(ts)) {
|
if (!mach->SessionExists(ts)) {
|
||||||
Session *sess = new Session(mach, ts);
|
Session *sess = new Session(mach, ts);
|
||||||
sess->really_set_first(qint64(ts) * 1000L);
|
sess->really_set_first(qint64(ts) * 1000L);
|
||||||
@ -676,13 +686,13 @@ bool SleepStyleLoader::OpenSummary(Machine *mach, const QString & filename)
|
|||||||
|
|
||||||
SessDate.insert(date, sess);
|
SessDate.insert(date, sess);
|
||||||
|
|
||||||
if (mode || minPressSeen != maxPressSeen || minPressSeen != CPAP_PressureSet) {
|
if ((maxPressSeen == CPAPpressSet) && (pct95PressSeen == CPAPpressSet)) {
|
||||||
|
sess->settings[CPAP_Mode] = (int)MODE_CPAP;
|
||||||
|
sess->settings[CPAP_Pressure] = CPAPpressSet / 10.0;
|
||||||
|
} else {
|
||||||
sess->settings[CPAP_Mode] = (int)MODE_APAP;
|
sess->settings[CPAP_Mode] = (int)MODE_APAP;
|
||||||
sess->settings[CPAP_PressureMin] = minPressSet / 10.0;
|
sess->settings[CPAP_PressureMin] = minPressSet / 10.0;
|
||||||
sess->settings[CPAP_PressureMax] = maxPressSet / 10.0;
|
sess->settings[CPAP_PressureMax] = maxPressSet / 10.0;
|
||||||
} else {
|
|
||||||
sess->settings[CPAP_Mode] = (int)MODE_CPAP;
|
|
||||||
sess->settings[CPAP_Pressure] = CPAPpressSet / 10.0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (EPRLevel == 0)
|
if (EPRLevel == 0)
|
||||||
@ -718,19 +728,21 @@ bool SleepStyleLoader::OpenDetail(Machine *mach, const QString & filename)
|
|||||||
{
|
{
|
||||||
Q_UNUSED(mach);
|
Q_UNUSED(mach);
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "SS DET Opening Detail" << filename;
|
qDebug() << "SS DET Opening Detail" << filename;
|
||||||
|
#endif
|
||||||
QByteArray header;
|
QByteArray header;
|
||||||
QFile file(filename);
|
QFile file(filename);
|
||||||
|
|
||||||
if (!file.open(QFile::ReadOnly)) {
|
if (!file.open(QFile::ReadOnly)) {
|
||||||
qDebug() << "SS DET Couldn't open" << filename;
|
qWarning() << "SS DET Couldn't open" << filename;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
header = file.read(0x200);
|
header = file.read(0x200);
|
||||||
|
|
||||||
if (header.size() != 0x200) {
|
if (header.size() != 0x200) {
|
||||||
qDebug() << "SS DET short file" << filename;
|
qWarning() << "SS DET short file" << filename;
|
||||||
file.close();
|
file.close();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -754,8 +766,9 @@ bool SleepStyleLoader::OpenDetail(Machine *mach, const QString & filename)
|
|||||||
htxt >> type; // SPSAAN etc with 4th character being A (Auto) or C (CPAP)
|
htxt >> type; // SPSAAN etc with 4th character being A (Auto) or C (CPAP)
|
||||||
htxt >> unknownident; // Constant, but has different value when version number is different.
|
htxt >> unknownident; // Constant, but has different value when version number is different.
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug() << "SS DET file header" << h1 << version << fname << serial << model << type << unknownident;
|
qDebug() << "SS DET file header" << h1 << version << fname << serial << model << type << unknownident;
|
||||||
|
#endif
|
||||||
// Read session indices
|
// Read session indices
|
||||||
QByteArray index = file.read(0x800);
|
QByteArray index = file.read(0x800);
|
||||||
if (index.size()!=0x800) {
|
if (index.size()!=0x800) {
|
||||||
@ -793,15 +806,16 @@ bool SleepStyleLoader::OpenDetail(Machine *mach, const QString & filename)
|
|||||||
in >> unknownIndex;
|
in >> unknownIndex;
|
||||||
totalrecs += recs; // Number of data records for this session
|
totalrecs += recs; // Number of data records for this session
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
qDebug().noquote() << "SS DET block timestamp" << ts << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss") << "start" << strt << "records" << recs << "unknown" << unknownIndex;
|
qDebug().noquote() << "SS DET block timestamp" << ts << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss") << "start" << strt << "records" << recs << "unknown" << unknownIndex;
|
||||||
|
#endif
|
||||||
if (Sessions.contains(ts)) {
|
if (Sessions.contains(ts)) {
|
||||||
times.push_back(ts);
|
times.push_back(ts);
|
||||||
start.push_back(strt);
|
start.push_back(strt);
|
||||||
records.push_back(recs);
|
records.push_back(recs);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
qDebug() << "SS DET session not found" << ts;
|
qDebug() << "SS DET session not found" << ts << QDateTime::fromSecsSinceEpoch(ts).toString("MM/dd/yyyy hh:mm:ss") << "start" << strt << "records" << recs << "unknown" << unknownIndex;;
|
||||||
} while (!in.atEnd());
|
} while (!in.atEnd());
|
||||||
|
|
||||||
QByteArray databytes = file.readAll();
|
QByteArray databytes = file.readAll();
|
||||||
@ -815,7 +829,7 @@ bool SleepStyleLoader::OpenDetail(Machine *mach, const QString & filename)
|
|||||||
quint8 *data = (quint8 *)databytes.data();
|
quint8 *data = (quint8 *)databytes.data();
|
||||||
|
|
||||||
qint64 ti;
|
qint64 ti;
|
||||||
quint8 pressure, leak, a1, a2, a3, a4, a5, a6, a7;
|
quint8 pressure, leak, a1, a2, a3, a4, a5, a6;
|
||||||
// quint8 sa1, sa2; // The two sense awake bits per 2 minutes
|
// quint8 sa1, sa2; // The two sense awake bits per 2 minutes
|
||||||
SessionID sessid;
|
SessionID sessid;
|
||||||
Session *sess;
|
Session *sess;
|
||||||
@ -864,9 +878,6 @@ bool SleepStyleLoader::OpenDetail(Machine *mach, const QString & filename)
|
|||||||
// TODO: Confirm that the bits are in the right order
|
// TODO: Confirm that the bits are in the right order
|
||||||
a6 = (a3 >> 6) << 4 | ((a4 >> 6) << 2) | (a5 >> 6);
|
a6 = (a3 >> 6) << 4 | ((a4 >> 6) << 2) | (a5 >> 6);
|
||||||
|
|
||||||
// See if extra bits from the first two fields are used at any time (see debug later)
|
|
||||||
a7 = ((a1 >> 6) << 2) | (a2 >> 6);
|
|
||||||
|
|
||||||
bitmask = 1;
|
bitmask = 1;
|
||||||
for (int k = 0; k < 6; k++) { // There are 6 flag sets per 2 minutes
|
for (int k = 0; k < 6; k++) { // There are 6 flag sets per 2 minutes
|
||||||
// TODO: Modify if all four channels are to be reported separately
|
// TODO: Modify if all four channels are to be reported separately
|
||||||
@ -881,13 +892,17 @@ bool SleepStyleLoader::OpenDetail(Machine *mach, const QString & filename)
|
|||||||
ti += 20000L; // Increment 20 seconds
|
ti += 20000L; // Increment 20 seconds
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef DEBUGSS
|
||||||
// Debug print non-zero flags
|
// Debug print non-zero flags
|
||||||
|
// See if extra bits from the first two fields are used at any time (see debug later)
|
||||||
|
quint8 a7 = ((a1 >> 6) << 2) | (a2 >> 6);
|
||||||
if (a1 != 0 || a2 != 0 || a3 != 0 || a4 != 0 || a5 != 0 || a6 != 0 || a7 != 0) {
|
if (a1 != 0 || a2 != 0 || a3 != 0 || a4 != 0 || a5 != 0 || a6 != 0 || a7 != 0) {
|
||||||
qDebug() << "SS DET events" << QDateTime::fromSecsSinceEpoch(ti/1000).toString("MM/dd/yyyy hh:mm:ss")
|
qDebug() << "SS DET events" << QDateTime::fromSecsSinceEpoch(ti/1000).toString("MM/dd/yyyy hh:mm:ss")
|
||||||
<< "pressure" << pressure
|
<< "pressure" << pressure
|
||||||
<< "leak" << leak
|
<< "leak" << leak
|
||||||
<< "flags" << a1 << a2 << a3 << a4 << a5 << a6 << "unknown" << a7;
|
<< "flags" << a1 << a2 << a3 << a4 << a5 << a6 << "unknown" << a7;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
idx += 7; //was 5;
|
idx += 7; //was 5;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user