From 2053e2bfeca6db115cd1655e001f03189b0bd331 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Mon, 11 Jul 2011 00:23:07 +1000 Subject: [PATCH] Start of System One ASV stuff --- Graphs/gLineChart.cpp | 7 +- Graphs/graphwindow.cpp | 1 - SleepLib/loader_plugins/prs1_loader.cpp | 487 ++++++++++++++++----- SleepLib/loader_plugins/prs1_loader.h | 6 +- SleepLib/loader_plugins/sleep_database.cpp | 5 + SleepLib/loader_plugins/sleep_database.h | 33 ++ SleepLib/machine_common.h | 3 + SleepLib/machine_loader.cpp | 196 ++++++++- SleepLib/machine_loader.h | 33 +- SleepyHeadQT.pro | 6 +- daily.cpp | 7 +- 11 files changed, 656 insertions(+), 128 deletions(-) create mode 100644 SleepLib/loader_plugins/sleep_database.cpp create mode 100644 SleepLib/loader_plugins/sleep_database.h diff --git a/Graphs/gLineChart.cpp b/Graphs/gLineChart.cpp index 4b766031..f0184b58 100644 --- a/Graphs/gLineChart.cpp +++ b/Graphs/gLineChart.cpp @@ -77,7 +77,10 @@ void gLineChart::Plot(gGraphWindow & w,float scrx,float scry) } return; } - + bool square_plot=m_square_plot; + if (num_points>500) { + square_plot=false; + } bool accel=m_accelerate; double sr; int dp,sam; @@ -197,7 +200,7 @@ void gLineChart::Plot(gGraphWindow & w,float scrx,float scry) // better to do it here than in the main loop. np <<= 2; - if (!accel && m_square_plot) + if (!accel && square_plot) np <<= 1; // double it again if (np>=maxverts) { diff --git a/Graphs/graphwindow.cpp b/Graphs/graphwindow.cpp index 9679953c..33c59158 100644 --- a/Graphs/graphwindow.cpp +++ b/Graphs/graphwindow.cpp @@ -555,7 +555,6 @@ void gGraphWindow::OnMouseLeftDown(QMouseEvent * event) } void gGraphWindow::dropEvent ( QDropEvent * event ) { - int frong=23; assert(splitter!=NULL); //m_dragGraph=false; } diff --git a/SleepLib/loader_plugins/prs1_loader.cpp b/SleepLib/loader_plugins/prs1_loader.cpp index 9f6a5a5d..b0a64d57 100644 --- a/SleepLib/loader_plugins/prs1_loader.cpp +++ b/SleepLib/loader_plugins/prs1_loader.cpp @@ -17,6 +17,13 @@ License: GPL #include "prs1_loader.h" #include "SleepLib/session.h" + +const int PRS1_MAGIC_NUMBER=2; +const int PRS1_SUMMARY_FILE=1; +const int PRS1_EVENT_FILE=2; +const int PRS1_WAVEFORM_FILE=5; + + //******************************************************************************************** /// IMPORTANT!!! //******************************************************************************************** @@ -277,8 +284,6 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile) if (s->second[0].isEmpty()) continue; Session *sess=new Session(m,session); - if (session==0x112) - int q=0; if (!OpenSummary(sess,s->second[0])) { qWarning() << "PRS1Loader: Could'nt open summary file " << s->second[0]; @@ -379,14 +384,14 @@ int PRS1Loader::OpenMachine(Machine *m,QString path,Profile *profile) bool PRS1Loader::OpenSummary(Session *session,QString filename) { - int size,sequence,seconds,br; + int size,sequence,seconds,br,htype,version; qint64 timestamp; unsigned char header[24]; unsigned char ext,sum; //qDebug() << "Opening PRS1 Summary " << filename; QFile f(filename); - //,wxT("rb")); + if (!f.open(QIODevice::ReadOnly)) return false; @@ -397,7 +402,7 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename) br=f.read((char *)header,hl); - if (header[0]!=header[5]) + if (header[0]!=PRS1_MAGIC_NUMBER) return false; sequence=size=timestamp=seconds=ext=0; @@ -405,13 +410,15 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename) timestamp=(header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11]; size=(header[2] << 8) | header[1]; ext=header[6]; + htype=header[3]; // 00 = normal // 01=waveform // could be a bool? + version=header[4]; - if (ext!=1) + if (ext!=PRS1_SUMMARY_FILE) return false; - //size|=(header[3]<<16) | (header[4]<<24); // the jury is still out on the 32bitness of one. doesn't matter here anyway. - size-=(hl+2); + + // Calculate header checksum and compare to verify header sum=0; for (int i=0; iset_first(date); + //session->set_last(date); double max; session->summary[CPAP_PressureMin]=(double)buffer[0x03]/10.0; session->summary[CPAP_PressureMax]=max=(double)buffer[0x04]/10.0; - session->summary[CPAP_RampTime]=(int)buffer[0x06]; // Minutes. Convert to seconds/hours here? - session->summary[CPAP_RampStartingPressure]=(double)buffer[0x07]/10.0; + int offset=0; + if (buffer[0x05]!=0) { // This is a time value for ASV stuff + // non zero adds extra fields.. + offset=4; + } + + session->summary[CPAP_RampTime]=(int)buffer[offset+0x06]; // Minutes. Convert to seconds/hours here? + session->summary[CPAP_RampStartingPressure]=(double)buffer[offset+0x07]/10.0; if (max>0) { // Ignoring bipap until I see some more data. session->summary[CPAP_Mode]=(int)MODE_APAP; } else session->summary[CPAP_Mode]=(int)MODE_CPAP; // This is incorrect.. - if (buffer[0x08] & 0x80) { // Flex Setting - if (buffer[0x08] & 0x08) { + if (buffer[offset+0x08] & 0x80) { // Flex Setting + if (buffer[offset+0x08] & 0x08) { if (max>0) session->summary[CPAP_PressureReliefType]=(int)PR_AFLEX; else session->summary[CPAP_PressureReliefType]=(int)PR_CFLEXPLUS; } else session->summary[CPAP_PressureReliefType]=(int)PR_CFLEX; } else session->summary[CPAP_PressureReliefType]=(int)PR_NONE; - session->summary[CPAP_PressureReliefSetting]=(int)buffer[0x08] & 3; - session->summary[CPAP_HumidifierSetting]=(int)buffer[0x09]&0x0f; - session->summary[CPAP_HumidifierStatus]=(buffer[0x09]&0x80)==0x80; - session->summary[PRS1_SystemLockStatus]=(buffer[0x0a]&0x80)==0x80; - session->summary[PRS1_SystemResistanceStatus]=(buffer[0x0a]&0x40)==0x40; - session->summary[PRS1_SystemResistanceSetting]=(int)buffer[0x0a]&7; - session->summary[PRS1_HoseDiameter]=(int)((buffer[0x0a]&0x08)?15:22); - session->summary[PRS1_AutoOff]=(buffer[0x0c]&0x10)==0x10; - session->summary[PRS1_MaskAlert]=(buffer[0x0c]&0x08)==0x08; - session->summary[PRS1_ShowAHI]=(buffer[0x0c]&0x04)==0x04; + session->summary[CPAP_PressureReliefSetting]=(int)buffer[offset+0x08] & 3; + session->summary[CPAP_HumidifierSetting]=(int)buffer[offset+0x09]&0x0f; + session->summary[CPAP_HumidifierStatus]=(buffer[offset+0x09]&0x80)==0x80; + session->summary[PRS1_SystemLockStatus]=(buffer[offset+0x0a]&0x80)==0x80; + session->summary[PRS1_SystemResistanceStatus]=(buffer[offset+0x0a]&0x40)==0x40; + session->summary[PRS1_SystemResistanceSetting]=(int)buffer[offset+0x0a]&7; + session->summary[PRS1_HoseDiameter]=(int)((buffer[offset+0x0a]&0x08)?15:22); + session->summary[PRS1_AutoOff]=(buffer[offset+0x0c]&0x10)==0x10; + session->summary[PRS1_MaskAlert]=(buffer[offset+0x0c]&0x08)==0x08; + session->summary[PRS1_ShowAHI]=(buffer[offset+0x0c]&0x04)==0x04; - unsigned char * b=&buffer[0x14]; + unsigned char * b=&buffer[offset+0x14]; quint16 bb=*(quint16*)b; unsigned duration=bb;// | (buffer[0x15] << 8); session->summary[CPAP_Duration]=(int)duration; //qDebug() << "ID: " << session->session() << " " << duration; //float hours=float(duration)/3600.0; //session->set_hours(hours); - if (!duration) return false; + if (!duration) + return false; session->set_last(date+qint64(duration)*1000L); - session->summary[CPAP_PressureMinAchieved]=buffer[0x16]/10.0; - session->summary[CPAP_PressureMaxAchieved]=buffer[0x17]/10.0; - session->summary[CPAP_PressureAverage]=buffer[0x18]/10.0; - session->summary[CPAP_PressurePercentValue]=buffer[0x19]/10.0; + session->summary[CPAP_PressureMinAchieved]=buffer[offset+0x16]/10.0; + session->summary[CPAP_PressureMaxAchieved]=buffer[offset+0x17]/10.0; + session->summary[CPAP_PressureAverage]=buffer[offset+0x18]/10.0; + session->summary[CPAP_PressurePercentValue]=buffer[offset+0x19]/10.0; session->summary[CPAP_PressurePercentName]=90.0; if (max==0) { @@ -486,16 +502,17 @@ bool PRS1Loader::OpenSummary(Session *session,QString filename) } if (size==0x4d) { - session->summary[CPAP_Obstructive]=(int)buffer[0x1C] | (buffer[0x1D] << 8); - session->summary[CPAP_ClearAirway]=(int)buffer[0x20] | (buffer[0x21] << 8); - session->summary[CPAP_Hypopnea]=(int)buffer[0x2A] | (buffer[0x2B] << 8); - session->summary[CPAP_RERA]=(int)buffer[0x2E] | (buffer[0x2F] << 8); - session->summary[CPAP_FlowLimit]=(int)buffer[0x30] | (buffer[0x31] << 8); + session->summary[CPAP_Obstructive]=(int)buffer[offset+0x1C] | (buffer[offset+0x1D] << 8); + session->summary[CPAP_ClearAirway]=(int)buffer[offset+0x20] | (buffer[offset+0x21] << 8); + session->summary[CPAP_Hypopnea]=(int)buffer[offset+0x2A] | (buffer[offset+0x2B] << 8); + session->summary[CPAP_RERA]=(int)buffer[offset+0x2E] | (buffer[offset+0x2F] << 8); + session->summary[CPAP_FlowLimit]=(int)buffer[offset+0x30] | (buffer[offset+0x31] << 8); } return true; } -bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp) +// v2 event parser. +bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp,int version) { MachineCode Codes[]={ PRS1_Unknown00, PRS1_Unknown01, CPAP_Pressure, CPAP_EAP, PRS1_PressurePulse, CPAP_RERA, CPAP_Obstructive, CPAP_ClearAirway, @@ -504,29 +521,43 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64 }; int ncodes=sizeof(Codes)/sizeof(MachineCode); - EventDataType data[4]; + EventDataType data[10]; qint64 start=timestamp; qint64 t=timestamp; qint64 tt; int pos=0; int cnt=0; - short delta; + short delta,duration; while (pos=ncodes) { + qDebug() << "Illegal PRS1 code " << hex << int(code) << " appeared at " << hex << pos+16; + return false; + } + //assert(codeAddEvent(new Event(t,cpapcode, data,0)); + break; + case 0x00: // Unknown (ASV Pressure value) // could this be RLE? + // offset? + data[0]=buffer[pos++]; + session->AddEvent(new Event(t,cpapcode, data,1)); + break; case 0x02: // Pressure data[0]=buffer[pos++]/10.0; session->AddEvent(new Event(t,cpapcode, data,1)); @@ -536,41 +567,57 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64 session->AddEvent(new Event(t,cpapcode, data,1)); break; + case 0x0a: // Hypopnea case 0x05: // RERA case 0x06: // Obstructive Apoanea case 0x07: // Clear Airway - case 0x0a: // Hypopnea case 0x0c: // Flow Limitation - data[0]=buffer[pos]; - - tt-=buffer[pos++]*1000; // Subtract Time Offset - + data[0]=buffer[pos++]; + tt-=data[0]*1000; // Subtract Time Offset session->AddEvent(new Event(tt,cpapcode,data,1)); break; + //case 0x0b: // ASV Codes + // fc++; + // data[0]=buffer[pos++]; + // tt-=data[0]*1000; // Subtract Time Offset + // fc++; + //data[1]=buffer[pos++]; + //session->AddEvent(new Event(tt,cpapcode,data,2)); + //break; + //case 0x08: // ASV Codes + //case 0x09: // ASV Codes + // data[0]=buffer[pos]; + // tt-=buffer[pos++]*1000; // Subtract Time Offset + // session->AddEvent(new Event(tt,cpapcode,data,1)); + // break; case 0x0d: // Vibratory Snore session->AddEvent(new Event(t,cpapcode, data,0)); break; case 0x03: // BIPAP Pressure - case 0x0b: // Unknown + data[0]=buffer[pos++]; + data[1]=buffer[pos++]; + data[0]/=10.0; + data[1]/=10.0; + session->AddEvent(new Event(t,CPAP_EAP, data, 1)); + session->AddEvent(new Event(t,CPAP_IAP, &data[1], 1)); + break; case 0x11: // Leak Rate data[0]=buffer[pos++]; data[1]=buffer[pos++]; - if (code==0x11) { - session->AddEvent(new Event(t,cpapcode, data,1)); - session->AddEvent(new Event(t,CPAP_Snore,&data[1],1)); - if (data[1]>0) { - session->AddEvent(new Event(t,PRS1_VSnore2, &data[1],1)); - } - } else if (code==0x03) { - data[0]/=10.0; - data[1]/=10.0; - session->AddEvent(new Event(t,CPAP_EAP, data, 1)); - session->AddEvent(new Event(t,CPAP_IAP, &data[1], 1)); - } else { - session->AddEvent(new Event(t,cpapcode, data, 2)); + session->AddEvent(new Event(t,cpapcode, data,1)); + session->AddEvent(new Event(t,CPAP_Snore,&data[1],1)); + if (data[1]>0) { + session->AddEvent(new Event(t,PRS1_VSnore2, &data[1],1)); } break; case 0x0e: // Unknown + + data[0]=buffer[pos++]; // << 8) | buffer[pos]; + data[1]=buffer[pos++]; + data[2]=buffer[pos++]; + session->AddEvent(new Event(t,cpapcode, data, 3)); + break; + case 0x10: // Unknown data[0]=buffer[pos++]; // << 8) | buffer[pos]; data[1]=buffer[pos++]; @@ -593,21 +640,181 @@ bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,qint64 break; default: // ERROR!!! - qWarning() << "Some new fandangled PRS1 code detected:" << code; + qWarning() << "Some new fandangled PRS1 code detected " << hex << int(code) << " at " << pos+15; return false; } } return true; } +bool PRS1Loader::Parse002ASV(Session *session,unsigned char *buffer,int size,qint64 timestamp,int version) +{ + MachineCode Codes[]={ + PRS1_Unknown00, PRS1_Unknown01, CPAP_Pressure, CPAP_EAP, PRS1_PressurePulse, CPAP_Obstructive, CPAP_ClearAirway, CPAP_Hypopnea, + PRS1_Unknown08, CPAP_FlowLimit, PRS1_Unknown0A, CPAP_CSR, PRS1_Unknown0C, CPAP_VSnore, PRS1_Unknown0E, PRS1_Unknown0F, PRS1_Unknown10, + CPAP_Leak, PRS1_Unknown12 + }; + + int ncodes=sizeof(Codes)/sizeof(MachineCode); + + EventDataType data[10]; + + qint64 start=timestamp; + qint64 t=timestamp; + qint64 tt; + int pos=0; + int cnt=0; + short delta,duration; + while (pos=ncodes) { + qDebug() << "Illegal PRS1 code " << hex << int(code) << " appeared at " << hex << pos+16; + return false; + } + //assert(codeAddEvent(new Event(t,cpapcode, data,0)); + break; + case 0x00: // Unknown (ASV Pressure value) // could this be RLE? + // offset? + data[0]=buffer[pos++]; + fc++; + if (!buffer[pos-1]) { + data[1]=buffer[pos++]; + fc++; + } + if (!buffer[pos-1]) { + data[2]=buffer[pos++]; + fc++; + } + session->AddEvent(new Event(t,cpapcode, data,2)); + break; + case 0x02: // Pressure + data[0]=buffer[pos++]/10.0; + session->AddEvent(new Event(t,cpapcode, data,1)); + break; + case 0x04: // Pressure Pulse + data[0]=buffer[pos++]; + session->AddEvent(new Event(t,cpapcode, data,1)); + break; + + case 0x0a: + code=0x0a; + case 0x05: + case 0x06: + case 0x07: + case 0x0c: + data[0]=buffer[pos++]; + tt-=data[0]*1000; // Subtract Time Offset + session->AddEvent(new Event(tt,cpapcode,data,1)); + break; + case 0x0b: // Cheyne Stokes + data[0]=((unsigned char *)buffer)[pos+1]<<8 | ((unsigned char *)buffer)[pos]; + data[0]*=2; + pos+=2; + data[1]=((unsigned char *)buffer)[pos]; //|buffer[pos+1] << 8 + pos+=1; + //tt-=delta; + tt-=data[1]*1000; + session->AddEvent(new Event(tt,cpapcode, data, 2)); + //tt+=delta; + break; + case 0x08: // ASV Codes + case 0x09: // ASV Codes + data[0]=buffer[pos]; + tt-=buffer[pos++]*1000; // Subtract Time Offset + session->AddEvent(new Event(tt,cpapcode,data,1)); + break; + case 0x0d: // All the other ASV graph stuff. + data[0]=buffer[pos++]; + if (buffer[pos]<=0x12) { // short type of 0d record + // Type?? + session->AddEvent(new Event(t,cpapcode, data,1)); + } else { + data[1]=buffer[pos++]; + data[2]=buffer[pos++]; + data[3]=buffer[pos++]; + data[4]=buffer[pos++]; // ?? + data[5]=buffer[pos++]; // Patient Triggered Breaths data[6]=buffer[pos++]; // Patient Triggered Breaths + data[6]=buffer[pos++]; // Patient Triggered Breaths data[6]=buffer[pos++]; // Patient Triggered Breaths + data[7]=buffer[pos++]; // Patient Triggered Breaths + data[8]=buffer[pos++]; // Patient Triggered Breaths + data[9]=buffer[pos++]; // Patient Triggered Breaths + //session->AddEvent(new Event(t,cpapcode, data,6)); + } + break; + case 0x03: // BIPAP Pressure + data[0]=buffer[pos++]; + data[1]=buffer[pos++]; + data[0]/=10.0; + data[1]/=10.0; + session->AddEvent(new Event(t,CPAP_EAP, data, 1)); + session->AddEvent(new Event(t,CPAP_IAP, &data[1], 1)); + break; + case 0x11: // Leak Rate + data[0]=buffer[pos++]; + session->AddEvent(new Event(t,cpapcode, data,1)); + break; + case 0x0e: // Unknown + data[0]=buffer[pos++]; // << 8) | buffer[pos]; + session->AddEvent(new Event(t,cpapcode, data, 1)); + break; + + case 0x10: // Unknown + data[0]=buffer[pos++]; // << 8) | buffer[pos]; + data[1]=buffer[pos++]; + data[2]=buffer[pos++]; + session->AddEvent(new Event(t,cpapcode, data, 3)); + break; + case 0x0f: + data[0]=buffer[pos+1]<<8 | buffer[pos]; + pos+=2; + data[1]=buffer[pos]; //|buffer[pos+1] << 8 + pos+=1; + tt-=data[1]*1000; + session->AddEvent(new Event(tt,cpapcode, data, 2)); + break; + case 0x12: // Summary + data[0]=buffer[pos++]; + data[1]=buffer[pos++]; + data[2]=buffer[pos+1]<<8 | buffer[pos]; + pos+=2; + session->AddEvent(new Event(t,cpapcode, data,3)); + break; + default: + // ERROR!!! + qWarning() << "Some new fandangled PRS1 code detected " << hex << int(code) << " at " << pos+15; + return false; + } + } + return true; +} + + bool PRS1Loader::OpenEvents(Session *session,QString filename) { - int size,sequence,seconds,br; + int size,sequence,seconds,br,version; qint64 timestamp; unsigned char header[24]; // use m_buffer? - unsigned char ext; + unsigned char ext,htype; - //wxLogMessage(wxT("Opening PRS1 Events ")+filename); QFile f(filename); if (!f.open(QIODevice::ReadOnly)) return false; @@ -616,7 +823,7 @@ bool PRS1Loader::OpenEvents(Session *session,QString filename) br=f.read((char *)header,hl); - if (header[0]!=header[5]) + if (header[0]!=PRS1_MAGIC_NUMBER) return false; sequence=size=timestamp=seconds=ext=0; @@ -624,8 +831,10 @@ bool PRS1Loader::OpenEvents(Session *session,QString filename) timestamp=(header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11]; size=(header[2] << 8) | header[1]; ext=header[6]; + htype=header[3]; // 00 = normal // 01=waveform // could be a bool? + version=header[4];// | header[4]; - if (ext!=2) + if (ext!=PRS1_EVENT_FILE) // 2 == Event file return false; //size|=(header[3]<<16) | (header[4]<<24); // the jury is still out on the 32bitness of one. doesn't matter here anyway. @@ -641,35 +850,46 @@ bool PRS1Loader::OpenEvents(Session *session,QString filename) if (br=max_load_buffer_size) { qWarning("max_load_buffer_size is too small in PRS1 Loader"); if (cnt==0) @@ -721,8 +966,6 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename) break; } - //wavedata.push_back(buffer); - //wavesize.push_back(size); cnt++; duration+=seconds; @@ -740,30 +983,52 @@ bool PRS1Loader::OpenWaveforms(Session *session,QString filename) if (samples==0) return false; - //double rate=double(duration)/double(samples); + // Create the buffers for real sample data + SampleFormat * data[numsignals]; + SampleFormat min[numsignals],max[numsignals]; + bool first[numsignals]; + int pos[numsignals]; - // Convert to SampleFormat - SampleFormat *data=new SampleFormat [samples]; - - SampleFormat min=0,max=0; - bool first=true; + int samplespersignal=samples/numsignals; // TODO: divide by type? + int ichunksize=0; // interleave chunk size + for (int i=0;ic) min[s]=c; + if (max[s]c) min=c; - if (maxAddWaveform(w); - + if (numsignals>1) { + Waveform *w=new Waveform(start,CPAP_MaskPressure,data[1],pos[1],duration,min[1],max[1]); + session->AddWaveform(w); + } return true; } diff --git a/SleepLib/loader_plugins/prs1_loader.h b/SleepLib/loader_plugins/prs1_loader.h index f1a49a98..461954f4 100644 --- a/SleepLib/loader_plugins/prs1_loader.h +++ b/SleepLib/loader_plugins/prs1_loader.h @@ -33,7 +33,7 @@ public: }; -const int max_load_buffer_size=384*1024; +const int max_load_buffer_size=1024*1024; const QString prs1_class_name="PRS1"; @@ -57,7 +57,9 @@ protected: bool OpenSummary(Session *session,QString filename); bool OpenEvents(Session *session,QString filename); bool OpenWaveforms(Session *session,QString filename); - bool Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp); + bool Parse002(Session *session,unsigned char *buffer,int size,qint64 timestamp,int version); + bool Parse002ASV(Session *session,unsigned char *buffer,int size,qint64 timestamp,int version); + unsigned char * m_buffer; }; diff --git a/SleepLib/loader_plugins/sleep_database.cpp b/SleepLib/loader_plugins/sleep_database.cpp new file mode 100644 index 00000000..73bfd29e --- /dev/null +++ b/SleepLib/loader_plugins/sleep_database.cpp @@ -0,0 +1,5 @@ +#include "sleep_database.h" + +/*database::database() +{ +}*/ diff --git a/SleepLib/loader_plugins/sleep_database.h b/SleepLib/loader_plugins/sleep_database.h new file mode 100644 index 00000000..8641bcc7 --- /dev/null +++ b/SleepLib/loader_plugins/sleep_database.h @@ -0,0 +1,33 @@ +#ifndef DATABASE_H +#define DATABASE_H + +//******************************************************************************************** +/// IMPORTANT!!! +//******************************************************************************************** +// Please INCREMENT the following value when making changes to this loaders implementation. +// +const int sleeplib_data_version=1; +// +//******************************************************************************************** + +/*#include +#include "SleepLib/profiles.h" +const QString sleeplib_class_name="SleepDB"; + +class SleepDatabase : public MachineLoader +{ +public: + SleepDatabase(); + virtual ~SleepDatabase(); + virtual bool Open(QString & path,Profile *profile); + virtual int Version() { return sleeplib_data_version; }; + virtual const QString & ClassName() { return sleeplib_class_name; }; + Machine *CreateMachine(QString serial,Profile *profile); + + static void Register(); +protected: + map MachList; + +};*/ + +#endif // DATABASE_H diff --git a/SleepLib/machine_common.h b/SleepLib/machine_common.h index 1780bbe1..d4aad891 100644 --- a/SleepLib/machine_common.h +++ b/SleepLib/machine_common.h @@ -51,6 +51,9 @@ enum MachineCode//:qint16 PRS1_Unknown00, PRS1_Unknown01, PRS1_Unknown08, PRS1_Unknown09, PRS1_Unknown0B, PRS1_Unknown0E, PRS1_Unknown10, PRS1_Unknown12, PRS1_SystemLockStatus, PRS1_SystemResistanceStatus, PRS1_SystemResistanceSetting, PRS1_HoseDiameter, PRS1_AutoOff, PRS1_MaskAlert, PRS1_ShowAHI, + // ASV Unknown Codes + PRS1_Unknown0A,PRS1_Unknown0C, PRS1_Unknown0F, + // Oximeter Codes OXI_Pulse=0x2000, OXI_SPO2, OXI_Plethy, OXI_Signal2, OXI_SignalGood, OXI_PulseAverage, OXI_PulseMin, OXI_PulseMax, OXI_SPO2Average, OXI_SPO2Min, OXI_SPO2Max, diff --git a/SleepLib/machine_loader.cpp b/SleepLib/machine_loader.cpp index 59ffeebe..3ed98000 100644 --- a/SleepLib/machine_loader.cpp +++ b/SleepLib/machine_loader.cpp @@ -4,11 +4,20 @@ License: GPL *********************************************************************/ +#include +#include +#include "tinyxml/tinyxml.h" + #include "machine_loader.h" - +// This crap moves to Profile list m_loaders; +list GetLoaders() +{ + return m_loaders; +} + void RegisterLoader(MachineLoader *loader) { m_loaders.push_back(loader); @@ -23,15 +32,192 @@ void DestroyLoaders() MachineLoader::MachineLoader() { - } +/*MachineLoader::MachineLoader(Profile * profile, QString & classname, MachineType type) +:m_profile(profile), m_class(classname), m_type(type) +{ + assert(m_profile!=NULL); + assert(!m_classname.isEmpty()); +}*/ MachineLoader::~MachineLoader() { - + for (vector::iterator m=m_machlist.begin();m!=m_machlist.end();m++) { + delete *m; + } } +/*const QString machine_profile_name="MachineList.xml"; -list GetLoaders() +void MachineLoader::LoadMachineList() { - return m_loaders; + QString filename=(*profile)["ProfileDirectory"]+"/"+m_classname+"/"+machine_profile_name; + + QFile f(filename); + if (!f.exists()) { + qDebug() << "XML file does not exist" << filename; + return; + } + TiXmlDocument xml(filename.toLatin1()); + if (!xml.LoadFile()) { + qDebug() << "Couldn't read XML file " << filename; + return; + } + TiXmlHandle hDoc(&xml); + TiXmlElement * pElem; + //TiXmlHandle hRoot(0); + pElem=hDoc.FirstChildElement().Element(); + + if (!pElem) { + qDebug("MachineList is empty."); + return; + } + + //hRoot=TiXmlHandle(pElem); + //pElem=hRoot.FirstChild("MachineList").FirstChild().Element(); + + if (pElem->Value()!="MachineList") { + qDebug() << "MachineLoader::LoadMachineList expected a MachineList"; + } + + int mt; + + Machine *mach; + pElem->QueryIntAttribute("type",&mt); + MachineType m_type=(MachineType)mt; + QString m_class=pElem->Attribute("class"); + + TiXmlElement *elem; + elem=pElem->FirstChildElement(); + if (!elem) { + qDebug("Machine is empty."); + return; + } + + int m_id; + for(; elem; elem=elem->NextSiblingElement()) { + QString pKey=elem->Value(); + if (!pKey=="Machine") continue; + + elem->QueryIntAttribute("id",&m_id); + + mach=CreateMachine(m_id); + + TiXmlElement *e=elem->FirstChildElement(); + for (; e; e=e->NextSiblingElement()) { + QString pKey=e->Value(); + mach->properties[pKey]=e->GetText(); + } +// QString filename=(*profile)["ProfileDirectory"]+"/"+m_classname+"/"+mach->hexid(); +// mach->LoadSummaries(filename); + } } +void MachineLoader::StoreMachineList() +{ + QString filename=(*profile)["ProfileDirectory"]+"/"+m_classname+"/"+machine_profile_name; + + TiXmlDocument xml; + TiXmlElement* msg; + TiXmlComment * comment; + TiXmlDeclaration *decl=new TiXmlDeclaration( "1.0", "", "" ); + xml.LinkEndChild(decl); + TiXmlElement *root=new TiXmlElement("MachineList"); + + char *cc=m_class.toLatin1().data(); + root->SetAttribute("type",(int)m_type); + root->SetAttribute("class",cc); + + xml.LinkEndChild(root); + + if (!m_comment.isEmpty()) { + comment = new TiXmlComment(); + comment->SetValue((QString(" ")+m_comment+QString(" ")).toLatin1()); + root->LinkEndChild(comment); + } + + for (int i=0;iSetAttribute("id",m->id()); + + for (map::iterator j=i->second->properties.begin(); j!=i->second->properties.end(); j++) { + TiXmlElement *mp=new TiXmlElement(j->first.toLatin1()); + mp->LinkEndChild(new TiXmlText(j->second.toLatin1())); + me->LinkEndChild(mp); + } + root->LinkEndChild(me); + } + xml.SaveFile(p_filename.toLatin1()); +} +void MachineLoader::LoadSummary(Machine *m, QString &filename) +{ + QFile f(filename); + if (!f.exists()) + return; + f.open(QIODevice::ReadOnly); + if (!f.isReadable()) + return; + + +} +void MachineLoader::LoadSummaries(Machine *m) +{ + QString path=(*profile)["ProfileDirectory"]+"/"+m_classname+"/"+mach->hexid(); + QDir dir(path); + if (!dir.exists() || !dir.isReadable()) + return false; + + dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks); + dir.setSorting(QDir::Name); + + QString fullpath,ext_s,sesstr; + int ext; + SessionID sessid; + bool ok; + QMap sessions; + QFileInfoList list=dir.entryInfoList(); + for (int i=0;i m_machlist; + QString m_class; + MachineType m_type; + Profile * m_profile; }; void RegisterLoader(MachineLoader *loader); diff --git a/SleepyHeadQT.pro b/SleepyHeadQT.pro index 9f1ebf55..0ae9d257 100644 --- a/SleepyHeadQT.pro +++ b/SleepyHeadQT.pro @@ -58,7 +58,8 @@ SOURCES += main.cpp\ Graphs/gCandleStick.cpp \ Graphs/gBarChart.cpp \ SleepLib/loader_plugins/resmed_loader.cpp \ - Graphs/gpiechart.cpp + Graphs/gpiechart.cpp \ + SleepLib/loader_plugins/sleep_database.cpp HEADERS += \ SleepLib/binary_file.h \ @@ -94,7 +95,8 @@ HEADERS += \ Graphs/gCandleStick.h \ Graphs/gBarChart.h \ SleepLib/loader_plugins/resmed_loader.h \ - Graphs/gpiechart.h + Graphs/gpiechart.h \ + SleepLib/loader_plugins/sleep_database.h FORMS += \ daily.ui \ diff --git a/daily.cpp b/daily.cpp index 99b85d8c..82df8744 100644 --- a/daily.cpp +++ b/daily.cpp @@ -97,7 +97,7 @@ Daily::Daily(QWidget *parent,QGLContext *context) : PRD->AddLayer(new gXAxis()); PRD->AddLayer(new gYAxis()); //PRD->AddLayer(new gFooBar()); - bool square=false; + bool square=true; PRD->AddLayer(new gLineChart(prd,QColor("dark green"),4096,false,false,square)); PRD->AddLayer(new gLineChart(pressure_iap,Qt::blue,4096,false,true,square)); PRD->AddLayer(new gLineChart(pressure_eap,Qt::red,4096,false,true,square)); @@ -116,14 +116,14 @@ Daily::Daily(QWidget *parent,QGLContext *context) : AddCPAPData(frw=new WaveData(CPAP_FlowRate,1000000)); //FlowRate - // AddCPAPData(mpw=new WaveData(CPAP_MaskPressure,700000)); //FlowRate + //AddCPAPData(mpw=new WaveData(CPAP_MaskPressure,700000)); //FlowRate // Holy crap resmed stuff is huge.. FRW=new gGraphWindow(gSplitter,tr("Flow Rate"),SF); //FRW->AddLayer(new gFooBar()); FRW->AddLayer(new gYAxis()); FRW->AddLayer(new gXAxis()); FRW->AddLayer(new gLineOverlayBar(flags[0],QColor("light green"),"CSR")); - //FRW->AddLayer(new gLineChart(mpw,Qt::blue,700000,true)); + // FRW->AddLayer(new gLineChart(mpw,Qt::blue,700000,true)); gLineChart *g=new gLineChart(frw,Qt::black,4000,true); g->ReportEmpty(true); FRW->AddLayer(g); @@ -428,7 +428,6 @@ void Daily::UpdateEventsTree(QTreeWidget *tree,Day *day) // Note this is not so evil on PRS1. if (code==CPAP_Pressure) continue; - if (code==CPAP_Snore) continue; if (code==PRS1_Unknown12) continue; QTreeWidgetItem *mcr;