From 2a326c816f7410d676258ef5b5a2883f4b3f5189 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Mon, 31 May 2021 20:24:09 -0400 Subject: [PATCH] Move PRS1 F0V6 parsing into separate parser file. No change in functionality. Use git blame dd9a087 to follow the history before this refactoring. --- oscar/SleepLib/loader_plugins/prs1_loader.cpp | 811 +----------------- oscar/SleepLib/loader_plugins/prs1_loader.h | 2 + oscar/SleepLib/loader_plugins/prs1_parser.cpp | 28 + oscar/SleepLib/loader_plugins/prs1_parser.h | 30 + .../loader_plugins/prs1_parser_xpap.cpp | 773 +++++++++++++++++ oscar/oscar.pro | 1 + oscar/tests/prs1tests.cpp | 6 - 7 files changed, 837 insertions(+), 814 deletions(-) create mode 100644 oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp index b14d4b59..8e261d90 100644 --- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp @@ -192,7 +192,7 @@ static crc32_t CRC32wchar(const unsigned char *data, size_t data_len, crc32_t cr } -static QString ts(qint64 msecs) +QString ts(qint64 msecs) { // TODO: make this UTC so that tests don't vary by where they're run return QDateTime::fromMSecsSinceEpoch(msecs).toString(Qt::ISODate); @@ -206,18 +206,6 @@ static QString hex(int i) // TODO: See the LogUnexpectedMessage TODO about generalizing this for other loaders. -// Right now this macro assumes that it's called within a method that has a "loader" member -// that points to the PRS1Loader* instance that's calling it. -#define UNEXPECTED_VALUE(SRC, VALS) { \ - QString message = QString("%1:%2: %3 = %4 != %5").arg(__func__).arg(__LINE__).arg(#SRC).arg(SRC).arg(VALS); \ - qWarning() << this->sessionid << message; \ - loader->LogUnexpectedMessage(message); \ - } -#define CHECK_VALUE(SRC, VAL) if ((SRC) != (VAL)) UNEXPECTED_VALUE(SRC, VAL) -#define CHECK_VALUES(SRC, VAL1, VAL2) if ((SRC) != (VAL1) && (SRC) != (VAL2)) UNEXPECTED_VALUE(SRC, #VAL1 " or " #VAL2) -// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails -#define HEX(SRC) { qWarning() << this->sessionid << QString("%1:%2: %3 = %4").arg(__func__).arg(__LINE__).arg(#SRC).arg((SRC & 0xFF), 2, 16, QChar('0')); } - void PRS1Loader::LogUnexpectedMessage(const QString & message) { m_importMutex.lock(); @@ -226,12 +214,6 @@ void PRS1Loader::LogUnexpectedMessage(const QString & message) } -enum FlexMode { FLEX_None, FLEX_CFlex, FLEX_CFlexPlus, FLEX_AFlex, FLEX_RiseTime, FLEX_BiFlex, FLEX_PFlex, FLEX_Flex, FLEX_Unknown = -1 }; - -enum BackupBreathMode { PRS1Backup_Off, PRS1Backup_Auto, PRS1Backup_Fixed }; - -enum HumidMode { HUMID_Fixed, HUMID_Adaptive, HUMID_HeatedTube, HUMID_Passover, HUMID_Error }; - ChannelID PRS1_Mode = 0; ChannelID PRS1_TimedBreath = 0, PRS1_HumidMode = 0, PRS1_TubeTemp = 0; ChannelID PRS1_FlexLock = 0, PRS1_TubeLock = 0, PRS1_RampType = 0; @@ -1468,30 +1450,6 @@ static const QHash> PRS1ImportChannelMap //******************************************************************************************** -static QString DumpEvent(int t, int code, const unsigned char* data, int size) -{ - int s = t; - int h = s / 3600; s -= h * 3600; - int m = s / 60; s -= m * 60; - QString dump = QString("%1:%2:%3 ") - .arg(h, 2, 10, QChar('0')) - .arg(m, 2, 10, QChar('0')) - .arg(s, 2, 10, QChar('0')); - dump = dump + " " + hex(code) + ":"; - for (int i = 0; i < size; i++) { - dump = dump + QString(" %1").arg(data[i]); - } - return dump; -} -#define DUMP_EVENT() qWarning() << this->sessionid << DumpEvent(t, code, data + pos, size - (pos - startpos)) + " @ " + hex(startpos-1) - - -void PRS1DataChunk::AddEvent(PRS1ParsedEvent* const event) -{ - m_parsedData.push_back(event); -} - - static const QVector ParsedEventsF5V3 = { PRS1EPAPSetEvent::TYPE, PRS1TimedBreathEvent::TYPE, @@ -3170,7 +3128,7 @@ void SmoothEventList(Session * session, EventList * ev, ChannelID code) #endif -static const QVector ParsedEventsF0V23 = { +const QVector ParsedEventsF0V23 = { PRS1PressureSetEvent::TYPE, PRS1IPAPSetEvent::TYPE, PRS1EPAPSetEvent::TYPE, @@ -3380,7 +3338,7 @@ bool PRS1DataChunk::ParseEventsF0V23() } -static const QVector ParsedEventsF0V4 = { +const QVector ParsedEventsF0V4 = { PRS1PressureSetEvent::TYPE, PRS1IPAPSetEvent::TYPE, PRS1EPAPSetEvent::TYPE, @@ -3589,235 +3547,6 @@ bool PRS1DataChunk::ParseEventsF0V4() } -static const QVector ParsedEventsF0V6 = { - PRS1PressureSetEvent::TYPE, - PRS1IPAPSetEvent::TYPE, - PRS1EPAPSetEvent::TYPE, - PRS1AutoPressureSetEvent::TYPE, - PRS1PressurePulseEvent::TYPE, - PRS1RERAEvent::TYPE, - PRS1ObstructiveApneaEvent::TYPE, - PRS1ClearAirwayEvent::TYPE, - PRS1HypopneaEvent::TYPE, - PRS1FlowLimitationEvent::TYPE, - PRS1VibratorySnoreEvent::TYPE, - PRS1VariableBreathingEvent::TYPE, - PRS1PeriodicBreathingEvent::TYPE, - PRS1LargeLeakEvent::TYPE, - PRS1TotalLeakEvent::TYPE, - PRS1SnoreEvent::TYPE, - PRS1PressureAverageEvent::TYPE, - PRS1FlexPressureAverageEvent::TYPE, - PRS1SnoresAtPressureEvent::TYPE, -}; - -// DreamStation family 0 CPAP/APAP machines (400X-700X, 400G-502G) -// Originally derived from F5V3 parsing + (incomplete) F0V234 parsing + sample data -bool PRS1DataChunk::ParseEventsF0V6() -{ - if (this->family != 0 || this->familyVersion != 6) { - qWarning() << "ParseEventsF0V6 called with family" << this->family << "familyVersion" << this->familyVersion; - return false; - } - const unsigned char * data = (unsigned char *)this->m_data.constData(); - int chunk_size = this->m_data.size(); - static const int minimum_sizes[] = { 2, 3, 4, 3, 3, 3, 3, 3, 3, 2, 3, 4, 3, 2, 5, 5, 5, 5, 4, 3, 3, 3 }; - static const int ncodes = sizeof(minimum_sizes) / sizeof(int); - - if (chunk_size < 1) { - // This does occasionally happen. - qDebug() << this->sessionid << "Empty event data"; - return false; - } - - bool ok = true; - int pos = 0, startpos; - int code, size; - int t = 0; - int elapsed, duration, value; - bool is_bilevel = false; - do { - code = data[pos++]; - if (!this->hblock.contains(code)) { - qWarning() << this->sessionid << "missing hblock entry for event" << code; - ok = false; - break; - } - size = this->hblock[code]; - if (code < ncodes) { - // make sure the handlers below don't go past the end of the buffer - if (size < minimum_sizes[code]) { - qWarning() << this->sessionid << "event" << code << "too small" << size << "<" << minimum_sizes[code]; - ok = false; - break; - } - } // else if it's past ncodes, we'll log its information below (rather than handle it) - if (pos + size > chunk_size) { - qWarning() << this->sessionid << "event" << code << "@" << pos << "longer than remaining chunk"; - ok = false; - break; - } - startpos = pos; - if (code != 0x12) { // This one event has no timestamp - t += data[pos] | (data[pos+1] << 8); - pos += 2; - } - - switch (code) { - //case 0x00: // never seen - case 0x01: // Pressure adjustment - // Matches pressure setting, both initial and when ramp button pressed. - // Based on waveform reports, it looks like the pressure graph is drawn by - // interpolating between these pressure adjustments, by 0.5 cmH2O spaced evenly between - // adjustments. E.g. 6 at 28:11 and 7.3 at 29:05 results in the following dots: - // 6 at 28:11, 6.5 around 28:30, 7.0 around 28:50, 7(.3) at 29:05. That holds until - // subsequent "adjustment" of 7.3 at 30:09 followed by 8.0 at 30:19. - this->AddEvent(new PRS1PressureSetEvent(t, data[pos])); - break; - case 0x02: // Pressure adjustment (bi-level) - // See notes above on interpolation. - this->AddEvent(new PRS1IPAPSetEvent(t, data[pos+1])); - this->AddEvent(new PRS1EPAPSetEvent(t, data[pos])); // EPAP needs to be added second to calculate PS - is_bilevel = true; - break; - case 0x03: // Auto-CPAP starting pressure - // Most of the time this occurs, it's at the start and end of a session with - // the same pressure at both. Occasionally an additional event shows up in the - // middle of a session, and then the pressure at the end matches that. - // In these cases, the new pressure corresponds to the next night's starting - // pressure for auto-CPAP. It does not appear to have any effect on the current - // night's pressure, unless there's a substantial gap between sessions, in - // which case the next session may use the new starting pressure. - //CHECK_VALUE(data[pos], 40); - // TODO: What does this mean in bi-level mode? - // See F0V4 event 3 for comparison. TODO: See if there's an Opti-Start label on F0V6 reports. - this->AddEvent(new PRS1AutoPressureSetEvent(t, data[pos])); - break; - case 0x04: // Pressure Pulse - duration = data[pos]; // TODO: is this a duration? - this->AddEvent(new PRS1PressurePulseEvent(t, duration)); - break; - case 0x05: // RERA - elapsed = data[pos]; // based on sample waveform, the RERA is over after this - this->AddEvent(new PRS1RERAEvent(t - elapsed, 0)); - break; - case 0x06: // Obstructive Apnea - // OA events are instantaneous flags with no duration: reviewing waveforms - // shows that the time elapsed between the flag and reporting often includes - // non-apnea breathing. - elapsed = data[pos]; - this->AddEvent(new PRS1ObstructiveApneaEvent(t - elapsed, 0)); - break; - case 0x07: // Clear Airway Apnea - // CA events are instantaneous flags with no duration: reviewing waveforms - // shows that the time elapsed between the flag and reporting often includes - // non-apnea breathing. - elapsed = data[pos]; - this->AddEvent(new PRS1ClearAirwayEvent(t - elapsed, 0)); - break; - //case 0x08: // never seen - //case 0x09: // never seen - //case 0x0a: // Hypopnea, see 0x15 - case 0x0b: // Hypopnea - // TODO: How is this hypopnea different from events 0xa, 0x14 and 0x15? - // TODO: What is the first byte? - //data[pos+0]; // unknown first byte? - elapsed = data[pos+1]; // based on sample waveform, the hypopnea is over after this - this->AddEvent(new PRS1HypopneaEvent(t - elapsed, 0)); - break; - case 0x0c: // Flow Limitation - // TODO: We should revisit whether this is elapsed or duration once (if) - // we start calculating flow limitations ourselves. Flow limitations aren't - // as obvious as OA/CA when looking at a waveform. - elapsed = data[pos]; - this->AddEvent(new PRS1FlowLimitationEvent(t - elapsed, 0)); - break; - case 0x0d: // Vibratory Snore - // VS events are instantaneous flags with no duration, drawn on the official waveform. - // The current thinking is that these are the snores that cause a change in auto-titrating - // pressure. The snoring statistics below seem to be a total count. It's unclear whether - // the trigger for pressure change is severity or count or something else. - // no data bytes - this->AddEvent(new PRS1VibratorySnoreEvent(t, 0)); - break; - case 0x0e: // Variable Breathing? - duration = 2 * (data[pos] | (data[pos+1] << 8)); - elapsed = data[pos+2]; // this is always 60 seconds unless it's at the end, so it seems like elapsed - CHECK_VALUES(elapsed, 60, 0); - this->AddEvent(new PRS1VariableBreathingEvent(t - elapsed - duration, duration)); - break; - case 0x0f: // Periodic Breathing - // PB events are reported some time after they conclude, and they do have a reported duration. - duration = 2 * (data[pos] | (data[pos+1] << 8)); - elapsed = data[pos+2]; - this->AddEvent(new PRS1PeriodicBreathingEvent(t - elapsed - duration, duration)); - break; - case 0x10: // Large Leak - // LL events are reported some time after they conclude, and they do have a reported duration. - duration = 2 * (data[pos] | (data[pos+1] << 8)); - elapsed = data[pos+2]; - this->AddEvent(new PRS1LargeLeakEvent(t - elapsed - duration, duration)); - break; - case 0x11: // Statistics - this->AddEvent(new PRS1TotalLeakEvent(t, data[pos])); - this->AddEvent(new PRS1SnoreEvent(t, data[pos+1])); - - value = data[pos+2]; - if (is_bilevel) { - // For bi-level modes, this appears to be the time-weighted average of EPAP and IPAP actually provided. - this->AddEvent(new PRS1PressureAverageEvent(t, value)); - } else { - // For single-pressure modes, this appears to be the average effective "EPAP" provided by Flex. - // - // Sample data shows this value around 10.3 cmH2O for a prescribed pressure of 12.0 (C-Flex+ 3). - // That's too low for an average pressure over time, but could easily be an average commanded EPAP. - // When flex mode is off, this is exactly the current CPAP set point. - this->AddEvent(new PRS1FlexPressureAverageEvent(t, value)); - } - this->AddEvent(new PRS1IntervalBoundaryEvent(t)); - break; - case 0x12: // Snore count per pressure - // Some sessions (with lots of ramps) have multiple of these, each with a - // different pressure. The total snore count across all of them matches the - // total found in the stats event. - if (data[pos] != 0) { - CHECK_VALUES(data[pos], 1, 2); // 0 = CPAP pressure, 1 = bi-level EPAP, 2 = bi-level IPAP - } - //CHECK_VALUE(data[pos+1], 0x78); // pressure - //CHECK_VALUE(data[pos+2], 1); // 16-bit snore count - //CHECK_VALUE(data[pos+3], 0); - value = (data[pos+2] | (data[pos+3] << 8)); - this->AddEvent(new PRS1SnoresAtPressureEvent(t, data[pos], data[pos+1], value)); - break; - //case 0x13: // never seen - case 0x0a: // Hypopnea - // TODO: Why does this hypopnea have a different event code? - // fall through - case 0x14: // Hypopnea, new to F0V6 - // TODO: Why does this hypopnea have a different event code? - // fall through - case 0x15: // Hypopnea, new to F0V6 - // TODO: We should revisit whether this is elapsed or duration once (if) - // we start calculating hypopneas ourselves. Their official definition - // is 40% reduction in flow lasting at least 10s. - duration = data[pos]; - this->AddEvent(new PRS1HypopneaEvent(t - duration, 0)); - break; - default: - DUMP_EVENT(); - UNEXPECTED_VALUE(code, "known event code"); - this->AddEvent(new PRS1UnknownDataEvent(m_data, startpos-1, size+1)); - break; - } - pos = startpos + size; - } while (ok && pos < chunk_size); - - this->duration = t; - - return ok; -} - - // TODO: This really should be in some kind of class hierarchy, once we figure out // the right one. const QVector & GetSupportedEvents(const PRS1DataChunk* chunk) @@ -6780,540 +6509,6 @@ void PRS1DataChunk::ParseHumidifierSettingV3(unsigned char byte1, unsigned char } -// The below is based on a combination of the old mainblock parsing for fileVersion == 3 -// in ParseSummary() and the switch statements of ParseSummaryF0V6. -// -// Both compliance and summary files (at least for 200X and 400X machines) seem to have -// the same length for this slice, so maybe the settings are the same? At least 0x0a -// looks like a pressure in compliance files. -bool PRS1DataChunk::ParseSettingsF0V6(const unsigned char* data, int size) -{ - static const QMap expected_lengths = { {0x0c,3}, {0x0d,2}, {0x0e,2}, {0x0f,4}, {0x10,3}, {0x35,2} }; - bool ok = true; - - PRS1Mode cpapmode = PRS1_MODE_UNKNOWN; - FlexMode flexmode = FLEX_Unknown; - - int pressure = 0; - int imin_ps = 0; - int imax_ps = 0; - int min_pressure = 0; - int max_pressure = 0; - - // Parse the nested data structure which contains settings - int pos = 0; - do { - int code = data[pos++]; - int len = data[pos++]; - - int expected_len = 1; - if (expected_lengths.contains(code)) { - expected_len = expected_lengths[code]; - } - //CHECK_VALUE(len, expected_len); - if (len < expected_len) { - qWarning() << this->sessionid << "setting" << code << "too small" << len << "<" << expected_len; - ok = false; - break; - } - if (pos + len > size) { - qWarning() << this->sessionid << "setting" << code << "@" << pos << "longer than remaining slice"; - ok = false; - break; - } - - switch (code) { - case 0: // Device Mode - CHECK_VALUE(pos, 2); // always first? - CHECK_VALUE(len, 1); - switch (data[pos]) { - case 0: cpapmode = PRS1_MODE_CPAP; break; - case 1: cpapmode = PRS1_MODE_BILEVEL; break; - case 2: cpapmode = PRS1_MODE_AUTOCPAP; break; - case 3: cpapmode = PRS1_MODE_AUTOBILEVEL; break; - case 4: cpapmode = PRS1_MODE_CPAPCHECK; break; - default: - UNEXPECTED_VALUE(data[pos], "known device mode"); - break; - } - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_CPAP_MODE, (int) cpapmode)); - break; - case 1: // ??? - CHECK_VALUES(len, 1, 2); - if (data[pos] != 0 && data[pos] != 3) { - 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 - switch (data[pos+1]) { - case 0x00: // 0x00 seen with EZ-Start disabled, no auto-trial, with CPAP-Check on 400X110 - case 0x10: // 0x10 seen with EZ-Start enabled, Opti-Start off on 500X110 - case 0x20: // 0x20 seen with Opti-Start enabled - case 0x30: // 0x30 seen with both Opti-Start and EZ-Start enabled on 500X110 - case 0x40: // 0x40 seen with Auto-Trial - case 0x80: // 0x80 seen with EZ-Start and CPAP-Check+ on 500X150 - break; - default: - UNEXPECTED_VALUE(data[pos+1], "[0,0x10,0x20,0x30,0x40,0x80]") - } - } - break; - case 0x0a: // CPAP pressure setting - CHECK_VALUE(len, 1); - CHECK_VALUE(cpapmode, PRS1_MODE_CPAP); - pressure = data[pos]; - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE, pressure)); - break; - case 0x0c: // CPAP-Check pressure setting - CHECK_VALUE(len, 3); - CHECK_VALUE(cpapmode, PRS1_MODE_CPAPCHECK); - min_pressure = data[pos]; // Min Setting on pressure graph - max_pressure = data[pos+1]; // Max Setting on pressure graph - pressure = data[pos+2]; // CPAP on pressure graph and CPAP-Check Pressure on settings detail - // This seems to be the initial pressure. If the pressure changes mid-session, the pressure - // graph will show either the changed pressure or the majority pressure, not sure which. - // The time of change is most likely in the events file. See slice 6 for ending pressure. - //CHECK_VALUE(pressure, 0x5a); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE, pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure)); - break; - case 0x0d: // AutoCPAP pressure setting - CHECK_VALUE(len, 2); - CHECK_VALUE(cpapmode, PRS1_MODE_AUTOCPAP); - min_pressure = data[pos]; - max_pressure = data[pos+1]; - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure)); - break; - case 0x0e: // Bi-Level pressure setting - CHECK_VALUE(len, 2); - CHECK_VALUE(cpapmode, PRS1_MODE_BILEVEL); - min_pressure = data[pos]; - max_pressure = data[pos+1]; - imin_ps = max_pressure - min_pressure; - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_EPAP, min_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_IPAP, max_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PS, imin_ps)); - break; - case 0x0f: // Auto Bi-Level pressure setting - CHECK_VALUE(len, 4); - CHECK_VALUE(cpapmode, PRS1_MODE_AUTOBILEVEL); - min_pressure = data[pos]; - max_pressure = data[pos+1]; - imin_ps = data[pos+2]; - imax_ps = data[pos+3]; - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_EPAP_MIN, min_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_IPAP_MAX, max_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PS_MIN, imin_ps)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PS_MAX, imax_ps)); - break; - case 0x10: // Auto-Trial mode - // This is not encoded as a separate mode as in F0V4, but instead as an auto-trial - // duration on top of the CPAP or CPAP-Check mode. Reports show Auto-CPAP results, - // but curiously report the use of C-Flex+, even though Auto-CPAP uses A-Flex. - CHECK_VALUE(len, 3); - CHECK_VALUES(cpapmode, PRS1_MODE_CPAP, PRS1_MODE_CPAPCHECK); - if (data[pos] != 30 && data[pos] != 9) { - CHECK_VALUES(data[pos], 5, 25); // Auto-Trial Duration - } - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_AUTO_TRIAL, data[pos])); - // If we want C-Flex+ to be reported as A-Flex, we can set cpapmode = PRS1_MODE_AUTOTRIAL here. - // (Note that the setting event has already been added above, which is why ImportSummary needs - // to adjust it when it sees this setting.) - min_pressure = data[pos+1]; - max_pressure = data[pos+2]; - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure)); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure)); - break; - case 0x2a: // EZ-Start - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0x00, 0x80); // both seem to mean enabled - // 0x80 is CPAP Mode - EZ-Start in pressure detail chart, 0x00 is just CPAP mode with no EZ-Start pressure - // TODO: How to represent which one is active in practice? Should this always be "true" since - // either value means that the setting is enabled? - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_EZ_START, data[pos] != 0)); - break; - case 0x42: // EZ-Start enabled for Auto-CPAP? - // Seen on 500X110 before 0x2b when EZ-Start is enabled on Auto-CPAP - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0x00, 0x80); // both seem to mean enabled, 0x00 appears when Opti-Start is used instead - // TODO: How to represent which one is active in practice? Should this always be "true" since - // either value means that the setting is 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" - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RAMP_TYPE, data[pos] != 0)); - break; - case 0x2c: // Ramp Time - CHECK_VALUE(len, 1); - if (data[pos] != 0) { // 0 == ramp off, and ramp pressure setting doesn't appear - if (data[pos] < 5 || data[pos] > 45) UNEXPECTED_VALUE(data[pos], "5-45"); - } - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RAMP_TIME, data[pos])); - break; - case 0x2d: // Ramp Pressure - CHECK_VALUE(len, 1); - this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_RAMP_PRESSURE, data[pos])); - break; - case 0x2e: // Flex mode - CHECK_VALUE(len, 1); - switch (data[pos]) { - case 0: - flexmode = FLEX_None; - break; - case 0x80: - switch (cpapmode) { - case PRS1_MODE_CPAP: - case PRS1_MODE_CPAPCHECK: - case PRS1_MODE_AUTOCPAP: - //case PRS1_MODE_AUTOTRIAL: - flexmode = FLEX_CFlex; - break; - case PRS1_MODE_BILEVEL: - case PRS1_MODE_AUTOBILEVEL: - flexmode = FLEX_BiFlex; - break; - default: - HEX(flexmode); - UNEXPECTED_VALUE(cpapmode, "untested mode"); - break; - } - break; - case 0x90: // C-Flex+ or A-Flex, depending on machine mode - switch (cpapmode) { - case PRS1_MODE_CPAP: - case PRS1_MODE_CPAPCHECK: - flexmode = FLEX_CFlexPlus; - break; - case PRS1_MODE_AUTOCPAP: - flexmode = FLEX_AFlex; - break; - default: - UNEXPECTED_VALUE(cpapmode, "cpap or apap"); - break; - } - break; - case 0xA0: // Rise Time - flexmode = FLEX_RiseTime; - switch (cpapmode) { - case PRS1_MODE_BILEVEL: - case PRS1_MODE_AUTOBILEVEL: - break; - default: - HEX(flexmode); - UNEXPECTED_VALUE(cpapmode, "autobilevel"); - break; - } - break; - case 0xB0: // P-Flex - flexmode = FLEX_PFlex; - switch (cpapmode) { - case PRS1_MODE_AUTOCPAP: - break; - default: - HEX(flexmode); - UNEXPECTED_VALUE(cpapmode, "apap"); - break; - } - break; - default: - UNEXPECTED_VALUE(data[pos], "known flex mode"); - break; - } - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_MODE, flexmode)); - break; - case 0x2f: // Flex lock - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0, 0x80); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_LOCK, data[pos] != 0)); - break; - case 0x30: // Flex level - CHECK_VALUE(len, 1); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_LEVEL, data[pos])); - if (flexmode == FLEX_PFlex) { - CHECK_VALUE(data[pos], 4); // No number appears on reports. - } - if (flexmode == FLEX_RiseTime) { - if (data[pos] < 1 || data[pos] > 3) UNEXPECTED_VALUE(data[pos], "1-3"); - } - break; - case 0x35: // Humidifier setting - CHECK_VALUE(len, 2); - this->ParseHumidifierSettingV3(data[pos], data[pos+1], true); - break; - case 0x36: // Mask Resistance Lock - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0, 0x80); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_LOCK, data[pos] != 0)); - break; - case 0x38: // Mask Resistance - CHECK_VALUE(len, 1); - if (data[pos] != 0) { // 0 == mask resistance off - if (data[pos] < 1 || data[pos] > 5) UNEXPECTED_VALUE(data[pos], "1-5"); - } - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_SETTING, data[pos])); - break; - case 0x39: // Tubing Type Lock - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0, 0x80); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_TUBING_LOCK, data[pos] != 0)); - break; - case 0x3b: // Tubing Type - CHECK_VALUE(len, 1); - if (data[pos] != 0) { - CHECK_VALUES(data[pos], 2, 1); // 15HT = 2, 15 = 1, 22 = 0 - } - this->ParseTubingTypeV3(data[pos]); - break; - case 0x40: // new to 400G, also seen on 500X110, alternate tubing type? appears after 0x39 and before 0x3c - CHECK_VALUE(len, 1); - if (data[pos] > 3) UNEXPECTED_VALUE(data[pos], "0-3"); // 0 = 22mm, 1 = 15mm, 2 = 15HT, 3 = 12mm - this->ParseTubingTypeV3(data[pos]); - break; - case 0x3c: // View Optional Screens - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0, 0x80); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_SHOW_AHI, data[pos] != 0)); - break; - case 0x3e: // Auto On - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0, 0x80); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_AUTO_ON, data[pos] != 0)); - break; - case 0x3f: // Auto Off - CHECK_VALUE(len, 1); - CHECK_VALUES(data[pos], 0, 0x80); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_AUTO_OFF, data[pos] != 0)); - break; - case 0x43: // new to 502G, sessions 3-8, Auto-Trial is off, Opti-Start is missing - CHECK_VALUE(len, 1); - CHECK_VALUE(data[pos], 0x3C); - break; - case 0x44: // new to 502G, sessions 3-8, Auto-Trial is off, Opti-Start is missing - CHECK_VALUE(len, 1); - CHECK_VALUE(data[pos], 0xFF); - break; - case 0x45: // Target Time, specific to DreamStation Go - CHECK_VALUE(len, 1); - // Included in the data, but not shown on reports when humidifier is in Fixed mode. - // According to the FAQ, this setting is only available in Adaptive mode. - if (data[pos] < 40 || data[pos] > 100) { // 4.0 through 10.0 hours in 0.5-hour increments - CHECK_VALUES(data[pos], 0, 1); // Off and Auto - } - this->AddEvent(new PRS1ScaledSettingEvent(PRS1_SETTING_HUMID_TARGET_TIME, data[pos], 0.1)); - break; - default: - UNEXPECTED_VALUE(code, "known setting"); - qDebug() << "Unknown setting:" << hex << code << "in" << this->sessionid << "at" << pos; - this->AddEvent(new PRS1UnknownDataEvent(QByteArray((const char*) data, size), pos, len)); - break; - } - - pos += len; - } while (ok && pos + 2 <= size); - - return ok; -} - - -bool PRS1DataChunk::ParseSummaryF0V6(void) -{ - if (this->family != 0 || this->familyVersion != 6) { - qWarning() << "ParseSummaryF0V6 called with family" << this->family << "familyVersion" << this->familyVersion; - return false; - } - const unsigned char * data = (unsigned char *)this->m_data.constData(); - int chunk_size = this->m_data.size(); - static const int minimum_sizes[] = { 1, 0x2b, 9, 4, 2, 4, 1, 4, 0x1b, 2, 4, 0x0b, 1, 2, 6 }; - static const int ncodes = sizeof(minimum_sizes) / sizeof(int); - // NOTE: The sizes contained in hblock can vary, even within a single machine, as can the length of hblock itself! - - // TODO: hardcoding this is ugly, think of a better approach - if (chunk_size < minimum_sizes[0] + minimum_sizes[1] + minimum_sizes[2]) { - qWarning() << this->sessionid << "summary data too short:" << chunk_size; - return false; - } - if (chunk_size < 59) UNEXPECTED_VALUE(chunk_size, ">= 59"); - - bool ok = true; - int pos = 0; - int code, size; - int tt = 0; - while (ok && pos < chunk_size) { - code = data[pos++]; - if (!this->hblock.contains(code)) { - qWarning() << this->sessionid << "missing hblock entry for" << code; - ok = false; - break; - } - size = this->hblock[code]; - if (code < ncodes) { - // make sure the handlers below don't go past the end of the buffer - if (size < minimum_sizes[code]) { - UNEXPECTED_VALUE(size, minimum_sizes[code]); - qWarning() << this->sessionid << "slice" << code << "too small" << size << "<" << minimum_sizes[code]; - if (code != 1) { // Settings are variable-length, so shorter settings slices aren't fatal. - ok = false; - break; - } - } - } // else if it's past ncodes, we'll log its information below (rather than handle it) - if (pos + size > chunk_size) { - qWarning() << this->sessionid << "slice" << code << "@" << pos << "longer than remaining chunk"; - ok = false; - break; - } - - switch (code) { - case 0: // Equipment On - CHECK_VALUE(pos, 1); // Always first? - //CHECK_VALUES(data[pos], 1, 7); // or 3? - if (size == 4) { // 400G has 3 more bytes? - //CHECK_VALUE(data[pos+1], 0); // or 2, 14, 4, etc. - //CHECK_VALUES(data[pos+2], 8, 65); // or 1 - //CHECK_VALUES(data[pos+3], 0, 20); // or 21, 22, etc. - } - break; - case 1: // Settings - ok = this->ParseSettingsF0V6(data + pos, size); - break; - case 3: // Mask On - tt += data[pos] | (data[pos+1] << 8); - this->AddEvent(new PRS1ParsedSliceEvent(tt, MaskOn)); - this->ParseHumidifierSettingV3(data[pos+2], data[pos+3]); - break; - case 4: // Mask Off - tt += data[pos] | (data[pos+1] << 8); - this->AddEvent(new PRS1ParsedSliceEvent(tt, MaskOff)); - break; - case 8: // vs. 7 in compliance, always follows mask off (except when there's a 5, see below), also longer - // Maybe statistics of some kind, given the pressure stats that seem to appear before it on AutoCPAP machines? - //CHECK_VALUES(data[pos], 0x02, 0x01); // probably 16-bit value - CHECK_VALUE(data[pos+1], 0x00); - //CHECK_VALUES(data[pos+2], 0x0d, 0x0a); // probably 16-bit value, maybe OA count? - CHECK_VALUE(data[pos+3], 0x00); - //CHECK_VALUES(data[pos+4], 0x09, 0x0b); // probably 16-bit value - CHECK_VALUE(data[pos+5], 0x00); - //CHECK_VALUES(data[pos+6], 0x1e, 0x35); // probably 16-bit value - CHECK_VALUE(data[pos+7], 0x00); - //CHECK_VALUES(data[pos+8], 0x8c, 0x4c); // 16-bit value, not sure what - //CHECK_VALUE(data[pos+9], 0x00); - //CHECK_VALUES(data[pos+0xa], 0xbb, 0x00); // 16-bit minutes in large leak - //CHECK_VALUE(data[pos+0xb], 0x00); - //CHECK_VALUES(data[pos+0xc], 0x15, 0x02); // 16-bit minutes in PB - //CHECK_VALUE(data[pos+0xd], 0x00); - //CHECK_VALUES(data[pos+0xe], 0x01, 0x00); // 16-bit VS count - //CHECK_VALUE(data[pos+0xf], 0x00); - //CHECK_VALUES(data[pos+0x10], 0x21, 5); // probably 16-bit value, maybe H count? - CHECK_VALUE(data[pos+0x11], 0x00); - //CHECK_VALUES(data[pos+0x12], 0x13, 0); // 16-bit value, not sure what - //CHECK_VALUE(data[pos+0x13], 0x00); - //CHECK_VALUES(data[pos+0x14], 0x05, 0); // probably 16-bit value, maybe RE count? - CHECK_VALUE(data[pos+0x15], 0x00); - //CHECK_VALUE(data[pos+0x16], 0x00, 4); // probably a 16-bit value, PB or FL count? - CHECK_VALUE(data[pos+0x17], 0x00); - //CHECK_VALUES(data[pos+0x18], 0x69, 0x23); - //CHECK_VALUES(data[pos+0x19], 0x44, 0x18); - //CHECK_VALUES(data[pos+0x1a], 0x80, 0x49); - if (size >= 0x1f) { // 500X is only 0x1b long! - //CHECK_VALUES(data[pos+0x1b], 0x00, 6); - CHECK_VALUE(data[pos+0x1c], 0x00); - //CHECK_VALUES(data[pos+0x1d], 0x0c, 0x0d); - //CHECK_VALUES(data[pos+0x1e], 0x31, 0x3b); - // TODO: 400G and 500G has 8 more bytes? - // TODO: 400G sometimes has another 4 on top of that? - } - break; - case 2: // Equipment Off - tt += data[pos] | (data[pos+1] << 8); - this->AddEvent(new PRS1ParsedSliceEvent(tt, EquipmentOff)); - //CHECK_VALUE(data[pos+2], 0x08); // 0x01 - //CHECK_VALUE(data[pos+3], 0x14); // 0x12 - //CHECK_VALUE(data[pos+4], 0x01); // 0x00 - //CHECK_VALUE(data[pos+5], 0x22); // 0x28 - //CHECK_VALUE(data[pos+6], 0x02); // sometimes 1, 0 - CHECK_VALUE(data[pos+7], 0x00); // 0x00 - CHECK_VALUE(data[pos+8], 0x00); // 0x00 - if (size == 0x0c) { // 400G has 3 more bytes, seem to match Equipment On bytes - //CHECK_VALUE(data[pos+1], 0); - //CHECK_VALUES(data[pos+2], 8, 65); - //CHECK_VALUE(data[pos+3], 0); - } - break; - case 0x09: // Time Elapsed (event 4 in F0V4) - tt += data[pos] | (data[pos+1] << 8); - break; - case 0x0a: // Humidifier setting change - tt += data[pos] | (data[pos+1] << 8); // This adds to the total duration (otherwise it won't match report) - this->ParseHumidifierSettingV3(data[pos+2], data[pos+3]); - break; - case 0x0d: // ??? - // seen on one 500G multiple times - //CHECK_VALUE(data[pos], 0); // 16-bit value - //CHECK_VALUE(data[pos+1], 0); - break; - case 0x0e: - // only seen once on 400G, many times on 500G - //CHECK_VALUES(data[pos], 0, 6); // 16-bit value - //CHECK_VALUE(data[pos+1], 0); - //CHECK_VALUES(data[pos+2], 7, 9); - //CHECK_VALUES(data[pos+3], 7, 15); - //CHECK_VALUES(data[pos+4], 7, 12); - //CHECK_VALUES(data[pos+5], 0, 3); - break; - case 0x05: - // AutoCPAP-related? First appeared on 500X, follows 4, before 8, look like pressure values - //CHECK_VALUE(data[pos], 0x4b); // maybe min pressure? (matches ramp pressure, see ramp on pressure graph) - //CHECK_VALUE(data[pos+1], 0x5a); // maybe max pressure? (close to max on pressure graph, time at pressure graph) - //CHECK_VALUE(data[pos+2], 0x5a); // seems to match Average 90% Pressure - //CHECK_VALUE(data[pos+3], 0x58); // seems to match Average CPAP - break; - case 0x07: - // AutoBiLevel-related? First appeared on 700X, follows 4, before 8, looks like pressure values - //CHECK_VALUE(data[pos], 0x50); // maybe min IPAP or max titrated EPAP? (matches time at pressure graph, auto bi-level summary) - //CHECK_VALUE(data[pos+1], 0x64); // maybe max IPAP or max titrated IPAP? (matches time at pressure graph, auto bi-level summary) - //CHECK_VALUE(data[pos+2], 0x4b); // seems to match 90% EPAP - //CHECK_VALUE(data[pos+3], 0x64); // seems to match 90% IPAP - break; - case 0x0b: - // CPAP-Check related, follows Mask On in CPAP-Check mode - tt += data[pos] | (data[pos+1] << 8); // This adds to the total duration (otherwise it won't match report) - //CHECK_VALUE(data[pos+2], 0); // probably 16-bit value - CHECK_VALUE(data[pos+3], 0); - //CHECK_VALUE(data[pos+4], 0); // probably 16-bit value - CHECK_VALUE(data[pos+5], 0); - //CHECK_VALUE(data[pos+6], 0); // probably 16-bit value - CHECK_VALUE(data[pos+7], 0); - //CHECK_VALUE(data[pos+8], 0); // probably 16-bit value - CHECK_VALUE(data[pos+9], 0); - //CHECK_VALUES(data[pos+0xa], 20, 60); // or 0? 44 when changed pressure mid-session? - break; - case 0x06: - // Maybe starting pressure? follows 4, before 8, looks like a pressure value, seen with CPAP-Check and EZ-Start - // Maybe ending pressure: matches ending CPAP-Check pressure if it changes mid-session. - // 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; - } - pos += size; - } - - this->duration = tt; - - return ok; -} - - // Originally based on ParseSummaryF0V6, with changes observed in ASV sample data // based on size, slices 0-5 look similar, and it looks like F0V6 slides 8-B are equivalent to 6-9 // diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.h b/oscar/SleepLib/loader_plugins/prs1_loader.h index d21d7dc0..b20b6fed 100644 --- a/oscar/SleepLib/loader_plugins/prs1_loader.h +++ b/oscar/SleepLib/loader_plugins/prs1_loader.h @@ -44,6 +44,8 @@ const int max_load_buffer_size = 1024 * 1024; #endif const QString prs1_class_name = STR_MACH_PRS1; +QString ts(qint64 msecs); + /*! \struct PRS1Waveform \brief Used in PRS1 Waveform Parsing */ struct PRS1Waveform { diff --git a/oscar/SleepLib/loader_plugins/prs1_parser.cpp b/oscar/SleepLib/loader_plugins/prs1_parser.cpp index 9a907d9d..b372fa2c 100644 --- a/oscar/SleepLib/loader_plugins/prs1_parser.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_parser.cpp @@ -317,3 +317,31 @@ QMap PRS1SnoresAtPressureEvent::contents(void) out["count"] = QString::number(m_count); return out; } + + +// MARK: - + +QString PRS1DataChunk::DumpEvent(int t, int code, const unsigned char* data, int size) +{ + int s = t; + int h = s / 3600; s -= h * 3600; + int m = s / 60; s -= m * 60; + QString dump = QString("%1:%2:%3 ") + .arg(h, 2, 10, QChar('0')) + .arg(m, 2, 10, QChar('0')) + .arg(s, 2, 10, QChar('0')); + dump = dump + " " + hex(code) + ":"; + for (int i = 0; i < size; i++) { + dump = dump + QString(" %1").arg(data[i]); + } + return dump; +} + + +void PRS1DataChunk::AddEvent(PRS1ParsedEvent* const event) +{ + m_parsedData.push_back(event); +} + + + diff --git a/oscar/SleepLib/loader_plugins/prs1_parser.h b/oscar/SleepLib/loader_plugins/prs1_parser.h index 69fac30c..4dfa5367 100644 --- a/oscar/SleepLib/loader_plugins/prs1_parser.h +++ b/oscar/SleepLib/loader_plugins/prs1_parser.h @@ -626,7 +626,37 @@ protected: //! \brief Parse a settings slice from a .000 and .001 file bool ParseSettingsF3V6(const unsigned char* data, int size); + +protected: + QString DumpEvent(int t, int code, const unsigned char* data, int size); }; +#define DUMP_EVENT() qWarning() << this->sessionid << DumpEvent(t, code, data + pos, size - (pos - startpos)) + " @ " + QString("0x") + QString::number(startpos-1, 16).toUpper() + +// TODO: See the LogUnexpectedMessage TODO about generalizing this for other loaders. +// Right now this macro assumes that it's called within a method that has a "loader" member +// that points to the PRS1Loader* instance that's calling it. +#define UNEXPECTED_VALUE(SRC, VALS) { \ + QString message = QString("%1:%2: %3 = %4 != %5").arg(__func__).arg(__LINE__).arg(#SRC).arg(SRC).arg(VALS); \ + qWarning() << this->sessionid << message; \ + loader->LogUnexpectedMessage(message); \ + } +#define CHECK_VALUE(SRC, VAL) if ((SRC) != (VAL)) UNEXPECTED_VALUE(SRC, VAL) +#define CHECK_VALUES(SRC, VAL1, VAL2) if ((SRC) != (VAL1) && (SRC) != (VAL2)) UNEXPECTED_VALUE(SRC, #VAL1 " or " #VAL2) +// for more than 2 values, just write the test manually and use UNEXPECTED_VALUE if it fails +#define HEX(SRC) { qWarning() << this->sessionid << QString("%1:%2: %3 = %4").arg(__func__).arg(__LINE__).arg(#SRC).arg((SRC & 0xFF), 2, 16, QChar('0')); } + +enum FlexMode { FLEX_None, FLEX_CFlex, FLEX_CFlexPlus, FLEX_AFlex, FLEX_RiseTime, FLEX_BiFlex, FLEX_PFlex, FLEX_Flex, FLEX_Unknown = -1 }; + +enum BackupBreathMode { PRS1Backup_Off, PRS1Backup_Auto, PRS1Backup_Fixed }; + +enum HumidMode { HUMID_Fixed, HUMID_Adaptive, HUMID_HeatedTube, HUMID_Passover, HUMID_Error }; + + +extern const QVector ParsedEventsF0V23; +extern const QVector ParsedEventsF0V4; +extern const QVector ParsedEventsF0V6; + + #endif // PRS1PARSER_H diff --git a/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp b/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp new file mode 100644 index 00000000..8736531d --- /dev/null +++ b/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp @@ -0,0 +1,773 @@ +/* PRS1 Parsing for CPAP and BIPAP (Family 0) + * + * Copyright (c) 2019-2021 The OSCAR Team + * Portions copyright (c) 2011-2018 Mark Watkins + * + * 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 source code + * for more details. */ + +#include "prs1_parser.h" +#include "prs1_loader.h" + +bool PRS1DataChunk::ParseSummaryF0V6(void) +{ + if (this->family != 0 || this->familyVersion != 6) { + qWarning() << "ParseSummaryF0V6 called with family" << this->family << "familyVersion" << this->familyVersion; + return false; + } + const unsigned char * data = (unsigned char *)this->m_data.constData(); + int chunk_size = this->m_data.size(); + static const int minimum_sizes[] = { 1, 0x2b, 9, 4, 2, 4, 1, 4, 0x1b, 2, 4, 0x0b, 1, 2, 6 }; + static const int ncodes = sizeof(minimum_sizes) / sizeof(int); + // NOTE: The sizes contained in hblock can vary, even within a single machine, as can the length of hblock itself! + + // TODO: hardcoding this is ugly, think of a better approach + if (chunk_size < minimum_sizes[0] + minimum_sizes[1] + minimum_sizes[2]) { + qWarning() << this->sessionid << "summary data too short:" << chunk_size; + return false; + } + if (chunk_size < 59) UNEXPECTED_VALUE(chunk_size, ">= 59"); + + bool ok = true; + int pos = 0; + int code, size; + int tt = 0; + while (ok && pos < chunk_size) { + code = data[pos++]; + if (!this->hblock.contains(code)) { + qWarning() << this->sessionid << "missing hblock entry for" << code; + ok = false; + break; + } + size = this->hblock[code]; + if (code < ncodes) { + // make sure the handlers below don't go past the end of the buffer + if (size < minimum_sizes[code]) { + UNEXPECTED_VALUE(size, minimum_sizes[code]); + qWarning() << this->sessionid << "slice" << code << "too small" << size << "<" << minimum_sizes[code]; + if (code != 1) { // Settings are variable-length, so shorter settings slices aren't fatal. + ok = false; + break; + } + } + } // else if it's past ncodes, we'll log its information below (rather than handle it) + if (pos + size > chunk_size) { + qWarning() << this->sessionid << "slice" << code << "@" << pos << "longer than remaining chunk"; + ok = false; + break; + } + + switch (code) { + case 0: // Equipment On + CHECK_VALUE(pos, 1); // Always first? + //CHECK_VALUES(data[pos], 1, 7); // or 3? + if (size == 4) { // 400G has 3 more bytes? + //CHECK_VALUE(data[pos+1], 0); // or 2, 14, 4, etc. + //CHECK_VALUES(data[pos+2], 8, 65); // or 1 + //CHECK_VALUES(data[pos+3], 0, 20); // or 21, 22, etc. + } + break; + case 1: // Settings + ok = this->ParseSettingsF0V6(data + pos, size); + break; + case 3: // Mask On + tt += data[pos] | (data[pos+1] << 8); + this->AddEvent(new PRS1ParsedSliceEvent(tt, MaskOn)); + this->ParseHumidifierSettingV3(data[pos+2], data[pos+3]); + break; + case 4: // Mask Off + tt += data[pos] | (data[pos+1] << 8); + this->AddEvent(new PRS1ParsedSliceEvent(tt, MaskOff)); + break; + case 8: // vs. 7 in compliance, always follows mask off (except when there's a 5, see below), also longer + // Maybe statistics of some kind, given the pressure stats that seem to appear before it on AutoCPAP machines? + //CHECK_VALUES(data[pos], 0x02, 0x01); // probably 16-bit value + CHECK_VALUE(data[pos+1], 0x00); + //CHECK_VALUES(data[pos+2], 0x0d, 0x0a); // probably 16-bit value, maybe OA count? + CHECK_VALUE(data[pos+3], 0x00); + //CHECK_VALUES(data[pos+4], 0x09, 0x0b); // probably 16-bit value + CHECK_VALUE(data[pos+5], 0x00); + //CHECK_VALUES(data[pos+6], 0x1e, 0x35); // probably 16-bit value + CHECK_VALUE(data[pos+7], 0x00); + //CHECK_VALUES(data[pos+8], 0x8c, 0x4c); // 16-bit value, not sure what + //CHECK_VALUE(data[pos+9], 0x00); + //CHECK_VALUES(data[pos+0xa], 0xbb, 0x00); // 16-bit minutes in large leak + //CHECK_VALUE(data[pos+0xb], 0x00); + //CHECK_VALUES(data[pos+0xc], 0x15, 0x02); // 16-bit minutes in PB + //CHECK_VALUE(data[pos+0xd], 0x00); + //CHECK_VALUES(data[pos+0xe], 0x01, 0x00); // 16-bit VS count + //CHECK_VALUE(data[pos+0xf], 0x00); + //CHECK_VALUES(data[pos+0x10], 0x21, 5); // probably 16-bit value, maybe H count? + CHECK_VALUE(data[pos+0x11], 0x00); + //CHECK_VALUES(data[pos+0x12], 0x13, 0); // 16-bit value, not sure what + //CHECK_VALUE(data[pos+0x13], 0x00); + //CHECK_VALUES(data[pos+0x14], 0x05, 0); // probably 16-bit value, maybe RE count? + CHECK_VALUE(data[pos+0x15], 0x00); + //CHECK_VALUE(data[pos+0x16], 0x00, 4); // probably a 16-bit value, PB or FL count? + CHECK_VALUE(data[pos+0x17], 0x00); + //CHECK_VALUES(data[pos+0x18], 0x69, 0x23); + //CHECK_VALUES(data[pos+0x19], 0x44, 0x18); + //CHECK_VALUES(data[pos+0x1a], 0x80, 0x49); + if (size >= 0x1f) { // 500X is only 0x1b long! + //CHECK_VALUES(data[pos+0x1b], 0x00, 6); + CHECK_VALUE(data[pos+0x1c], 0x00); + //CHECK_VALUES(data[pos+0x1d], 0x0c, 0x0d); + //CHECK_VALUES(data[pos+0x1e], 0x31, 0x3b); + // TODO: 400G and 500G has 8 more bytes? + // TODO: 400G sometimes has another 4 on top of that? + } + break; + case 2: // Equipment Off + tt += data[pos] | (data[pos+1] << 8); + this->AddEvent(new PRS1ParsedSliceEvent(tt, EquipmentOff)); + //CHECK_VALUE(data[pos+2], 0x08); // 0x01 + //CHECK_VALUE(data[pos+3], 0x14); // 0x12 + //CHECK_VALUE(data[pos+4], 0x01); // 0x00 + //CHECK_VALUE(data[pos+5], 0x22); // 0x28 + //CHECK_VALUE(data[pos+6], 0x02); // sometimes 1, 0 + CHECK_VALUE(data[pos+7], 0x00); // 0x00 + CHECK_VALUE(data[pos+8], 0x00); // 0x00 + if (size == 0x0c) { // 400G has 3 more bytes, seem to match Equipment On bytes + //CHECK_VALUE(data[pos+1], 0); + //CHECK_VALUES(data[pos+2], 8, 65); + //CHECK_VALUE(data[pos+3], 0); + } + break; + case 0x09: // Time Elapsed (event 4 in F0V4) + tt += data[pos] | (data[pos+1] << 8); + break; + case 0x0a: // Humidifier setting change + tt += data[pos] | (data[pos+1] << 8); // This adds to the total duration (otherwise it won't match report) + this->ParseHumidifierSettingV3(data[pos+2], data[pos+3]); + break; + case 0x0d: // ??? + // seen on one 500G multiple times + //CHECK_VALUE(data[pos], 0); // 16-bit value + //CHECK_VALUE(data[pos+1], 0); + break; + case 0x0e: + // only seen once on 400G, many times on 500G + //CHECK_VALUES(data[pos], 0, 6); // 16-bit value + //CHECK_VALUE(data[pos+1], 0); + //CHECK_VALUES(data[pos+2], 7, 9); + //CHECK_VALUES(data[pos+3], 7, 15); + //CHECK_VALUES(data[pos+4], 7, 12); + //CHECK_VALUES(data[pos+5], 0, 3); + break; + case 0x05: + // AutoCPAP-related? First appeared on 500X, follows 4, before 8, look like pressure values + //CHECK_VALUE(data[pos], 0x4b); // maybe min pressure? (matches ramp pressure, see ramp on pressure graph) + //CHECK_VALUE(data[pos+1], 0x5a); // maybe max pressure? (close to max on pressure graph, time at pressure graph) + //CHECK_VALUE(data[pos+2], 0x5a); // seems to match Average 90% Pressure + //CHECK_VALUE(data[pos+3], 0x58); // seems to match Average CPAP + break; + case 0x07: + // AutoBiLevel-related? First appeared on 700X, follows 4, before 8, looks like pressure values + //CHECK_VALUE(data[pos], 0x50); // maybe min IPAP or max titrated EPAP? (matches time at pressure graph, auto bi-level summary) + //CHECK_VALUE(data[pos+1], 0x64); // maybe max IPAP or max titrated IPAP? (matches time at pressure graph, auto bi-level summary) + //CHECK_VALUE(data[pos+2], 0x4b); // seems to match 90% EPAP + //CHECK_VALUE(data[pos+3], 0x64); // seems to match 90% IPAP + break; + case 0x0b: + // CPAP-Check related, follows Mask On in CPAP-Check mode + tt += data[pos] | (data[pos+1] << 8); // This adds to the total duration (otherwise it won't match report) + //CHECK_VALUE(data[pos+2], 0); // probably 16-bit value + CHECK_VALUE(data[pos+3], 0); + //CHECK_VALUE(data[pos+4], 0); // probably 16-bit value + CHECK_VALUE(data[pos+5], 0); + //CHECK_VALUE(data[pos+6], 0); // probably 16-bit value + CHECK_VALUE(data[pos+7], 0); + //CHECK_VALUE(data[pos+8], 0); // probably 16-bit value + CHECK_VALUE(data[pos+9], 0); + //CHECK_VALUES(data[pos+0xa], 20, 60); // or 0? 44 when changed pressure mid-session? + break; + case 0x06: + // Maybe starting pressure? follows 4, before 8, looks like a pressure value, seen with CPAP-Check and EZ-Start + // Maybe ending pressure: matches ending CPAP-Check pressure if it changes mid-session. + // 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; + } + pos += size; + } + + this->duration = tt; + + return ok; +} + + +// The below is based on a combination of the old mainblock parsing for fileVersion == 3 +// in ParseSummary() and the switch statements of ParseSummaryF0V6. +// +// Both compliance and summary files (at least for 200X and 400X machines) seem to have +// the same length for this slice, so maybe the settings are the same? At least 0x0a +// looks like a pressure in compliance files. +bool PRS1DataChunk::ParseSettingsF0V6(const unsigned char* data, int size) +{ + static const QMap expected_lengths = { {0x0c,3}, {0x0d,2}, {0x0e,2}, {0x0f,4}, {0x10,3}, {0x35,2} }; + bool ok = true; + + PRS1Mode cpapmode = PRS1_MODE_UNKNOWN; + FlexMode flexmode = FLEX_Unknown; + + int pressure = 0; + int imin_ps = 0; + int imax_ps = 0; + int min_pressure = 0; + int max_pressure = 0; + + // Parse the nested data structure which contains settings + int pos = 0; + do { + int code = data[pos++]; + int len = data[pos++]; + + int expected_len = 1; + if (expected_lengths.contains(code)) { + expected_len = expected_lengths[code]; + } + //CHECK_VALUE(len, expected_len); + if (len < expected_len) { + qWarning() << this->sessionid << "setting" << code << "too small" << len << "<" << expected_len; + ok = false; + break; + } + if (pos + len > size) { + qWarning() << this->sessionid << "setting" << code << "@" << pos << "longer than remaining slice"; + ok = false; + break; + } + + switch (code) { + case 0: // Device Mode + CHECK_VALUE(pos, 2); // always first? + CHECK_VALUE(len, 1); + switch (data[pos]) { + case 0: cpapmode = PRS1_MODE_CPAP; break; + case 1: cpapmode = PRS1_MODE_BILEVEL; break; + case 2: cpapmode = PRS1_MODE_AUTOCPAP; break; + case 3: cpapmode = PRS1_MODE_AUTOBILEVEL; break; + case 4: cpapmode = PRS1_MODE_CPAPCHECK; break; + default: + UNEXPECTED_VALUE(data[pos], "known device mode"); + break; + } + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_CPAP_MODE, (int) cpapmode)); + break; + case 1: // ??? + CHECK_VALUES(len, 1, 2); + if (data[pos] != 0 && data[pos] != 3) { + 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 + switch (data[pos+1]) { + case 0x00: // 0x00 seen with EZ-Start disabled, no auto-trial, with CPAP-Check on 400X110 + case 0x10: // 0x10 seen with EZ-Start enabled, Opti-Start off on 500X110 + case 0x20: // 0x20 seen with Opti-Start enabled + case 0x30: // 0x30 seen with both Opti-Start and EZ-Start enabled on 500X110 + case 0x40: // 0x40 seen with Auto-Trial + case 0x80: // 0x80 seen with EZ-Start and CPAP-Check+ on 500X150 + break; + default: + UNEXPECTED_VALUE(data[pos+1], "[0,0x10,0x20,0x30,0x40,0x80]") + } + } + break; + case 0x0a: // CPAP pressure setting + CHECK_VALUE(len, 1); + CHECK_VALUE(cpapmode, PRS1_MODE_CPAP); + pressure = data[pos]; + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE, pressure)); + break; + case 0x0c: // CPAP-Check pressure setting + CHECK_VALUE(len, 3); + CHECK_VALUE(cpapmode, PRS1_MODE_CPAPCHECK); + min_pressure = data[pos]; // Min Setting on pressure graph + max_pressure = data[pos+1]; // Max Setting on pressure graph + pressure = data[pos+2]; // CPAP on pressure graph and CPAP-Check Pressure on settings detail + // This seems to be the initial pressure. If the pressure changes mid-session, the pressure + // graph will show either the changed pressure or the majority pressure, not sure which. + // The time of change is most likely in the events file. See slice 6 for ending pressure. + //CHECK_VALUE(pressure, 0x5a); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE, pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure)); + break; + case 0x0d: // AutoCPAP pressure setting + CHECK_VALUE(len, 2); + CHECK_VALUE(cpapmode, PRS1_MODE_AUTOCPAP); + min_pressure = data[pos]; + max_pressure = data[pos+1]; + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure)); + break; + case 0x0e: // Bi-Level pressure setting + CHECK_VALUE(len, 2); + CHECK_VALUE(cpapmode, PRS1_MODE_BILEVEL); + min_pressure = data[pos]; + max_pressure = data[pos+1]; + imin_ps = max_pressure - min_pressure; + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_EPAP, min_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_IPAP, max_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PS, imin_ps)); + break; + case 0x0f: // Auto Bi-Level pressure setting + CHECK_VALUE(len, 4); + CHECK_VALUE(cpapmode, PRS1_MODE_AUTOBILEVEL); + min_pressure = data[pos]; + max_pressure = data[pos+1]; + imin_ps = data[pos+2]; + imax_ps = data[pos+3]; + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_EPAP_MIN, min_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_IPAP_MAX, max_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PS_MIN, imin_ps)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PS_MAX, imax_ps)); + break; + case 0x10: // Auto-Trial mode + // This is not encoded as a separate mode as in F0V4, but instead as an auto-trial + // duration on top of the CPAP or CPAP-Check mode. Reports show Auto-CPAP results, + // but curiously report the use of C-Flex+, even though Auto-CPAP uses A-Flex. + CHECK_VALUE(len, 3); + CHECK_VALUES(cpapmode, PRS1_MODE_CPAP, PRS1_MODE_CPAPCHECK); + if (data[pos] != 30 && data[pos] != 9) { + CHECK_VALUES(data[pos], 5, 25); // Auto-Trial Duration + } + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_AUTO_TRIAL, data[pos])); + // If we want C-Flex+ to be reported as A-Flex, we can set cpapmode = PRS1_MODE_AUTOTRIAL here. + // (Note that the setting event has already been added above, which is why ImportSummary needs + // to adjust it when it sees this setting.) + min_pressure = data[pos+1]; + max_pressure = data[pos+2]; + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MIN, min_pressure)); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_PRESSURE_MAX, max_pressure)); + break; + case 0x2a: // EZ-Start + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0x00, 0x80); // both seem to mean enabled + // 0x80 is CPAP Mode - EZ-Start in pressure detail chart, 0x00 is just CPAP mode with no EZ-Start pressure + // TODO: How to represent which one is active in practice? Should this always be "true" since + // either value means that the setting is enabled? + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_EZ_START, data[pos] != 0)); + break; + case 0x42: // EZ-Start enabled for Auto-CPAP? + // Seen on 500X110 before 0x2b when EZ-Start is enabled on Auto-CPAP + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0x00, 0x80); // both seem to mean enabled, 0x00 appears when Opti-Start is used instead + // TODO: How to represent which one is active in practice? Should this always be "true" since + // either value means that the setting is 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" + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RAMP_TYPE, data[pos] != 0)); + break; + case 0x2c: // Ramp Time + CHECK_VALUE(len, 1); + if (data[pos] != 0) { // 0 == ramp off, and ramp pressure setting doesn't appear + if (data[pos] < 5 || data[pos] > 45) UNEXPECTED_VALUE(data[pos], "5-45"); + } + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_RAMP_TIME, data[pos])); + break; + case 0x2d: // Ramp Pressure + CHECK_VALUE(len, 1); + this->AddEvent(new PRS1PressureSettingEvent(PRS1_SETTING_RAMP_PRESSURE, data[pos])); + break; + case 0x2e: // Flex mode + CHECK_VALUE(len, 1); + switch (data[pos]) { + case 0: + flexmode = FLEX_None; + break; + case 0x80: + switch (cpapmode) { + case PRS1_MODE_CPAP: + case PRS1_MODE_CPAPCHECK: + case PRS1_MODE_AUTOCPAP: + //case PRS1_MODE_AUTOTRIAL: + flexmode = FLEX_CFlex; + break; + case PRS1_MODE_BILEVEL: + case PRS1_MODE_AUTOBILEVEL: + flexmode = FLEX_BiFlex; + break; + default: + HEX(flexmode); + UNEXPECTED_VALUE(cpapmode, "untested mode"); + break; + } + break; + case 0x90: // C-Flex+ or A-Flex, depending on machine mode + switch (cpapmode) { + case PRS1_MODE_CPAP: + case PRS1_MODE_CPAPCHECK: + flexmode = FLEX_CFlexPlus; + break; + case PRS1_MODE_AUTOCPAP: + flexmode = FLEX_AFlex; + break; + default: + UNEXPECTED_VALUE(cpapmode, "cpap or apap"); + break; + } + break; + case 0xA0: // Rise Time + flexmode = FLEX_RiseTime; + switch (cpapmode) { + case PRS1_MODE_BILEVEL: + case PRS1_MODE_AUTOBILEVEL: + break; + default: + HEX(flexmode); + UNEXPECTED_VALUE(cpapmode, "autobilevel"); + break; + } + break; + case 0xB0: // P-Flex + flexmode = FLEX_PFlex; + switch (cpapmode) { + case PRS1_MODE_AUTOCPAP: + break; + default: + HEX(flexmode); + UNEXPECTED_VALUE(cpapmode, "apap"); + break; + } + break; + default: + UNEXPECTED_VALUE(data[pos], "known flex mode"); + break; + } + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_MODE, flexmode)); + break; + case 0x2f: // Flex lock + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0, 0x80); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_LOCK, data[pos] != 0)); + break; + case 0x30: // Flex level + CHECK_VALUE(len, 1); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_FLEX_LEVEL, data[pos])); + if (flexmode == FLEX_PFlex) { + CHECK_VALUE(data[pos], 4); // No number appears on reports. + } + if (flexmode == FLEX_RiseTime) { + if (data[pos] < 1 || data[pos] > 3) UNEXPECTED_VALUE(data[pos], "1-3"); + } + break; + case 0x35: // Humidifier setting + CHECK_VALUE(len, 2); + this->ParseHumidifierSettingV3(data[pos], data[pos+1], true); + break; + case 0x36: // Mask Resistance Lock + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0, 0x80); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_LOCK, data[pos] != 0)); + break; + case 0x38: // Mask Resistance + CHECK_VALUE(len, 1); + if (data[pos] != 0) { // 0 == mask resistance off + if (data[pos] < 1 || data[pos] > 5) UNEXPECTED_VALUE(data[pos], "1-5"); + } + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_SETTING, data[pos])); + break; + case 0x39: // Tubing Type Lock + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0, 0x80); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_TUBING_LOCK, data[pos] != 0)); + break; + case 0x3b: // Tubing Type + CHECK_VALUE(len, 1); + if (data[pos] != 0) { + CHECK_VALUES(data[pos], 2, 1); // 15HT = 2, 15 = 1, 22 = 0 + } + this->ParseTubingTypeV3(data[pos]); + break; + case 0x40: // new to 400G, also seen on 500X110, alternate tubing type? appears after 0x39 and before 0x3c + CHECK_VALUE(len, 1); + if (data[pos] > 3) UNEXPECTED_VALUE(data[pos], "0-3"); // 0 = 22mm, 1 = 15mm, 2 = 15HT, 3 = 12mm + this->ParseTubingTypeV3(data[pos]); + break; + case 0x3c: // View Optional Screens + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0, 0x80); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_SHOW_AHI, data[pos] != 0)); + break; + case 0x3e: // Auto On + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0, 0x80); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_AUTO_ON, data[pos] != 0)); + break; + case 0x3f: // Auto Off + CHECK_VALUE(len, 1); + CHECK_VALUES(data[pos], 0, 0x80); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_AUTO_OFF, data[pos] != 0)); + break; + case 0x43: // new to 502G, sessions 3-8, Auto-Trial is off, Opti-Start is missing + CHECK_VALUE(len, 1); + CHECK_VALUE(data[pos], 0x3C); + break; + case 0x44: // new to 502G, sessions 3-8, Auto-Trial is off, Opti-Start is missing + CHECK_VALUE(len, 1); + CHECK_VALUE(data[pos], 0xFF); + break; + case 0x45: // Target Time, specific to DreamStation Go + CHECK_VALUE(len, 1); + // Included in the data, but not shown on reports when humidifier is in Fixed mode. + // According to the FAQ, this setting is only available in Adaptive mode. + if (data[pos] < 40 || data[pos] > 100) { // 4.0 through 10.0 hours in 0.5-hour increments + CHECK_VALUES(data[pos], 0, 1); // Off and Auto + } + this->AddEvent(new PRS1ScaledSettingEvent(PRS1_SETTING_HUMID_TARGET_TIME, data[pos], 0.1)); + break; + default: + UNEXPECTED_VALUE(code, "known setting"); + qDebug() << "Unknown setting:" << hex << code << "in" << this->sessionid << "at" << pos; + this->AddEvent(new PRS1UnknownDataEvent(QByteArray((const char*) data, size), pos, len)); + break; + } + + pos += len; + } while (ok && pos + 2 <= size); + + return ok; +} + + +const QVector ParsedEventsF0V6 = { + PRS1PressureSetEvent::TYPE, + PRS1IPAPSetEvent::TYPE, + PRS1EPAPSetEvent::TYPE, + PRS1AutoPressureSetEvent::TYPE, + PRS1PressurePulseEvent::TYPE, + PRS1RERAEvent::TYPE, + PRS1ObstructiveApneaEvent::TYPE, + PRS1ClearAirwayEvent::TYPE, + PRS1HypopneaEvent::TYPE, + PRS1FlowLimitationEvent::TYPE, + PRS1VibratorySnoreEvent::TYPE, + PRS1VariableBreathingEvent::TYPE, + PRS1PeriodicBreathingEvent::TYPE, + PRS1LargeLeakEvent::TYPE, + PRS1TotalLeakEvent::TYPE, + PRS1SnoreEvent::TYPE, + PRS1PressureAverageEvent::TYPE, + PRS1FlexPressureAverageEvent::TYPE, + PRS1SnoresAtPressureEvent::TYPE, +}; + +// DreamStation family 0 CPAP/APAP machines (400X-700X, 400G-502G) +// Originally derived from F5V3 parsing + (incomplete) F0V234 parsing + sample data +bool PRS1DataChunk::ParseEventsF0V6() +{ + if (this->family != 0 || this->familyVersion != 6) { + qWarning() << "ParseEventsF0V6 called with family" << this->family << "familyVersion" << this->familyVersion; + return false; + } + const unsigned char * data = (unsigned char *)this->m_data.constData(); + int chunk_size = this->m_data.size(); + static const int minimum_sizes[] = { 2, 3, 4, 3, 3, 3, 3, 3, 3, 2, 3, 4, 3, 2, 5, 5, 5, 5, 4, 3, 3, 3 }; + static const int ncodes = sizeof(minimum_sizes) / sizeof(int); + + if (chunk_size < 1) { + // This does occasionally happen. + qDebug() << this->sessionid << "Empty event data"; + return false; + } + + bool ok = true; + int pos = 0, startpos; + int code, size; + int t = 0; + int elapsed, duration, value; + bool is_bilevel = false; + do { + code = data[pos++]; + if (!this->hblock.contains(code)) { + qWarning() << this->sessionid << "missing hblock entry for event" << code; + ok = false; + break; + } + size = this->hblock[code]; + if (code < ncodes) { + // make sure the handlers below don't go past the end of the buffer + if (size < minimum_sizes[code]) { + qWarning() << this->sessionid << "event" << code << "too small" << size << "<" << minimum_sizes[code]; + ok = false; + break; + } + } // else if it's past ncodes, we'll log its information below (rather than handle it) + if (pos + size > chunk_size) { + qWarning() << this->sessionid << "event" << code << "@" << pos << "longer than remaining chunk"; + ok = false; + break; + } + startpos = pos; + if (code != 0x12) { // This one event has no timestamp + t += data[pos] | (data[pos+1] << 8); + pos += 2; + } + + switch (code) { + //case 0x00: // never seen + case 0x01: // Pressure adjustment + // Matches pressure setting, both initial and when ramp button pressed. + // Based on waveform reports, it looks like the pressure graph is drawn by + // interpolating between these pressure adjustments, by 0.5 cmH2O spaced evenly between + // adjustments. E.g. 6 at 28:11 and 7.3 at 29:05 results in the following dots: + // 6 at 28:11, 6.5 around 28:30, 7.0 around 28:50, 7(.3) at 29:05. That holds until + // subsequent "adjustment" of 7.3 at 30:09 followed by 8.0 at 30:19. + this->AddEvent(new PRS1PressureSetEvent(t, data[pos])); + break; + case 0x02: // Pressure adjustment (bi-level) + // See notes above on interpolation. + this->AddEvent(new PRS1IPAPSetEvent(t, data[pos+1])); + this->AddEvent(new PRS1EPAPSetEvent(t, data[pos])); // EPAP needs to be added second to calculate PS + is_bilevel = true; + break; + case 0x03: // Auto-CPAP starting pressure + // Most of the time this occurs, it's at the start and end of a session with + // the same pressure at both. Occasionally an additional event shows up in the + // middle of a session, and then the pressure at the end matches that. + // In these cases, the new pressure corresponds to the next night's starting + // pressure for auto-CPAP. It does not appear to have any effect on the current + // night's pressure, unless there's a substantial gap between sessions, in + // which case the next session may use the new starting pressure. + //CHECK_VALUE(data[pos], 40); + // TODO: What does this mean in bi-level mode? + // See F0V4 event 3 for comparison. TODO: See if there's an Opti-Start label on F0V6 reports. + this->AddEvent(new PRS1AutoPressureSetEvent(t, data[pos])); + break; + case 0x04: // Pressure Pulse + duration = data[pos]; // TODO: is this a duration? + this->AddEvent(new PRS1PressurePulseEvent(t, duration)); + break; + case 0x05: // RERA + elapsed = data[pos]; // based on sample waveform, the RERA is over after this + this->AddEvent(new PRS1RERAEvent(t - elapsed, 0)); + break; + case 0x06: // Obstructive Apnea + // OA events are instantaneous flags with no duration: reviewing waveforms + // shows that the time elapsed between the flag and reporting often includes + // non-apnea breathing. + elapsed = data[pos]; + this->AddEvent(new PRS1ObstructiveApneaEvent(t - elapsed, 0)); + break; + case 0x07: // Clear Airway Apnea + // CA events are instantaneous flags with no duration: reviewing waveforms + // shows that the time elapsed between the flag and reporting often includes + // non-apnea breathing. + elapsed = data[pos]; + this->AddEvent(new PRS1ClearAirwayEvent(t - elapsed, 0)); + break; + //case 0x08: // never seen + //case 0x09: // never seen + //case 0x0a: // Hypopnea, see 0x15 + case 0x0b: // Hypopnea + // TODO: How is this hypopnea different from events 0xa, 0x14 and 0x15? + // TODO: What is the first byte? + //data[pos+0]; // unknown first byte? + elapsed = data[pos+1]; // based on sample waveform, the hypopnea is over after this + this->AddEvent(new PRS1HypopneaEvent(t - elapsed, 0)); + break; + case 0x0c: // Flow Limitation + // TODO: We should revisit whether this is elapsed or duration once (if) + // we start calculating flow limitations ourselves. Flow limitations aren't + // as obvious as OA/CA when looking at a waveform. + elapsed = data[pos]; + this->AddEvent(new PRS1FlowLimitationEvent(t - elapsed, 0)); + break; + case 0x0d: // Vibratory Snore + // VS events are instantaneous flags with no duration, drawn on the official waveform. + // The current thinking is that these are the snores that cause a change in auto-titrating + // pressure. The snoring statistics below seem to be a total count. It's unclear whether + // the trigger for pressure change is severity or count or something else. + // no data bytes + this->AddEvent(new PRS1VibratorySnoreEvent(t, 0)); + break; + case 0x0e: // Variable Breathing? + duration = 2 * (data[pos] | (data[pos+1] << 8)); + elapsed = data[pos+2]; // this is always 60 seconds unless it's at the end, so it seems like elapsed + CHECK_VALUES(elapsed, 60, 0); + this->AddEvent(new PRS1VariableBreathingEvent(t - elapsed - duration, duration)); + break; + case 0x0f: // Periodic Breathing + // PB events are reported some time after they conclude, and they do have a reported duration. + duration = 2 * (data[pos] | (data[pos+1] << 8)); + elapsed = data[pos+2]; + this->AddEvent(new PRS1PeriodicBreathingEvent(t - elapsed - duration, duration)); + break; + case 0x10: // Large Leak + // LL events are reported some time after they conclude, and they do have a reported duration. + duration = 2 * (data[pos] | (data[pos+1] << 8)); + elapsed = data[pos+2]; + this->AddEvent(new PRS1LargeLeakEvent(t - elapsed - duration, duration)); + break; + case 0x11: // Statistics + this->AddEvent(new PRS1TotalLeakEvent(t, data[pos])); + this->AddEvent(new PRS1SnoreEvent(t, data[pos+1])); + + value = data[pos+2]; + if (is_bilevel) { + // For bi-level modes, this appears to be the time-weighted average of EPAP and IPAP actually provided. + this->AddEvent(new PRS1PressureAverageEvent(t, value)); + } else { + // For single-pressure modes, this appears to be the average effective "EPAP" provided by Flex. + // + // Sample data shows this value around 10.3 cmH2O for a prescribed pressure of 12.0 (C-Flex+ 3). + // That's too low for an average pressure over time, but could easily be an average commanded EPAP. + // When flex mode is off, this is exactly the current CPAP set point. + this->AddEvent(new PRS1FlexPressureAverageEvent(t, value)); + } + this->AddEvent(new PRS1IntervalBoundaryEvent(t)); + break; + case 0x12: // Snore count per pressure + // Some sessions (with lots of ramps) have multiple of these, each with a + // different pressure. The total snore count across all of them matches the + // total found in the stats event. + if (data[pos] != 0) { + CHECK_VALUES(data[pos], 1, 2); // 0 = CPAP pressure, 1 = bi-level EPAP, 2 = bi-level IPAP + } + //CHECK_VALUE(data[pos+1], 0x78); // pressure + //CHECK_VALUE(data[pos+2], 1); // 16-bit snore count + //CHECK_VALUE(data[pos+3], 0); + value = (data[pos+2] | (data[pos+3] << 8)); + this->AddEvent(new PRS1SnoresAtPressureEvent(t, data[pos], data[pos+1], value)); + break; + //case 0x13: // never seen + case 0x0a: // Hypopnea + // TODO: Why does this hypopnea have a different event code? + // fall through + case 0x14: // Hypopnea, new to F0V6 + // TODO: Why does this hypopnea have a different event code? + // fall through + case 0x15: // Hypopnea, new to F0V6 + // TODO: We should revisit whether this is elapsed or duration once (if) + // we start calculating hypopneas ourselves. Their official definition + // is 40% reduction in flow lasting at least 10s. + duration = data[pos]; + this->AddEvent(new PRS1HypopneaEvent(t - duration, 0)); + break; + default: + DUMP_EVENT(); + UNEXPECTED_VALUE(code, "known event code"); + this->AddEvent(new PRS1UnknownDataEvent(m_data, startpos-1, size+1)); + break; + } + pos = startpos + size; + } while (ok && pos < chunk_size); + + this->duration = t; + + return ok; +} diff --git a/oscar/oscar.pro b/oscar/oscar.pro index 8a9b15b3..caf01a7e 100644 --- a/oscar/oscar.pro +++ b/oscar/oscar.pro @@ -299,6 +299,7 @@ SOURCES += \ SleepLib/loader_plugins/mseries_loader.cpp \ SleepLib/loader_plugins/prs1_loader.cpp \ SleepLib/loader_plugins/prs1_parser.cpp \ + SleepLib/loader_plugins/prs1_parser_xpap.cpp \ SleepLib/loader_plugins/resmed_loader.cpp \ SleepLib/loader_plugins/resmed_EDFinfo.cpp \ SleepLib/loader_plugins/somnopose_loader.cpp \ diff --git a/oscar/tests/prs1tests.cpp b/oscar/tests/prs1tests.cpp index b92184d1..d53a3b88 100644 --- a/oscar/tests/prs1tests.cpp +++ b/oscar/tests/prs1tests.cpp @@ -127,12 +127,6 @@ void PRS1Tests::testSessionsToYaml() // ==================================================================================================== -static QString ts(qint64 msecs) -{ - // TODO: make this UTC so that tests don't vary by where they're run - return QDateTime::fromMSecsSinceEpoch(msecs).toString(Qt::ISODate); -} - static QString dur(qint64 msecs) { qint64 s = msecs / 1000L;