Chase down PRS1 unordered time warnings and add an explanatory TODO comment.

Also fix some whitespace.

No functional changes.
This commit is contained in:
sawinglogz 2019-11-19 16:00:08 -05:00
parent 60ec8711dc
commit 46c9d4f0c8
2 changed files with 111 additions and 96 deletions

View File

@ -63,6 +63,12 @@ EventDataType EventList::data2(quint32 i)
return EventDataType(m_data2[i]); return EventDataType(m_data2[i]);
} }
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);
}
void EventList::AddEvent(qint64 time, EventStoreType data) void EventList::AddEvent(qint64 time, EventStoreType data)
{ {
// Apply gain & offset // Apply gain & offset
@ -85,7 +91,7 @@ void EventList::AddEvent(qint64 time, EventStoreType data)
if (m_first > time) { if (m_first > time) {
// Crud.. Update all the previous records // Crud.. Update all the previous records
// This really shouldn't happen. // This really shouldn't happen.
qDebug() << "Unordered time detected in AddEvent()" << m_count << m_first << time << data; qDebug() << "Unordered time detected in AddEvent()" << m_count << ts(m_first) << ts(time) << data;
qint32 delta = (m_first - time); qint32 delta = (m_first - time);

View File

@ -2889,109 +2889,118 @@ bool PRS1Import::ImportEventChunk(PRS1DataChunk* event)
void PRS1Import::ImportEvent(qint64 t, PRS1ParsedEvent* e) void PRS1Import::ImportEvent(qint64 t, PRS1ParsedEvent* e)
{ {
qint64 duration; qint64 duration;
// TODO: Filter out duplicate/overlapping PB and RE events.
//
// These actually get reported by the machines, but they cause "unordered time" warnings
// and they throw off the session statistics. Even official reports show the wrong stats,
// for example counting each of 3 duplicate PBs towards the total time in PB.
//
// It's not clear whether filtering can reasonably be done here or whether it needs
// to be done in ImportEventChunk.
const QVector<ChannelID*> & channels = PRS1ImportChannelMap[e->m_type]; const QVector<ChannelID*> & channels = PRS1ImportChannelMap[e->m_type];
ChannelID channel = NoChannel, PS, VS2, LEAK; ChannelID channel = NoChannel, PS, VS2, LEAK;
if (channels.count() > 0) { if (channels.count() > 0) {
channel = *channels.at(0); channel = *channels.at(0);
} }
switch (e->m_type) { switch (e->m_type) {
case PRS1PressureSetEvent::TYPE: // currentPressure is used to calculate unintentional leak, not just PS case PRS1PressureSetEvent::TYPE: // currentPressure is used to calculate unintentional leak, not just PS
case PRS1IPAPSetEvent::TYPE: case PRS1IPAPSetEvent::TYPE:
case PRS1IPAPAverageEvent::TYPE: case PRS1IPAPAverageEvent::TYPE:
AddEvent(channel, t, e->m_value, e->m_gain); AddEvent(channel, t, e->m_value, e->m_gain);
m_currentPressure = e->m_value; m_currentPressure = e->m_value;
break; break;
case PRS1EPAPSetEvent::TYPE: case PRS1EPAPSetEvent::TYPE:
AddEvent(channel, t, e->m_value, e->m_gain); AddEvent(channel, t, e->m_value, e->m_gain);
if (m_calcPSfromSet) { if (m_calcPSfromSet) {
PS = *(PRS1ImportChannelMap[PRS1IPAPSetEvent::TYPE].at(1)); PS = *(PRS1ImportChannelMap[PRS1IPAPSetEvent::TYPE].at(1));
AddEvent(PS, t, m_currentPressure - e->m_value, e->m_gain); // Pressure Support
}
break;
case PRS1EPAPAverageEvent::TYPE:
PS = *channels.at(1);
AddEvent(channel, t, e->m_value, e->m_gain);
AddEvent(PS, t, m_currentPressure - e->m_value, e->m_gain); // Pressure Support AddEvent(PS, t, m_currentPressure - e->m_value, e->m_gain); // Pressure Support
break; }
break;
case PRS1EPAPAverageEvent::TYPE:
PS = *channels.at(1);
AddEvent(channel, t, e->m_value, e->m_gain);
AddEvent(PS, t, m_currentPressure - e->m_value, e->m_gain); // Pressure Support
break;
case PRS1TimedBreathEvent::TYPE: case PRS1TimedBreathEvent::TYPE:
// The duration appears to correspond to the length of the timed breath in seconds when multiplied by 0.1 (100ms)! // The duration appears to correspond to the length of the timed breath in seconds when multiplied by 0.1 (100ms)!
// TODO: consider changing parsers to use milliseconds for time, since it turns out there's at least one way // TODO: consider changing parsers to use milliseconds for time, since it turns out there's at least one way
// they can express durations less than 1 second. // they can express durations less than 1 second.
// TODO: consider allowing OSCAR to record millisecond durations so that the display will say "2.1" instead of "21" or "2". // TODO: consider allowing OSCAR to record millisecond durations so that the display will say "2.1" instead of "21" or "2".
duration = e->m_duration * 100L; // for now do this here rather than in parser, since parser events don't use milliseconds duration = e->m_duration * 100L; // for now do this here rather than in parser, since parser events don't use milliseconds
AddEvent(*channels.at(0), t - duration, e->m_duration * 0.1F, 0.1F); // TODO: a gain of 0.1 should render this unnecessary, but gain doesn't seem to work currently AddEvent(*channels.at(0), t - duration, e->m_duration * 0.1F, 0.1F); // TODO: a gain of 0.1 should render this unnecessary, but gain doesn't seem to work currently
break; break;
case PRS1ObstructiveApneaEvent::TYPE: case PRS1ObstructiveApneaEvent::TYPE:
case PRS1ClearAirwayEvent::TYPE: case PRS1ClearAirwayEvent::TYPE:
case PRS1HypopneaEvent::TYPE: case PRS1HypopneaEvent::TYPE:
case PRS1FlowLimitationEvent::TYPE: case PRS1FlowLimitationEvent::TYPE:
AddEvent(channel, t, e->m_duration, e->m_gain); AddEvent(channel, t, e->m_duration, e->m_gain);
break; break;
case PRS1PeriodicBreathingEvent::TYPE: case PRS1PeriodicBreathingEvent::TYPE:
case PRS1LargeLeakEvent::TYPE: case PRS1LargeLeakEvent::TYPE:
case PRS1UnknownDurationEvent::TYPE: case PRS1UnknownDurationEvent::TYPE:
// TODO: The graphs silently treat the timestamp of a span as an end time rather than start (see gFlagsLine::paint). // TODO: The graphs silently treat the timestamp of a span as an end time rather than start (see gFlagsLine::paint).
// Decide whether to preserve that behavior or change it universally and update either this code or comment. // Decide whether to preserve that behavior or change it universally and update either this code or comment.
duration = e->m_duration * 1000L; duration = e->m_duration * 1000L;
AddEvent(channel, t + duration, e->m_duration, e->m_gain); AddEvent(channel, t + duration, e->m_duration, e->m_gain);
break; break;
case PRS1TotalLeakEvent::TYPE: case PRS1TotalLeakEvent::TYPE:
AddEvent(channel, t, e->m_value, e->m_gain);
// F0 up through F0V6 doesn't appear to report unintentional leak.
// TODO: decide whether to keep this here, shouldn't keep it here just because it's "quicker".
// TODO: compare this value for the reported value for F5V1 and higher?
// TODO: Fix this for 0.125 gain: it assumes 0.1 (dividing by 10.0)...
// Or omit, because machines with 0.125 gain report unintentional leak directly.
if (m_calcLeaks) { // Much Quicker doing this here than the recalc method.
EventDataType leak = e->m_value;
leak -= (((m_currentPressure/10.0f) - 4.0) * m_ppm + m_lpm4);
if (leak < 0) leak = 0;
LEAK = *channels.at(1);
AddEvent(LEAK, t, leak, 1);
}
break;
case PRS1SnoreEvent::TYPE: // snore count that shows up in flags but not waveform
// TODO: The numeric snore graph is the right way to present this information,
// but it needs to be shifted left 2 minutes, since it's not a starting value
// but a past statistic.
AddEvent(channel, t, e->m_value, e->m_gain); // Snore count
if (e->m_value > 0) {
// TODO: currently these get drawn on our waveforms, but they probably shouldn't,
// since they don't have a precise timestamp. They should continue to be drawn
// on the flags overview. See the comment in ImportEventChunk regarding flags
// for numeric channels.
VS2 = *channels.at(1);
AddEvent(VS2, t, 0, 1);
}
break;
case PRS1VibratorySnoreEvent::TYPE: // real VS marker on waveform
// TODO: These don't need to be drawn separately on the flag overview, since
// they're presumably included in the overall snore count statistic. They should
// continue to be drawn on the waveform, due to their precise timestamp.
AddEvent(channel, t, e->m_value, e->m_gain);
break;
default:
if (channels.count() == 1) {
// For most events, simply pass the value through to the mapped channel.
AddEvent(channel, t, e->m_value, e->m_gain); AddEvent(channel, t, e->m_value, e->m_gain);
// F0 up through F0V6 doesn't appear to report unintentional leak. } else if (channels.count() > 1) {
// TODO: decide whether to keep this here, shouldn't keep it here just because it's "quicker". // Anything mapped to more than one channel must have a case statement above.
// TODO: compare this value for the reported value for F5V1 and higher? qWarning() << "Missing import handler for PRS1 event type" << (int) e->m_type;
// TODO: Fix this for 0.125 gain: it assumes 0.1 (dividing by 10.0)...
// Or omit, because machines with 0.125 gain report unintentional leak directly.
if (m_calcLeaks) { // Much Quicker doing this here than the recalc method.
EventDataType leak = e->m_value;
leak -= (((m_currentPressure/10.0f) - 4.0) * m_ppm + m_lpm4);
if (leak < 0) leak = 0;
LEAK = *channels.at(1);
AddEvent(LEAK, t, leak, 1);
}
break; break;
} else {
case PRS1SnoreEvent::TYPE: // snore count that shows up in flags but not waveform // Not imported, no channels mapped to this event
// TODO: The numeric snore graph is the right way to present this information, // These will show up in chunk YAML and any user alerts will be driven by the parser.
// but it needs to be shifted left 2 minutes, since it's not a starting value }
// but a past statistic. break;
AddEvent(channel, t, e->m_value, e->m_gain); // Snore count }
if (e->m_value > 0) {
// TODO: currently these get drawn on our waveforms, but they probably shouldn't,
// since they don't have a precise timestamp. They should continue to be drawn
// on the flags overview. See the comment in ImportEventChunk regarding flags
// for numeric channels.
VS2 = *channels.at(1);
AddEvent(VS2, t, 0, 1);
}
break;
case PRS1VibratorySnoreEvent::TYPE: // real VS marker on waveform
// TODO: These don't need to be drawn separately on the flag overview, since
// they're presumably included in the overall snore count statistic. They should
// continue to be drawn on the waveform, due to their precise timestamp.
AddEvent(channel, t, e->m_value, e->m_gain);
break;
default:
if (channels.count() == 1) {
// For most events, simply pass the value through to the mapped channel.
AddEvent(channel, t, e->m_value, e->m_gain);
} else if (channels.count() > 1) {
// Anything mapped to more than one channel must have a case statement above.
qWarning() << "Missing import handler for PRS1 event type" << (int) e->m_type;
break;
} else {
// Not imported, no channels mapped to this event
// These will show up in chunk YAML and any user alerts will be driven by the parser.
}
break;
}
} }