From c9b774a47acfe2e1da14e68a4cf206f060eba553 Mon Sep 17 00:00:00 2001
From: Mark Watkins
Date: Sat, 27 Feb 2016 13:37:56 +1000
Subject: [PATCH] Dreamstation 0x14 flag is missing hypopnea, change CSR to PB
on P.R. machines
---
.../SleepLib/loader_plugins/prs1_loader.cpp | 28 ++++++++++-------
sleepyhead/SleepLib/machine_common.cpp | 2 +-
sleepyhead/SleepLib/machine_common.h | 2 +-
sleepyhead/SleepLib/schema.cpp | 8 ++++-
sleepyhead/daily.cpp | 4 +--
sleepyhead/docs/release_notes.html | 10 ++++---
sleepyhead/reports.cpp | 3 ++
sleepyhead/statistics.cpp | 30 +++++++++++++++++++
8 files changed, 67 insertions(+), 20 deletions(-)
diff --git a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp
index 0215c742..2dc78baf 100644
--- a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp
+++ b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp
@@ -730,7 +730,7 @@ bool PRS1Import::ParseF5Events()
{
ChannelID Codes[] = {
PRS1_00, PRS1_01, CPAP_Pressure, CPAP_EPAP, CPAP_PressurePulse, CPAP_Obstructive,
- CPAP_ClearAirway, CPAP_Hypopnea, PRS1_08, CPAP_FlowLimit, PRS1_0A, CPAP_CSR,
+ CPAP_ClearAirway, CPAP_Hypopnea, PRS1_08, CPAP_FlowLimit, PRS1_0A, CPAP_PB,
PRS1_0C, CPAP_VSnore, PRS1_0E, PRS1_0F,
CPAP_LargeLeak, // Large leak apparently
CPAP_LeakTotal, PRS1_12
@@ -741,7 +741,7 @@ bool PRS1Import::ParseF5Events()
EventList *OA = session->AddEventList(CPAP_Obstructive, EVL_Event);
EventList *HY = session->AddEventList(CPAP_Hypopnea, EVL_Event);
- EventList *CSR = session->AddEventList(CPAP_CSR, EVL_Event);
+ EventList *PB = session->AddEventList(CPAP_PB, EVL_Event);
EventList *LEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event);
EventList *LL = session->AddEventList(CPAP_LargeLeak, EVL_Event);
EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event);
@@ -928,14 +928,14 @@ bool PRS1Import::ParseF5Events()
//tt-=delta;
tt -= qint64(data[1]) * 1000L;
- if (!CSR) {
- if (!(CSR = session->AddEventList(cpapcode, EVL_Event))) {
- qDebug() << "!CSR addeventlist exit";
+ if (!PB) {
+ if (!(PB = session->AddEventList(cpapcode, EVL_Event))) {
+ qDebug() << "!PB addeventlist exit";
return false;
}
}
- CSR->AddEvent(tt, data[0]);
+ PB->AddEvent(tt, data[0]);
break;
case 0x0c:
@@ -1086,7 +1086,7 @@ bool PRS1Import::ParseF0Events()
EventList *OA = session->AddEventList(CPAP_Obstructive, EVL_Event);
EventList *HY = session->AddEventList(CPAP_Hypopnea, EVL_Event);
- EventList *CSR = session->AddEventList(CPAP_CSR, EVL_Event);
+ EventList *PB = session->AddEventList(CPAP_PB, EVL_Event);
EventList *LEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event);
EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event);
@@ -1141,6 +1141,7 @@ bool PRS1Import::ParseF0Events()
if (code != 0x12) {
delta = buffer[pos + 1] << 8 | buffer[pos];
pos += 2;
+
t += qint64(delta) * 1000L;
tt = t;
}
@@ -1232,8 +1233,10 @@ bool PRS1Import::ParseF0Events()
break;
case 0x04: // Pressure Pulse
+ data[0] = buffer[pos++];
+ tt = t - (qint64(data[0]) * 1000L);
- PP->AddEvent(t, buffer[pos++]);
+ PP->AddEvent(tt, data[0]);
break;
case 0x05: // RERA
@@ -1297,7 +1300,7 @@ bool PRS1Import::ParseF0Events()
//qDebug() << hex << data[0] << data[1] << data[2];
//session->AddEvent(new Event(t,cpapcode, 0, data, 3));
//tt-=data[1]*1000;
- //session->AddEvent(new Event(t,CPAP_CSR, data, 2));
+ //session->AddEvent(new Event(t,CPAP_PB, data, 2));
break;
case 0x0f: // Cheyne Stokes Respiration
@@ -1305,7 +1308,7 @@ bool PRS1Import::ParseF0Events()
pos += 2;
data[1] = buffer[pos++];
tt = t - qint64(data[1]) * 1000L;
- CSR->AddEvent(tt, data[0]);
+ PB->AddEvent(tt, data[0]);
break;
case 0x10: // Large Leak
data[0] = buffer[pos + 1] << 8 | buffer[pos];
@@ -1349,9 +1352,12 @@ bool PRS1Import::ParseF0Events()
break;
case 0x14: // DreamStation unknown code
+ data[0] = buffer[pos++];
tt = t - (qint64(data[0]) * 1000L);
+ HY->AddEvent(tt, data[0]);
+// tt = t - (qint64(data[0]) * 1000L);
- pos++;
+// pos++;
break;
default:
diff --git a/sleepyhead/SleepLib/machine_common.cpp b/sleepyhead/SleepLib/machine_common.cpp
index c4eaecf6..e1310a33 100644
--- a/sleepyhead/SleepLib/machine_common.cpp
+++ b/sleepyhead/SleepLib/machine_common.cpp
@@ -15,7 +15,7 @@ ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CPAP_EPAP
CPAP_PS, CPAP_Mode, CPAP_AHI,
CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive,
CPAP_Hypopnea,
- CPAP_ClearAirway, CPAP_Apnea, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore,
+ CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore,
CPAP_VSnore2,
CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure,
CPAP_MaskPressureHi,
diff --git a/sleepyhead/SleepLib/machine_common.h b/sleepyhead/SleepLib/machine_common.h
index 77234aad..434f5049 100644
--- a/sleepyhead/SleepLib/machine_common.h
+++ b/sleepyhead/SleepLib/machine_common.h
@@ -139,7 +139,7 @@ extern ChannelID CPAP_IPAP, CPAP_IPAPLo, CPAP_IPAPHi, CPAP_EPAP, CPAP_EPAPLo, CP
CPAP_Mode, CPAP_AHI,
CPAP_PressureMin, CPAP_PressureMax, CPAP_Ramp, CPAP_RampTime, CPAP_RampPressure, CPAP_Obstructive,
CPAP_Hypopnea,
- CPAP_ClearAirway, CPAP_Apnea, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore,
+ CPAP_ClearAirway, CPAP_Apnea, CPAP_PB, CPAP_CSR, CPAP_LeakFlag, CPAP_ExP, CPAP_NRI, CPAP_VSnore,
CPAP_VSnore2,
CPAP_RERA, CPAP_PressurePulse, CPAP_FlowLimit, CPAP_SensAwake, CPAP_FlowRate, CPAP_MaskPressure,
CPAP_MaskPressureHi,
diff --git a/sleepyhead/SleepLib/schema.cpp b/sleepyhead/SleepLib/schema.cpp
index 12fe35b6..57482dbd 100644
--- a/sleepyhead/SleepLib/schema.cpp
+++ b/sleepyhead/SleepLib/schema.cpp
@@ -57,6 +57,7 @@ QHash Scopes;
bool schema_initialized = false;
void setOrders() {
+ schema::channel[CPAP_PB].setOrder(1);
schema::channel[CPAP_CSR].setOrder(1);
schema::channel[CPAP_Ramp].setOrder(2);
schema::channel[CPAP_LargeLeak].setOrder(2);
@@ -183,9 +184,14 @@ void init()
// Flags
schema::channel.add(GRP_CPAP, new Channel(CPAP_CSR = 0x1000, SPAN, MT_CPAP, SESSION, "CSR",
QObject::tr("Cheyne Stokes Respiration"),
- QObject::tr("An abnormal period of Periodic Breathing/Cheyne Stokes Respiration"),
+ QObject::tr("An abnormal period of Cheyne Stokes Respiration"),
QObject::tr("CSR"), STR_UNIT_Percentage, DEFAULT, COLOR_CSR));
+ schema::channel.add(GRP_CPAP, new Channel(CPAP_PB = 0x1028, SPAN, MT_CPAP, SESSION, "PB",
+ QObject::tr("Periodic Breathing"),
+ QObject::tr("An abnormal period of Periodic Breathing"),
+ QObject::tr("PB"), STR_UNIT_Percentage, DEFAULT, COLOR_CSR));
+
schema::channel.add(GRP_CPAP, new Channel(CPAP_ClearAirway = 0x1001, FLAG, MT_CPAP, SESSION,
"ClearAirway", QObject::tr("Clear Airway"),
diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp
index 55be7593..a25db8f8 100644
--- a/sleepyhead/daily.cpp
+++ b/sleepyhead/daily.cpp
@@ -664,8 +664,8 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day)
for (quint32 o=0;oRelease Notes
Greetings!
-I know it's been a while, but here is new and improved SleepyHead build :)
-
-Warning: This is a test build, not all features are complete or will work 100% as intended yet!
+Warning: This is a pre-release test build, not all features are complete or will work 100% as intended yet!
+This one is specifically to test CMS50i on Mac, and nothing else.
+Important: Back up your data folder manually, by making a copy the SleepyHeadData folder, as attempting to roll back later may break things.
+DreamStation support is not ready yet, so please, I *don't* want feedback from DreamStation users testing this particular build.(You're of course welcome to try, but you are likely to be a little disappointed.)
+Please don't share this with anyone else or reupload anywhere, it's not a public release.
Sleep Well, and good luck!
JediMark
Changes and fixes in v0.9.8-2
-DreamStation support (Thanks for your assistance JediBob!)
+Initial DreamStation support (Thanks for your assistance JediBob!)
AirCurve pressure reporting glitches
Fix F&P Import hang
Pinch to zoom support
diff --git a/sleepyhead/reports.cpp b/sleepyhead/reports.cpp
index 564ae4a8..ee82f041 100644
--- a/sleepyhead/reports.cpp
+++ b/sleepyhead/reports.cpp
@@ -210,6 +210,7 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
float hours = day->hours(MT_CPAP);
ahi /= hours;
float csr = (100.0 / hours) * (day->sum(CPAP_CSR) / 3600.0);
+ float pb = (100.0 / hours) * (day->sum(CPAP_PB) / 3600.0);
float uai = day->count(CPAP_Apnea) / hours;
float oai = day->count(CPAP_Obstructive) / hours;
float hi = (day->count(CPAP_ExP) + day->count(CPAP_Hypopnea)) / hours;
@@ -270,6 +271,8 @@ void Report::PrintReport(gGraphView *gv, QString name, QDate date)
2);
if (cpap->loaderName() == STR_MACH_PRS1) {
+
+ // FIXME: PB / CSR are now separate flags
stats += QObject::tr("REI=%1 VSI=%2 FLI=%3 PB/CSR=%4%%")
.arg(rei, 0, 'f', 2).arg(vsi, 0, 'f', 2)
.arg(fli, 0, 'f', 2).arg(csr, 0, 'f', 2);
diff --git a/sleepyhead/statistics.cpp b/sleepyhead/statistics.cpp
index f053a48c..71c0a276 100644
--- a/sleepyhead/statistics.cpp
+++ b/sleepyhead/statistics.cpp
@@ -1959,6 +1959,36 @@ void Statistics::UpdateRecordsBox()
html += "
";
}
}
+ if (p_profile->hasChannel(CPAP_PB)) {
+ ahilist.clear();
+ for (QDate date = first; date <= last; date = date.addDays(1)) {
+ Day * day = p_profile->GetGoodDay(date, MT_CPAP);
+ if (!day) continue;
+
+ float leak = day->calcPON(CPAP_PB);
+ ahilist.insert(leak, date);
+ }
+
+ if (ahilist.size() > (show_records * 2)) {
+ html += ""+QObject::tr("Worst PB")+"
";
+
+ it = ahilist.end() - 1;
+ it_end = ahilist.begin();
+ for (int i=0; (i < show_records) && (it != it_end); ++i, --it) {
+
+ if (it.key() > 0) {
+ html += QString("").arg(it.value().toString(Qt::ISODate))
+ +QObject::tr("Date: %1 PB: %2%").arg(it.value().toString(Qt::SystemLocaleShortDate)).arg(it.key(), 0, 'f', 2) + "
";
+ cnt++;
+ }
+ }
+ if (cnt == 0) {
+ html+= ""+QObject::tr("No PB on record")+"
";
+ }
+
+ html += "
";
+ }
+ }
} else {
html += "
"+QObject::tr("Want more information?")+"
";