2014-08-17 12:56:05 +00:00
|
|
|
/* Custom CPAP/Oximetry Calculations Header
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
2024-01-13 20:27:48 +00:00
|
|
|
* Copyright (c) 2019-2024 The OSCAR Team
|
2024-02-01 00:14:19 +00:00
|
|
|
* Copyright (C) 2011-2018 Mark Watkins
|
2014-04-09 21:01:57 +00:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
2018-06-04 20:48:38 +00:00
|
|
|
* License. See the file COPYING in the main directory of the source code
|
|
|
|
* for more details. */
|
2014-04-09 21:01:57 +00:00
|
|
|
|
2011-11-28 01:39:28 +00:00
|
|
|
#ifndef CALCS_H
|
|
|
|
#define CALCS_H
|
|
|
|
|
|
|
|
#include "day.h"
|
|
|
|
|
2012-01-09 03:43:42 +00:00
|
|
|
//! param samples Number of samples
|
|
|
|
//! width number of surrounding samples to consider
|
|
|
|
//! percentile fractional percentage, between 0 and 1
|
2014-04-17 05:58:57 +00:00
|
|
|
void percentileFilter(EventDataType *input, EventDataType *output, int samples, int width,
|
|
|
|
EventDataType percentile);
|
|
|
|
void xpassFilter(EventDataType *input, EventDataType *output, int samples, EventDataType weight);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
enum FilterType { FilterNone = 0, FilterPercentile, FilterXPass };
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
struct Filter {
|
2014-04-17 05:58:57 +00:00
|
|
|
Filter(FilterType t, EventDataType p1, EventDataType p2, EventDataType p3) {
|
|
|
|
type = t;
|
|
|
|
param1 = p1;
|
|
|
|
param2 = p2;
|
|
|
|
param3 = p3;
|
2012-01-09 03:43:42 +00:00
|
|
|
}
|
|
|
|
Filter() {
|
2014-04-17 05:58:57 +00:00
|
|
|
type = FilterNone;
|
|
|
|
param1 = 0;
|
|
|
|
param2 = 0;
|
|
|
|
param3 = 0;
|
2012-01-09 03:43:42 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
Filter(const Filter ©) {
|
|
|
|
type = copy.type;
|
|
|
|
param1 = copy.param1;
|
|
|
|
param2 = copy.param2;
|
|
|
|
param3 = copy.param3;
|
2012-01-09 03:43:42 +00:00
|
|
|
}
|
2023-03-30 21:11:28 +00:00
|
|
|
Filter& operator=(const Filter ©) = default;
|
|
|
|
~Filter() {};
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
FilterType type;
|
|
|
|
EventDataType param1;
|
|
|
|
EventDataType param2;
|
|
|
|
EventDataType param3;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct BreathPeak {
|
2014-04-17 05:58:57 +00:00
|
|
|
BreathPeak() { min = 0; max = 0; start = 0; middle = 0; end = 0; } // peakmin=0; peakmax=0; }
|
|
|
|
BreathPeak(EventDataType _min, EventDataType _max, qint32 _start, qint32 _middle,
|
|
|
|
qint32 _end) {//, qint64 _peakmin, qint64 _peakmax) {
|
|
|
|
min = _min;
|
|
|
|
max = _max;
|
|
|
|
start = _start;
|
|
|
|
middle = _middle;
|
|
|
|
end = _end;
|
2012-01-09 15:38:41 +00:00
|
|
|
//peakmax=_peakmax;
|
|
|
|
//peakmin=_peakmin;
|
2012-01-09 03:43:42 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
BreathPeak(const BreathPeak ©) {
|
|
|
|
min = copy.min;
|
|
|
|
max = copy.max;
|
|
|
|
start = copy.start;
|
|
|
|
middle = copy.middle;
|
|
|
|
end = copy.end;
|
2012-01-09 15:38:41 +00:00
|
|
|
//peakmin=copy.peakmin;
|
|
|
|
//peakmax=copy.peakmax;
|
2012-01-09 03:43:42 +00:00
|
|
|
}
|
2014-04-17 05:58:57 +00:00
|
|
|
int samplelength() { return end - start; }
|
|
|
|
int upperLength() { return middle - start; }
|
|
|
|
int lowerLength() { return end - middle; }
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
EventDataType min; // peak value
|
|
|
|
EventDataType max; // peak value
|
|
|
|
qint32 start; // beginning zero cross
|
|
|
|
qint32 middle; // ending zero cross
|
|
|
|
qint32 end; // ending zero cross
|
2012-01-09 15:38:41 +00:00
|
|
|
//qint64 peakmin; // min peak index
|
|
|
|
//qint64 peakmax; // max peak index
|
2012-01-09 03:43:42 +00:00
|
|
|
};
|
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
bool operator<(const BreathPeak &p1, const BreathPeak &p2);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
2014-04-17 05:58:57 +00:00
|
|
|
const int num_filter_buffers = 2;
|
2012-01-09 03:43:42 +00:00
|
|
|
|
2023-11-14 23:42:53 +00:00
|
|
|
const int max_filter_buf_size_entries = 0x20C000 ; // was 2097152 == 0x200000;
|
2023-10-24 23:31:10 +00:00
|
|
|
const int max_filter_buf_size_malloc = max_filter_buf_size_entries * sizeof(EventDataType);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
//! \brief Class to process Flow Rate waveform data
|
2014-04-17 05:58:57 +00:00
|
|
|
class FlowParser
|
|
|
|
{
|
|
|
|
public:
|
2012-01-09 03:43:42 +00:00
|
|
|
FlowParser();
|
|
|
|
~FlowParser();
|
|
|
|
|
|
|
|
//! \brief Clears the (input) filter chain
|
|
|
|
void clearFilters();
|
|
|
|
|
|
|
|
//! \brief Applies the filter chain to input, with supplied number of samples
|
2014-04-17 05:58:57 +00:00
|
|
|
EventDataType *applyFilters(EventDataType *input, int samples);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
//! \brief Add the filter
|
2014-04-17 05:58:57 +00:00
|
|
|
void addFilter(FilterType ft, EventDataType p1 = 0, EventDataType p2 = 0, EventDataType p3 = 0) {
|
|
|
|
m_filters.push_back(Filter(ft, p1, p2, p3));
|
2012-01-09 03:43:42 +00:00
|
|
|
}
|
|
|
|
//! \brief Opens the flow rate EventList, applies the input filter chain, and calculates peaks
|
2014-04-17 05:58:57 +00:00
|
|
|
void openFlow(Session *session, EventList *flow);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
//! \brief Calculates the upper and lower breath peaks
|
2014-04-17 05:58:57 +00:00
|
|
|
void calcPeaks(EventDataType *input, int samples);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
// Minute vent needs Resp & TV calcs made here..
|
|
|
|
void calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool calcMv);
|
2012-01-09 08:05:20 +00:00
|
|
|
void flagEvents();
|
2014-07-20 05:24:49 +00:00
|
|
|
void flagUserEvents(ChannelID code, EventDataType restriction, EventDataType duration);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
|
|
|
/*void calcTidalVolume();
|
|
|
|
void calcRespRate();
|
|
|
|
void calcMinuteVent(); */
|
|
|
|
|
|
|
|
|
|
|
|
QList<Filter> m_filters;
|
2014-04-17 05:58:57 +00:00
|
|
|
protected:
|
2012-01-09 03:43:42 +00:00
|
|
|
QVector<BreathPeak> breaths;
|
|
|
|
|
|
|
|
int m_samples;
|
2014-04-17 05:58:57 +00:00
|
|
|
EventList *m_flow;
|
|
|
|
Session *m_session;
|
2012-01-09 03:43:42 +00:00
|
|
|
EventDataType m_gain;
|
|
|
|
EventDataType m_rate;
|
|
|
|
EventDataType m_minutes;
|
|
|
|
//! \brief The filtered waveform
|
2014-04-17 05:58:57 +00:00
|
|
|
EventDataType *m_filtered;
|
2012-01-09 03:43:42 +00:00
|
|
|
//! \brief BreathPeak's start on positive cycle?
|
|
|
|
bool m_startsUpper;
|
2014-04-17 05:58:57 +00:00
|
|
|
private:
|
|
|
|
EventDataType *m_buffers[num_filter_buffers];
|
2012-01-09 03:43:42 +00:00
|
|
|
};
|
|
|
|
|
2014-07-20 13:25:57 +00:00
|
|
|
bool SearchApnea(Session *session, qint64 time, double dur);
|
2012-01-09 03:43:42 +00:00
|
|
|
|
2011-12-18 16:39:36 +00:00
|
|
|
//! \brief Calculate Respiratory Rate, Tidal Volume & Minute Ventilation for PRS1 data
|
2014-04-23 13:19:56 +00:00
|
|
|
void calcRespRate(Session *session, FlowParser *flowparser = nullptr);
|
2011-12-18 16:39:36 +00:00
|
|
|
|
|
|
|
//! \brief Calculates the sliding window AHI graph
|
2011-12-02 11:40:47 +00:00
|
|
|
int calcAHIGraph(Session *session);
|
2011-12-18 16:39:36 +00:00
|
|
|
|
|
|
|
//! \brief Calculates AHI for a session between start & end (a support function for the sliding window graph)
|
2014-04-17 05:58:57 +00:00
|
|
|
EventDataType calcAHI(Session *session, qint64 start = -1, qint64 end = -1);
|
2011-11-28 01:39:28 +00:00
|
|
|
|
2022-02-27 16:01:46 +00:00
|
|
|
//! \brief Scans for leaks over Redline and flags as large leaks, unless device provided them already
|
2014-08-05 22:14:51 +00:00
|
|
|
void flagLargeLeaks(Session *session);
|
|
|
|
|
2011-12-18 16:39:36 +00:00
|
|
|
//! \brief Leaks calculations for PRS1
|
2011-11-30 12:32:16 +00:00
|
|
|
int calcLeaks(Session *session);
|
|
|
|
|
2011-12-18 16:39:36 +00:00
|
|
|
//! \brief Calculate Pulse change flagging, according to preferences
|
2011-11-28 04:05:09 +00:00
|
|
|
int calcPulseChange(Session *session);
|
2011-12-18 16:39:36 +00:00
|
|
|
|
|
|
|
//! \brief Calculate SPO2 Drop flagging, according to preferences
|
2011-11-28 04:05:09 +00:00
|
|
|
int calcSPO2Drop(Session *session);
|
|
|
|
|
2011-11-28 01:39:28 +00:00
|
|
|
|
|
|
|
#endif // CALCS_H
|