diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp index 40d037c0..e0bbd5da 100644 --- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp @@ -2023,42 +2023,6 @@ void SmoothEventList(Session * session, EventList * ev, ChannelID code) #endif -// TODO: This really should be in some kind of class hierarchy, once we figure out -// the right one. -const QVector & GetSupportedEvents(const PRS1DataChunk* chunk) -{ - static const QVector none; - - switch (chunk->family) { - case 0: - switch (chunk->familyVersion) { - case 2: return ParsedEventsF0V23; break; - case 3: return ParsedEventsF0V23; break; - case 4: return ParsedEventsF0V4; break; - case 6: return ParsedEventsF0V6; break; - } - break; - case 3: - switch (chunk->familyVersion) { - case 0: return ParsedEventsF3V0; break; - case 3: return ParsedEventsF3V3; break; - case 6: return ParsedEventsF3V6; break; - } - break; - case 5: - switch (chunk->familyVersion) { - case 0: return ParsedEventsF5V0; break; - case 1: return ParsedEventsF5V1; break; - case 2: return ParsedEventsF5V2; break; - case 3: return ParsedEventsF5V3; break; - } - break; - } - qWarning() << "Missing supported event list for family" << chunk->family << "version" << chunk->familyVersion; - return none; -} - - CPAPMode PRS1Import::importMode(int prs1mode) { CPAPMode mode = MODE_UNKNOWN; @@ -2191,462 +2155,6 @@ bool PRS1Import::ImportCompliance() } -bool PRS1DataChunk::ParseCompliance(void) -{ - switch (this->family) { - case 0: - switch (this->familyVersion) { - case 2: - case 3: - return this->ParseComplianceF0V23(); - case 4: - return this->ParseComplianceF0V4(); - case 5: - return this->ParseComplianceF0V5(); - case 6: - return this->ParseComplianceF0V6(); - } - default: - ; - } - - qWarning() << "unexpected compliance family" << this->family << "familyVersion" << this->familyVersion; - return false; -} - - -// F0V4 confirmed: -// B3 0A = HT=5, H=3, HT -// A3 0A = HT=5, H=2, HT -// 33 0A = HT=4, H=3, HT -// 23 4A = HT=4, H=2, HT -// B3 09 = HT=3, H=3, HT -// A4 09 = HT=3, H=2, HT -// A3 49 = HT=3, H=2, HT -// 22 09 = HT=2, H=2, HT -// 33 09 = HT=2, H=3, HT -// 21 09 = HT=2, H=2, HT -// 13 09 = HT=2, H=1, HT -// B5 08 = HT=1, H=3, HT -// 03 08 = HT=off, HT; data=tube t=0,h=0 -// 05 24 = H=5, S1 -// 95 06 = H=5, S1 -// 95 05 = H=5, S1 -// 94 05 = H=4, S1 -// 04 24 = H=4, S1 -// A3 05 = H=3, S1 -// 92 05 = H=2, S1 -// A2 05 = H=2, S1 -// 01 24 = H=1, S1 -// 90 05 = H=off, S1 -// 30 05 = H=off, S1 -// 95 41 = H=5, Classic -// A4 61 = H=4, Classic -// A3 61 = H=3, Classic -// A2 61 = H=2, Classic -// A1 61 = H=1, Classic -// 90 41 = H=Off, Classic; data=classic h=0 -// 94 11 = H=3, S1, no data [note that bits encode H=4, so no data falls back to H=3] -// 93 11 = H=3, S1, no data -// 04 30 = H=3, S1, no data - -// F0V5 confirmed: -// 00 60 = H=Off, Classic -// 02 60 = H=2, Classic -// 05 60 = H=5, Classic -// 00 70 = H=Off, no data in chart - -// F5V1 confirmed: -// A0 4A = HT=5, H=2, HT -// B1 09 = HT=3, H=3, HT -// 91 09 = HT=3, H=1, HT -// 32 09 = HT=2, H=3, HT -// B2 08 = HT=1, H=3, HT -// 00 48 = HT=off, data=tube t=0,h=0 -// 95 05 = H=5, S1 -// 94 05 = H=4, S1 -// 93 05 = H=3, S1 -// 92 05 = H=2, S1 -// 91 05 = H=1, S1 -// 90 05 = H=Off, S1 -// 95 41 = H=5, Classic -// 94 41 = H=4, Classic -// 93 41 = H=3, Classic -// 92 41 = H=2, Classic -// 01 60 = H=1, Classic -// 00 60 = H=Off, Classic -// 00 70 = H=3, S1, no data [no data ignores Classic mode, H bits, falls back to S1 H=3] - -// F5V2 confirmed: -// 00 48 = HT=off, data=tube t=0,h=0 -// 93 09 = HT=3, H=1, HT -// 00 10 = H=3, S1, no data - -// XX XX = 60-Series Humidifier bytes -// 7 = humidity level without tube [on tube disconnect / system one with 22mm hose / classic] : 0 = humidifier off -// 8 = [never seen] -// 3 = humidity level with tube -// 4 = maybe part of humidity level? [never seen] -// 8 3 = tube temperature (high bit of humid 1 is low bit of temp) -// 4 = "System One" mode (valid even when humidifier is off) -// 8 = heated tube present -// 10 = no data in chart, maybe no humidifier attached? Seems to fall back on System One = 3 despite other (humidity level and S1) bits. -// 20 = unknown, something tube related since whenever it's set tubepresent is false -// 40 = "Classic" mode (valid even when humidifier is off, ignored when heated tube is present) -// 80 = [never seen] - -void PRS1DataChunk::ParseHumidifierSetting60Series(unsigned char humid1, unsigned char humid2, bool add_setting) -{ - int humidlevel = humid1 & 7; // Ignored when heated tube is present: humidifier setting on tube disconnect is always reported as 3 - if (humidlevel > 5) UNEXPECTED_VALUE(humidlevel, "<= 5"); - CHECK_VALUE(humid1 & 8, 0); // never seen - int tubehumidlevel = (humid1 >> 4) & 7; // This mask is a best guess based on other masks. - if (tubehumidlevel > 5) UNEXPECTED_VALUE(tubehumidlevel, "<= 5"); - CHECK_VALUE(tubehumidlevel & 4, 0); // never seen, but would clarify whether above mask is correct - - int tubetemp = (humid1 >> 7) | ((humid2 & 3) << 1); - if (tubetemp > 5) UNEXPECTED_VALUE(tubetemp, "<= 5"); - - CHECK_VALUE(humid2 & 0x80, 0); // never seen - bool humidclassic = (humid2 & 0x40) != 0; // Set on classic mode reports; evidently ignored (sometimes set!) when tube is present - //bool no_tube? = (humid2 & 0x20) != 0; // Something tube related: whenever it is set, tube is never present (inverse is not true) - bool no_data = (humid2 & 0x10) != 0; // As described in chart, settings still show up - int tubepresent = (humid2 & 0x08) != 0; - bool humidsystemone = (humid2 & 0x04) != 0; // Set on "System One" humidification mode reports when tubepresent is false - if (humidsystemone && tubepresent) { - // On a 560P, we've observed a spurious tubepresent bit being set during two sessions. - // Those sessions (and the ones that followed) used a 22mm hose. - CHECK_VALUE(add_setting, false); // We've only seen this appear during a session, not in the initial settings. - tubepresent = false; - } - - // When no_data, reports always say "System One" with humidity level 3, regardless of humidlevel and humidsystemone - - if (humidsystemone + tubepresent + no_data == 0) CHECK_VALUE(humidclassic, true); // Always set when everything else is off - if (humidsystemone + tubepresent + no_data > 1) UNEXPECTED_VALUE(humid2, "one bit set"); // Only one of these ever seems to be set at a time - if (tubepresent && tubetemp == 0) CHECK_VALUE(tubehumidlevel, 0); // When the heated tube is off, tube humidity seems to be 0 - - if (tubepresent) humidclassic = false; // Classic mode bit is evidently ignored when tube is present - if (no_data) humidclassic = false; // Classic mode bit is evidently ignored when tube is present - - //qWarning() << this->sessionid << (humidclassic ? "C" : ".") << (humid2 & 0x20 ? "?" : ".") << (tubepresent ? "T" : ".") << (no_data ? "X" : ".") << (humidsystemone ? "1" : "."); - /* - if (tubepresent) { - if (tubetemp) { - qWarning() << this->sessionid << "tube temp" << tubetemp << "tube humidity" << tubehumidlevel << (humidclassic ? "classic" : "systemone") << "humidity" << humidlevel; - } else { - qWarning() << this->sessionid << "heated tube off" << (humidclassic ? "classic" : "systemone") << "humidity" << humidlevel; - } - } else { - qWarning() << this->sessionid << (humidclassic ? "classic" : "systemone") << "humidity" << humidlevel; - } - */ - HumidMode humidmode = HUMID_Fixed; - if (tubepresent) { - humidmode = HUMID_HeatedTube; - } else { - if (humidsystemone + humidclassic > 1) UNEXPECTED_VALUE(humid2, "fixed or adaptive"); - if (humidsystemone) humidmode = HUMID_Adaptive; - } - - if (add_setting) { - bool humidifier_present = (no_data == 0); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_STATUS, humidifier_present)); - if (humidifier_present) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_MODE, humidmode)); - if (humidmode == HUMID_HeatedTube) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HEATED_TUBE_TEMP, tubetemp)); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, tubehumidlevel)); - } else { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, humidlevel)); - } - } - } - - // Check for previously unseen data that we expect to be normal: - if (this->family == 0) { - // F0V4 - if (tubetemp && (tubehumidlevel < 1 || tubehumidlevel > 3)) UNEXPECTED_VALUE(tubehumidlevel, "1-3"); - } else if (this->familyVersion == 1) { - // F5V1 - if (tubepresent) { - // all tube temperatures seen - if (tubetemp) { - if (tubehumidlevel == 0 || tubehumidlevel > 3) UNEXPECTED_VALUE(tubehumidlevel, "1-3"); - } - } - } else if (this->familyVersion == 2) { - // F5V2 - if (tubepresent) { - CHECK_VALUES(tubetemp, 0, 3); - if (tubetemp) { - CHECK_VALUE(tubehumidlevel, 1); - } - } - CHECK_VALUE(humidsystemone, false); - CHECK_VALUE(humidclassic, false); - } -} - - -// Humid F0V2 confirmed -// 0x00 = Off (presumably no humidifier present) -// 0x80 = Off -// 0x81 = 1 -// 0x82 = 2 -// 0x83 = 3 -// 0x84 = 4 -// 0x85 = 5 - -// Humid F3V0 confirmed -// 0x03 = 3 (but no humidification shown on hours of usage chart) -// 0x04 = 4 (but no humidification shown on hours of usage chart) -// 0x80 = Off -// 0x81 = 1 -// 0x82 = 2 -// 0x83 = 3 -// 0x84 = 4 -// 0x85 = 5 - -// Humid F5V0 confirmed -// 0x00 = Off (presumably no humidifier present) -// 0x80 = Off -// 0x81 = 1, bypass = no -// 0x82 = 2, bypass = no -// 0x83 = 3, bypass = no -// 0x84 = 4, bypass = no -// 0x85 = 5, bypass = no -// 0xA0 = Off, bypass = yes - -void PRS1DataChunk::ParseHumidifierSetting50Series(int humid, bool add_setting) -{ - if (humid & (0x40 | 0x10 | 0x08)) UNEXPECTED_VALUE(humid, "known bits"); - if (humid & 0x20) { - if (this->family == 5) { - CHECK_VALUE(humid, 0xA0); // only example of bypass set, unsure whether it can appear otherwise - } else { - CHECK_VALUE(humid & 0x20, 0); // only ever seen on 950P, where "Bypass System One humidification" is "Yes" - } - } - - bool humidifier_present = ((humid & 0x80) != 0); // humidifier connected - int humidlevel = humid & 7; // humidification level - - HumidMode humidmode = HUMID_Fixed; // 50-Series didn't have adaptive or heated tube humidification - if (add_setting) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_STATUS, humidifier_present)); - if (humidifier_present) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_MODE, humidmode)); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, humidlevel)); - } - } - - // Check for truly unexpected values: - if (humidlevel > 5) UNEXPECTED_VALUE(humidlevel, "<= 5"); - //if (!humidifier_present) CHECK_VALUES(humidlevel, 0, 1); // Some machines appear to encode the humidlevel setting even when the humidifier is not present. -} - - -void PRS1DataChunk::ParseTubingTypeV3(unsigned char type) -{ - int diam; - switch (type) { - case 0: diam = 22; break; - case 1: diam = 15; break; - case 2: diam = 15; break; // 15HT, though the reports only say "15" for DreamStation models - case 3: diam = 12; break; // seen on DreamStation Go models - default: - UNEXPECTED_VALUE(type, "known tubing type"); - return; - } - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HOSE_DIAMETER, diam)); -} - - -// F0V6 confirmed -// 90 B0 = HT=3!,H=3!,data=none [no humidifier appears to ignore HT and H bits and show HT=3,H=3 in details] -// 8C 6C = HT=3, H=3, data=none -// 80 00 = nothing listed in details, data=none, only seen on 400G and 502G -// 54 B4 = HT=5, H=5, data=tube -// 50 90 = HT=4, H=4, data=tube -// 4C 6C = HT=3, H=3, data=tube -// 48 68 = HT=3, H=2, data=tube -// 40 60 = HT=3, H=Off, data=tube t=3,h=0 -// 50 50 = HT=2, H=4, data=tube -// 4C 4C = HT=2, H=3, data=tube -// 50 30 = HT=1, H=4, data=tube -// 4C 0C = HT=off, H=3, data=tube t=0,h=3 -// 34 74 = HT=3, H=5, data=adaptive (5) -// 50 B0 = HT=5, H=4, adaptive -// 30 B0 = HT=3, H=4, data=adaptive (4) -// 30 50 = HT=3, H=4, data=adaptive (4) -// 30 10 = HT=3!,H=4, data=adaptive (4) [adaptive mode appears to ignore HT bits and show HT=3 in details] -// 30 70 = HT=3, H=4, data=adaptive (4) -// 2C 6C = HT=3, H=3, data=adaptive (3) -// 28 08 = H=2, data=adaptive (2), no details (400G) -// 28 48 = HT=3!,H=2, data=adaptive (2) [adaptive mode appears to ignore HT bits and show HT=3 in details] -// 28 68 = HT=3, H=2, data=adaptive (2) -// 24 64 = HT=3, H=1, data=adaptive (1) -// 20 60 = HT=3, H=off, data=adaptive (0) -// 14 74 = HT=3, H=5, data=fixed (5) -// 10 70 = HT=3, H=4, data=fixed (4) -// 0C 6C = HT=3, H=3, data=fixed (3) -// 08 48 = HT=3, H=2, data=fixed (2) -// 08 68 = HT=3, H=2, data=fixed (2) -// 04 64 = HT=3, H=1, data=fixed (1) -// 00 00 = HT=3, H=off, data=fixed (0) - -// F5V3 confirmed: -// 90 70 = HT=3, H=3, adaptive, data=no data -// 54 14 = HT=Off, H=5, adaptive, data=tube t=0,h=5 -// 54 34 = HT=1, H=5, adaptive, data=tube t=1,h=5 -// 50 70 = HT=3, H=4, adaptive, data=tube t=3,h=4 -// 4C 6C = HT=3, H=3, adaptive, data=tube t=3,h=3 -// 4C 4C = HT=2, H=3, adaptive, data=tube t=2,h=3 -// 4C 2C = HT=1, H=3, adaptive, data=tube t=1,h=3 -// 4C 0C = HT=off, H=3, adaptive, data=tube t=0,h=3 -// 48 08 = HT=off, H=2, adaptive, data=tube t=0,h=2 -// 44 04 = HT=off, H=1, adaptive, data=tube t=0,h=1 -// 40 00 = HT=off,H=off, adaptive, data=tube t=0,h=0 -// 34 74 = HT=3, H=5, adaptive, data=s1 (5) -// 30 70 = HT=3, H=4, adaptive, data=s1 (4) -// 2C 6C = HT=3, H=3, adaptive, data=s1 (3) -// 28 68 = HT=3, H=2, adaptive, data=s1 (2) -// 24 64 = HT=3, H=1, adaptive, data=s1 (1) - -// F3V6 confirmed: -// 84 24 = HT=3, H=3, disconnect=adaptive, data=no data -// 50 90 = HT=4, H=4, disconnect=adaptive, data=tube t=4,h=4 -// 44 84 = HT=4, H=1, disconnect=adaptive, data=tube t=4,h=1 -// 40 80 = HT=4, H=Off,disconnect=adaptive, data=tube t=4,h=0 -// 4C 6C = HT=3, H=3, disconnect=adaptive, data=tube t=3,h=3 -// 48 68 = HT=3, H=2, disconnect=adaptive, data=tube t=3,h=2 -// 44 44 = HT=2, H=1, disconnect=adaptive, data=tube t=2,h=1 -// 48 28 = HT=1, H=2, disconnect=adaptive, data=tube t=1,h=2 -// 54 14 = HT=Off,H=5, disconnect=adaptive data=tube t=0,h=5 -// 34 14 = HT=3, H=5, disconnect=adaptive, data=s1 (5) -// 30 70 = HT=3, H=4, disconnect=adaptive, data=s1 (4) -// 2C 6C = HT=3, H=3, disconnect=adaptive, data=s1 (3) -// 28 08 = HT=3, H=2, disconnect=adaptive, data=s1 (2) -// 20 20 = HT=3, H=Off, disconnect=adaptive, data=s1 (0) -// 14 14 = HT=3, H=3, disconnect=fixed, data=classic (5) -// 10 10 = HT=3, H=4, disconnect=fixed, data=classic (4) [fixed mode appears to ignore HT bits and show HT=3 in details] -// 0C 0C = HT=3, H=3, disconnect=fixed, data=classic (3) -// 08 08 = HT=3, H=2, disconnect=fixed, data=classic (2) -// 04 64 = HT=3, H=1, disconnect=fixed, data=classic (1) - -// The data is consistent among all fileVersion 3 models: F0V6, F5V3, F3V6. -// -// NOTE: F5V3 and F3V6 charts report the "Adaptive" setting as "System One" and the "Fixed" -// setting as "Classic", despite labeling the settings "Adaptive" and "Fixed" just like F0V6. -// F0V6 is consistent and labels both settings and chart as "Adaptive" and "Fixed". -// -// 400G and 502G appear to omit the humidifier settings in their details, though they -// do support humidifiers, and will show the humidification in the charts. - -void PRS1DataChunk::ParseHumidifierSettingV3(unsigned char byte1, unsigned char byte2, bool add_setting) -{ - bool humidifier_present = true; - bool humidfixed = false; // formerly called "Classic" - bool humidadaptive = false; // formerly called "System One" - bool tubepresent = false; - bool passover = false; - bool error = false; - - // Byte 1: 0x90 (no humidifier data), 0x50 (15ht, tube 4/5, humid 4), 0x54 (15ht, tube 5, humid 5) 0x4c (15ht, tube temp 3, humidifier 3) - // 0x0c (15, tube 3, humid 3, fixed) - // 0b1001 0000 no humidifier data - // 0b0101 0000 tube 4 and 5, humidifier 4 - // 0b0101 0100 15ht, tube 5, humidifier 5 - // 0b0100 1100 15ht, tube 3, humidifier 3 - // 0b1011 0000 15, tube 3, humidifier 3, "Error" on humidification chart with asterisk at 4 - // 0b0111 0000 15, tube 3, humidifier 3, "Passover" on humidification chart with notch at 4 - // 842 = humidifier status - // 1 84 = humidifier setting - // ?? - CHECK_VALUE(byte1 & 3, 0); - int humid = byte1 >> 5; - switch (humid) { - case 0: humidfixed = true; break; // fixed, ignores tubetemp bits and reports tubetemp=3 - case 1: humidadaptive = true; break; // adaptive, ignores tubetemp bits and reports tubetemp=3 - case 2: tubepresent = true; break; // heated tube - case 3: passover = true; break; // passover mode (only visible in chart) - case 4: humidifier_present = false; break; // no humidifier, reports tubetemp=3 and humidlevel=3 - case 5: error = true; break; // "Error" in humidification chart, reports tubetemp=3 and humidlevel=3 in settings - default: - UNEXPECTED_VALUE(humid, "known value"); - break; - } - int humidlevel = (byte1 >> 2) & 7; - - // Byte 2: 0xB4 (15ht, tube 5, humid 5), 0xB0 (15ht, tube 5, humid 4), 0x90 (tube 4, humid 4), 0x6C (15ht, tube temp 3, humidifier 3) - // 0x80? - // 0b1011 0100 15ht, tube 5, humidifier 5 - // 0b1011 0000 15ht, tube 5, humidifier 4 - // 0b1001 0000 tube 4, humidifier 4 - // 0b0110 1100 15ht, tube 3, humidifier 3 - // 842 = tube temperature - // 1 84 = humidity level when using heated tube, thus far always identical to humidlevel - // ?? - CHECK_VALUE(byte2 & 3, 0); - int tubehumidlevel = (byte2 >> 2) & 7; - CHECK_VALUE(humidlevel, tubehumidlevel); // thus far always the same - int tubetemp = (byte2 >> 5) & 7; - if (humidifier_present) { - if (humidlevel > 5 || humidlevel < 0) UNEXPECTED_VALUE(humidlevel, "0-5"); // 0=off is valid when a humidifier is attached - if (humid == 2) { // heated tube - if (tubetemp > 5 || tubetemp < 0) UNEXPECTED_VALUE(tubetemp, "0-5"); // TODO: maybe this is only if heated tube? 0=off is valid even in heated tube mode - } - } - - // TODO: move this up into the switch statement above, given how many modes there now are. - HumidMode humidmode = HUMID_Fixed; - if (tubepresent) { - humidmode = HUMID_HeatedTube; - } else if (humidadaptive) { - humidmode = HUMID_Adaptive; - } else if (passover) { - humidmode = HUMID_Passover; - } else if (error) { - humidmode = HUMID_Error; - } - - if (add_setting) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_STATUS, humidifier_present)); - if (humidifier_present) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_MODE, humidmode)); - if (humidmode == HUMID_HeatedTube) { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HEATED_TUBE_TEMP, tubetemp)); - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, tubehumidlevel)); - } else { - this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, humidlevel)); - } - } - } - - // Check for previously unseen data that we expect to be normal: - if (family == 0) { - // All variations seen. - } else if (family == 5) { - if (tubepresent) { - // All tube temperature and humidity levels seen. - } else if (humidadaptive) { - // All humidity levels seen. - } else if (humidfixed) { - if (humidlevel < 3) UNEXPECTED_VALUE(humidlevel, "3-5"); - } - } else if (family == 3) { - if (tubepresent) { - // All tube temperature and humidity levels seen. - } else if (humidadaptive) { - // All humidity levels seen. - } else if (humidfixed) { - // All humidity levels seen. - } - } -} - - void PRS1Import::AddSlice(qint64 start, PRS1ParsedEvent* e) { // Cache all slices and incrementally calculate their durations. @@ -2868,78 +2376,6 @@ bool PRS1Import::ImportSummary() } -bool PRS1DataChunk::ParseSummary() -{ - switch (this->family) { - case 0: - if (this->familyVersion == 6) { - return this->ParseSummaryF0V6(); - } else if (this->familyVersion == 4) { - return this->ParseSummaryF0V4(); - } else { - return this->ParseSummaryF0V23(); - } - case 3: - switch (this->familyVersion) { - case 0: return this->ParseSummaryF3V03(); - case 3: return this->ParseSummaryF3V03(); - case 6: return this->ParseSummaryF3V6(); - } - break; - case 5: - if (this->familyVersion == 1) { - return this->ParseSummaryF5V012(); - } else if (this->familyVersion == 0) { - return this->ParseSummaryF5V012(); - } else if (this->familyVersion == 2) { - return this->ParseSummaryF5V012(); - } else if (this->familyVersion == 3) { - return this->ParseSummaryF5V3(); - } - default: - ; - } - - qWarning() << "unexpected family" << this->family << "familyVersion" << this->familyVersion; - return false; -} - - -// TODO: The nested switch statement below just begs for per-version subclasses. -bool PRS1DataChunk::ParseEvents() -{ - bool ok = false; - switch (this->family) { - case 0: - switch (this->familyVersion) { - case 2: ok = this->ParseEventsF0V23(); break; - case 3: ok = this->ParseEventsF0V23(); break; - case 4: ok = this->ParseEventsF0V4(); break; - case 6: ok = this->ParseEventsF0V6(); break; - } - break; - case 3: - switch (this->familyVersion) { - case 0: ok = this->ParseEventsF3V03(); break; - case 3: ok = this->ParseEventsF3V03(); break; - case 6: ok = this->ParseEventsF3V6(); break; - } - break; - case 5: - switch (this->familyVersion) { - case 0: ok = this->ParseEventsF5V0(); break; - case 1: ok = this->ParseEventsF5V1(); break; - case 2: ok = this->ParseEventsF5V2(); break; - case 3: ok = this->ParseEventsF5V3(); break; - } - break; - default: - qDebug() << "Unknown PRS1 family" << this->family << "familyVersion" << this->familyVersion; - } - return ok; -} - - bool PRS1Import::ImportEvents() { bool ok = true; diff --git a/oscar/SleepLib/loader_plugins/prs1_parser.cpp b/oscar/SleepLib/loader_plugins/prs1_parser.cpp index b372fa2c..f8d2dd92 100644 --- a/oscar/SleepLib/loader_plugins/prs1_parser.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_parser.cpp @@ -9,6 +9,7 @@ #include "prs1_parser.h" +#include "prs1_loader.h" #include @@ -53,6 +54,7 @@ const PRS1ParsedEventType PRS1ApneaAlarmEvent::TYPE; //const PRS1ParsedEventType PRS1LowMinuteVentilationAlarmEvent::TYPE; +//******************************************************************************************** // MARK: Render parsed events as text static QString hex(int i) @@ -319,7 +321,141 @@ QMap PRS1SnoresAtPressureEvent::contents(void) } +//******************************************************************************************** // MARK: - +// MARK: Chunk parsing + +bool PRS1DataChunk::ParseCompliance(void) +{ + switch (this->family) { + case 0: + switch (this->familyVersion) { + case 2: + case 3: + return this->ParseComplianceF0V23(); + case 4: + return this->ParseComplianceF0V4(); + case 5: + return this->ParseComplianceF0V5(); + case 6: + return this->ParseComplianceF0V6(); + } + default: + ; + } + + qWarning() << "unexpected compliance family" << this->family << "familyVersion" << this->familyVersion; + return false; +} + + +bool PRS1DataChunk::ParseSummary() +{ + switch (this->family) { + case 0: + if (this->familyVersion == 6) { + return this->ParseSummaryF0V6(); + } else if (this->familyVersion == 4) { + return this->ParseSummaryF0V4(); + } else { + return this->ParseSummaryF0V23(); + } + case 3: + switch (this->familyVersion) { + case 0: return this->ParseSummaryF3V03(); + case 3: return this->ParseSummaryF3V03(); + case 6: return this->ParseSummaryF3V6(); + } + break; + case 5: + if (this->familyVersion == 1) { + return this->ParseSummaryF5V012(); + } else if (this->familyVersion == 0) { + return this->ParseSummaryF5V012(); + } else if (this->familyVersion == 2) { + return this->ParseSummaryF5V012(); + } else if (this->familyVersion == 3) { + return this->ParseSummaryF5V3(); + } + default: + ; + } + + qWarning() << "unexpected family" << this->family << "familyVersion" << this->familyVersion; + return false; +} + + +// TODO: The nested switch statement below just begs for per-version subclasses. +bool PRS1DataChunk::ParseEvents() +{ + bool ok = false; + switch (this->family) { + case 0: + switch (this->familyVersion) { + case 2: ok = this->ParseEventsF0V23(); break; + case 3: ok = this->ParseEventsF0V23(); break; + case 4: ok = this->ParseEventsF0V4(); break; + case 6: ok = this->ParseEventsF0V6(); break; + } + break; + case 3: + switch (this->familyVersion) { + case 0: ok = this->ParseEventsF3V03(); break; + case 3: ok = this->ParseEventsF3V03(); break; + case 6: ok = this->ParseEventsF3V6(); break; + } + break; + case 5: + switch (this->familyVersion) { + case 0: ok = this->ParseEventsF5V0(); break; + case 1: ok = this->ParseEventsF5V1(); break; + case 2: ok = this->ParseEventsF5V2(); break; + case 3: ok = this->ParseEventsF5V3(); break; + } + break; + default: + qDebug() << "Unknown PRS1 family" << this->family << "familyVersion" << this->familyVersion; + } + 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) +{ + static const QVector none; + + switch (chunk->family) { + case 0: + switch (chunk->familyVersion) { + case 2: return ParsedEventsF0V23; break; + case 3: return ParsedEventsF0V23; break; + case 4: return ParsedEventsF0V4; break; + case 6: return ParsedEventsF0V6; break; + } + break; + case 3: + switch (chunk->familyVersion) { + case 0: return ParsedEventsF3V0; break; + case 3: return ParsedEventsF3V3; break; + case 6: return ParsedEventsF3V6; break; + } + break; + case 5: + switch (chunk->familyVersion) { + case 0: return ParsedEventsF5V0; break; + case 1: return ParsedEventsF5V1; break; + case 2: return ParsedEventsF5V2; break; + case 3: return ParsedEventsF5V3; break; + } + break; + } + qWarning() << "Missing supported event list for family" << chunk->family << "version" << chunk->familyVersion; + return none; +} + QString PRS1DataChunk::DumpEvent(int t, int code, const unsigned char* data, int size) { @@ -345,3 +481,439 @@ void PRS1DataChunk::AddEvent(PRS1ParsedEvent* const event) +//******************************************************************************************** +// MARK: - +// MARK: Parse settings shared by multiple families + +// Humid F0V2 confirmed +// 0x00 = Off (presumably no humidifier present) +// 0x80 = Off +// 0x81 = 1 +// 0x82 = 2 +// 0x83 = 3 +// 0x84 = 4 +// 0x85 = 5 + +// Humid F3V0 confirmed +// 0x03 = 3 (but no humidification shown on hours of usage chart) +// 0x04 = 4 (but no humidification shown on hours of usage chart) +// 0x80 = Off +// 0x81 = 1 +// 0x82 = 2 +// 0x83 = 3 +// 0x84 = 4 +// 0x85 = 5 + +// Humid F5V0 confirmed +// 0x00 = Off (presumably no humidifier present) +// 0x80 = Off +// 0x81 = 1, bypass = no +// 0x82 = 2, bypass = no +// 0x83 = 3, bypass = no +// 0x84 = 4, bypass = no +// 0x85 = 5, bypass = no +// 0xA0 = Off, bypass = yes + +void PRS1DataChunk::ParseHumidifierSetting50Series(int humid, bool add_setting) +{ + if (humid & (0x40 | 0x10 | 0x08)) UNEXPECTED_VALUE(humid, "known bits"); + if (humid & 0x20) { + if (this->family == 5) { + CHECK_VALUE(humid, 0xA0); // only example of bypass set, unsure whether it can appear otherwise + } else { + CHECK_VALUE(humid & 0x20, 0); // only ever seen on 950P, where "Bypass System One humidification" is "Yes" + } + } + + bool humidifier_present = ((humid & 0x80) != 0); // humidifier connected + int humidlevel = humid & 7; // humidification level + + HumidMode humidmode = HUMID_Fixed; // 50-Series didn't have adaptive or heated tube humidification + if (add_setting) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_STATUS, humidifier_present)); + if (humidifier_present) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_MODE, humidmode)); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, humidlevel)); + } + } + + // Check for truly unexpected values: + if (humidlevel > 5) UNEXPECTED_VALUE(humidlevel, "<= 5"); + //if (!humidifier_present) CHECK_VALUES(humidlevel, 0, 1); // Some machines appear to encode the humidlevel setting even when the humidifier is not present. +} + + + + +// F0V4 confirmed: +// B3 0A = HT=5, H=3, HT +// A3 0A = HT=5, H=2, HT +// 33 0A = HT=4, H=3, HT +// 23 4A = HT=4, H=2, HT +// B3 09 = HT=3, H=3, HT +// A4 09 = HT=3, H=2, HT +// A3 49 = HT=3, H=2, HT +// 22 09 = HT=2, H=2, HT +// 33 09 = HT=2, H=3, HT +// 21 09 = HT=2, H=2, HT +// 13 09 = HT=2, H=1, HT +// B5 08 = HT=1, H=3, HT +// 03 08 = HT=off, HT; data=tube t=0,h=0 +// 05 24 = H=5, S1 +// 95 06 = H=5, S1 +// 95 05 = H=5, S1 +// 94 05 = H=4, S1 +// 04 24 = H=4, S1 +// A3 05 = H=3, S1 +// 92 05 = H=2, S1 +// A2 05 = H=2, S1 +// 01 24 = H=1, S1 +// 90 05 = H=off, S1 +// 30 05 = H=off, S1 +// 95 41 = H=5, Classic +// A4 61 = H=4, Classic +// A3 61 = H=3, Classic +// A2 61 = H=2, Classic +// A1 61 = H=1, Classic +// 90 41 = H=Off, Classic; data=classic h=0 +// 94 11 = H=3, S1, no data [note that bits encode H=4, so no data falls back to H=3] +// 93 11 = H=3, S1, no data +// 04 30 = H=3, S1, no data + +// F0V5 confirmed: +// 00 60 = H=Off, Classic +// 02 60 = H=2, Classic +// 05 60 = H=5, Classic +// 00 70 = H=Off, no data in chart + +// F5V1 confirmed: +// A0 4A = HT=5, H=2, HT +// B1 09 = HT=3, H=3, HT +// 91 09 = HT=3, H=1, HT +// 32 09 = HT=2, H=3, HT +// B2 08 = HT=1, H=3, HT +// 00 48 = HT=off, data=tube t=0,h=0 +// 95 05 = H=5, S1 +// 94 05 = H=4, S1 +// 93 05 = H=3, S1 +// 92 05 = H=2, S1 +// 91 05 = H=1, S1 +// 90 05 = H=Off, S1 +// 95 41 = H=5, Classic +// 94 41 = H=4, Classic +// 93 41 = H=3, Classic +// 92 41 = H=2, Classic +// 01 60 = H=1, Classic +// 00 60 = H=Off, Classic +// 00 70 = H=3, S1, no data [no data ignores Classic mode, H bits, falls back to S1 H=3] + +// F5V2 confirmed: +// 00 48 = HT=off, data=tube t=0,h=0 +// 93 09 = HT=3, H=1, HT +// 00 10 = H=3, S1, no data + +// XX XX = 60-Series Humidifier bytes +// 7 = humidity level without tube [on tube disconnect / system one with 22mm hose / classic] : 0 = humidifier off +// 8 = [never seen] +// 3 = humidity level with tube +// 4 = maybe part of humidity level? [never seen] +// 8 3 = tube temperature (high bit of humid 1 is low bit of temp) +// 4 = "System One" mode (valid even when humidifier is off) +// 8 = heated tube present +// 10 = no data in chart, maybe no humidifier attached? Seems to fall back on System One = 3 despite other (humidity level and S1) bits. +// 20 = unknown, something tube related since whenever it's set tubepresent is false +// 40 = "Classic" mode (valid even when humidifier is off, ignored when heated tube is present) +// 80 = [never seen] + +void PRS1DataChunk::ParseHumidifierSetting60Series(unsigned char humid1, unsigned char humid2, bool add_setting) +{ + int humidlevel = humid1 & 7; // Ignored when heated tube is present: humidifier setting on tube disconnect is always reported as 3 + if (humidlevel > 5) UNEXPECTED_VALUE(humidlevel, "<= 5"); + CHECK_VALUE(humid1 & 8, 0); // never seen + int tubehumidlevel = (humid1 >> 4) & 7; // This mask is a best guess based on other masks. + if (tubehumidlevel > 5) UNEXPECTED_VALUE(tubehumidlevel, "<= 5"); + CHECK_VALUE(tubehumidlevel & 4, 0); // never seen, but would clarify whether above mask is correct + + int tubetemp = (humid1 >> 7) | ((humid2 & 3) << 1); + if (tubetemp > 5) UNEXPECTED_VALUE(tubetemp, "<= 5"); + + CHECK_VALUE(humid2 & 0x80, 0); // never seen + bool humidclassic = (humid2 & 0x40) != 0; // Set on classic mode reports; evidently ignored (sometimes set!) when tube is present + //bool no_tube? = (humid2 & 0x20) != 0; // Something tube related: whenever it is set, tube is never present (inverse is not true) + bool no_data = (humid2 & 0x10) != 0; // As described in chart, settings still show up + int tubepresent = (humid2 & 0x08) != 0; + bool humidsystemone = (humid2 & 0x04) != 0; // Set on "System One" humidification mode reports when tubepresent is false + if (humidsystemone && tubepresent) { + // On a 560P, we've observed a spurious tubepresent bit being set during two sessions. + // Those sessions (and the ones that followed) used a 22mm hose. + CHECK_VALUE(add_setting, false); // We've only seen this appear during a session, not in the initial settings. + tubepresent = false; + } + + // When no_data, reports always say "System One" with humidity level 3, regardless of humidlevel and humidsystemone + + if (humidsystemone + tubepresent + no_data == 0) CHECK_VALUE(humidclassic, true); // Always set when everything else is off + if (humidsystemone + tubepresent + no_data > 1) UNEXPECTED_VALUE(humid2, "one bit set"); // Only one of these ever seems to be set at a time + if (tubepresent && tubetemp == 0) CHECK_VALUE(tubehumidlevel, 0); // When the heated tube is off, tube humidity seems to be 0 + + if (tubepresent) humidclassic = false; // Classic mode bit is evidently ignored when tube is present + if (no_data) humidclassic = false; // Classic mode bit is evidently ignored when tube is present + + //qWarning() << this->sessionid << (humidclassic ? "C" : ".") << (humid2 & 0x20 ? "?" : ".") << (tubepresent ? "T" : ".") << (no_data ? "X" : ".") << (humidsystemone ? "1" : "."); + /* + if (tubepresent) { + if (tubetemp) { + qWarning() << this->sessionid << "tube temp" << tubetemp << "tube humidity" << tubehumidlevel << (humidclassic ? "classic" : "systemone") << "humidity" << humidlevel; + } else { + qWarning() << this->sessionid << "heated tube off" << (humidclassic ? "classic" : "systemone") << "humidity" << humidlevel; + } + } else { + qWarning() << this->sessionid << (humidclassic ? "classic" : "systemone") << "humidity" << humidlevel; + } + */ + HumidMode humidmode = HUMID_Fixed; + if (tubepresent) { + humidmode = HUMID_HeatedTube; + } else { + if (humidsystemone + humidclassic > 1) UNEXPECTED_VALUE(humid2, "fixed or adaptive"); + if (humidsystemone) humidmode = HUMID_Adaptive; + } + + if (add_setting) { + bool humidifier_present = (no_data == 0); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_STATUS, humidifier_present)); + if (humidifier_present) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_MODE, humidmode)); + if (humidmode == HUMID_HeatedTube) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HEATED_TUBE_TEMP, tubetemp)); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, tubehumidlevel)); + } else { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, humidlevel)); + } + } + } + + // Check for previously unseen data that we expect to be normal: + if (this->family == 0) { + // F0V4 + if (tubetemp && (tubehumidlevel < 1 || tubehumidlevel > 3)) UNEXPECTED_VALUE(tubehumidlevel, "1-3"); + } else if (this->familyVersion == 1) { + // F5V1 + if (tubepresent) { + // all tube temperatures seen + if (tubetemp) { + if (tubehumidlevel == 0 || tubehumidlevel > 3) UNEXPECTED_VALUE(tubehumidlevel, "1-3"); + } + } + } else if (this->familyVersion == 2) { + // F5V2 + if (tubepresent) { + CHECK_VALUES(tubetemp, 0, 3); + if (tubetemp) { + CHECK_VALUE(tubehumidlevel, 1); + } + } + CHECK_VALUE(humidsystemone, false); + CHECK_VALUE(humidclassic, false); + } +} + + +// F0V6 confirmed +// 90 B0 = HT=3!,H=3!,data=none [no humidifier appears to ignore HT and H bits and show HT=3,H=3 in details] +// 8C 6C = HT=3, H=3, data=none +// 80 00 = nothing listed in details, data=none, only seen on 400G and 502G +// 54 B4 = HT=5, H=5, data=tube +// 50 90 = HT=4, H=4, data=tube +// 4C 6C = HT=3, H=3, data=tube +// 48 68 = HT=3, H=2, data=tube +// 40 60 = HT=3, H=Off, data=tube t=3,h=0 +// 50 50 = HT=2, H=4, data=tube +// 4C 4C = HT=2, H=3, data=tube +// 50 30 = HT=1, H=4, data=tube +// 4C 0C = HT=off, H=3, data=tube t=0,h=3 +// 34 74 = HT=3, H=5, data=adaptive (5) +// 50 B0 = HT=5, H=4, adaptive +// 30 B0 = HT=3, H=4, data=adaptive (4) +// 30 50 = HT=3, H=4, data=adaptive (4) +// 30 10 = HT=3!,H=4, data=adaptive (4) [adaptive mode appears to ignore HT bits and show HT=3 in details] +// 30 70 = HT=3, H=4, data=adaptive (4) +// 2C 6C = HT=3, H=3, data=adaptive (3) +// 28 08 = H=2, data=adaptive (2), no details (400G) +// 28 48 = HT=3!,H=2, data=adaptive (2) [adaptive mode appears to ignore HT bits and show HT=3 in details] +// 28 68 = HT=3, H=2, data=adaptive (2) +// 24 64 = HT=3, H=1, data=adaptive (1) +// 20 60 = HT=3, H=off, data=adaptive (0) +// 14 74 = HT=3, H=5, data=fixed (5) +// 10 70 = HT=3, H=4, data=fixed (4) +// 0C 6C = HT=3, H=3, data=fixed (3) +// 08 48 = HT=3, H=2, data=fixed (2) +// 08 68 = HT=3, H=2, data=fixed (2) +// 04 64 = HT=3, H=1, data=fixed (1) +// 00 00 = HT=3, H=off, data=fixed (0) + +// F5V3 confirmed: +// 90 70 = HT=3, H=3, adaptive, data=no data +// 54 14 = HT=Off, H=5, adaptive, data=tube t=0,h=5 +// 54 34 = HT=1, H=5, adaptive, data=tube t=1,h=5 +// 50 70 = HT=3, H=4, adaptive, data=tube t=3,h=4 +// 4C 6C = HT=3, H=3, adaptive, data=tube t=3,h=3 +// 4C 4C = HT=2, H=3, adaptive, data=tube t=2,h=3 +// 4C 2C = HT=1, H=3, adaptive, data=tube t=1,h=3 +// 4C 0C = HT=off, H=3, adaptive, data=tube t=0,h=3 +// 48 08 = HT=off, H=2, adaptive, data=tube t=0,h=2 +// 44 04 = HT=off, H=1, adaptive, data=tube t=0,h=1 +// 40 00 = HT=off,H=off, adaptive, data=tube t=0,h=0 +// 34 74 = HT=3, H=5, adaptive, data=s1 (5) +// 30 70 = HT=3, H=4, adaptive, data=s1 (4) +// 2C 6C = HT=3, H=3, adaptive, data=s1 (3) +// 28 68 = HT=3, H=2, adaptive, data=s1 (2) +// 24 64 = HT=3, H=1, adaptive, data=s1 (1) + +// F3V6 confirmed: +// 84 24 = HT=3, H=3, disconnect=adaptive, data=no data +// 50 90 = HT=4, H=4, disconnect=adaptive, data=tube t=4,h=4 +// 44 84 = HT=4, H=1, disconnect=adaptive, data=tube t=4,h=1 +// 40 80 = HT=4, H=Off,disconnect=adaptive, data=tube t=4,h=0 +// 4C 6C = HT=3, H=3, disconnect=adaptive, data=tube t=3,h=3 +// 48 68 = HT=3, H=2, disconnect=adaptive, data=tube t=3,h=2 +// 44 44 = HT=2, H=1, disconnect=adaptive, data=tube t=2,h=1 +// 48 28 = HT=1, H=2, disconnect=adaptive, data=tube t=1,h=2 +// 54 14 = HT=Off,H=5, disconnect=adaptive data=tube t=0,h=5 +// 34 14 = HT=3, H=5, disconnect=adaptive, data=s1 (5) +// 30 70 = HT=3, H=4, disconnect=adaptive, data=s1 (4) +// 2C 6C = HT=3, H=3, disconnect=adaptive, data=s1 (3) +// 28 08 = HT=3, H=2, disconnect=adaptive, data=s1 (2) +// 20 20 = HT=3, H=Off, disconnect=adaptive, data=s1 (0) +// 14 14 = HT=3, H=3, disconnect=fixed, data=classic (5) +// 10 10 = HT=3, H=4, disconnect=fixed, data=classic (4) [fixed mode appears to ignore HT bits and show HT=3 in details] +// 0C 0C = HT=3, H=3, disconnect=fixed, data=classic (3) +// 08 08 = HT=3, H=2, disconnect=fixed, data=classic (2) +// 04 64 = HT=3, H=1, disconnect=fixed, data=classic (1) + +// The data is consistent among all fileVersion 3 models: F0V6, F5V3, F3V6. +// +// NOTE: F5V3 and F3V6 charts report the "Adaptive" setting as "System One" and the "Fixed" +// setting as "Classic", despite labeling the settings "Adaptive" and "Fixed" just like F0V6. +// F0V6 is consistent and labels both settings and chart as "Adaptive" and "Fixed". +// +// 400G and 502G appear to omit the humidifier settings in their details, though they +// do support humidifiers, and will show the humidification in the charts. + +void PRS1DataChunk::ParseHumidifierSettingV3(unsigned char byte1, unsigned char byte2, bool add_setting) +{ + bool humidifier_present = true; + bool humidfixed = false; // formerly called "Classic" + bool humidadaptive = false; // formerly called "System One" + bool tubepresent = false; + bool passover = false; + bool error = false; + + // Byte 1: 0x90 (no humidifier data), 0x50 (15ht, tube 4/5, humid 4), 0x54 (15ht, tube 5, humid 5) 0x4c (15ht, tube temp 3, humidifier 3) + // 0x0c (15, tube 3, humid 3, fixed) + // 0b1001 0000 no humidifier data + // 0b0101 0000 tube 4 and 5, humidifier 4 + // 0b0101 0100 15ht, tube 5, humidifier 5 + // 0b0100 1100 15ht, tube 3, humidifier 3 + // 0b1011 0000 15, tube 3, humidifier 3, "Error" on humidification chart with asterisk at 4 + // 0b0111 0000 15, tube 3, humidifier 3, "Passover" on humidification chart with notch at 4 + // 842 = humidifier status + // 1 84 = humidifier setting + // ?? + CHECK_VALUE(byte1 & 3, 0); + int humid = byte1 >> 5; + switch (humid) { + case 0: humidfixed = true; break; // fixed, ignores tubetemp bits and reports tubetemp=3 + case 1: humidadaptive = true; break; // adaptive, ignores tubetemp bits and reports tubetemp=3 + case 2: tubepresent = true; break; // heated tube + case 3: passover = true; break; // passover mode (only visible in chart) + case 4: humidifier_present = false; break; // no humidifier, reports tubetemp=3 and humidlevel=3 + case 5: error = true; break; // "Error" in humidification chart, reports tubetemp=3 and humidlevel=3 in settings + default: + UNEXPECTED_VALUE(humid, "known value"); + break; + } + int humidlevel = (byte1 >> 2) & 7; + + // Byte 2: 0xB4 (15ht, tube 5, humid 5), 0xB0 (15ht, tube 5, humid 4), 0x90 (tube 4, humid 4), 0x6C (15ht, tube temp 3, humidifier 3) + // 0x80? + // 0b1011 0100 15ht, tube 5, humidifier 5 + // 0b1011 0000 15ht, tube 5, humidifier 4 + // 0b1001 0000 tube 4, humidifier 4 + // 0b0110 1100 15ht, tube 3, humidifier 3 + // 842 = tube temperature + // 1 84 = humidity level when using heated tube, thus far always identical to humidlevel + // ?? + CHECK_VALUE(byte2 & 3, 0); + int tubehumidlevel = (byte2 >> 2) & 7; + CHECK_VALUE(humidlevel, tubehumidlevel); // thus far always the same + int tubetemp = (byte2 >> 5) & 7; + if (humidifier_present) { + if (humidlevel > 5 || humidlevel < 0) UNEXPECTED_VALUE(humidlevel, "0-5"); // 0=off is valid when a humidifier is attached + if (humid == 2) { // heated tube + if (tubetemp > 5 || tubetemp < 0) UNEXPECTED_VALUE(tubetemp, "0-5"); // TODO: maybe this is only if heated tube? 0=off is valid even in heated tube mode + } + } + + // TODO: move this up into the switch statement above, given how many modes there now are. + HumidMode humidmode = HUMID_Fixed; + if (tubepresent) { + humidmode = HUMID_HeatedTube; + } else if (humidadaptive) { + humidmode = HUMID_Adaptive; + } else if (passover) { + humidmode = HUMID_Passover; + } else if (error) { + humidmode = HUMID_Error; + } + + if (add_setting) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_STATUS, humidifier_present)); + if (humidifier_present) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_MODE, humidmode)); + if (humidmode == HUMID_HeatedTube) { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HEATED_TUBE_TEMP, tubetemp)); + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, tubehumidlevel)); + } else { + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HUMID_LEVEL, humidlevel)); + } + } + } + + // Check for previously unseen data that we expect to be normal: + if (family == 0) { + // All variations seen. + } else if (family == 5) { + if (tubepresent) { + // All tube temperature and humidity levels seen. + } else if (humidadaptive) { + // All humidity levels seen. + } else if (humidfixed) { + if (humidlevel < 3) UNEXPECTED_VALUE(humidlevel, "3-5"); + } + } else if (family == 3) { + if (tubepresent) { + // All tube temperature and humidity levels seen. + } else if (humidadaptive) { + // All humidity levels seen. + } else if (humidfixed) { + // All humidity levels seen. + } + } +} + + +void PRS1DataChunk::ParseTubingTypeV3(unsigned char type) +{ + int diam; + switch (type) { + case 0: diam = 22; break; + case 1: diam = 15; break; + case 2: diam = 15; break; // 15HT, though the reports only say "15" for DreamStation models + case 3: diam = 12; break; // seen on DreamStation Go models + default: + UNEXPECTED_VALUE(type, "known tubing type"); + return; + } + this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_HOSE_DIAMETER, diam)); +} diff --git a/oscar/SleepLib/loader_plugins/prs1_parser_asv.cpp b/oscar/SleepLib/loader_plugins/prs1_parser_asv.cpp index e530c851..a5fb6526 100644 --- a/oscar/SleepLib/loader_plugins/prs1_parser_asv.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_parser_asv.cpp @@ -15,6 +15,9 @@ static QString hex(int i) return QString("0x") + QString::number(i, 16).toUpper(); } +//******************************************************************************************** +// MARK: - +// MARK: 50 and 60 Series // borrowed largely from ParseSummaryF0V4 bool PRS1DataChunk::ParseSummaryF5V012(void) @@ -990,6 +993,10 @@ bool PRS1DataChunk::ParseEventsF5V2(void) } +//******************************************************************************************** +// MARK: - +// MARK: DreamStation + // 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_parser_vent.cpp b/oscar/SleepLib/loader_plugins/prs1_parser_vent.cpp index 706f3cbb..a34e6251 100644 --- a/oscar/SleepLib/loader_plugins/prs1_parser_vent.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_parser_vent.cpp @@ -15,6 +15,10 @@ static QString hex(int i) return QString("0x") + QString::number(i, 16).toUpper(); } +//******************************************************************************************** +// MARK: - +// MARK: 50 and 60 Series + // borrowed largely from ParseSummaryF5V012 bool PRS1DataChunk::ParseSummaryF3V03(void) { @@ -595,6 +599,10 @@ bool PRS1DataChunk::ParseEventsF3V03(void) } +//******************************************************************************************** +// MARK: - +// MARK: DreamStation + // Originally based on ParseSummaryF5V3, with changes observed in ventilator sample data // // TODO: surely there will be a way to merge ParseSummary (FV3) loops and abstract the machine-specific diff --git a/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp b/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp index bf64bea0..c5643505 100644 --- a/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_parser_xpap.cpp @@ -10,6 +10,9 @@ #include "prs1_parser.h" #include "prs1_loader.h" +//******************************************************************************************** +// MARK: 50 Series + bool PRS1DataChunk::ParseComplianceF0V23(void) { if (this->family != 0 || (this->familyVersion != 2 && this->familyVersion != 3)) { @@ -643,6 +646,10 @@ bool PRS1DataChunk::ParseEventsF0V23() } +//******************************************************************************************** +// MARK: - +// MARK: 60 Series + bool PRS1DataChunk::ParseComplianceF0V4(void) { if (this->family != 0 || (this->familyVersion != 4)) { @@ -1417,6 +1424,10 @@ bool PRS1DataChunk::ParseComplianceF0V5(void) } +//******************************************************************************************** +// MARK: - +// MARK: DreamStation + // The below is based on fixing the fileVersion == 3 parsing in ParseSummary() based // on our understanding of slices from F0V23. The switch values come from sample files. bool PRS1DataChunk::ParseComplianceF0V6(void)