Split PRS1Import::ImportEvent from ImportEventChunk.

This will let us create "duplicate" statistical events at the end of
a session (and eventually slice) to reflect the end time of each
measurement channel.
This commit is contained in:
sawinglogz 2019-11-07 14:19:58 -05:00
parent db1df4eeef
commit ff630b07a6
2 changed files with 46 additions and 31 deletions

View File

@ -2625,26 +2625,25 @@ bool PRS1Import::AddEvent(ChannelID channel, qint64 t, float value, float gain)
bool PRS1Import::ImportEventChunk(PRS1DataChunk* event) bool PRS1Import::ImportEventChunk(PRS1DataChunk* event)
{ {
EventDataType currentPressure=0; m_currentPressure=0;
const QVector<PRS1ParsedEventType> & supported = GetSupportedEvents(event); const QVector<PRS1ParsedEventType> & supported = GetSupportedEvents(event);
// Calculate PS from IPAP/EPAP set events only when both are supported. This includes F0, but excludes // Calculate PS from IPAP/EPAP set events only when both are supported. This includes F0, but excludes
// F5, which only reports EPAP set events, but both IPAP/EPAP average, from which PS will be calculated. // F5, which only reports EPAP set events, but both IPAP/EPAP average, from which PS will be calculated.
bool calcPSfromSet = supported.contains(PRS1IPAPSetEvent::TYPE) && supported.contains(PRS1EPAPSetEvent::TYPE); m_calcPSfromSet = supported.contains(PRS1IPAPSetEvent::TYPE) && supported.contains(PRS1EPAPSetEvent::TYPE);
// Unintentional leak calculation, see zMaskProfile:calcLeak in calcs.cpp for explanation // Unintentional leak calculation, see zMaskProfile:calcLeak in calcs.cpp for explanation
bool calcLeaks = p_profile->cpap->calculateUnintentionalLeaks(); m_calcLeaks = p_profile->cpap->calculateUnintentionalLeaks();
if (calcLeaks) { if (m_calcLeaks) {
// Only needed for machines that don't support it directly. // Only needed for machines that don't support it directly.
calcLeaks = (supported.contains(PRS1LeakEvent::TYPE) == false); m_calcLeaks = (supported.contains(PRS1LeakEvent::TYPE) == false);
} }
EventDataType lpm4 = p_profile->cpap->custom4cmH2OLeaks(); m_lpm4 = p_profile->cpap->custom4cmH2OLeaks();
EventDataType lpm20 = p_profile->cpap->custom20cmH2OLeaks(); EventDataType lpm20 = p_profile->cpap->custom20cmH2OLeaks();
EventDataType lpm = lpm20 - lpm4; EventDataType lpm = lpm20 - m_lpm4;
EventDataType ppm = lpm / 16.0; m_ppm = lpm / 16.0;
qint64 duration;
qint64 t = qint64(event->timestamp) * 1000L; qint64 t = qint64(event->timestamp) * 1000L;
session->updateFirst(t); session->updateFirst(t);
@ -2696,6 +2695,30 @@ bool PRS1Import::ImportEventChunk(PRS1DataChunk* event)
break; break;
} }
ImportEvent(t, e);
}
if (!ok) {
return false;
}
// TODO: This needs to be special-cased for F3V3 due to its weird interval-based event format
// until there's a way for its parser to correctly set the timestamps for truncated
// intervals in sessions that don't end on a 2-minute boundary.
if (!(event->family == 3 && event->familyVersion == 3)) {
// If the last event has a non-zero duration, t will not reflect the full duration of the chunk, so update it.
t = qint64(event->timestamp + event->duration) * 1000L;
session->updateLast(t);
}
return true;
}
void PRS1Import::ImportEvent(qint64 t, PRS1ParsedEvent* e)
{
qint64 duration;
QVector<ChannelID*> channels = PRS1ImportChannelMap[e->m_type]; 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) {
@ -2707,19 +2730,19 @@ bool PRS1Import::ImportEventChunk(PRS1DataChunk* event)
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);
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 (calcPSfromSet) { if (m_calcPSfromSet) {
PS = *(PRS1ImportChannelMap[PRS1IPAPSetEvent::TYPE].at(1)); PS = *(PRS1ImportChannelMap[PRS1IPAPSetEvent::TYPE].at(1));
AddEvent(PS, t, 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: case PRS1EPAPAverageEvent::TYPE:
PS = *channels.at(1); PS = *channels.at(1);
AddEvent(channel, t, e->m_value, e->m_gain); AddEvent(channel, t, e->m_value, e->m_gain);
AddEvent(PS, t, 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 PRS1TimedBreathEvent::TYPE: case PRS1TimedBreathEvent::TYPE:
@ -2754,9 +2777,9 @@ bool PRS1Import::ImportEventChunk(PRS1DataChunk* event)
// TODO: compare this value for the reported value for F5V1 and higher? // 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)... // 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. // Or omit, because machines with 0.125 gain report unintentional leak directly.
if (calcLeaks) { // Much Quicker doing this here than the recalc method. if (m_calcLeaks) { // Much Quicker doing this here than the recalc method.
EventDataType leak = e->m_value; EventDataType leak = e->m_value;
leak -= (((currentPressure/10.0f) - 4.0) * ppm + lpm4); leak -= (((m_currentPressure/10.0f) - 4.0) * m_ppm + m_lpm4);
if (leak < 0) leak = 0; if (leak < 0) leak = 0;
LEAK = *channels.at(1); LEAK = *channels.at(1);
AddEvent(LEAK, t, leak, 1); AddEvent(LEAK, t, leak, 1);
@ -2797,22 +2820,6 @@ bool PRS1Import::ImportEventChunk(PRS1DataChunk* event)
} }
break; break;
} }
}
if (!ok) {
return false;
}
// TODO: This needs to be special-cased for F3V3 due to its weird interval-based event format
// until there's a way for its parser to correctly set the timestamps for truncated
// intervals in sessions that don't end on a 2-minute boundary.
if (!(event->family == 3 && event->familyVersion == 3)) {
// If the last event has a non-zero duration, t will not reflect the full duration of the chunk, so update it.
t = qint64(event->timestamp + event->duration) * 1000L;
session->updateLast(t);
}
return true;
} }

View File

@ -327,6 +327,14 @@ protected:
//! \brief Save parsed session data to the database //! \brief Save parsed session data to the database
void SaveSessionToDatabase(void); void SaveSessionToDatabase(void);
//! \brief Import a single event from a data chunk.
void ImportEvent(qint64 t, PRS1ParsedEvent* event);
// State that needs to persist between individual events:
EventDataType m_currentPressure;
bool m_calcPSfromSet;
bool m_calcLeaks;
EventDataType m_lpm4, m_ppm;
//! \brief Import a single data chunk from a .002 file containing event data. //! \brief Import a single data chunk from a .002 file containing event data.
bool ImportEventChunk(PRS1DataChunk* event); bool ImportEventChunk(PRS1DataChunk* event);
//! \brief Create all supported channels (except for on-demand ones that only get created if an event appears). //! \brief Create all supported channels (except for on-demand ones that only get created if an event appears).