/* Custom CPAP/Oximetry Calculations Header Copyright (c)2011 Mark Watkins License: GPL */ #include #include #include "calcs.h" #include "profiles.h" extern double round(double number); bool SearchApnea(Session *session, qint64 time, qint64 dist) { if (session->SearchEvent(CPAP_Obstructive,time,dist)) return true; if (session->SearchEvent(CPAP_Apnea,time,dist)) return true; if (session->SearchEvent(CPAP_ClearAirway,time,dist)) return true; if (session->SearchEvent(CPAP_Hypopnea,time,dist)) return true; return false; } // Sort BreathPeak by peak index bool operator<(const BreathPeak & p1, const BreathPeak & p2) { return p1.start < p2.start; } //! \brief Filters input to output with a percentile filter with supplied width. //! \param samples Number of samples //! \param width number of surrounding samples to consider //! \param percentile fractional percentage, between 0 and 1 void percentileFilter(EventDataType * input, EventDataType * output, int samples, int width, EventDataType percentile) { if (samples<=0) return; if (percentile>1) percentile=1; QVector buf(width); int s,e; int z1=width/2; int z2=z1+(width % 2); int nm1=samples-1; //int j; // Scan through all of input for (int k=0;k < samples;k++) { s=k-z1; e=k+z2; // Cap bounds if (s < 0) s=0; if (e > nm1) e=nm1; // int j=0; for (int i=s; i < e; i++) { buf[j++]=input[i]; } j--; EventDataType val=j * percentile; EventDataType fl=floor(val); // If even percentile, or already max value.. if ((val==fl) || (j>=width-1)) { nth_element(buf.begin(),buf.begin()+j,buf.begin()+width-1); val=buf[j]; } else { // Percentile lies between two points, interpolate. double v1,v2; nth_element(buf.begin(),buf.begin()+j,buf.begin()+width-1); v1=buf[j]; nth_element(buf.begin(),buf.begin()+j+1,buf.begin()+width-1); v2=buf[j+1]; val = v1 + (v2-v1)*(val-fl); } output[k]=val; } } void xpassFilter(EventDataType * input, EventDataType * output, int samples, EventDataType weight) { // prime the first value output[0]=input[0]; for (int i=1;igain(); m_rate=flow->rate(); m_samples=flow->count(); EventStoreType *inraw=flow->rawData(); // Make sure we won't overflow internal buffers if (m_samples > max_filter_buf_size) { qDebug() << "Error: Sample size exceeds max_filter_buf_size in FlowParser::openFlow().. Capping!!!"; m_samples=max_filter_buf_size; } // Begin with the second internal buffer EventDataType * buf=m_filtered; // Apply gain to waveform EventStoreType *eptr=inraw+m_samples; // Convert from store type to floats.. for (; inraw < eptr; inraw++) { *buf++ = EventDataType(*inraw) * m_gain; } // Apply the rest of the filters chain buf=applyFilters(m_filtered, m_samples); calcPeaks(m_filtered, m_samples); } // Calculates breath upper & lower peaks for a chunk of EventList data void FlowParser::calcPeaks(EventDataType * input, int samples) { if (samples<=0) return; EventDataType min=0,max=0, c, lastc=0; EventDataType zeroline=0; double rate=m_flow->rate(); double flowstart=m_flow->first(); double lasttime,time; double peakmax=flowstart, peakmin=flowstart; time=lasttime=flowstart; breaths.clear(); // Estimate storage space needed using typical average breaths per minute. m_minutes=double(m_flow->last() - m_flow->first()) / 60000.0; const double avgbpm=20; int guestimate=m_minutes*avgbpm; breaths.reserve(guestimate); // Prime min & max, and see which side of the zero line we are starting from. c=input[0]; min=max=c; lastc=c; m_startsUpper=(c >= zeroline); qint32 start=0,middle=0;//,end=0; int sps=1000/m_rate; int len=0, lastk=0; //lastlen=0 qint64 sttime=time;//, ettime=time; // For each samples, find the peak upper and lower breath components for (int k=0; k < samples; k++) { c=input[k]; if (c >= zeroline) { // Did we just cross the zero line going up? if (lastc < zeroline) { // This helps filter out dirty breaths.. len=k-start; if ((max>3) && ((max-min) > 8) && (len>sps) && (middle > start)) { // peak detection may not be needed.. breaths.push_back(BreathPeak(min, max, start, middle, k)); //, peakmin, peakmax)); // Set max for start of the upper breath cycle max=c; peakmax=time; // Starting point of next breath cycle start=k; sttime=time; } } else if (c > max) { // Update upper breath peak max=c; peakmax=time; } } if (c < zeroline) { // Did we just cross the zero line going down? if (lastc >= zeroline) { // Set min for start of the lower breath cycle min=c; peakmin=time; middle=k; } else if (c < min) { // Update lower breath peak min=c; peakmin=time; } } lasttime=time; time+=rate; lastc=c; lastk=k; } } //! \brief Calculate Respiratory Rate, TidalVolume, Minute Ventilation, Ti & Te.. // These are grouped together because, a) it's faster, and b) some of these calculations rely on others. void FlowParser::calc(bool calcResp, bool calcTv, bool calcTi, bool calcTe, bool calcMv) { if (!m_session) { return; } // Don't even bother if only a few breaths in this chunk const int lowthresh=4; int nm=breaths.size(); if (nm < lowthresh) return; const qint64 minute=60000; double start=m_flow->first(); double time=start; int bs,be,bm; double st,et,mt; ///////////////////////////////////////////////////////////////////////////////// // Respiratory Rate setup ///////////////////////////////////////////////////////////////////////////////// EventDataType minrr,maxrr; EventList * RR=NULL; quint32 * rr_tptr=NULL; EventStoreType * rr_dptr=NULL; if (calcResp) { RR=m_session->AddEventList(CPAP_RespRate,EVL_Event); minrr=RR->Min(), maxrr=RR->Max(); RR->setFirst(time+minute); RR->getData().resize(nm); RR->getTime().resize(nm); rr_tptr=RR->rawTime(); rr_dptr=RR->rawData(); } int rr_count=0; double len, st2,et2,adj, stmin, b, rr=0; int idx; ///////////////////////////////////////////////////////////////////////////////// // Inspiratory / Expiratory Time setup ///////////////////////////////////////////////////////////////////////////////// double lastte2=0,lastti2=0,lastte=0, lastti=0, te, ti, ti1, te1,c; EventList * Te=NULL, * Ti=NULL; if (calcTi) { Ti=m_session->AddEventList(CPAP_Ti,EVL_Event); Ti->setGain(0.02); } if (calcTe) { Te=m_session->AddEventList(CPAP_Te,EVL_Event); Te->setGain(0.02); } ///////////////////////////////////////////////////////////////////////////////// // Tidal Volume setup ///////////////////////////////////////////////////////////////////////////////// EventList * TV; EventDataType mintv, maxtv, tv; double val1, val2; quint32 * tv_tptr; EventStoreType * tv_dptr; int tv_count=0; if (calcTv) { TV=m_session->AddEventList(CPAP_TidalVolume,EVL_Event); mintv=TV->Min(), maxtv=TV->Max(); TV->setFirst(start); TV->getData().resize(nm); TV->getTime().resize(nm); tv_tptr=TV->rawTime(); tv_dptr=TV->rawData(); } ///////////////////////////////////////////////////////////////////////////////// // Minute Ventilation setup ///////////////////////////////////////////////////////////////////////////////// EventList * MV=NULL; EventDataType mv; if (calcMv) { MV=m_session->AddEventList(CPAP_MinuteVent,EVL_Event); MV->setGain(0.125); } EventDataType sps=(1000.0/m_rate); // Samples Per Second qint32 timeval=0; // Time relative to start for (idx=0;idxAddEvent(mt,ti1); lastti2=lastti; lastti=ti; } ///////////////////////////////////////////////////////////////////// // Calculate Expiratory Time (Te) for this breath ///////////////////////////////////////////////////////////////////// if (calcTe) { te=((et-mt)/1000.0)*50.0; // Average last three values.. te1=(lastte2+lastte+te)/3.0; Te->AddEvent(mt,te1); lastte2=lastte; lastte=te; } ///////////////////////////////////////////////////////////////////// // Calculate TidalVolume ///////////////////////////////////////////////////////////////////// if (calcTv) { val1=0, val2=0; // Scan the upper breath for (int j=bs;j maxtv) maxtv=tv; *tv_tptr++ = timeval; *tv_dptr++ = tv / 20.0; tv_count++; } ///////////////////////////////////////////////////////////////////// // Respiratory Rate Calculations ///////////////////////////////////////////////////////////////////// if (calcResp) { stmin=et-minute; if (stmin < start) stmin=start; len=et-stmin; rr=0; if (len >= minute) { //et2=et; // Step back through last minute and count breaths for (int i=idx;i>=0;i--) { st2=start + double(breaths[i].start) * m_rate; et2=start + double(breaths[i].end) * m_rate; if (et2 < stmin) break; len=et2-st2; if (st2 < stmin) { // Partial breath st2=stmin; adj=et2 - st2; b=(1.0 / len) * adj; } else b=1; rr+=b; } // Calculate min & max if (rr < minrr) minrr=rr; if (rr > maxrr) maxrr=rr; // Add manually.. (much quicker) *rr_tptr++ = timeval; // Use the same gains as ResMed.. *rr_dptr++ = rr * 5.0; rr_count++; //rr->AddEvent(et,br * 50.0); } } if (calcMv && calcResp && calcTv) { mv=(tv/1000.0) * rr; MV->AddEvent(et,mv * 8.0); } } ///////////////////////////////////////////////////////////////////// // Respiratory Rate post filtering ///////////////////////////////////////////////////////////////////// if (calcResp) { RR->setGain(0.2); RR->setMin(minrr); RR->setMax(maxrr); RR->setFirst(start); RR->setLast(et); RR->setCount(rr_count); } ///////////////////////////////////////////////////////////////////// // Tidal Volume post filtering ///////////////////////////////////////////////////////////////////// if (calcTv) { TV->setGain(20); TV->setMin(mintv); TV->setMax(maxtv); TV->setFirst(start); TV->setLast(et); TV->setCount(tv_count); } } void FlowParser::flagEvents() { int numbreaths=breaths.size(); EventDataType val,mx,mn; QVector br; QVector bstart; QVector bend; //QVector bvalue; bstart.reserve(numbreaths*2); bend.reserve(numbreaths*2); //bvalue.reserve(numbreaths*2); br.reserve(numbreaths*2); double start=m_flow->first(); // double sps=1000.0/m_rate; double st,mt,et, dur; qint64 len; bool allowDuplicates=PROFILE.cpap->userEventDuplicates(); for (int i=0;iAddEventList(CPAP_UserFlag2,EVL_Event); EventList * uf3=m_session->AddEventList(CPAP_UserFlag3,EVL_Event); const EventDataType perc=0.6; int idx=float(br.size())*perc; nth_element(br.begin(),br.begin()+idx,br.end()); EventDataType peak=br[idx];//*(br.begin()+idx); EventDataType cutoffval=peak * (PROFILE.cpap->userFlowRestriction()/100.0); int bs,bm,be, bs1, bm1, be1; for (int i=0;i cutoffval) { bs1=bs; for (;bs1 cutoffval) { break; } } bm1=bm; for (;bm1>bs;bm1--) { if (qAbs(m_filtered[bm1]) > cutoffval) { break; } } if (bm1>=bs1) { bstart.push_back(bs1); bend.push_back(bm1); } // } // if (qAbs(mn) > cutoffval) { bm1=bm; for (;bm1 cutoffval) { break; } } be1=be; for (;be1>bm;be1--) { if (qAbs(m_filtered[be1]) > cutoffval) { break; } } if (be1>=bm1) { bstart.push_back(bm1); bend.push_back(be1); } // } st=start + bs1 * m_rate; et=start + be1 * m_rate; uf2->AddEvent(st,0); uf3->AddEvent(et,0); } EventDataType duration=PROFILE.cpap->userEventDuration(); //double lastst=start, lastet=start; //EventDataType v; int bsize=bstart.size(); EventList * uf1=NULL; for (int i=0;i=duration) { if (allowDuplicates || !SearchApnea(m_session,et-len/2,15000)) { if (!uf1) { uf1=m_session->AddEventList(CPAP_UserFlag1,EVL_Event); } uf1->AddEvent(et-len/2,dur); } } } } void calcRespRate(Session *session, FlowParser * flowparser) { if (session->machine()->GetType()!=MT_CPAP) return; // if (session->machine()->GetClass()!=STR_MACH_PRS1) return; if (!session->eventlist.contains(CPAP_FlowRate)) { //qDebug() << "calcRespRate called without FlowRate waveform available"; return; //need flow waveform } bool trashfp; if (!flowparser) { flowparser=new FlowParser(); trashfp=true; //qDebug() << "calcRespRate called without valid FlowParser object.. using a slow throw-away!"; //return; } else { trashfp=false; } bool calcResp=!session->eventlist.contains(CPAP_RespRate); bool calcTv=!session->eventlist.contains(CPAP_TidalVolume); bool calcTi=!session->eventlist.contains(CPAP_Ti); bool calcTe=!session->eventlist.contains(CPAP_Te); bool calcMv=!session->eventlist.contains(CPAP_MinuteVent); int z=(calcResp ? 1 : 0) + (calcTv ? 1 : 0) + (calcMv ? 1 : 0); // If any of these three missing, remove all, and switch all on if (z>0 && z<3) { if (!calcResp && !calcTv && !calcMv) calcTv=calcMv=calcResp=true; QVector & list=session->eventlist[CPAP_RespRate]; for (int i=0;ieventlist[CPAP_RespRate].clear(); QVector & list2=session->eventlist[CPAP_TidalVolume]; for (int i=0;ieventlist[CPAP_TidalVolume].clear(); QVector & list3=session->eventlist[CPAP_MinuteVent]; for (int i=0;ieventlist[CPAP_MinuteVent].clear(); } flowparser->clearFilters(); // No filters works rather well with the new peak detection algorithm.. // Although the output could use filtering. //flowparser->addFilter(FilterPercentile,7,0.5); //flowparser->addFilter(FilterPercentile,5,0.5); //flowparser->addFilter(FilterXPass,0.5); EventList *flow; for (int ws=0; ws < session->eventlist[CPAP_FlowRate].size(); ws++) { flow=session->eventlist[CPAP_FlowRate][ws]; if (flow->count() > 20) { flowparser->openFlow(session, flow); flowparser->calc(calcResp, calcTv, calcTi ,calcTe, calcMv); flowparser->flagEvents(); } } if (trashfp) { delete flowparser; } } EventDataType calcAHI(Session *session,qint64 start, qint64 end) { bool rdi=PROFILE.general->calculateRDI(); double hours,ahi,cnt; if (start<0) { // much faster.. hours=session->hours(); cnt=session->count(CPAP_Obstructive) +session->count(CPAP_Hypopnea) +session->count(CPAP_ClearAirway) +session->count(CPAP_Apnea); if (rdi) { cnt+=session->count(CPAP_RERA); } ahi=cnt/hours; } else { hours=double(end-start)/3600000L; if (hours==0) return 0; cnt=session->rangeCount(CPAP_Obstructive,start,end) +session->rangeCount(CPAP_Hypopnea,start,end) +session->rangeCount(CPAP_ClearAirway,start,end) +session->rangeCount(CPAP_Apnea,start,end); if (rdi) { cnt+=session->rangeCount(CPAP_RERA,start,end); } ahi=cnt/hours; } return ahi; } int calcAHIGraph(Session *session) { bool calcrdi=session->machine()->GetClass()=="PRS1"; //PROFILE.general->calculateRDI() const qint64 window_step=30000; // 30 second windows double window_size=p_profile->cpap->AHIWindow(); qint64 window_size_ms=window_size*60000L; bool zeroreset=p_profile->cpap->AHIReset(); if (session->machine()->GetType()!=MT_CPAP) return 0; bool hasahi=session->eventlist.contains(CPAP_AHI); bool hasrdi=session->eventlist.contains(CPAP_RDI); if (hasahi && hasrdi) return 0; // abort if already there if (!(!hasahi && !hasrdi)) { session->destroyEvent(CPAP_AHI); session->destroyEvent(CPAP_RDI); } if (!session->channelExists(CPAP_Obstructive) && !session->channelExists(CPAP_Hypopnea) && !session->channelExists(CPAP_Apnea) && !session->channelExists(CPAP_ClearAirway) && !session->channelExists(CPAP_RERA) ) return 0; qint64 first=session->first(), last=session->last(), f; EventList *AHI=new EventList(EVL_Event); AHI->setGain(0.02); session->eventlist[CPAP_AHI].push_back(AHI); EventList *RDI=NULL; if (calcrdi) { RDI=new EventList(EVL_Event); RDI->setGain(0.02); session->eventlist[CPAP_RDI].push_back(RDI); } EventDataType ahi,rdi; qint64 ti=first,lastti=first; double avgahi=0; double avgrdi=0; int cnt=0; double events; double hours=(window_size/60.0); if (zeroreset) { // I personally don't see the point of resetting each hour. do { // For each window, in 30 second increments for (qint64 t=ti;t < ti+window_size_ms; t+=window_step) { if (t > last) break; events=session->rangeCount(CPAP_Obstructive,ti,t) +session->rangeCount(CPAP_Hypopnea,ti,t) +session->rangeCount(CPAP_ClearAirway,ti,t) +session->rangeCount(CPAP_Apnea,ti,t); ahi = events / hours; AHI->AddEvent(t,ahi * 50); avgahi+=ahi; if (calcrdi) { events+=session->rangeCount(CPAP_RERA,ti,t); rdi=events / hours; RDI->AddEvent(t,rdi * 50); avgrdi+=rdi; } cnt++; } lastti=ti; ti+=window_size_ms; } while (tirangeCount(CPAP_Obstructive,f,ti) +session->rangeCount(CPAP_Hypopnea,f,ti) +session->rangeCount(CPAP_ClearAirway,f,ti) +session->rangeCount(CPAP_Apnea,f,ti); ahi=events/hours; avgahi+=ahi; AHI->AddEvent(ti,ahi * 50); if (calcrdi) { events+=session->rangeCount(CPAP_RERA,f,ti); rdi=events/hours; RDI->AddEvent(ti,rdi * 50); avgrdi+=rdi; } cnt++; lastti=ti; ti+=window_step; } } AHI->AddEvent(lastti,0); if (calcrdi) RDI->AddEvent(lastti,0); if (!cnt) { avgahi=0; avgrdi=0; } else { avgahi/=double(cnt); avgrdi/=double(cnt); } cnt++; session->setAvg(CPAP_AHI,avgahi); if (calcrdi) session->setAvg(CPAP_RDI,avgrdi); return cnt; } int calcLeaks(Session *session) { if (session->machine()->GetType()!=MT_CPAP) return 0; if (session->eventlist.contains(CPAP_Leak)) return 0; // abort if already there if (!session->eventlist.contains(CPAP_LeakTotal)) return 0; // can't calculate without this.. if (session->settings[CPAP_Mode].toInt()>MODE_APAP) return 0; // Don't bother calculating when in APAP mode const qint64 winsize=3600000; // 5 minute window //qint64 first=session->first(), last=session->last(), f; EventList *leak=new EventList(EVL_Event); session->eventlist[CPAP_Leak].push_back(leak); const int rbsize=128; EventDataType rbuf[rbsize],tmp,median; qint64 rtime[rbsize],ti; int rpos=0; int tcnt=0; QVector med; qint64 start; quint32 * tptr; EventStoreType * dptr; EventDataType gain; for (int i=0;ieventlist[CPAP_LeakTotal].size();i++) { EventList & el=*session->eventlist[CPAP_LeakTotal][i]; gain=el.gain(); dptr=el.rawData(); tptr=el.rawTime(); start=el.first(); for (unsigned j=0;j ti-winsize) // if fits in time window, add to the list med.push_back(rbuf[k]); } int idx=float(med.size() * 0.0); if (idx>=med.size()) idx--; nth_element(med.begin(),med.begin()+idx,med.end()); median=tmp-(*(med.begin()+idx)); if (median<0) median=0; leak->AddEvent(ti,median); rpos=rpos % rbsize; } } return leak->count(); } int calcPulseChange(Session *session) { if (session->eventlist.contains(OXI_PulseChange)) return 0; QHash >::iterator it=session->eventlist.find(OXI_Pulse); if (it==session->eventlist.end()) return 0; EventDataType val,val2,change,tmp; qint64 time,time2; qint64 window=PROFILE.oxi->pulseChangeDuration(); window*=1000; change=PROFILE.oxi->pulseChangeBPM(); EventList *pc=new EventList(EVL_Event,1,0,0,0,0,true); pc->setFirst(session->first(OXI_Pulse)); qint64 lastt; EventDataType lv=0; int li=0; int max; for (int e=0;e time+window) break; val2=el.data(j); tmp=qAbs(val2-val); if (tmp > lv) { lastt=time2; if (tmp>max) max=tmp; //lv=tmp; li=j; } } if (lastt>0) { qint64 len=(lastt-time)/1000.0; pc->AddEvent(lastt,len,tmp); i=li; } } } if (pc->count()==0) { delete pc; return 0; } session->eventlist[OXI_PulseChange].push_back(pc); session->setMin(OXI_PulseChange,pc->Min()); session->setMax(OXI_PulseChange,pc->Max()); session->setCount(OXI_PulseChange,pc->count()); session->setFirst(OXI_PulseChange,pc->first()); session->setLast(OXI_PulseChange,pc->last()); return pc->count(); } int calcSPO2Drop(Session *session) { if (session->eventlist.contains(OXI_SPO2Drop)) return 0; QHash >::iterator it=session->eventlist.find(OXI_SPO2); if (it==session->eventlist.end()) return 0; EventDataType val,val2,change,tmp; qint64 time,time2; qint64 window=PROFILE.oxi->spO2DropDuration(); window*=1000; change=PROFILE.oxi->spO2DropPercentage(); EventList *pc=new EventList(EVL_Event,1,0,0,0,0,true); qint64 lastt; EventDataType lv=0; int li=0; // Fix me.. Time scale varies. //const unsigned ringsize=30; //EventDataType ring[ringsize]={0}; //qint64 rtime[ringsize]={0}; //int rp=0; int min; int cnt=0; tmp=0; qint64 start=0; // Calculate median baseline QList med; for (int e=0;e0) med.push_back(val); if (!start) start=time; if (time > start+3600000) break; // just look at the first hour tmp+=val; cnt++; } } qSort(med); int midx=float(med.size())*0.90; if (midx>med.size()-1) midx=med.size()-1; EventDataType baseline=med[midx]; session->settings[OXI_SPO2Drop]=baseline; //EventDataType baseline=round(tmp/EventDataType(cnt)); EventDataType current; qDebug() << "Calculated baseline" << baseline; for (int e=0;e time-300000) { // only look at recent entries.. tmp+=ring[j]; cnt++; } } if (!cnt) { unsigned j=abs((rp-1) % ringsize); tmp=(ring[j]+val)/2; } else tmp/=EventDataType(cnt); */ val=baseline; lastt=0; lv=val; min=val; for (unsigned j=i;j time+window) break; val2=el.data(j); if (val2 > baseline-change) break; lastt=time2; li=j+1; } if (lastt>0) { qint64 len=(lastt-time); if (len>=window) { pc->AddEvent(lastt,len/1000,val-min); i=li; } } } } if (pc->count()==0) { delete pc; return 0; } session->eventlist[OXI_SPO2Drop].push_back(pc); session->setMin(OXI_SPO2Drop,pc->Min()); session->setMax(OXI_SPO2Drop,pc->Max()); session->setCount(OXI_SPO2Drop,pc->count()); session->setFirst(OXI_SPO2Drop,pc->first()); session->setLast(OXI_SPO2Drop,pc->last()); return pc->count(); }