From 7182c847861e5ce997273f943699d515732aec65 Mon Sep 17 00:00:00 2001 From: Mark Watkins Date: Mon, 21 Nov 2011 20:20:11 +1000 Subject: [PATCH] DeVilbiss Intellipap Loader --- SleepLib/day.cpp | 2 +- SleepLib/loader_plugins/intellipap_loader.cpp | 263 +++++++++++++++++- SleepLib/machine_common.h | 1 + SleepLib/session.cpp | 1 + daily.cpp | 1 + docs/channels.xml | 1 + docs/index.html | 4 +- mainwindow.cpp | 16 +- mainwindow.h | 4 + mainwindow.ui | 13 + 10 files changed, 288 insertions(+), 18 deletions(-) diff --git a/SleepLib/day.cpp b/SleepLib/day.cpp index 0d145228..401bf95c 100644 --- a/SleepLib/day.cpp +++ b/SleepLib/day.cpp @@ -22,7 +22,7 @@ Day::~Day() MachineType Day::machine_type() { return machine->GetType(); -}; +} void Day::AddSession(Session *s) { diff --git a/SleepLib/loader_plugins/intellipap_loader.cpp b/SleepLib/loader_plugins/intellipap_loader.cpp index c90b23a0..79de32d1 100644 --- a/SleepLib/loader_plugins/intellipap_loader.cpp +++ b/SleepLib/loader_plugins/intellipap_loader.cpp @@ -11,9 +11,12 @@ It does not seem to record multiple days, graph data is overwritten each time.. */ #include +#include #include "intellipap_loader.h" +extern QProgressBar *qprogress; + Intellipap::Intellipap(Profile *p,MachineID id) :CPAP(p,id) { @@ -41,6 +44,7 @@ IntellipapLoader::~IntellipapLoader() int IntellipapLoader::Open(QString & path,Profile *profile) { // Check for SL directory + // Check for DV5MFirm.bin? QString newpath; QString dirtag="SL"; @@ -51,41 +55,272 @@ int IntellipapLoader::Open(QString & path,Profile *profile) newpath=path+QDir::separator()+dirtag; } - unsigned char buf[27]; - QString filename=newpath+QDir::separator()+"U"; + QString filename; + + ////////////////////////// + // Parse the Settings File + ////////////////////////// + filename=newpath+QDir::separator()+"SET1"; QFile f(filename); if (!f.exists()) return 0; + f.open(QFile::ReadOnly); + QTextStream tstream(&f); - QHash Sessions; + QHash lookup; + lookup["Sn"]="Serial"; + lookup["Mn"]="Model"; + lookup["Mo"]="PAPMode"; // 0=cpap, 1=auto + //lookup["Pn"]="Pn"; + lookup["Pu"]="MaxPressure"; + lookup["Pl"]="MinPressure"; + //lookup["Ds"]="Ds"; + //lookup["Pc"]="Pc"; + lookup["Pd"]="RampPressure"; + lookup["Dt"]="RampTime"; + //lookup["Ld"]="Ld"; + //lookup["Lh"]="Lh"; + //lookup["FC"]="FC"; + //lookup["FE"]="FE"; + //lookup["FL"]="FL"; + lookup["A%"]="ApneaThreshold"; + lookup["Ad"]="ApneaDuration"; + lookup["H%"]="HypopneaThreshold"; + lookup["Hd"]="HypopneaDuration"; + //lookup["Pi"]="Pi"; //080 + //lookup["Pe"]="Pe"; //WF + //lookup["Ri"]="SmartFlexIRnd"; //1 + //lookup["Re"]="SmartFlexERnd"; //2 + //lookup["Bu"]="Bu"; //WF + //lookup["Ie"]="Ie"; //20 + //lookup["Se"]="Se"; //05 + //lookup["Si"]="Si"; //05 + //lookup["Mi"]="Mi"; //0 + //lookup["Uh"]="Uh"; //0000.0 + //lookup["Up"]="Up"; //0000.0 + //lookup["Er"]="ErrorCode"; // E00 + //lookup["El"]="LastErrorCode"; // E00 00/00/0000 + //lookup["Hp"]="Hp"; //1 + //lookup["Hs"]="Hs"; //02 + //lookup["Lu"]="LowUseThreshold"; // defaults to 0 (4 hours) + //lookup["Sf"]="SmartFlex"; + //lookup["Sm"]="SmartFlexMode"; + lookup["Ks=s"]="Ks_s"; + lookup["Ks=i"]="Ks_i"; - quint32 ts1, ts2, length; - unsigned char cs; + QHash set1; + QHash::iterator hi; + while (1) { + QString line=tstream.readLine(); + if ((line.length()<=2) || + (line.isNull())) break; + QString key=line.section("\t",0,0).trimmed(); + hi=lookup.find(key); + if (hi!=lookup.end()) { + key=hi.value(); + } + + QString value=line.section("\t",1).trimmed(); + set1[key]=value; + qDebug() << key << "=" << value; + } + + Machine *mach=NULL; + if (set1.contains("Serial")) { + mach=CreateMachine(set1["Serial"],profile); + } + if (!mach) { + qDebug() << "Couldn't get Intellipap machine record"; + return 0; + } + + // Refresh properties data.. + for (QHash::iterator i=set1.begin();i!=set1.end();i++) { + mach->properties[i.key()]=i.value(); + } + + f.close(); + + ////////////////////////// + // Parse the Session Index + ////////////////////////// + unsigned char buf[27]; + filename=newpath+QDir::separator()+"U"; + f.setFileName(filename); + if (!f.exists()) return 0; + + QVector SessionStart; + QVector SessionEnd; + QHash Sessions; + + quint32 ts1, ts2;//, length; + //unsigned char cs; f.open(QFile::ReadOnly); int cnt=0; - QDateTime epoch(QDate(2000,1,1),QTime(0,0,0),Qt::UTC); + QDateTime epoch(QDate(2002,1,1),QTime(0,0,0),Qt::UTC); // Intellipap Epoch int ep=epoch.toTime_t(); do { cnt=f.read((char *)buf,9); ts1=(buf[0] << 24) | (buf[1] << 16) | (buf[2] << 8) | buf[3]; ts2=(buf[4] << 24) | (buf[5] << 16) | (buf[6] << 8) | buf[7]; - Sessions[ts1]=ts2; - QDateTime d1=QDateTime::fromTime_t(ts1+ep); - QDateTime d2=QDateTime::fromTime_t(ts2+ep); - - length=ts2-ts1; - cs=buf[8]; - qDebug() << d1 << d2 << "Length: (" << length << ")"; + ts1+=ep; + ts2+=ep; + SessionStart.append(ts1); + SessionEnd.append(ts2); + //cs=buf[8]; } while (cnt>0); + qDebug() << "U file logs" << SessionStart.size() << "sessions."; f.close(); + ////////////////////////// + // Parse the Session Data + ////////////////////////// filename=newpath+QDir::separator()+"L"; f.setFileName(filename); if (!f.exists()) return 0; f.open(QFile::ReadOnly); + long size=f.size(); + int recs=size/26; + m_buffer=new unsigned char [size]; + if (size!=f.read((char *)m_buffer,size)) { + qDebug() << "Couldn't read 'L' data"<< filename; + return 0; + } - // Check for DV5MFirm.bin + Session *sess; + SessionID sid; + for (int i=0;iSessionExists(sid)) { + // knock out the already imported sessions.. + SessionStart[i]=0; + SessionEnd[i]=0; + } else if (!Sessions.contains(sid)) { + sess=Sessions[sid]=new Session(mach,sid); + sess->SetChanged(true); + sess->AddEventList(CPAP_IPAP,EVL_Event); + sess->AddEventList(CPAP_EPAP,EVL_Event); + sess->AddEventList(CPAP_Pressure,EVL_Event); + + sess->AddEventList(CPAP_Te,EVL_Event); + sess->AddEventList(CPAP_Ti,EVL_Event); + + sess->AddEventList(CPAP_Leak,EVL_Event); + sess->AddEventList(CPAP_MaxLeak,EVL_Event); + //sess->AddEventList(CPAP_AHI,EVL_Event); + sess->AddEventList(CPAP_TidalVolume,EVL_Event); + sess->AddEventList(CPAP_RespRate,EVL_Event); + sess->AddEventList(CPAP_Snore,EVL_Event); + } else { + // If there is a double up, null out the earlier session + // otherwise there will be a crash on shutdown. + for (int z=0;z=(quint32)sid) && (ts1eventlist[CPAP_Pressure][0]->AddEvent(time,m_buffer[pos+0xd]/10.0); // 0x0d + sess->eventlist[CPAP_EPAP][0]->AddEvent(time,m_buffer[pos+0x13]/10.0); + sess->eventlist[CPAP_IPAP][0]->AddEvent(time,m_buffer[pos+0x14]/10.0); + + sess->eventlist[CPAP_Leak][0]->AddEvent(time,m_buffer[pos+0x7]); //correct + sess->eventlist[CPAP_MaxLeak][0]->AddEvent(time,m_buffer[pos+0x6]); //correct + + sess->eventlist[CPAP_RespRate][0]->AddEvent(time,m_buffer[pos+0xa]); // 0x0a is correct + sess->eventlist[CPAP_Te][0]->AddEvent(time,m_buffer[pos+0xf]); + sess->eventlist[CPAP_Ti][0]->AddEvent(time,m_buffer[pos+0xc]); + + sess->eventlist[CPAP_Snore][0]->AddEvent(time,m_buffer[pos+0x5]); //4/5?? + + if (m_buffer[pos+0x5]>0) { + if (!sess->eventlist.contains(CPAP_VSnore)) { + sess->AddEventList(CPAP_VSnore,EVL_Event); + } + sess->eventlist[CPAP_VSnore][0]->AddEvent(time,m_buffer[pos+0x5]); + } + + if (m_buffer[pos+0x10]>0) { + if (!sess->eventlist.contains(CPAP_Obstructive)) { + sess->AddEventList(CPAP_Obstructive,EVL_Event); + } + sess->eventlist[CPAP_Obstructive][0]->AddEvent(time,m_buffer[pos+0x10]); + } + if (m_buffer[pos+0x11]>0) { + if (!sess->eventlist.contains(CPAP_Hypopnea)) { + sess->AddEventList(CPAP_Hypopnea,EVL_Event); + } + sess->eventlist[CPAP_Hypopnea][0]->AddEvent(time,m_buffer[pos+0x11]); + } + if (m_buffer[pos+0x12]>0) { + if (!sess->eventlist.contains(CPAP_Apnea)) { + sess->AddEventList(CPAP_Apnea,EVL_Event); + } + sess->eventlist[CPAP_Apnea][0]->AddEvent(time,m_buffer[pos+0x12]); + } + quint16 tv=(m_buffer[pos+0x8] << 8) | m_buffer[pos+0x9]; // correct + sess->eventlist[CPAP_TidalVolume][0]->AddEvent(time,tv); + break; + } + + } + pos+=26; + } + for (int i=0;ieventlist.size()==0) { + // delete sess; + // continue; + //} + quint64 first=quint64(sid)*1000L; + quint64 last=quint64(SessionEnd[i])*1000L; + quint64 len=last-first; + //if (len>0) { + //if (!sess->first()) { + sess->set_first(first); + sess->set_last(last); + // } + sess->UpdateSummaries(); + mach->AddSession(sess,profile); + /*} else { + delete sess; + }*/ + } + } + mach->properties["DataVersion"]=QString().sprintf("%i",intellipap_data_version); + + mach->Save(); + + delete [] m_buffer; + + if (qprogress) qprogress->setValue(100); + + f.close(); return 1; } diff --git a/SleepLib/machine_common.h b/SleepLib/machine_common.h index cdf49b60..b05e3538 100644 --- a/SleepLib/machine_common.h +++ b/SleepLib/machine_common.h @@ -87,6 +87,7 @@ const QString CPAP_RespRate="RespRate"; const QString CPAP_TidalVolume="TidalVolume"; const QString CPAP_PTB="PTB"; const QString CPAP_Leak="Leak"; +const QString CPAP_MaxLeak="MaxLeak"; const QString CPAP_FLG="FLG"; const QString CPAP_IE="IE"; const QString CPAP_Te="Te"; diff --git a/SleepLib/session.cpp b/SleepLib/session.cpp index f2a0b4c3..0eb2d455 100644 --- a/SleepLib/session.cpp +++ b/SleepLib/session.cpp @@ -414,6 +414,7 @@ void Session::UpdateSummaries() //sum(id); // avg calculates this and cnt. min(id); max(id); + count(id); if ((id==CPAP_FlowRate) || (id==CPAP_MaskPressure)) continue; cph(id); diff --git a/daily.cpp b/daily.cpp index 6d87830d..83934d82 100644 --- a/daily.cpp +++ b/daily.cpp @@ -178,6 +178,7 @@ Daily::Daily(QWidget *parent,gGraphView * shared, MainWindow *mw) PRD->AddLayer(AddCPAP(new gLineChart(CPAP_IPAP,Qt::red,square))); LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_Leak,Qt::darkYellow,square))); + LEAK->AddLayer(AddCPAP(new gLineChart(CPAP_MaxLeak,Qt::darkRed,square))); SNORE->AddLayer(AddCPAP(new gLineChart(CPAP_Snore,Qt::darkGray,true))); PTB->AddLayer(AddCPAP(new gLineChart(CPAP_PTB,Qt::gray,square))); diff --git a/docs/channels.xml b/docs/channels.xml index f215c4fb..b2d3d050 100644 --- a/docs/channels.xml +++ b/docs/channels.xml @@ -45,6 +45,7 @@ One id code per item + diff --git a/docs/index.html b/docs/index.html index e01d8b4a..cb6a0c13 100644 --- a/docs/index.html +++ b/docs/index.html @@ -19,7 +19,9 @@ p,a,td,body { font-size: 14px }

I don't recommend using this built in "web browser" to do any major surfing in, it will work, but it's mainly meant as a help browser. (It doesn't support SSL encryption.)

-

Here are the release notes for this version, in case you missed them.
+

+Here is a link to the SleepyHead Wiki
+Here are the release notes for this version, in case you missed them.
Plus a few usage notes, and some important information for Mac users.
SleepyHead's Project Website on SourceForge

The authors' boring and neglected Personal Blog

diff --git a/mainwindow.cpp b/mainwindow.cpp index 68d45c28..eefcb77e 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -103,8 +103,8 @@ MainWindow::MainWindow(QWidget *parent) : if (!PROFILE.Exists("EnableMultithreading")) PROFILE["EnableMultithreading"]=QThread::idealThreadCount()>1; if (!PROFILE.Exists("MemoryHog")) PROFILE["MemoryHog"]=false; - if (!PROFILE.Exists("EnableGraphSnapshots")) PROFILE["EnableGraphSnapshots"]=false; - if (!PROFILE.Exists("SquareWavePlots")) PROFILE["SquareWavePlots"]=true; + if (!PROFILE.Exists("EnableGraphSnapshots")) PROFILE["EnableGraphSnapshots"]=true; + if (!PROFILE.Exists("SquareWavePlots")) PROFILE["SquareWavePlots"]=false; if (!PROFILE.Exists("EnableOximetry")) PROFILE["EnableOximetry"]=false; if (!PROFILE.Exists("LinkGroups")) PROFILE["LinkGroups"]=false; if (!PROFILE.Exists("AlwaysShowOverlayBars")) PROFILE["AlwaysShowOverlayBars"]=0; @@ -568,3 +568,15 @@ void MainWindow::on_actionExp_ort_triggered() if (ex.exec()==ExportCSV::Accepted) { } } + +void MainWindow::on_actionOnline_Users_Guide_triggered() +{ + ui->webView->load(QUrl("http://sourceforge.net/apps/mediawiki/sleepyhead/index.php?title=SleepyHead_Users_Guide")); + ui->tabWidget->setCurrentIndex(0); +} + +void MainWindow::on_action_Frequently_Asked_Questions_triggered() +{ + ui->webView->load(QUrl("http://sourceforge.net/apps/mediawiki/sleepyhead/index.php?title=Frequently_Asked_Questions")); + ui->tabWidget->setCurrentIndex(0); +} diff --git a/mainwindow.h b/mainwindow.h index bf490e5e..b2237a00 100644 --- a/mainwindow.h +++ b/mainwindow.h @@ -106,6 +106,10 @@ private slots: void on_actionExp_ort_triggered(); + void on_actionOnline_Users_Guide_triggered(); + + void on_action_Frequently_Asked_Questions_triggered(); + private: Ui::MainWindow *ui; Daily * daily; diff --git a/mainwindow.ui b/mainwindow.ui index d642fe7d..73adf4eb 100644 --- a/mainwindow.ui +++ b/mainwindow.ui @@ -601,6 +601,9 @@ &Help + + + @@ -747,6 +750,16 @@ Exp&ort + + + Online Users &Guide + + + + + &Frequently Asked Questions + +