2011-06-28 02:21:38 +00:00
|
|
|
/*
|
|
|
|
|
|
|
|
SleepLib ResMed Loader Implementation
|
|
|
|
|
|
|
|
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
|
|
|
License: GPL
|
|
|
|
*/
|
|
|
|
|
2011-07-30 00:36:31 +00:00
|
|
|
|
|
|
|
#include <QApplication>
|
2011-06-28 02:21:38 +00:00
|
|
|
#include <QString>
|
|
|
|
#include <QDateTime>
|
|
|
|
#include <QDir>
|
|
|
|
#include <QFile>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QProgressBar>
|
2011-07-01 10:10:44 +00:00
|
|
|
#include <QDebug>
|
2011-07-27 09:21:53 +00:00
|
|
|
#include <cmath>
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
#include "resmed_loader.h"
|
|
|
|
#include "SleepLib/session.h"
|
|
|
|
|
2011-06-30 04:55:20 +00:00
|
|
|
extern QProgressBar *qprogress;
|
2011-07-31 20:24:43 +00:00
|
|
|
QHash<int,QString> RMS9ModelMap;
|
2011-06-30 04:55:20 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
EDFParser::EDFParser(QString name)
|
|
|
|
{
|
|
|
|
buffer=NULL;
|
|
|
|
Open(name);
|
|
|
|
}
|
|
|
|
EDFParser::~EDFParser()
|
|
|
|
{
|
2011-07-31 20:24:43 +00:00
|
|
|
QVector<EDFSignal *>::iterator s;
|
2011-06-29 14:19:38 +00:00
|
|
|
for (s=edfsignals.begin();s!=edfsignals.end();s++) {
|
|
|
|
if ((*s)->data) delete [] (*s)->data;
|
|
|
|
delete *s;
|
|
|
|
}
|
|
|
|
if (buffer) delete [] buffer;
|
|
|
|
}
|
2011-06-29 16:19:57 +00:00
|
|
|
qint16 EDFParser::Read16()
|
|
|
|
{
|
|
|
|
unsigned char *buf=(unsigned char *)buffer;
|
|
|
|
if (pos>=filesize) return 0;
|
|
|
|
qint16 res=*(qint16 *)&buf[pos];
|
|
|
|
//qint16 res=(buf[pos] ^128)<< 8 | buf[pos+1] ^ 128;
|
|
|
|
pos+=2;
|
|
|
|
return res;
|
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
QString EDFParser::Read(int si)
|
|
|
|
{
|
|
|
|
QString str;
|
|
|
|
if (pos>=filesize) return "";
|
|
|
|
for (int i=0;i<si;i++) {
|
|
|
|
str+=buffer[pos++];
|
|
|
|
}
|
|
|
|
return str.trimmed();
|
|
|
|
}
|
|
|
|
bool EDFParser::Parse()
|
|
|
|
{
|
|
|
|
bool ok;
|
|
|
|
QString temp,temp2;
|
2011-07-02 14:35:50 +00:00
|
|
|
|
|
|
|
version=QString::fromAscii(header.version,8).toLong(&ok);
|
2011-06-29 14:19:38 +00:00
|
|
|
if (!ok)
|
|
|
|
return false;
|
|
|
|
|
2011-07-02 14:35:50 +00:00
|
|
|
//patientident=QString::fromAscii(header.patientident,80);
|
2011-07-05 05:37:11 +00:00
|
|
|
recordingident=QString::fromAscii(header.recordingident,80); // Serial number is in here..
|
|
|
|
int snp=recordingident.indexOf("SRN=");
|
2011-07-02 14:35:50 +00:00
|
|
|
serialnumber.clear();
|
2011-07-05 05:37:11 +00:00
|
|
|
/*char * idx=index(header.recordingident,'=');
|
|
|
|
idx++;
|
2011-07-02 14:35:50 +00:00
|
|
|
for (int i=0;i<16;++i) {
|
|
|
|
if (*idx==0x20) break;
|
|
|
|
serialnumber+=*idx;
|
|
|
|
++idx;
|
2011-07-05 05:37:11 +00:00
|
|
|
} */
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2011-07-05 05:37:11 +00:00
|
|
|
for (int i=snp+4;i<recordingident.length();i++) {
|
|
|
|
if (recordingident[i]==' ')
|
|
|
|
break;
|
|
|
|
serialnumber+=recordingident[i];
|
|
|
|
}
|
2011-07-03 06:10:45 +00:00
|
|
|
QDateTime startDate=QDateTime::fromString(QString::fromAscii(header.datetime,16),"dd.MM.yyHH.mm.ss");
|
2011-07-02 14:35:50 +00:00
|
|
|
QDate d2=startDate.date();
|
2011-06-29 16:19:57 +00:00
|
|
|
if (d2.year()<2000) {
|
|
|
|
d2.setYMD(d2.year()+100,d2.month(),d2.day());
|
2011-07-02 14:35:50 +00:00
|
|
|
startDate.setDate(d2);
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2011-07-02 14:35:50 +00:00
|
|
|
if (!startDate.isValid()) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Invalid date time retreieved parsing EDF File " << filename;
|
2011-06-29 16:19:57 +00:00
|
|
|
return false;
|
|
|
|
}
|
2011-07-02 14:35:50 +00:00
|
|
|
startdate=startDate.toMSecsSinceEpoch();
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-07-21 03:38:17 +00:00
|
|
|
//qDebug() << startDate.toString("yyyy-MM-dd HH:mm:ss");
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-07-02 14:35:50 +00:00
|
|
|
num_header_bytes=QString::fromAscii(header.num_header_bytes,8).toLong(&ok);
|
2011-06-29 14:19:38 +00:00
|
|
|
if (!ok)
|
|
|
|
return false;
|
2011-07-02 14:35:50 +00:00
|
|
|
//reserved44=QString::fromAscii(header.reserved,44);
|
|
|
|
num_data_records=QString::fromAscii(header.num_data_records,8).toLong(&ok);
|
2011-06-29 14:19:38 +00:00
|
|
|
if (!ok)
|
|
|
|
return false;
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2011-07-02 14:35:50 +00:00
|
|
|
dur_data_record=QString::fromAscii(header.dur_data_records,8).toDouble(&ok)*1000.0;
|
2011-06-29 14:19:38 +00:00
|
|
|
if (!ok)
|
|
|
|
return false;
|
2011-07-02 14:35:50 +00:00
|
|
|
num_signals=QString::fromAscii(header.num_signals,4).toLong(&ok);
|
2011-06-29 14:19:38 +00:00
|
|
|
if (!ok)
|
|
|
|
return false;
|
|
|
|
|
2011-07-03 02:43:50 +00:00
|
|
|
enddate=startdate+dur_data_record*qint64(num_data_records);
|
2011-07-03 03:53:09 +00:00
|
|
|
// if (dur_data_record==0)
|
|
|
|
// return false;
|
2011-07-02 14:35:50 +00:00
|
|
|
|
|
|
|
// this could be loaded quicker by transducer_type[signal] etc..
|
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
for (int i=0;i<num_signals;i++) {
|
|
|
|
EDFSignal *signal=new EDFSignal;
|
|
|
|
edfsignals.push_back(signal);
|
|
|
|
signal->data=NULL;
|
|
|
|
edfsignals[i]->label=Read(16);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->transducer_type=Read(80);
|
2011-07-27 09:21:53 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->physical_dimension=Read(8);
|
2011-07-27 09:21:53 +00:00
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->physical_minimum=Read(8).toDouble(&ok);
|
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->physical_maximum=Read(8).toDouble(&ok);
|
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->digital_minimum=Read(8).toDouble(&ok);
|
|
|
|
for (int i=0;i<num_signals;i++) {
|
|
|
|
EDFSignal & e=*edfsignals[i];
|
|
|
|
e.digital_maximum=Read(8).toDouble(&ok);
|
|
|
|
e.gain=(e.physical_maximum-e.physical_minimum)/(e.digital_maximum-e.digital_minimum);
|
|
|
|
e.offset=0;
|
|
|
|
}
|
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->prefiltering=Read(80);
|
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->nr=Read(8).toLong(&ok);
|
|
|
|
for (int i=0;i<num_signals;i++) edfsignals[i]->reserved=Read(32);
|
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
// allocate the buffers
|
2011-06-29 14:19:38 +00:00
|
|
|
for (int i=0;i<num_signals;i++) {
|
|
|
|
//qDebug//cout << "Reading signal " << signals[i]->label << endl;
|
2011-06-29 16:19:57 +00:00
|
|
|
EDFSignal & sig=*edfsignals[i];
|
|
|
|
|
|
|
|
long recs=sig.nr * num_data_records;
|
|
|
|
if (num_data_records<0)
|
|
|
|
return false;
|
|
|
|
sig.data=new qint16 [recs];
|
|
|
|
sig.pos=0;
|
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
for (int x=0;x<num_data_records;x++) {
|
|
|
|
for (int i=0;i<num_signals;i++) {
|
|
|
|
EDFSignal & sig=*edfsignals[i];
|
2011-06-30 10:56:22 +00:00
|
|
|
memcpy((char *)&sig.data[sig.pos],(char *)&buffer[pos],sig.nr*2);
|
|
|
|
sig.pos+=sig.nr;
|
|
|
|
pos+=sig.nr*2;
|
2011-07-02 14:35:50 +00:00
|
|
|
// big endian will screw up without this..
|
2011-06-30 10:56:22 +00:00
|
|
|
/*for (int j=0;j<sig.nr;j++) {
|
2011-06-29 16:19:57 +00:00
|
|
|
qint16 t=Read16();
|
|
|
|
sig.data[sig.pos++]=t;
|
2011-06-30 10:56:22 +00:00
|
|
|
} */
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool EDFParser::Open(QString name)
|
|
|
|
{
|
|
|
|
QFile f(name);
|
|
|
|
if (!f.open(QIODevice::ReadOnly)) return false;
|
|
|
|
if (!f.isReadable()) return false;
|
|
|
|
filename=name;
|
|
|
|
filesize=f.size();
|
2011-07-02 14:35:50 +00:00
|
|
|
datasize=filesize-EDFHeaderSize;
|
|
|
|
if (datasize<0) return false;
|
|
|
|
f.read((char *)&header,EDFHeaderSize);
|
2011-07-01 10:10:44 +00:00
|
|
|
//qDebug() << "Opening " << name;
|
2011-07-02 14:35:50 +00:00
|
|
|
buffer=new char [datasize];
|
|
|
|
f.read(buffer,datasize);
|
2011-06-29 14:19:38 +00:00
|
|
|
f.close();
|
|
|
|
pos=0;
|
2011-06-29 16:19:57 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
|
|
|
|
2011-06-28 02:21:38 +00:00
|
|
|
ResmedLoader::ResmedLoader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
ResmedLoader::~ResmedLoader()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Machine *ResmedLoader::CreateMachine(QString serial,Profile *profile)
|
|
|
|
{
|
2011-07-24 16:34:53 +00:00
|
|
|
if (!profile) return NULL;
|
2011-07-31 20:24:43 +00:00
|
|
|
QVector<Machine *> ml=profile->GetMachines(MT_CPAP);
|
2011-06-28 02:21:38 +00:00
|
|
|
bool found=false;
|
2011-07-31 20:24:43 +00:00
|
|
|
QVector<Machine *>::iterator i;
|
|
|
|
for (i=ml.begin(); i!=ml.end(); i++) {
|
2011-06-29 14:19:38 +00:00
|
|
|
if (((*i)->GetClass()==resmed_class_name) && ((*i)->properties["Serial"]==serial)) {
|
2011-06-28 02:21:38 +00:00
|
|
|
ResmedList[serial]=*i; //static_cast<CPAP *>(*i);
|
|
|
|
found=true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-07-31 20:24:43 +00:00
|
|
|
if (found) return *i;
|
2011-06-28 02:21:38 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Create ResMed Machine" << serial;
|
2011-06-28 02:21:38 +00:00
|
|
|
Machine *m=new CPAP(profile,0);
|
2011-06-29 14:19:38 +00:00
|
|
|
m->SetClass(resmed_class_name);
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
ResmedList[serial]=m;
|
|
|
|
profile->AddMachine(m);
|
|
|
|
|
|
|
|
m->properties["Serial"]=serial;
|
2011-06-29 14:19:38 +00:00
|
|
|
m->properties["Brand"]="ResMed";
|
2011-07-24 17:18:10 +00:00
|
|
|
QString a;
|
|
|
|
a.sprintf("%i",resmed_data_version);
|
|
|
|
m->properties["DataVersion"]=a;
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
return m;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-07-02 14:35:50 +00:00
|
|
|
long event_cnt=0;
|
|
|
|
|
2011-07-15 13:30:41 +00:00
|
|
|
int ResmedLoader::Open(QString & path,Profile *profile)
|
2011-06-28 02:21:38 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
QString newpath;
|
|
|
|
|
|
|
|
QString dirtag="DATALOG";
|
|
|
|
if (path.endsWith("/"+dirtag)) {
|
2011-07-15 13:30:41 +00:00
|
|
|
return 0; // id10t user..
|
2011-06-29 19:06:49 +00:00
|
|
|
//newpath=path;
|
2011-06-28 02:21:38 +00:00
|
|
|
} else {
|
|
|
|
newpath=path+"/"+dirtag;
|
|
|
|
}
|
2011-06-29 19:06:49 +00:00
|
|
|
QString idfile=path+"/Identification.tgt";
|
|
|
|
QFile f(idfile);
|
2011-07-31 20:24:43 +00:00
|
|
|
QHash<QString,QString> idmap;
|
2011-06-29 19:06:49 +00:00
|
|
|
if (f.open(QIODevice::ReadOnly)) {
|
|
|
|
if (!f.isReadable())
|
2011-07-15 13:30:41 +00:00
|
|
|
return 0;
|
2011-06-30 10:56:22 +00:00
|
|
|
|
2011-06-29 19:06:49 +00:00
|
|
|
while (!f.atEnd()) {
|
|
|
|
QString line=f.readLine().trimmed();
|
|
|
|
QString key,value;
|
|
|
|
if (!line.isEmpty()) {
|
|
|
|
key=line.section(" ",0,0);
|
|
|
|
value=line.section(" ",1);
|
|
|
|
key=key.section("#",1);
|
|
|
|
idmap[key]=value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-06-28 02:21:38 +00:00
|
|
|
QDir dir(newpath);
|
|
|
|
|
|
|
|
if ((!dir.exists() || !dir.isReadable()))
|
|
|
|
return 0;
|
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "ResmedLoader::Open newpath=" << newpath;
|
2011-06-29 14:19:38 +00:00
|
|
|
dir.setFilter(QDir::Files | QDir::Hidden | QDir::NoSymLinks);
|
2011-06-28 02:21:38 +00:00
|
|
|
dir.setSorting(QDir::Name);
|
|
|
|
QFileInfoList flist=dir.entryInfoList();
|
2011-08-01 08:22:37 +00:00
|
|
|
QMap<SessionID,QVector<QString> > sessfiles;
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
QString ext,rest,datestr,s,codestr;
|
|
|
|
SessionID sessionid;
|
|
|
|
QDateTime date;
|
2011-06-30 10:56:22 +00:00
|
|
|
QString filename;
|
2011-06-30 04:55:20 +00:00
|
|
|
int size=flist.size();
|
|
|
|
for (int i=0;i<size;i++) {
|
2011-06-28 02:21:38 +00:00
|
|
|
QFileInfo fi=flist.at(i);
|
2011-06-30 10:56:22 +00:00
|
|
|
filename=fi.fileName();
|
2011-06-29 14:19:38 +00:00
|
|
|
ext=filename.section(".",1).toLower();
|
|
|
|
if (ext!="edf") continue;
|
2011-06-29 16:19:57 +00:00
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
rest=filename.section(".",0,0);
|
|
|
|
datestr=filename.section("_",0,1);
|
|
|
|
date=QDateTime::fromString(datestr,"yyyyMMdd_HHmmss");
|
|
|
|
sessionid=date.toTime_t();
|
2011-06-29 21:23:00 +00:00
|
|
|
// Resmed bugs up on the session filenames.. 1 second either way
|
|
|
|
if (sessfiles.find(sessionid)==sessfiles.end()) {
|
2011-08-01 08:22:37 +00:00
|
|
|
if (sessfiles.find(sessionid+2)!=sessfiles.end()) sessionid+=2;
|
|
|
|
else if (sessfiles.find(sessionid+1)!=sessfiles.end()) sessionid++;
|
|
|
|
else if (sessfiles.find(sessionid-1)!=sessfiles.end()) sessionid--;
|
|
|
|
else if (sessfiles.find(sessionid-2)!=sessfiles.end()) sessionid-=2;
|
2011-06-29 21:23:00 +00:00
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
sessfiles[sessionid].push_back(fi.canonicalFilePath());
|
2011-06-30 04:55:20 +00:00
|
|
|
if (qprogress) qprogress->setValue((float(i+1)/float(size)*33.0));
|
2011-07-30 00:36:31 +00:00
|
|
|
QApplication::processEvents();
|
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
Machine *m=NULL;
|
|
|
|
|
2011-06-30 10:56:22 +00:00
|
|
|
QString fn;
|
2011-07-03 03:53:09 +00:00
|
|
|
Session *sess;
|
2011-06-30 04:55:20 +00:00
|
|
|
int cnt=0;
|
|
|
|
size=sessfiles.size();
|
2011-08-01 08:22:37 +00:00
|
|
|
for (QMap<SessionID,QVector<QString> >::iterator si=sessfiles.begin();si!=sessfiles.end();si++) {
|
2011-07-31 20:24:43 +00:00
|
|
|
sessionid=si.key();
|
2011-07-01 10:10:44 +00:00
|
|
|
//qDebug() << "Parsing Session " << sessionid;
|
2011-06-29 16:19:57 +00:00
|
|
|
bool done=false;
|
|
|
|
bool first=true;
|
2011-07-03 03:53:09 +00:00
|
|
|
sess=NULL;
|
2011-07-31 20:24:43 +00:00
|
|
|
for (int i=0;i<si.value().size();++i) {
|
|
|
|
fn=si.value()[i].section("_",-1).toLower();
|
|
|
|
EDFParser edf(si.value()[i]);
|
2011-07-01 10:10:44 +00:00
|
|
|
//qDebug() << "Parsing File " << i << " " << edf.filesize;
|
2011-06-29 16:19:57 +00:00
|
|
|
|
|
|
|
if (!edf.Parse())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (first) { // First EDF file parsed, check if this data set is already imported
|
|
|
|
m=CreateMachine(edf.serialnumber,profile);
|
2011-07-31 20:24:43 +00:00
|
|
|
for (QHash<QString,QString>::iterator i=idmap.begin();i!=idmap.end();i++) {
|
|
|
|
if (i.key()=="SRN") {
|
|
|
|
if (edf.serialnumber!=i.value()) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "edf Serial number doesn't match Identification.tgt";
|
2011-06-29 19:06:49 +00:00
|
|
|
}
|
2011-07-31 20:24:43 +00:00
|
|
|
} else if (i.key()=="PNA") {
|
|
|
|
m->properties["Model"]=i.value();
|
|
|
|
} else if (i.key()=="PCD") {
|
2011-07-03 04:22:58 +00:00
|
|
|
bool ok;
|
2011-07-31 20:24:43 +00:00
|
|
|
int j=i.value().toInt(&ok);
|
2011-07-03 04:22:58 +00:00
|
|
|
if (RMS9ModelMap.find(j)!=RMS9ModelMap.end()) {
|
|
|
|
m->properties["SubModel"]=RMS9ModelMap[j];
|
|
|
|
}
|
2011-06-29 19:06:49 +00:00
|
|
|
} else {
|
2011-07-31 20:24:43 +00:00
|
|
|
m->properties[i.key()]=i.value();
|
2011-06-29 19:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 16:19:57 +00:00
|
|
|
if (m->SessionExists(sessionid)) {
|
|
|
|
done=true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sess=new Session(m,sessionid);
|
2011-07-03 03:53:09 +00:00
|
|
|
first=false;
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
|
|
|
if (!done) {
|
2011-07-01 10:10:44 +00:00
|
|
|
if (fn=="eve.edf") LoadEVE(sess,edf);
|
|
|
|
else if (fn=="pld.edf") LoadPLD(sess,edf);
|
|
|
|
else if (fn=="brp.edf") LoadBRP(sess,edf);
|
|
|
|
else if (fn=="sad.edf") LoadSAD(sess,edf);
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2011-07-03 03:53:09 +00:00
|
|
|
//if (first) {
|
|
|
|
//first=false;
|
|
|
|
//}
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-07-03 03:53:09 +00:00
|
|
|
if (qprogress) qprogress->setValue(33.0+(float(++cnt)/float(size)*33.0));
|
2011-07-30 00:36:31 +00:00
|
|
|
QApplication::processEvents();
|
|
|
|
|
2011-07-03 03:53:09 +00:00
|
|
|
if (!sess) continue;
|
|
|
|
if (!sess->first()) {
|
|
|
|
delete sess;
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
sess->SetChanged(true);
|
|
|
|
m->AddSession(sess,profile); // Adding earlier than I really like here..
|
|
|
|
}
|
2011-07-02 14:35:50 +00:00
|
|
|
if (!done && sess) {
|
2011-07-31 20:24:43 +00:00
|
|
|
ChannelID e[]={
|
|
|
|
CPAP_Obstructive, CPAP_Hypopnea, CPAP_ClearAirway, CPAP_Apnea
|
|
|
|
};
|
|
|
|
for (unsigned i=0;i<sizeof(e)/sizeof(ChannelID);i++) {
|
|
|
|
|
|
|
|
// Merge this crap together where possible
|
|
|
|
sess->count(e[i]);
|
|
|
|
sess->max(e[i]);
|
|
|
|
sess->min(e[i]);
|
|
|
|
sess->avg(e[i]);
|
2011-08-01 08:58:46 +00:00
|
|
|
//sess->p90(e[i]);
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->cph(e[i]);
|
|
|
|
sess->sph(e[i]);
|
|
|
|
}
|
|
|
|
sess->setCph(CPAP_AHI,sess->cph(CPAP_Obstructive)+sess->cph(CPAP_Hypopnea)+sess->cph(CPAP_ClearAirway)+sess->cph(CPAP_Apnea));
|
|
|
|
sess->setSph(CPAP_AHI,sess->sph(CPAP_Obstructive)+sess->sph(CPAP_Hypopnea)+sess->sph(CPAP_ClearAirway)+sess->sph(CPAP_Apnea));
|
|
|
|
|
|
|
|
ChannelID a[]={
|
2011-08-01 08:58:46 +00:00
|
|
|
CPAP_Leak, CPAP_Snore, CPAP_EPAP,
|
2011-07-31 20:24:43 +00:00
|
|
|
CPAP_IPAP, CPAP_TidalVolume, CPAP_RespiratoryRate,
|
|
|
|
CPAP_PatientTriggeredBreaths,CPAP_MinuteVentilation,
|
|
|
|
CPAP_FlowLimitGraph, CPAP_PressureSupport,CPAP_Pressure
|
|
|
|
};
|
|
|
|
for (unsigned i=0;i<sizeof(a)/sizeof(ChannelID);i++) {
|
|
|
|
if (sess->eventlist.contains(a[i])) {
|
|
|
|
sess->min(a[i]);
|
|
|
|
sess->max(a[i]);
|
|
|
|
sess->avg(a[i]);
|
|
|
|
sess->wavg(a[i]);
|
2011-08-01 12:29:17 +00:00
|
|
|
sess->p90(a[i]);
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->cph(a[i]);
|
|
|
|
}
|
|
|
|
}
|
2011-08-01 08:58:46 +00:00
|
|
|
ChannelID b[]={
|
|
|
|
CPAP_FlowRate, CPAP_MaskPressure
|
|
|
|
};
|
|
|
|
for (unsigned i=0;i<sizeof(b)/sizeof(ChannelID);i++) {
|
|
|
|
if (sess->eventlist.contains(b[i])) {
|
|
|
|
sess->min(b[i]);
|
|
|
|
sess->max(b[i]);
|
|
|
|
sess->avg(b[i]);
|
|
|
|
//sess->wavg(b[i]);
|
|
|
|
//sess->p90(b[i]);
|
|
|
|
sess->cph(b[i]);
|
|
|
|
}
|
|
|
|
}
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->settings[CPAP_Mode]=MODE_APAP;
|
2011-06-30 09:37:24 +00:00
|
|
|
}
|
|
|
|
|
2011-06-28 02:21:38 +00:00
|
|
|
}
|
2011-07-02 16:10:20 +00:00
|
|
|
m->Save();
|
2011-06-30 04:55:20 +00:00
|
|
|
if (qprogress) qprogress->setValue(100);
|
2011-07-02 14:35:50 +00:00
|
|
|
qDebug() << "Total Events " << event_cnt;
|
2011-07-15 13:30:41 +00:00
|
|
|
return 1;
|
2011-06-28 02:21:38 +00:00
|
|
|
}
|
2011-06-29 17:58:28 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
bool ResmedLoader::LoadEVE(Session *sess,EDFParser &edf)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2011-07-27 09:21:53 +00:00
|
|
|
// EVEnt records have useless duration record.
|
|
|
|
|
2011-06-29 14:19:38 +00:00
|
|
|
QString t;
|
2011-06-30 10:56:22 +00:00
|
|
|
long recs;
|
2011-07-27 09:21:53 +00:00
|
|
|
double duration;
|
2011-06-30 10:56:22 +00:00
|
|
|
char * data;
|
|
|
|
char c;
|
|
|
|
long pos;
|
|
|
|
bool sign,ok;
|
|
|
|
double d;
|
2011-07-02 15:48:55 +00:00
|
|
|
double tt;
|
2011-07-31 20:24:43 +00:00
|
|
|
ChannelID code;
|
2011-07-01 10:10:44 +00:00
|
|
|
//Event *e;
|
2011-07-27 09:21:53 +00:00
|
|
|
//totaldur=edf.GetNumDataRecords()*edf.GetDuration();
|
2011-07-02 14:35:50 +00:00
|
|
|
|
2011-07-27 09:21:53 +00:00
|
|
|
EventList *EL[4]={NULL};
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateFirst(edf.startdate);
|
2011-07-03 03:53:09 +00:00
|
|
|
//if (edf.enddate>edf.startdate) sess->set_last(edf.enddate);
|
2011-06-29 14:19:38 +00:00
|
|
|
for (int s=0;s<edf.GetNumSignals();s++) {
|
2011-06-30 10:56:22 +00:00
|
|
|
recs=edf.edfsignals[s]->nr*edf.GetNumDataRecords()*2;
|
2011-07-02 09:49:53 +00:00
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
//qDebug() << edf.edfsignals[s]->label << " " << t;
|
2011-06-30 10:56:22 +00:00
|
|
|
data=(char *)edf.edfsignals[s]->data;
|
|
|
|
pos=0;
|
|
|
|
tt=edf.startdate;
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateFirst(tt);
|
2011-06-30 10:56:22 +00:00
|
|
|
duration=0;
|
2011-06-29 17:58:28 +00:00
|
|
|
while (pos<recs) {
|
|
|
|
c=data[pos];
|
|
|
|
if ((c!='+') && (c!='-'))
|
|
|
|
break;
|
|
|
|
if (data[pos++]=='+') sign=true; else sign=false;
|
|
|
|
t="";
|
|
|
|
c=data[pos];
|
|
|
|
do {
|
|
|
|
t+=c;
|
|
|
|
pos++;
|
|
|
|
c=data[pos];
|
|
|
|
} while ((c!=20) && (c!=21)); // start code
|
|
|
|
d=t.toDouble(&ok);
|
|
|
|
if (!ok) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Faulty EDF EVE file " << edf.filename;
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!sign) d=-d;
|
2011-07-27 09:21:53 +00:00
|
|
|
tt=edf.startdate+qint64(d*1000.0);
|
2011-06-29 17:58:28 +00:00
|
|
|
duration=0;
|
|
|
|
// First entry
|
|
|
|
|
|
|
|
if (data[pos]==21) {
|
|
|
|
pos++;
|
|
|
|
// get duration.
|
|
|
|
t="";
|
|
|
|
do {
|
|
|
|
t+=data[pos];
|
|
|
|
pos++;
|
|
|
|
} while ((data[pos]!=20) && (pos<recs)); // start code
|
2011-07-02 15:48:55 +00:00
|
|
|
duration=t.toDouble(&ok);
|
2011-06-29 17:58:28 +00:00
|
|
|
if (!ok) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Faulty EDF EVE file (at %" << pos << ") " << edf.filename;
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while ((data[pos]==20) && (pos<recs)) {
|
|
|
|
t="";
|
|
|
|
pos++;
|
|
|
|
if (data[pos]==0)
|
|
|
|
break;
|
|
|
|
if (data[pos]==20) {
|
|
|
|
pos++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
t+=tolower(data[pos++]);
|
|
|
|
} while ((data[pos]!=20) && (pos<recs)); // start code
|
|
|
|
if (!t.isEmpty()) {
|
2011-07-27 09:21:53 +00:00
|
|
|
//code=MC_UNKNOWN;
|
|
|
|
if (t=="obstructive apnea") {
|
|
|
|
code=CPAP_Obstructive;
|
|
|
|
if (!EL[0]) {
|
|
|
|
EL[0]=new EventList(code,EVL_Event);
|
|
|
|
sess->eventlist[code].push_back(EL[0]);
|
|
|
|
}
|
|
|
|
EL[0]->AddEvent(tt,duration);
|
|
|
|
} else if (t=="hypopnea") {
|
|
|
|
code=CPAP_Hypopnea;
|
|
|
|
if (!EL[1]) {
|
|
|
|
EL[1]=new EventList(code,EVL_Event);
|
|
|
|
sess->eventlist[code].push_back(EL[1]);
|
|
|
|
}
|
|
|
|
EL[1]->AddEvent(tt,duration);
|
|
|
|
} else if (t=="apnea") {
|
|
|
|
code=CPAP_Apnea;
|
|
|
|
if (!EL[2]) {
|
|
|
|
EL[2]=new EventList(code,EVL_Event);
|
|
|
|
sess->eventlist[code].push_back(EL[2]);
|
|
|
|
}
|
|
|
|
EL[2]->AddEvent(tt,duration);
|
|
|
|
} else if (t=="central apnea") {
|
|
|
|
code=CPAP_ClearAirway;
|
|
|
|
if (!EL[3]) {
|
|
|
|
EL[3]=new EventList(code,EVL_Event);
|
|
|
|
sess->eventlist[code].push_back(EL[3]);
|
|
|
|
}
|
|
|
|
EL[3]->AddEvent(tt,duration);
|
2011-06-29 17:58:28 +00:00
|
|
|
} else {
|
|
|
|
if (t!="recording starts") {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Unknown ResMed annotation field: " << t;
|
2011-06-29 17:58:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (pos>=recs) {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Short EDF EVE file" << edf.filename;
|
2011-06-29 17:58:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
// pos++;
|
|
|
|
}
|
|
|
|
while ((data[pos]==0) && pos<recs) pos++;
|
|
|
|
if (pos>=recs) break;
|
|
|
|
}
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateLast(tt);
|
2011-06-29 17:58:28 +00:00
|
|
|
// qDebug(data);
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-07-01 10:10:44 +00:00
|
|
|
bool ResmedLoader::LoadBRP(Session *sess,EDFParser &edf)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2011-06-29 16:19:57 +00:00
|
|
|
QString t;
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateFirst(edf.startdate);
|
2011-07-03 02:43:50 +00:00
|
|
|
qint64 duration=edf.GetNumDataRecords()*edf.GetDuration();
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateLast(edf.startdate+duration);
|
2011-07-03 02:43:50 +00:00
|
|
|
|
2011-06-29 16:19:57 +00:00
|
|
|
for (int s=0;s<edf.GetNumSignals();s++) {
|
2011-07-27 09:21:53 +00:00
|
|
|
EDFSignal & es=*edf.edfsignals[s];
|
|
|
|
//qDebug() << "BRP:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum;
|
2011-06-29 16:19:57 +00:00
|
|
|
long recs=edf.edfsignals[s]->nr*edf.GetNumDataRecords();
|
2011-07-31 20:24:43 +00:00
|
|
|
ChannelID code;
|
2011-07-27 09:21:53 +00:00
|
|
|
if (edf.edfsignals[s]->label=="Flow") {
|
|
|
|
code=CPAP_FlowRate;
|
|
|
|
} else if (edf.edfsignals[s]->label=="Mask Pres") {
|
2011-07-10 17:49:02 +00:00
|
|
|
code=CPAP_MaskPressure;
|
2011-07-24 17:18:10 +00:00
|
|
|
//for (int i=0;i<recs;i++) edf.edfsignals[s]->data[i]/=50.0;
|
2011-07-10 17:49:02 +00:00
|
|
|
} else {
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Unknown Signal " << edf.edfsignals[s]->label;
|
2011-06-29 17:58:28 +00:00
|
|
|
continue;
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
double rate=double(duration)/double(recs);
|
|
|
|
//es.gain=1;
|
|
|
|
EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
|
|
|
|
|
|
|
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
|
|
|
|
|
|
|
if (code==CPAP_MaskPressure) {
|
|
|
|
/*int v=ceil(a->max()/1);
|
|
|
|
a->setMax(v*1);
|
|
|
|
v=floor(a->min()/1);
|
|
|
|
a->setMin(v*1); */
|
|
|
|
} else if (code==CPAP_FlowRate) {
|
|
|
|
a->setMax(1);
|
|
|
|
a->setMin(-1);
|
|
|
|
}
|
2011-08-01 08:53:26 +00:00
|
|
|
sess->setMin(code,a->min());
|
|
|
|
sess->setMax(code,a->max());
|
2011-07-27 09:21:53 +00:00
|
|
|
sess->eventlist[code].push_back(a);
|
|
|
|
//delete edf.edfsignals[s]->data;
|
|
|
|
//edf.edfsignals[s]->data=NULL; // so it doesn't get deleted when edf gets trashed.
|
2011-06-29 16:19:57 +00:00
|
|
|
}
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-07-31 20:24:43 +00:00
|
|
|
EventList * ResmedLoader::ToTimeDelta(Session *sess,EDFParser &edf, EDFSignal & es, ChannelID code, long recs, qint64 duration,EventDataType min,EventDataType max)
|
2011-06-29 20:30:23 +00:00
|
|
|
{
|
|
|
|
bool first=true;
|
2011-07-02 14:35:50 +00:00
|
|
|
double rate=(duration/recs); // milliseconds per record
|
2011-07-02 15:48:55 +00:00
|
|
|
double tt=edf.startdate;
|
2011-07-27 09:21:53 +00:00
|
|
|
//sess->UpdateFirst(tt);
|
2011-06-29 20:30:23 +00:00
|
|
|
EventDataType c,last;
|
2011-07-27 09:21:53 +00:00
|
|
|
//if (gain==0) gain=1;
|
|
|
|
EventList *el=new EventList(code,EVL_Event,es.gain,es.offset,min,max);
|
|
|
|
sess->eventlist[code].push_back(el);
|
2011-06-29 20:30:23 +00:00
|
|
|
for (int i=0;i<recs;i++) {
|
2011-07-27 09:21:53 +00:00
|
|
|
c=es.data[i];
|
2011-06-29 20:30:23 +00:00
|
|
|
|
|
|
|
if (first) {
|
2011-07-27 09:21:53 +00:00
|
|
|
el->AddEvent(tt,c);
|
2011-06-29 20:30:23 +00:00
|
|
|
first=false;
|
|
|
|
} else {
|
|
|
|
if (last!=c) {
|
2011-07-27 09:21:53 +00:00
|
|
|
el->AddEvent(tt,c);
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
|
|
|
}
|
2011-07-02 14:35:50 +00:00
|
|
|
tt+=rate;
|
2011-06-29 20:30:23 +00:00
|
|
|
|
|
|
|
last=c;
|
|
|
|
}
|
2011-07-27 09:21:53 +00:00
|
|
|
el->AddEvent(tt,c);
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateLast(tt);
|
2011-07-27 09:21:53 +00:00
|
|
|
return el;
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
2011-07-01 10:10:44 +00:00
|
|
|
bool ResmedLoader::LoadSAD(Session *sess,EDFParser &edf)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2011-08-06 13:37:06 +00:00
|
|
|
QString t;
|
|
|
|
sess->updateFirst(edf.startdate);
|
|
|
|
qint64 duration=edf.GetNumDataRecords()*edf.GetDuration();
|
|
|
|
sess->updateLast(edf.startdate+duration);
|
|
|
|
|
|
|
|
for (int s=0;s<edf.GetNumSignals();s++) {
|
|
|
|
EDFSignal & es=*edf.edfsignals[s];
|
2011-08-07 01:26:28 +00:00
|
|
|
//qDebug() << "SAD:" << es.label << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum;
|
2011-08-06 13:37:06 +00:00
|
|
|
long recs=edf.edfsignals[s]->nr*edf.GetNumDataRecords();
|
|
|
|
ChannelID code;
|
|
|
|
if (edf.edfsignals[s]->label=="Pulse") {
|
|
|
|
code=CPAP_Pulse;
|
|
|
|
} else if (edf.edfsignals[s]->label=="SpO2") {
|
|
|
|
code=CPAP_SPO2;
|
|
|
|
} else {
|
|
|
|
qDebug() << "Unknown SAD.edf Signal " << edf.edfsignals[s]->label;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
bool hasdata=false;
|
|
|
|
for (int i=0;i<recs;i++) {
|
|
|
|
if (es.data[i]!=-1) {
|
|
|
|
hasdata=true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hasdata) {
|
|
|
|
EventList *a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
|
|
|
if (a) {
|
|
|
|
sess->setMin(code,a->min());
|
|
|
|
sess->setMax(code,a->max());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-06-29 20:30:23 +00:00
|
|
|
|
|
|
|
|
2011-07-01 10:10:44 +00:00
|
|
|
bool ResmedLoader::LoadPLD(Session *sess,EDFParser &edf)
|
2011-06-29 14:19:38 +00:00
|
|
|
{
|
2011-07-01 02:52:02 +00:00
|
|
|
// Is it save to assume the order does not change here?
|
|
|
|
enum PLDType { MaskPres=0, TherapyPres, ExpPress, Leak, RR, Vt, Mv, SnoreIndex, FFLIndex, U1, U2 };
|
|
|
|
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateFirst(edf.startdate);
|
2011-07-03 02:43:50 +00:00
|
|
|
qint64 duration=edf.GetNumDataRecords()*edf.GetDuration();
|
2011-07-31 20:24:43 +00:00
|
|
|
sess->updateLast(edf.startdate+duration);
|
2011-06-29 17:58:28 +00:00
|
|
|
QString t;
|
2011-07-21 03:35:59 +00:00
|
|
|
int emptycnt=0;
|
2011-08-01 08:53:26 +00:00
|
|
|
EventList *a;
|
|
|
|
double rate;
|
|
|
|
long recs;
|
|
|
|
ChannelID code;
|
2011-06-29 17:58:28 +00:00
|
|
|
for (int s=0;s<edf.GetNumSignals();s++) {
|
2011-07-27 09:21:53 +00:00
|
|
|
EDFSignal & es=*edf.edfsignals[s];
|
2011-08-01 08:53:26 +00:00
|
|
|
recs=es.nr*edf.GetNumDataRecords();
|
|
|
|
rate=double(duration)/double(recs);
|
2011-07-27 09:21:53 +00:00
|
|
|
//qDebug() << "EVE:" << es.digital_maximum << es.digital_minimum << es.physical_maximum << es.physical_minimum << es.gain;
|
|
|
|
if (es.label=="Snore Index") {
|
2011-06-29 20:30:23 +00:00
|
|
|
code=CPAP_Snore;
|
2011-07-27 09:21:53 +00:00
|
|
|
|
|
|
|
//EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,es.physical_minimum,es.physical_maximum,rate);
|
|
|
|
//sess->eventlist[code].push_back(a);
|
|
|
|
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
a->setMax(1);
|
|
|
|
a->setMin(0);
|
|
|
|
} else if (es.label=="Therapy Pres") {
|
2011-08-07 02:11:54 +00:00
|
|
|
code=CPAP_IPAP; //TherapyPressure;
|
2011-07-27 09:21:53 +00:00
|
|
|
//EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,es.physical_minimum,es.physical_maximum,rate);
|
|
|
|
//sess->eventlist[code].push_back(a);
|
|
|
|
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
} else if (es.label=="MV") {
|
2011-06-29 20:30:23 +00:00
|
|
|
code=CPAP_MinuteVentilation;
|
2011-07-27 09:21:53 +00:00
|
|
|
//EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,es.physical_minimum,es.physical_maximum,rate);
|
|
|
|
//sess->eventlist[code].push_back(a);
|
|
|
|
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
} else if (es.label=="RR") {
|
2011-06-30 04:55:20 +00:00
|
|
|
code=CPAP_RespiratoryRate;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=new EventList(code,EVL_Waveform,es.gain,es.offset,0,0,rate);
|
2011-07-27 09:21:53 +00:00
|
|
|
sess->eventlist[code].push_back(a);
|
|
|
|
a->AddWaveform(edf.startdate,es.data,recs,duration);
|
|
|
|
//ToTimeDelta(sess,edf,es, code,recs,duration);
|
|
|
|
} else if (es.label=="Vt") {
|
2011-06-30 04:55:20 +00:00
|
|
|
code=CPAP_TidalVolume;
|
2011-07-27 09:21:53 +00:00
|
|
|
//EventList *a=new EventList(code,EVL_Waveform,es.gain,es.offset,es.physical_minimum,es.physical_maximum,rate);
|
|
|
|
//sess->eventlist[code].push_back(a);
|
|
|
|
//a->AddWaveform(edf.startdate,es.data,recs,duration);
|
|
|
|
es.physical_maximum=es.physical_minimum=0;
|
|
|
|
es.gain*=1000.0;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
} else if (es.label=="Leak") {
|
2011-06-29 20:30:23 +00:00
|
|
|
code=CPAP_Leak;
|
2011-07-27 09:21:53 +00:00
|
|
|
// es.gain*=100.0;
|
|
|
|
//es.gain=1;//10.0;
|
|
|
|
es.offset=-0.5;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
a->setMax(1);
|
|
|
|
a->setMin(0);
|
|
|
|
} else if (es.label=="FFL Index") {
|
2011-06-30 12:07:11 +00:00
|
|
|
code=CPAP_FlowLimitGraph;
|
2011-07-27 09:21:53 +00:00
|
|
|
//es.gain=1;//10.0;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
a->setMax(1);
|
|
|
|
a->setMin(0);
|
|
|
|
} else if (es.label=="Mask Pres") {
|
2011-07-24 17:50:27 +00:00
|
|
|
code=CPAP_Pressure;
|
2011-07-27 09:21:53 +00:00
|
|
|
//es.gain=1;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
} else if (es.label=="Exp Press") {
|
2011-08-07 02:11:54 +00:00
|
|
|
code=CPAP_EPAP;//ExpiratoryPressure;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
|
2011-07-27 09:21:53 +00:00
|
|
|
} else if (es.label=="") {
|
2011-07-21 03:35:59 +00:00
|
|
|
if (emptycnt==0) {
|
2011-07-31 20:24:43 +00:00
|
|
|
code=RMS9_Empty1;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration);
|
2011-07-21 03:35:59 +00:00
|
|
|
} else if (emptycnt==1) {
|
2011-07-31 20:24:43 +00:00
|
|
|
code=RMS9_Empty2;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=ToTimeDelta(sess,edf,es, code,recs,duration);
|
2011-07-21 03:35:59 +00:00
|
|
|
} else {
|
2011-07-27 09:21:53 +00:00
|
|
|
qDebug() << "Unobserved Empty Signal " << es.label;
|
2011-07-21 03:35:59 +00:00
|
|
|
}
|
|
|
|
emptycnt++;
|
2011-06-30 09:37:24 +00:00
|
|
|
} else {
|
2011-07-27 09:21:53 +00:00
|
|
|
qDebug() << "Unobserved Signal " << es.label;
|
2011-08-01 08:53:26 +00:00
|
|
|
a=NULL;
|
|
|
|
}
|
|
|
|
if (a) {
|
|
|
|
sess->setMin(code,a->min());
|
|
|
|
sess->setMax(code,a->max());
|
2011-06-29 20:30:23 +00:00
|
|
|
}
|
2011-06-29 17:58:28 +00:00
|
|
|
}
|
2011-07-01 10:10:44 +00:00
|
|
|
return true;
|
2011-06-29 14:19:38 +00:00
|
|
|
}
|
2011-06-28 02:21:38 +00:00
|
|
|
|
|
|
|
void ResInitModelMap()
|
|
|
|
{
|
2011-07-03 04:22:58 +00:00
|
|
|
// Courtesy Troy Schultz
|
|
|
|
RMS9ModelMap[36001]="ResMed S9 Escape";
|
|
|
|
RMS9ModelMap[36002]="ResMed S9 Escape Auto";
|
|
|
|
RMS9ModelMap[36003]="ResMed S9 Elite";
|
|
|
|
RMS9ModelMap[36004]="ResMed S9 VPAP S";
|
|
|
|
RMS9ModelMap[36005]="ResMed S9 AutoSet";
|
|
|
|
RMS9ModelMap[36006]="ResMed S9 VPAP Auto";
|
|
|
|
RMS9ModelMap[36007]="ResMed S9 VPAP Adapt";
|
|
|
|
RMS9ModelMap[36008]="ResMed S9 VPAP ST";
|
|
|
|
// S8 Series
|
|
|
|
RMS9ModelMap[33007]="ResMed S8 Escape";
|
|
|
|
RMS9ModelMap[33039]="ResMed S8 Elite II";
|
|
|
|
RMS9ModelMap[33051]="ResMed S8 Escape II";
|
|
|
|
RMS9ModelMap[33064]="ResMed S8 Escape II AutoSet";
|
|
|
|
RMS9ModelMap[33064]="ResMed S8 Escape II AutoSet";
|
|
|
|
RMS9ModelMap[33129]="ResMed S8 AutoSet II";
|
2011-06-28 02:21:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
bool resmed_initialized=false;
|
|
|
|
void ResmedLoader::Register()
|
|
|
|
{
|
|
|
|
if (resmed_initialized) return;
|
2011-07-01 10:10:44 +00:00
|
|
|
qDebug() << "Registering ResmedLoader";
|
2011-06-28 02:21:38 +00:00
|
|
|
RegisterLoader(new ResmedLoader());
|
|
|
|
ResInitModelMap();
|
|
|
|
resmed_initialized=true;
|
|
|
|
}
|
|
|
|
|