mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-06 11:10:44 +00:00
Add initial support for PRS1 Dorma 501V.
This commit is contained in:
parent
53de4f0f49
commit
c8b10e31a7
@ -274,6 +274,8 @@ static const PRS1TestedModel s_PRS1TestedModels[] = {
|
|||||||
{ "660P", 0, 4, "BiPAP Pro (System One 60 Series)" },
|
{ "660P", 0, 4, "BiPAP Pro (System One 60 Series)" },
|
||||||
{ "760P", 0, 4, "BiPAP Auto (System One 60 Series)" },
|
{ "760P", 0, 4, "BiPAP Auto (System One 60 Series)" },
|
||||||
|
|
||||||
|
{ "501V", 0, 5, "Dorma 500 Auto (System One 60 Series)" }, // (brick)
|
||||||
|
|
||||||
{ "200X110", 0, 6, "DreamStation CPAP" }, // (brick)
|
{ "200X110", 0, 6, "DreamStation CPAP" }, // (brick)
|
||||||
{ "400G110", 0, 6, "DreamStation Go" },
|
{ "400G110", 0, 6, "DreamStation Go" },
|
||||||
{ "400X110", 0, 6, "DreamStation CPAP Pro" },
|
{ "400X110", 0, 6, "DreamStation CPAP Pro" },
|
||||||
@ -324,7 +326,7 @@ PRS1ModelInfo::PRS1ModelInfo()
|
|||||||
m_modelNames[model.model] = model.name;
|
m_modelNames[model.model] = model.name;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_bricks = { "251P", "261CA", "200X110" };
|
m_bricks = { "251P", "261CA", "200X110", "501V" };
|
||||||
}
|
}
|
||||||
|
|
||||||
bool PRS1ModelInfo::IsSupported(int family, int familyVersion) const
|
bool PRS1ModelInfo::IsSupported(int family, int familyVersion) const
|
||||||
@ -4359,6 +4361,8 @@ bool PRS1DataChunk::ParseCompliance(void)
|
|||||||
return this->ParseComplianceF0V23();
|
return this->ParseComplianceF0V23();
|
||||||
case 4:
|
case 4:
|
||||||
return this->ParseComplianceF0V4();
|
return this->ParseComplianceF0V4();
|
||||||
|
case 5:
|
||||||
|
return this->ParseComplianceF0V5();
|
||||||
case 6:
|
case 6:
|
||||||
return this->ParseComplianceF0V6();
|
return this->ParseComplianceF0V6();
|
||||||
}
|
}
|
||||||
@ -4681,8 +4685,12 @@ bool PRS1DataChunk::ParseSettingsF0V23(const unsigned char* data, int /*size*/)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool PRS1DataChunk::ParseSettingsF0V4(const unsigned char* data, int /*size*/)
|
bool PRS1DataChunk::ParseSettingsF0V45(const unsigned char* data, int size)
|
||||||
{
|
{
|
||||||
|
if (size < 0xd) {
|
||||||
|
qWarning() << "invalid size passed to ParseSettingsF0V45";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN;
|
PRS1Mode cpapmode = PRS1_MODE_UNKNOWN;
|
||||||
|
|
||||||
switch (data[0x02]) { // PRS1 mode
|
switch (data[0x02]) { // PRS1 mode
|
||||||
@ -4764,10 +4772,19 @@ bool PRS1DataChunk::ParseSettingsF0V4(const unsigned char* data, int /*size*/)
|
|||||||
}
|
}
|
||||||
|
|
||||||
quint8 flex = data[0x0a];
|
quint8 flex = data[0x0a];
|
||||||
|
if (this->familyVersion == 5) CHECK_VALUE(flex, 0xE1);
|
||||||
this->ParseFlexSettingF0V234(flex, cpapmode);
|
this->ParseFlexSettingF0V234(flex, cpapmode);
|
||||||
|
|
||||||
|
if (this->familyVersion == 5) {
|
||||||
|
CHECK_VALUES(data[0x0b], 0x00, 0x02);
|
||||||
|
CHECK_VALUE(data[0x0c], 0x60);
|
||||||
|
}
|
||||||
this->ParseHumidifierSetting60Series(data[0x0b], data[0x0c], true);
|
this->ParseHumidifierSetting60Series(data[0x0b], data[0x0c], true);
|
||||||
|
|
||||||
|
if (size <= 0xd) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
int resist_level = (data[0x0d] >> 3) & 7; // 0x18 resist=3, 0x11 resist=2, 0x28 resist=5
|
int resist_level = (data[0x0d] >> 3) & 7; // 0x18 resist=3, 0x11 resist=2, 0x28 resist=5
|
||||||
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_LOCK, (data[0x0d] & 0x40) != 0));
|
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_LOCK, (data[0x0d] & 0x40) != 0));
|
||||||
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_SETTING, resist_level));
|
this->AddEvent(new PRS1ParsedSettingEvent(PRS1_SETTING_MASK_RESIST_SETTING, resist_level));
|
||||||
@ -4831,6 +4848,10 @@ bool PRS1DataChunk::ParseSettingsF0V4(const unsigned char* data, int /*size*/)
|
|||||||
// 93 11 = H=3, S1, no data
|
// 93 11 = H=3, S1, no data
|
||||||
// 04 30 = H=3, S1, no data
|
// 04 30 = H=3, S1, no data
|
||||||
|
|
||||||
|
// F0V5 confirmed:
|
||||||
|
// 00 60 = H=Off, Classic
|
||||||
|
// 02 60 = H=2, Classic
|
||||||
|
|
||||||
// F5V1 confirmed:
|
// F5V1 confirmed:
|
||||||
// A0 4A = HT=5, H=2, HT
|
// A0 4A = HT=5, H=2, HT
|
||||||
// B1 09 = HT=3, H=3, HT
|
// B1 09 = HT=3, H=3, HT
|
||||||
@ -4964,6 +4985,103 @@ void PRS1DataChunk::ParseHumidifierSetting60Series(unsigned char humid1, unsigne
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Based on ParseComplianceF0V4, but this has shorter settings and stats following equipment off.
|
||||||
|
bool PRS1DataChunk::ParseComplianceF0V5(void)
|
||||||
|
{
|
||||||
|
if (this->family != 0 || (this->familyVersion != 5)) {
|
||||||
|
qWarning() << "ParseComplianceF0V5 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[] = { 0xf, 7, 4, 0xf };
|
||||||
|
static const int ncodes = sizeof(minimum_sizes) / sizeof(int);
|
||||||
|
// NOTE: These are fixed sizes, but are called minimum to more closely match the F0V6 parser.
|
||||||
|
|
||||||
|
bool ok = true;
|
||||||
|
int pos = 0;
|
||||||
|
int code, size;
|
||||||
|
int tt = 0;
|
||||||
|
while (ok && pos < chunk_size) {
|
||||||
|
code = data[pos++];
|
||||||
|
// There is no hblock prior to F0V6.
|
||||||
|
size = 0;
|
||||||
|
if (code < ncodes) {
|
||||||
|
// make sure the handlers below don't go past the end of the buffer
|
||||||
|
size = minimum_sizes[code];
|
||||||
|
} // 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_VALUE(data[pos], 0x31);
|
||||||
|
// F0V5 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
|
||||||
|
ok = ParseSettingsF0V45(data, 0x0d);
|
||||||
|
CHECK_VALUE(data[pos+0xd], 0);
|
||||||
|
CHECK_VALUE(data[pos+0xe], 0);
|
||||||
|
CHECK_VALUE(data[pos+0xf], 0);
|
||||||
|
break;
|
||||||
|
case 2: // Mask On
|
||||||
|
tt += data[pos] | (data[pos+1] << 8);
|
||||||
|
this->AddEvent(new PRS1ParsedSliceEvent(tt, MaskOn));
|
||||||
|
CHECK_VALUES(data[pos+2], 0, 2);
|
||||||
|
CHECK_VALUE(data[pos+3], 0x60);
|
||||||
|
this->ParseHumidifierSetting60Series(data[pos+2], data[pos+3]);
|
||||||
|
break;
|
||||||
|
case 3: // Mask Off
|
||||||
|
tt += data[pos] | (data[pos+1] << 8);
|
||||||
|
this->AddEvent(new PRS1ParsedSliceEvent(tt, MaskOff));
|
||||||
|
// F0V5 compliance has MaskOff stats unlike all other compliance.
|
||||||
|
// This is presumably because the 501V is an Auto-CPAP, so it needs to record titration data.
|
||||||
|
// 51s, 13940s, 13348s
|
||||||
|
CHECK_VALUES(data[pos+2], 40, 50); // min pressure
|
||||||
|
CHECK_VALUES(data[pos+3], 40, 150); // maybe max pressure?
|
||||||
|
//CHECK_VALUES(data[pos+4], 40, 150); // Average Device Pressure <= 90% of Time (report is time-weighted per slice)
|
||||||
|
//CHECK_VALUES(data[pos+5], 40, 108); // Auto CPAP Mean Pressure (report is time-weighted per slice)
|
||||||
|
// (not sure how "Peak Average Pressure" on report differs, maybe that's over all sessions or days?)
|
||||||
|
//CHECK_VALUES(data[pos+6], 0, 5); // Apnea or Hypopnea count (probably 16-bit), contributes to AHI
|
||||||
|
CHECK_VALUE(data[pos+7], 0);
|
||||||
|
//CHECK_VALUES(data[pos+8], 0, 6); // Apnea or Hypopnea count (probably 16-bit), contributes to AHI
|
||||||
|
CHECK_VALUE(data[pos+9], 0);
|
||||||
|
CHECK_VALUES(data[pos+10], 0, 2); // Average Large Leak minutes? (Maybe average only if there's more than one day?)
|
||||||
|
CHECK_VALUE(data[pos+11], 0);
|
||||||
|
//CHECK_VALUES(data[pos+12], 179, 50); // Average 90% Leak (report is time-weighted per slice)
|
||||||
|
//CHECK_VALUES(data[pos+13], 178, 32); // Average Total Leak (report is time-weighted per slice)
|
||||||
|
//CHECK_VALUES(data[pos+14], 180, 36); // Max leak (report shows max for all slices)
|
||||||
|
break;
|
||||||
|
case 1: // Equipment Off
|
||||||
|
tt += data[pos] | (data[pos+1] << 8);
|
||||||
|
this->AddEvent(new PRS1ParsedSliceEvent(tt, EquipmentOff));
|
||||||
|
CHECK_VALUE(data[pos+2], 1);
|
||||||
|
CHECK_VALUES(data[pos+3], 0x16, 0x13); // 22, 19
|
||||||
|
CHECK_VALUE(data[pos+4], 0);
|
||||||
|
CHECK_VALUES(data[pos+5], 0x2F, 0x26); // 47, 38
|
||||||
|
CHECK_VALUE(data[pos+6], 1);
|
||||||
|
break;
|
||||||
|
/* See ParseComplianceF0V4 if we encounter slices 4-8 */
|
||||||
|
default:
|
||||||
|
UNEXPECTED_VALUE(code, "known slice code");
|
||||||
|
ok = false; // unlike F0V6, we don't know the size of unknown slices, so we can't recover
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
pos += size;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ok && pos != chunk_size) {
|
||||||
|
qWarning() << this->sessionid << (this->size() - pos) << "trailing bytes";
|
||||||
|
}
|
||||||
|
|
||||||
|
this->duration = tt;
|
||||||
|
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
bool PRS1DataChunk::ParseComplianceF0V4(void)
|
bool PRS1DataChunk::ParseComplianceF0V4(void)
|
||||||
{
|
{
|
||||||
if (this->family != 0 || (this->familyVersion != 4)) {
|
if (this->family != 0 || (this->familyVersion != 4)) {
|
||||||
@ -4999,7 +5117,7 @@ bool PRS1DataChunk::ParseComplianceF0V4(void)
|
|||||||
CHECK_VALUE(pos, 1); // Always first
|
CHECK_VALUE(pos, 1); // Always first
|
||||||
CHECK_VALUES(data[pos], 1, 3);
|
CHECK_VALUES(data[pos], 1, 3);
|
||||||
// F0V4 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
|
// F0V4 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
|
||||||
ok = ParseSettingsF0V4(data, 0x0f);
|
ok = ParseSettingsF0V45(data, 0x11);
|
||||||
CHECK_VALUE(data[pos+0x11], 0);
|
CHECK_VALUE(data[pos+0x11], 0);
|
||||||
CHECK_VALUE(data[pos+0x12], 0);
|
CHECK_VALUE(data[pos+0x12], 0);
|
||||||
CHECK_VALUE(data[pos+0x13], 0);
|
CHECK_VALUE(data[pos+0x13], 0);
|
||||||
@ -5152,7 +5270,7 @@ bool PRS1DataChunk::ParseSummaryF0V4(void)
|
|||||||
//CHECK_VALUES(data[pos] & 0x0F, 3, 5); // TODO: what are these? 0 seems to be related to errors.
|
//CHECK_VALUES(data[pos] & 0x0F, 3, 5); // TODO: what are these? 0 seems to be related to errors.
|
||||||
}
|
}
|
||||||
// F0V4 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
|
// F0V4 doesn't have a separate settings record like F0V6 does, the settings just follow the EquipmentOn data.
|
||||||
ok = ParseSettingsF0V4(data, 0x0f);
|
ok = ParseSettingsF0V45(data, 0x11);
|
||||||
CHECK_VALUE(data[pos+0x11], 0);
|
CHECK_VALUE(data[pos+0x11], 0);
|
||||||
CHECK_VALUE(data[pos+0x12], 0);
|
CHECK_VALUE(data[pos+0x12], 0);
|
||||||
CHECK_VALUE(data[pos+0x13], 0);
|
CHECK_VALUE(data[pos+0x13], 0);
|
||||||
@ -6490,7 +6608,11 @@ bool PRS1DataChunk::ParseSummaryF5V012(void)
|
|||||||
// 0x8B = C-Flex+ 3 (CPAP mode)
|
// 0x8B = C-Flex+ 3 (CPAP mode)
|
||||||
// 0x8B = A-Flex 3 (AutoCPAP mode)
|
// 0x8B = A-Flex 3 (AutoCPAP mode)
|
||||||
|
|
||||||
|
// Flex F0V5 confirmed
|
||||||
|
// 0xE1 = Flex (AutoCPAP mode)
|
||||||
|
|
||||||
// 8 = enabled
|
// 8 = enabled
|
||||||
|
// 4 = lock
|
||||||
// 1 = rise time
|
// 1 = rise time
|
||||||
// 8 = C-Flex+ / A-Flex (depending on mode)
|
// 8 = C-Flex+ / A-Flex (depending on mode)
|
||||||
// 3 = level
|
// 3 = level
|
||||||
|
@ -141,6 +141,9 @@ public:
|
|||||||
//! \brief Parse a single data chunk from a .000 file containing compliance data for a P256x brick
|
//! \brief Parse a single data chunk from a .000 file containing compliance data for a P256x brick
|
||||||
bool ParseComplianceF0V4(void);
|
bool ParseComplianceF0V4(void);
|
||||||
|
|
||||||
|
//! \brief Parse a single data chunk from a .000 file containing compliance data for a x00V brick
|
||||||
|
bool ParseComplianceF0V5(void);
|
||||||
|
|
||||||
//! \brief Parse a single data chunk from a .000 file containing compliance data for a DreamStation 200X brick
|
//! \brief Parse a single data chunk from a .000 file containing compliance data for a DreamStation 200X brick
|
||||||
bool ParseComplianceF0V6(void);
|
bool ParseComplianceF0V6(void);
|
||||||
|
|
||||||
@ -168,7 +171,7 @@ public:
|
|||||||
//! \brief Parse a single data chunk from a .001 file containing summary data for a family 5 ASV family version 3 machine
|
//! \brief Parse a single data chunk from a .001 file containing summary data for a family 5 ASV family version 3 machine
|
||||||
bool ParseSummaryF5V3(void);
|
bool ParseSummaryF5V3(void);
|
||||||
|
|
||||||
//! \brief Parse a flex setting byte from a .000 or .001 containing compliance/summary data for CPAP/APAP family versions 2, 3, or 4
|
//! \brief Parse a flex setting byte from a .000 or .001 containing compliance/summary data for CPAP/APAP family versions 2, 3, 4, or 5
|
||||||
void ParseFlexSettingF0V234(quint8 flex, int prs1mode);
|
void ParseFlexSettingF0V234(quint8 flex, int prs1mode);
|
||||||
|
|
||||||
//! \brief Parse a flex setting byte from a .000 or .001 containing compliance/summary data for ASV family versions 0, 1, or 2
|
//! \brief Parse a flex setting byte from a .000 or .001 containing compliance/summary data for ASV family versions 0, 1, or 2
|
||||||
@ -241,7 +244,7 @@ protected:
|
|||||||
bool ParseSettingsF0V23(const unsigned char* data, int size);
|
bool ParseSettingsF0V23(const unsigned char* data, int size);
|
||||||
|
|
||||||
//! \brief Parse a settings slice from a .000 and .001 file
|
//! \brief Parse a settings slice from a .000 and .001 file
|
||||||
bool ParseSettingsF0V4(const unsigned char* data, int size);
|
bool ParseSettingsF0V45(const unsigned char* data, int size);
|
||||||
|
|
||||||
//! \brief Parse a settings slice from a .000 and .001 file
|
//! \brief Parse a settings slice from a .000 and .001 file
|
||||||
bool ParseSettingsF0V6(const unsigned char* data, int size);
|
bool ParseSettingsF0V6(const unsigned char* data, int size);
|
||||||
|
Loading…
Reference in New Issue
Block a user