Add infrastructure to allow import of parsed PRS1 events to be unified.

This commit adds a mapping from PRS1 events to the OSCAR channels that
should receive the data, along with the necessary supporting code to
allow for generic handling of PRS1 events as a result.

The F3V6 importer has minimal changes to partially use the new
plumbing, enough to make sure that it works and causes no change in
the imported data.
This commit is contained in:
sawinglogz 2019-10-20 17:46:18 -04:00
parent fd9c73f79f
commit ada3bbe891
2 changed files with 159 additions and 24 deletions

View File

@ -1496,8 +1496,71 @@ enum PRS1Mode {
PRS1_MODE_AUTOTRIAL, // "Auto-Trial" PRS1_MODE_AUTOTRIAL, // "Auto-Trial"
}; };
// Returns the set of all channels ever reported/supported by the parser for the given chunk.
const QVector<PRS1ParsedEventType> & GetSupportedEvents(const PRS1DataChunk* chunk); const QVector<PRS1ParsedEventType> & GetSupportedEvents(const PRS1DataChunk* chunk);
// The set of PRS1 "on-demand" channels that only get created on import if the session
// contains events of that type. Any channels not on this list always get created if
// they're reported/supported by the parser.
static const QVector<PRS1ParsedEventType> PRS1OnDemandChannels =
{
//PRS1TimedBreathEvent::TYPE, // TODO: TB could be on-demand
PRS1PressurePulseEvent::TYPE,
// Pressure initialized on-demand for F0 due to the possibility of bilevel vs. single pressure.
PRS1PressureSetEvent::TYPE,
PRS1IPAPSetEvent::TYPE,
PRS1EPAPSetEvent::TYPE,
};
// The channel ID (referenced by pointer because their values aren't initialized
// prior to runtime) to which a given PRS1 event should be added. Events with
// no channel IDs are silently dropped, and events with more than one channel ID
// must be handled specially.
static const QHash<PRS1ParsedEventType,QVector<ChannelID*>> PRS1ImportChannelMap =
{
{ PRS1ClearAirwayEvent::TYPE, { &CPAP_ClearAirway } },
{ PRS1ObstructiveApneaEvent::TYPE, { &CPAP_Obstructive } },
{ PRS1HypopneaEvent::TYPE, { &CPAP_Hypopnea } },
{ PRS1FlowLimitationEvent::TYPE, { &CPAP_FlowLimit } },
{ PRS1SnoreEvent::TYPE, { &CPAP_Snore, &CPAP_VSnore2 } }, // VSnore2 is calculated from snore count, used to flag nonzero intervals on overview
{ PRS1VibratorySnoreEvent::TYPE, { &CPAP_VSnore } },
{ PRS1RERAEvent::TYPE, { &CPAP_RERA } },
{ PRS1PeriodicBreathingEvent::TYPE, { &CPAP_PB } },
{ PRS1LargeLeakEvent::TYPE, { &CPAP_LargeLeak } },
{ PRS1TotalLeakEvent::TYPE, { &CPAP_LeakTotal } }, // TODO: should Leak be listed here since it's sometimes calculated from TotalLeak?
{ PRS1LeakEvent::TYPE, { &CPAP_Leak } },
{ PRS1RespiratoryRateEvent::TYPE, { &CPAP_RespRate } },
{ PRS1TidalVolumeEvent::TYPE, { &CPAP_TidalVolume } },
{ PRS1MinuteVentilationEvent::TYPE, { &CPAP_MinuteVent } },
{ PRS1PatientTriggeredBreathsEvent::TYPE, { &CPAP_PTB } },
{ PRS1TimedBreathEvent::TYPE, { &PRS1_TimedBreath } },
// TODO: Distinguish between pressure-set and average-pressure events
// F0 imports pressure-set events and ignores pressure average events.
// F5 imports average events and ignores EPAP set events.
// F3 imports average events and has no set events.
{ PRS1PressureSetEvent::TYPE, { &CPAP_Pressure } },
{ PRS1IPAPSetEvent::TYPE, { &CPAP_IPAP } },
{ PRS1EPAPSetEvent::TYPE, { &CPAP_EPAP, &CPAP_PS } }, // PS is calculated from IPAP and EPAP
{ PRS1PressureAverageEvent::TYPE, { &CPAP_Pressure } },
{ PRS1IPAPAverageEvent::TYPE, { &CPAP_IPAP } },
{ PRS1EPAPAverageEvent::TYPE, { &CPAP_EPAP, &CPAP_PS } }, // PS is calculated from IPAP and EPAP
{ PRS1IPAPLowEvent::TYPE, { &CPAP_IPAPLo } },
{ PRS1IPAPHighEvent::TYPE, { &CPAP_IPAPHi } },
{ PRS1Test1Event::TYPE, { &CPAP_Test1 } },
{ PRS1Test2Event::TYPE, { &CPAP_Test2 } },
{ PRS1PressurePulseEvent::TYPE, { &CPAP_PressurePulse } },
{ PRS1ApneaAlarmEvent::TYPE, { /* Not imported */ } },
{ PRS1SnoresAtPressureEvent::TYPE, { /* Not imoprted */ } },
{ PRS1AutoPressureSetEvent::TYPE, { /* Not imported */ } },
{ PRS1UnknownDurationEvent::TYPE, { /* Not imported */ } }, // TODO: import as PRS1_0E
};
//******************************************************************************************** //********************************************************************************************
static QString hex(int i) static QString hex(int i)
@ -2577,6 +2640,65 @@ bool PRS1DataChunk::ParseEventsF5V2(void)
} }
bool PRS1Import::CreateEventChannels(void)
{
m_importChannels.clear();
// Create all supported channels (except for on-demand ones that only get created if an event appears)
const QVector<PRS1ParsedEventType> & supported = GetSupportedEvents(event);
for (auto & e : supported) {
if (!PRS1OnDemandChannels.contains(e)) {
for (auto & pChannelID : PRS1ImportChannelMap[e]) {
if (!GetImportChannel(*pChannelID)) {
return false;
}
}
}
}
return true;
}
EventList* PRS1Import::GetImportChannel(ChannelID channel)
{
EventList* C = m_importChannels[channel];
if (C == nullptr) {
C = session->AddEventList(channel, EVL_Event);
// TODO: figure out why this can fail and prevent it
if (C == nullptr) {
qWarning() << "Unable to create event list for channel" << channel;
} else {
m_importChannels[channel] = C;
}
}
return C;
}
bool PRS1Import::AddEvent(ChannelID channel, qint64 t, float value, float gain)
{
EventList* C = GetImportChannel(channel);
if (C == nullptr) {
// Warnings are handled by GetImportChannel.
return false;
}
if (C->count() == 0) {
// Initialize the gain (here, since required channels are created with default gain).
C->setGain(gain);
} else {
// Any change in gain is a programming error.
if (gain != C->gain()) {
qWarning() << "gain mismatch for channel" << channel << "at" << ts(t);
}
}
// Add the event
C->AddEvent(t, value, gain);
// TODO: can this ever fail?
return true;
}
bool PRS1Import::ParseEventsF3V6() bool PRS1Import::ParseEventsF3V6()
{ {
float GAIN = PRS1PressureEvent::GAIN; float GAIN = PRS1PressureEvent::GAIN;
@ -2585,36 +2707,40 @@ bool PRS1Import::ParseEventsF3V6()
GAIN = 0.125F; // TODO: parameterize this somewhere better GAIN = 0.125F; // TODO: parameterize this somewhere better
} }
if (!CreateEventChannels()) {
return false;
}
// Required channels // Required channels
EventList *CA = session->AddEventList(CPAP_ClearAirway, EVL_Event); EventList *CA = GetImportChannel(CPAP_ClearAirway);
EventList *OA = session->AddEventList(CPAP_Obstructive, EVL_Event); EventList *OA = GetImportChannel(CPAP_Obstructive);
EventList *HY = session->AddEventList(CPAP_Hypopnea, EVL_Event); EventList *HY = GetImportChannel(CPAP_Hypopnea);
// No FL detection? // No FL detection?
EventList *SNORE = session->AddEventList(CPAP_Snore, EVL_Event); // snore count statistic EventList *SNORE = GetImportChannel(CPAP_Snore); // snore count statistic
// No individual VS events reported // No individual VS events reported
EventList *VS2 = session->AddEventList(CPAP_VSnore2, EVL_Event); // flags nonzero snore-count intervals on overview EventList *VS2 = GetImportChannel(CPAP_VSnore2); // flags nonzero snore-count intervals on overview
EventList *RE = session->AddEventList(CPAP_RERA, EVL_Event); EventList *RE = GetImportChannel(CPAP_RERA);
EventList *PB = session->AddEventList(CPAP_PB, EVL_Event); EventList *PB = GetImportChannel(CPAP_PB);
EventList *LL = session->AddEventList(CPAP_LargeLeak, EVL_Event); EventList *LL = GetImportChannel(CPAP_LargeLeak);
EventList *TOTLEAK = session->AddEventList(CPAP_LeakTotal, EVL_Event); EventList *TOTLEAK = GetImportChannel(CPAP_LeakTotal);
EventList *LEAK = session->AddEventList(CPAP_Leak, EVL_Event); EventList *LEAK = GetImportChannel(CPAP_Leak);
EventList *RR = session->AddEventList(CPAP_RespRate, EVL_Event); EventList *RR = GetImportChannel(CPAP_RespRate);
EventList *TV = session->AddEventList(CPAP_TidalVolume, EVL_Event, 10.0F); //EventList *TV = GetImportChannel(CPAP_TidalVolume);
EventList *MV = session->AddEventList(CPAP_MinuteVent, EVL_Event); EventList *MV = GetImportChannel(CPAP_MinuteVent);
EventList *PTB = session->AddEventList(CPAP_PTB, EVL_Event); EventList *PTB = GetImportChannel(CPAP_PTB);
// TB could be on-demand // TB could be on-demand
EventList *TB = session->AddEventList(PRS1_TimedBreath, EVL_Event); EventList *TB = GetImportChannel(PRS1_TimedBreath);
EventList *IPAP = session->AddEventList(CPAP_IPAP, EVL_Event, GAIN); //EventList *IPAP = GetImportChannel(CPAP_IPAP);
EventList *EPAP = session->AddEventList(CPAP_EPAP, EVL_Event, GAIN); //EventList *EPAP = GetImportChannel(CPAP_EPAP);
EventList *PS = session->AddEventList(CPAP_PS, EVL_Event, GAIN); //EventList *PS = GetImportChannel(CPAP_PS);
EventList *TMV = session->AddEventList(CPAP_Test1, EVL_Event); // ??? EventList *TMV = GetImportChannel(CPAP_Test1); // ???
EventList *FLOW = session->AddEventList(CPAP_Test2, EVL_Event); // ??? EventList *FLOW = GetImportChannel(CPAP_Test2); // ???
// On-demand channels // On-demand channels
EventList *PP = nullptr; EventList *PP = nullptr;
@ -2630,18 +2756,19 @@ bool PRS1Import::ParseEventsF3V6()
for (int i=0; i < event->m_parsedData.count(); i++) { for (int i=0; i < event->m_parsedData.count(); i++) {
PRS1ParsedEvent* e = event->m_parsedData.at(i); PRS1ParsedEvent* e = event->m_parsedData.at(i);
QVector<ChannelID*> channels = PRS1ImportChannelMap[e->m_type];
t = qint64(event->timestamp + e->m_start) * 1000L; t = qint64(event->timestamp + e->m_start) * 1000L;
switch (e->m_type) { switch (e->m_type) {
case PRS1ApneaAlarmEvent::TYPE: case PRS1ApneaAlarmEvent::TYPE:
break; // not imported or displayed break; // not imported or displayed
case PRS1IPAPAverageEvent::TYPE: case PRS1IPAPAverageEvent::TYPE:
IPAP->AddEvent(t, e->m_value); // TODO: This belongs in an average channel rather than setting channel. AddEvent(*channels.at(0), t, e->m_value, e->m_gain); // TODO: This belongs in an average channel rather than setting channel.
currentPressure = e->m_value; currentPressure = e->m_value;
break; break;
case PRS1EPAPAverageEvent::TYPE: case PRS1EPAPAverageEvent::TYPE:
EPAP->AddEvent(t, e->m_value); // TODO: This belongs in an average channel rather than setting channel. AddEvent(CPAP_EPAP, t, e->m_value, e->m_gain); // TODO: This belongs in an average channel rather than setting channel.
PS->AddEvent(t, currentPressure - e->m_value); // Pressure Support AddEvent(CPAP_PS, t, currentPressure - e->m_value, e->m_gain); // Pressure Support
break; 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)!
@ -2699,7 +2826,7 @@ bool PRS1Import::ParseEventsF3V6()
MV->AddEvent(t, e->m_value); MV->AddEvent(t, e->m_value);
break; break;
case PRS1TidalVolumeEvent::TYPE: case PRS1TidalVolumeEvent::TYPE:
TV->AddEvent(t, e->m_value); AddEvent(*channels.at(0), t, e->m_value, e->m_gain);
break; break;
case PRS1RERAEvent::TYPE: case PRS1RERAEvent::TYPE:
RE->AddEvent(t, e->m_value); RE->AddEvent(t, e->m_value);

View File

@ -321,6 +321,7 @@ protected:
PRS1Loader * loader; PRS1Loader * loader;
SessionID sessionid; SessionID sessionid;
Machine * mach; Machine * mach;
QHash<ChannelID,EventList*> m_importChannels; // map channel ID to the session's current EventList*
int summary_duration; int summary_duration;
@ -330,6 +331,13 @@ protected:
bool ParseSession(void); bool ParseSession(void);
//! \brief Save parsed session data to the database //! \brief Save parsed session data to the database
void SaveSessionToDatabase(void); void SaveSessionToDatabase(void);
//! \brief Create all supported channels (except for on-demand ones that only get created if an event appears).
bool CreateEventChannels(void);
//! \brief Get the EventList* for the import channel, creating it if necessary.
EventList* GetImportChannel(ChannelID channel);
//! \brief Import a single event to a channel, creating the channel if necessary.
bool AddEvent(ChannelID channel, qint64 t, float value, float gain);
}; };
/*! \class PRS1Loader /*! \class PRS1Loader