From e380b408fdfafdb3b3e5874edd626cba9a6cbbe3 Mon Sep 17 00:00:00 2001
From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com>
Date: Mon, 2 Dec 2019 17:30:28 -0500
Subject: [PATCH 1/5] Remove warnings about PRS1 oximetry being untested.
Also fine-tune various warnings for weird data, such as truncated
chunks and multiple sessions in a waveform file.
---
oscar/SleepLib/loader_plugins/prs1_loader.cpp | 51 ++++++++++++++++---
oscar/tests/prs1tests.cpp | 4 +-
2 files changed, 45 insertions(+), 10 deletions(-)
diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
index f4159b2c..2debd6aa 100644
--- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
@@ -955,7 +955,6 @@ void PRS1Loader::ScanFiles(const QStringList & paths, int sessionid_base, Machin
// All samples exhibiting this behavior are DreamStations.
task->m_wavefiles.append(fi.canonicalFilePath());
} else if (ext == 6) {
- qWarning() << fi.canonicalFilePath() << "oximetry is untested"; // TODO: mark as untested/unexpected
if (!task->oxifile.isEmpty()) {
qDebug() << sid << "already has oximetry file" << relativePath(task->oxifile)
<< "skipping" << relativePath(fi.canonicalFilePath());
@@ -1025,6 +1024,10 @@ void PRS1Loader::ScanFiles(const QStringList & paths, int sessionid_base, Machin
//qDebug() << chunkComparison(chunk, task->summary);
} else {
// Warn about any non-identical duplicate session IDs.
+ //
+ // This seems to happen with F5V1 slice 8, which is the only slice in a session,
+ // and which doesn't update the session ID, so the following slice 7 session
+ // (which can be hours later) has the same session ID. Neither affects import.
qWarning() << chunkComparison(chunk, task->summary);
}
delete chunk;
@@ -7138,13 +7141,27 @@ QList PRS1Import::CoalesceWaveformChunks(QList
QString session_s = fi.fileName().section(".", 0, -2);
qint32 sid = session_s.toInt(&numeric, m_sessionid_base);
if (!numeric || sid != chunk->sessionid) {
- qWarning() << chunk->m_path << chunk->sessionid << "session ID mismatch";
+ qWarning() << chunk->m_path << "@" << chunk->m_filepos << "session ID mismatch:" << chunk->sessionid;
}
if (lastchunk != nullptr) {
- // Waveform files shouldn't contain multiple sessions
+ // A handful of 960P waveform files have been observed to have multiple sessions.
+ //
+ // This breaks the current approach of deferring waveform parsing until the (multithreaded)
+ // import, since each session is in a separate import task and could be in a separate
+ // thread, or already imported by the time it is discovered that this file contains
+ // more than one session.
+ //
+ // For now, we just dump the chunks that don't belong to the session currently
+ // being imported in this thread, since this happens so rarely.
+ //
+ // TODO: Rework the import process to handle waveform data after compliance/summary/
+ // events (since we're no longer inferring session information from it) and add it to the
+ // newly imported sessions.
if (lastchunk->sessionid != chunk->sessionid) {
- qWarning() << "lastchunk->sessionid != chunk->sessionid in PRS1Loader::CoalesceWaveformChunks()";
+ qWarning() << chunk->m_path << "@" << chunk->m_filepos
+ << "session ID" << lastchunk->sessionid << "->" << chunk->sessionid
+ << ", skipping" << allchunks.size() - i << "remaining chunks";
// Free any remaining chunks
for (int j=i; j < allchunks.size(); ++j) {
chunk = allchunks.at(j);
@@ -7190,8 +7207,20 @@ QList PRS1Import::CoalesceWaveformChunks(QList
}
for (int n=0; n < num; n++) {
int interleave = chunk->waveformInfo.at(n).interleave;
- if (interleave != 5) {
- qDebug() << chunk->m_path << "interleave?" << interleave;
+ switch (chunk->ext) {
+ case 5: // flow data, 5 samples per second
+ if (interleave != 5) {
+ qDebug() << chunk->m_path << "interleave?" << interleave;
+ }
+ break;
+ case 6: // oximetry, 1 sample per second
+ if (interleave != 1) {
+ qDebug() << chunk->m_path << "interleave?" << interleave;
+ }
+ break;
+ default:
+ qWarning() << chunk->m_path << "unknown waveform?" << chunk->ext;
+ break;
}
}
@@ -7210,6 +7239,7 @@ bool PRS1Import::ParseOximetry()
for (int i=0; i < size; ++i) {
PRS1DataChunk * oxi = oximetry.at(i);
int num = oxi->waveformInfo.size();
+ CHECK_VALUE(num, 2);
int size = oxi->m_data.size();
if (size == 0) {
@@ -7643,8 +7673,15 @@ PRS1DataChunk* PRS1DataChunk::ParseNext(QFile & f)
// Make sure the calculated CRC over the entire chunk (header and data) matches the stored CRC.
if (chunk->calcCrc != chunk->storedCrc) {
- // corrupt data block.. bleh..
+ // Correupt data block, warn about it.
qWarning() << chunk->m_path << "@" << chunk->m_filepos << "block CRC calc" << hex << chunk->calcCrc << "!= stored" << hex << chunk->storedCrc;
+
+ // TODO: When this happens, it's usually because the chunk was truncated and another chunk header
+ // exists within the blockSize bytes. In theory it should be possible to rewing and resync by
+ // looking for another chunk header with the same fileVersion, htype, family, familyVersion, and
+ // ext (blockSize and other fields could vary).
+ //
+ // But this is quite rare, so for now we bail on the rest of the file.
break;
}
diff --git a/oscar/tests/prs1tests.cpp b/oscar/tests/prs1tests.cpp
index 1ccb3412..28752eb6 100644
--- a/oscar/tests/prs1tests.cpp
+++ b/oscar/tests/prs1tests.cpp
@@ -325,9 +325,7 @@ void parseAndEmitChunkYaml(const QString & path)
case 1: ok = chunk->ParseSummary(); break;
case 2: ok = chunk->ParseEvents(); break;
case 5: break; // skip flow/pressure waveforms
- case 6: // skip oximetry data (but log it)
- qWarning() << relative << "oximetry is untested"; // never encountered
- break;
+ case 6: break; // skip oximetry data
default:
qWarning() << relative << "unexpected file type";
break;
From a495defdce1bc7489dc45d50855cb74974eea63a Mon Sep 17 00:00:00 2001
From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com>
Date: Tue, 3 Dec 2019 19:04:09 -0500
Subject: [PATCH 2/5] Add support for P-Flex and add 501X120 to tested list.
---
oscar/SleepLib/loader_plugins/prs1_loader.cpp | 7 ++++++-
1 file changed, 6 insertions(+), 1 deletion(-)
diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
index 2debd6aa..389e3022 100644
--- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
@@ -204,7 +204,7 @@ static QString ts(qint64 msecs)
// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails
-enum FlexMode { FLEX_None, FLEX_CFlex, FLEX_CFlexPlus, FLEX_AFlex, FLEX_RiseTime, FLEX_BiFlex, FLEX_AVAPS, FLEX_Unknown };
+enum FlexMode { FLEX_None, FLEX_CFlex, FLEX_CFlexPlus, FLEX_AFlex, FLEX_RiseTime, FLEX_BiFlex, FLEX_AVAPS, FLEX_PFlex, FLEX_Unknown };
enum BackupBreathMode { PRS1Backup_Off, PRS1Backup_Auto, PRS1Backup_Fixed };
@@ -243,6 +243,7 @@ static const PRS1TestedModel s_PRS1TestedModels[] = {
{ "400X150", 0, 6, "DreamStation CPAP Pro" },
{ "500X110", 0, 6, "DreamStation Auto CPAP" },
{ "500X150", 0, 6, "DreamStation Auto CPAP" },
+ { "501X120", 0, 6, "DreamStation Auto CPAP with P-Flex" },
{ "500G110", 0, 6, "DreamStation Go Auto" },
{ "502G150", 0, 6, "DreamStation Go Auto" },
{ "600X110", 0, 6, "DreamStation BiPAP Pro" },
@@ -6138,6 +6139,9 @@ bool PRS1DataChunk::ParseSettingsF0V6(const unsigned char* data, int size)
break;
}
break;
+ case 0xB0: // P-Flex
+ flexmode = FLEX_PFlex; // TOOD: There's a level present in the settings, does it have any effect?
+ break;
default:
UNEXPECTED_VALUE(data[pos], "known flex mode");
break;
@@ -7995,6 +7999,7 @@ void PRS1Loader::initChannels()
chan->addOption(FLEX_CFlex, QObject::tr("C-Flex"));
chan->addOption(FLEX_CFlexPlus, QObject::tr("C-Flex+"));
chan->addOption(FLEX_AFlex, QObject::tr("A-Flex"));
+ chan->addOption(FLEX_PFlex, QObject::tr("P-Flex"));
chan->addOption(FLEX_RiseTime, QObject::tr("Rise Time"));
chan->addOption(FLEX_BiFlex, QObject::tr("Bi-Flex"));
chan->addOption(FLEX_AVAPS, QObject::tr("AVAPS"));
From 14aad44f2183e104b7201b67cb20e314235bde29 Mon Sep 17 00:00:00 2001
From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com>
Date: Sat, 7 Dec 2019 15:10:48 -0500
Subject: [PATCH 3/5] Add support for newly observed PRS1 EZ-Start in Auto-CPAP
mode.
---
oscar/SleepLib/loader_plugins/prs1_loader.cpp | 17 ++++++++++++++++-
1 file changed, 16 insertions(+), 1 deletion(-)
diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
index 389e3022..6e9a1724 100644
--- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp
+++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp
@@ -6033,7 +6033,11 @@ bool PRS1DataChunk::ParseSettingsF0V6(const unsigned char* data, int size)
CHECK_VALUES(data[pos], 1, 2); // 1 when EZ-Start is enabled? 2 when Auto-Trial? 3 when Auto-Trial is off or Opti-Start isn't off?
}
if (len == 2) { // 400G, 500G has extra byte
- CHECK_VALUES(data[pos+1], 0, 0x20); // Maybe related to Opti-Start?
+ if (data[pos+1]) {
+ // 0x20 seen with Opti-Start enabled
+ // 0x30 seen with both Opti-Start and EZ-Start enabled on 500X110
+ CHECK_VALUES(data[pos+1], 0x20, 0x30);
+ }
}
break;
case 0x0a: // CPAP pressure setting
@@ -6101,6 +6105,12 @@ bool PRS1DataChunk::ParseSettingsF0V6(const unsigned char* data, int size)
CHECK_VALUE(data[pos], 0x80); // EZ-Start enabled
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_EZ_START, data[pos] != 0));
break;
+ case 0x42: // EZ-Start for Auto-CPAP?
+ // Seen on 500X110 before 0x2b when EZ-Start is enabled on Auto-CPAP
+ CHECK_VALUE(len, 1);
+ CHECK_VALUE(data[pos], 0x80); // EZ-Start enabled
+ this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_EZ_START, data[pos] != 0));
+ break;
case 0x2b: // Ramp Type
CHECK_VALUE(len, 1);
CHECK_VALUES(data[pos], 0, 0x80); // 0 == "Linear", 0x80 = "SmartRamp"
@@ -6397,6 +6407,11 @@ bool PRS1DataChunk::ParseSummaryF0V6(void)
// TODO: The daily details will show when it changed, so maybe there's an event that indicates a pressure change.
//CHECK_VALUES(data[pos], 90, 60); // maybe CPAP-Check pressure, also matches EZ-Start Pressure
break;
+ case 0x0c:
+ // EZ-Start pressure for Auto-CPAP, seen on 500X110 following 4, before 8
+ // Appears to reflect the current session's EZ-Start pressure, though reported afterwards
+ //CHECK_VALUE(data[pos], 70, 80);
+ break;
default:
UNEXPECTED_VALUE(code, "known slice code");
break;
From 66575d17d4f743fb6ecff557064e57d439dc6f3e Mon Sep 17 00:00:00 2001
From: Guy Scharf
Date: Wed, 18 Dec 2019 23:34:53 -0700
Subject: [PATCH 4/5] Documentation for 1.1.0-beta-1
---
oscar/build_number.h | 2 +-
oscar/docs/release_notes.html | 36 +++++++++++++++++++++++++++++++++--
oscar/version.h | 2 +-
3 files changed, 36 insertions(+), 4 deletions(-)
diff --git a/oscar/build_number.h b/oscar/build_number.h
index 396b197e..4efaa269 100644
--- a/oscar/build_number.h
+++ b/oscar/build_number.h
@@ -1 +1 @@
-const int build_number = 4;
+const int build_number = 1;
diff --git a/oscar/docs/release_notes.html b/oscar/docs/release_notes.html
index f78c2466..d6d1d6d4 100644
--- a/oscar/docs/release_notes.html
+++ b/oscar/docs/release_notes.html
@@ -7,16 +7,48 @@ Which was written and copyright 2011-2018 © Mark Watkins
-Changes and fixes in OSCAR **AFTER** v1.1.0-testing-4
+Changes and fixes in OSCAR v1.1.0-beta-1
- Portions of OSCAR are © 2019 by The OSCAR Team
-- [new] Default and View/reset graphs use a different order for AVS and AVAPS CPAP modes
+- [new] Extensive overhaul of the Philips Respironics System One importer, resolving all known issues.
+- [new] The following Philips Respironics machines are now tested and fully supported:
+REMstar Plus (System One) (251P),
+REMstar Pro (System One) (450P, 451P),
+REMstar Auto (System One) (550P, 551P),
+BiPAP Auto (System One) (750P),
+BiPAP AutoSV Advanced System One (950P, 951P),
+REMstar Pro (System One 60 Series) (460P, 461P),
+REMstar Auto (System One 60 Series) (560P, 561P, 562P, 560PBT),
+BiPAP Pro (System One 60 Series) (660P),
+BiPAP Auto (System One 60 Series) (760P),
+BiPAP autoSV Advanced (System One 60 Series) (960P, 961P),
+BiPAP autoSV Advanced 30 (System One 60 Series) (960T),
+BiPAP S/T 30 (System One 60 Series) (1061T),
+BiPAP AVAPS 30 (System One 60 Series) (1160P),
+DreamStation CPAP (200X110),
+DreamStation CPAP Pro (400X110, 400X150),
+DreamStation Go (400G110),
+DreamStation Auto CPAP (500X110, 500X150),
+DreamStation Go Auto (500G110, 502G150),
+DreamStation BiPAP Pro (600X110),
+DreamStation Auto BiPAP (700X110),
+DreamStation BiPAP autoSV (900X110, 900X120),
+DreamStation BiPAP S/T 30 (1030X110),
+DreamStation BiPAP S/T 30 with AAM (1030X150),
+DreamStation BiPAP AVAPS 30 (1130X110),
+DreamStation BiPAP AVAPS 30 AE (1131X150)
+- [new] Update translation files and add new languages
+- [new] Allow user to reset graph order on Daily page to Standard or Advanced order (often useful for AVS and AVAPS CPAP modes)
- [new] Add preference setting to include serial number on machine settings list
- [fix] Place date, time, and Oscar version information in report footers
- [fix] Update identification of ResMed S9 machines on Welcome page
- [fix] Correct formatting of event number in Daily Events tab
- [fix] Correct timezone offset for somnopose imports
- [fix] Show a progress bar when setting Overview range to a large number of days
+- [fix] Make session bars on Daily page clearer by using a better color
+- [fix] Improve list of machines on Statistics page
+- [fix] Report Pressure when IPAP data is missing
+- [fix] Implement Refresh button on Profile page
diff --git a/oscar/version.h b/oscar/version.h
index d23b5769..f78b00b1 100644
--- a/oscar/version.h
+++ b/oscar/version.h
@@ -15,7 +15,7 @@
const int major_version = 1; // incompatible API changes
const int minor_version = 1; // new features that don't break things
const int revision_number = 0; // bugfixes, revisions
-const QString ReleaseStatus = "testing"; // testing/nightly/unstable, beta/untamed, rc/almost, r/stable
+const QString ReleaseStatus = "beta"; // testing/nightly/unstable, beta/untamed, rc/almost, r/stable
const QString VersionString = QString("%1.%2.%3-%4-%5").arg(major_version).arg(minor_version).arg(revision_number).arg(ReleaseStatus).arg(build_number);
const QString ShortVersionString = QString("%1.%2.%3").arg(major_version).arg(minor_version).arg(revision_number);
From 7c15e9bd9c6cde035b8847e5f47287f710516df7 Mon Sep 17 00:00:00 2001
From: Guy Scharf
Date: Tue, 24 Dec 2019 23:12:49 -0700
Subject: [PATCH 5/5] Add PressureSet, EPAPSet, and IPAPSet lines to Statistics
page
---
oscar/statistics.cpp | 11 +++++++++++
1 file changed, 11 insertions(+)
diff --git a/oscar/statistics.cpp b/oscar/statistics.cpp
index afede063..2a50e984 100644
--- a/oscar/statistics.cpp
+++ b/oscar/statistics.cpp
@@ -556,13 +556,24 @@ Statistics::Statistics(QObject *parent) :
rows.push_back(StatisticsRow("Pressure", SC_MIN, MT_CPAP));
rows.push_back(StatisticsRow("Pressure", SC_MAX, MT_CPAP));
rows.push_back(StatisticsRow("Pressure", SC_90P, MT_CPAP));
+ rows.push_back(StatisticsRow("PressureSet", SC_WAVG, MT_CPAP));
+ rows.push_back(StatisticsRow("PressureSet", SC_MIN, MT_CPAP));
+ rows.push_back(StatisticsRow("PressureSet", SC_MAX, MT_CPAP));
+ rows.push_back(StatisticsRow("PressureSet", SC_90P, MT_CPAP));
rows.push_back(StatisticsRow("EPAP", SC_WAVG, MT_CPAP));
rows.push_back(StatisticsRow("EPAP", SC_MIN, MT_CPAP));
rows.push_back(StatisticsRow("EPAP", SC_MAX, MT_CPAP));
+ rows.push_back(StatisticsRow("EPAPSet", SC_WAVG, MT_CPAP));
+ rows.push_back(StatisticsRow("EPAPSet", SC_MIN, MT_CPAP));
+ rows.push_back(StatisticsRow("EPAPSet", SC_MAX, MT_CPAP));
rows.push_back(StatisticsRow("IPAP", SC_WAVG, MT_CPAP));
rows.push_back(StatisticsRow("IPAP", SC_90P, MT_CPAP));
rows.push_back(StatisticsRow("IPAP", SC_MIN, MT_CPAP));
rows.push_back(StatisticsRow("IPAP", SC_MAX, MT_CPAP));
+ rows.push_back(StatisticsRow("IPAPSet", SC_WAVG, MT_CPAP));
+ rows.push_back(StatisticsRow("IPAPSet", SC_90P, MT_CPAP));
+ rows.push_back(StatisticsRow("IPAPSet", SC_MIN, MT_CPAP));
+ rows.push_back(StatisticsRow("IPAPSet", SC_MAX, MT_CPAP));
rows.push_back(StatisticsRow("", SC_HEADING, MT_OXIMETER)); // Just adds some space
rows.push_back(StatisticsRow(tr("Oximeter Statistics"), SC_HEADING, MT_OXIMETER));