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-11-28 01:39:28 +00:00
# include "SleepLib/calcs.h"
2011-06-28 02:21:38 +00:00
2012-01-05 06:54:07 +00:00
# ifdef DEBUG_EFFICIENCY
# include <QElapsedTimer> // only available in 4.8
# endif
2011-06-30 04:55:20 +00:00
extern QProgressBar * qprogress ;
2011-07-31 20:24:43 +00:00
QHash < int , QString > RMS9ModelMap ;
2011-09-23 03:54:48 +00:00
QHash < ChannelID , QVector < QString > > resmed_codes ;
2011-06-30 04:55:20 +00:00
2011-12-26 18:26:06 +00:00
// Looks up foreign language Signal names that match this channelID
2011-11-25 12:13:35 +00:00
EDFSignal * EDFParser : : lookupSignal ( ChannelID ch )
{
QHash < ChannelID , QVector < QString > > : : iterator ci ;
QHash < QString , EDFSignal * > : : iterator jj ;
ci = resmed_codes . find ( ch ) ;
if ( ci = = resmed_codes . end ( ) ) return NULL ;
for ( int i = 0 ; i < ci . value ( ) . size ( ) ; i + + ) {
jj = lookup . find ( ci . value ( ) [ i ] ) ;
if ( jj = = lookup . end ( ) ) continue ;
return jj . value ( ) ;
}
return NULL ;
}
2012-01-03 07:15:02 +00:00
EDFSignal * EDFParser : : lookupName ( QString name )
{
QHash < QString , EDFSignal * > : : iterator i = lookup . find ( name ) ;
if ( i ! = lookup . end ( ) ) return i . value ( ) ;
return NULL ;
}
2011-11-25 12:13:35 +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-12-07 12:23:19 +00:00
//startDate.toTimeSpec(Qt::UTC);
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-09-01 13:07:26 +00:00
startdate = qint64 ( startDate . toTime_t ( ) ) * 1000L ;
2011-12-07 12:23:19 +00:00
//startdate-=timezoneOffset();
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 ) ;
2011-09-17 12:39:00 +00:00
lookup [ edfsignals [ i ] - > label ] = signal ;
2011-06-29 14:19:38 +00:00
}
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-12-26 18:26:06 +00:00
// big endian will probably 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 )
{
2011-12-19 05:35:05 +00:00
//Urk.. This needs fixing for VC++, as it doesn't have packed attribute type..
2012-01-05 04:37:22 +00:00
if ( name . endsWith ( " .gz " ) ) {
filename = name . mid ( 0 , - 3 ) ;
QFile fi ( name ) ;
fi . open ( QFile : : ReadOnly ) ;
fi . seek ( fi . size ( ) - 4 ) ;
unsigned char ch [ 4 ] ;
fi . read ( ( char * ) ch , 4 ) ;
filesize = ch [ 0 ] | ( ch [ 1 ] < < 8 ) | ( ch [ 2 ] < < 16 ) | ( ch [ 3 ] < < 24 ) ;
datasize = filesize - EDFHeaderSize ;
if ( datasize < 0 ) return false ;
qDebug ( ) < < " Size of " < < name < < " uncompressed= " < < filesize ;
gzFile f = gzopen ( name . toAscii ( ) , " rb " ) ;
if ( ! f ) {
qDebug ( ) < < " EDFParser::Open() Couldn't open file " < < name ;
return false ;
}
gzread ( f , ( char * ) & header , EDFHeaderSize ) ;
buffer = new char [ datasize ] ;
gzbuffer ( f , 65536 * 2 ) ;
gzread ( f , buffer , datasize ) ;
gzclose ( f ) ;
} else {
QFile f ( name ) ;
if ( ! f . open ( QIODevice : : ReadOnly ) )
return false ;
filename = name ;
filesize = f . size ( ) ;
datasize = filesize - EDFHeaderSize ;
if ( datasize < 0 ) return false ;
f . read ( ( char * ) & header , EDFHeaderSize ) ;
//qDebug() << "Opening " << name;
buffer = new char [ datasize ] ;
f . read ( buffer , datasize ) ;
f . close ( ) ;
}
2011-06-29 14:19:38 +00:00
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-12-08 04:10:35 +00:00
QList < Machine * > ml = profile - > GetMachines ( MT_CPAP ) ;
2011-06-28 02:21:38 +00:00
bool found = false ;
2011-12-08 04:10:35 +00:00
QList < Machine * > : : iterator i ;
2011-07-31 20:24:43 +00:00
for ( i = ml . begin ( ) ; i ! = ml . end ( ) ; i + + ) {
2011-12-21 14:24:09 +00:00
if ( ( ( * i ) - > GetClass ( ) = = resmed_class_name ) & & ( ( * i ) - > properties [ STR_PROP_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 ) ;
2011-12-21 14:24:09 +00:00
m - > properties [ STR_PROP_Serial ] = serial ;
m - > properties [ STR_PROP_Brand ] = STR_MACH_ResMed ;
2012-01-05 04:37:22 +00:00
m - > properties [ STR_PROP_DataVersion ] = QString : : number ( resmed_data_version ) ;
QString path = " { " + STR_GEN_DataFolder + " }/ " + m - > GetClass ( ) + " _ " + serial + " / " ;
m - > properties [ STR_PROP_Path ] = path ;
m - > properties [ STR_PROP_BackupPath ] = path + " Backup/ " ;
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
{
2011-12-30 23:02:45 +00:00
const QString datalog = " DATALOG " ;
const QString idfile = " Identification. " ;
const QString strfile = " STR. " ;
const QString ext_TGT = " tgt " ;
const QString ext_CRC = " crc " ;
const QString ext_EDF = " edf " ;
2012-01-05 04:37:22 +00:00
const QString ext_gz = " .gz " ;
2011-12-30 23:02:45 +00:00
QString serial ; // Serial number
QString key , value ;
QString line ;
2011-06-28 02:21:38 +00:00
QString newpath ;
2011-12-30 23:02:45 +00:00
QString filename ;
2011-06-28 02:21:38 +00:00
2011-12-30 23:02:45 +00:00
QHash < QString , QString > idmap ; // Temporary properties hash
// Strip off end "/" if any
if ( path . endsWith ( " / " ) )
path = path . section ( " / " , 0 , - 2 ) ;
// Strip off DATALOG from path, and set newpath to the path contianing DATALOG
if ( path . endsWith ( datalog ) ) {
newpath = path + " / " ;
path = path . section ( " / " , 0 , - 2 ) ;
2011-06-28 02:21:38 +00:00
} else {
2011-12-30 23:02:45 +00:00
newpath = path + " / " + datalog + " / " ;
2011-06-28 02:21:38 +00:00
}
2011-12-30 23:02:45 +00:00
// Add separator back
path + = " / " ;
// Check DATALOG folder exists and is readable
if ( ! QDir ( ) . exists ( newpath ) )
return 0 ;
///////////////////////////////////////////////////////////////////////////////////
// Parse Identification.tgt file (containing serial number and machine information)
///////////////////////////////////////////////////////////////////////////////////
filename = path + idfile + ext_TGT ;
QFile f ( filename ) ;
// Abort if this file is dodgy..
if ( ! f . exists ( ) | | ! f . open ( QIODevice : : ReadOnly ) )
return 0 ;
// Parse # entries into idmap.
while ( ! f . atEnd ( ) ) {
line = f . readLine ( ) . trimmed ( ) ;
if ( ! line . isEmpty ( ) ) {
key = line . section ( " " , 0 , 0 ) ;
value = line . section ( " " , 1 ) ;
key = key . section ( " # " , 1 ) ;
if ( key = = " SRN " ) {
key = STR_PROP_Serial ;
serial = value ;
2011-06-29 19:06:49 +00:00
}
2011-12-30 23:02:45 +00:00
idmap [ key ] = value ;
}
}
f . close ( ) ;
// Abort if no serial number
if ( serial . isEmpty ( ) ) {
qDebug ( ) < < " S9 Data card has no valid serial number in Indentification.tgt " ;
return 0 ;
}
// Early check for STR.edf file, so we can early exit before creating faulty machine record.
QString strpath = path + strfile + ext_EDF ; // STR.edf file
f . setFileName ( strpath ) ;
2012-01-05 04:37:22 +00:00
if ( ! f . exists ( ) ) { // No STR.edf.. Do we have a STR.edf.gz?
strpath + = ext_gz ;
f . setFileName ( strpath ) ;
if ( ! f . exists ( ) ) {
qDebug ( ) < < " Missing STR.edf file " ;
return 0 ;
}
2011-12-30 23:02:45 +00:00
}
///////////////////////////////////////////////////////////////////////////////////
// Create machine object (unless it's already registered)
///////////////////////////////////////////////////////////////////////////////////
Machine * m = CreateMachine ( serial , profile ) ;
2012-01-05 04:37:22 +00:00
bool create_backups = PROFILE . session - > backupCardData ( ) ;
bool compress_backups = PROFILE . session - > compressBackupData ( ) ;
QString backup_path = PROFILE . Get ( m - > properties [ STR_PROP_BackupPath ] ) ;
if ( backup_path . isEmpty ( ) )
backup_path = PROFILE . Get ( m - > properties [ STR_PROP_Path ] ) + " Backup/ " ;
if ( path = = backup_path ) {
create_backups = false ;
}
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
// Parse the idmap into machine objects properties, (overwriting any old values)
///////////////////////////////////////////////////////////////////////////////////
for ( QHash < QString , QString > : : iterator i = idmap . begin ( ) ; i ! = idmap . end ( ) ; i + + ) {
m - > properties [ i . key ( ) ] = i . value ( ) ;
if ( i . key ( ) = = " PCD " ) { // Lookup Product Code for real model string
bool ok ;
2011-12-31 06:54:51 +00:00
int j = i . value ( ) . toInt ( & ok ) ;
2011-12-30 23:02:45 +00:00
if ( ok ) m - > properties [ STR_PROP_Model ] = RMS9ModelMap [ j ] ;
2011-06-29 19:06:49 +00:00
}
}
2011-09-17 12:39:00 +00:00
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
// Open and Parse STR.edf file
///////////////////////////////////////////////////////////////////////////////////
EDFParser stredf ( strpath ) ;
2011-09-17 12:39:00 +00:00
if ( ! stredf . Parse ( ) ) {
qDebug ( ) < < " Faulty file " < < strfile ;
return 0 ;
}
2011-12-30 23:02:45 +00:00
if ( stredf . serialnumber ! = serial ) {
qDebug ( ) < < " Identification.tgt Serial number doesn't match STR.edf! " ;
}
// Creating early as we need the object
QDir dir ( newpath ) ;
2012-01-05 04:37:22 +00:00
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
// Create the backup folder for storing a copy of everything in..
2012-01-05 04:37:22 +00:00
// (Unless we are importing from this backup folder)
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
2012-01-05 04:37:22 +00:00
if ( create_backups ) {
if ( ! dir . exists ( backup_path ) ) {
if ( ! dir . mkpath ( backup_path + datalog ) ) {
qDebug ( ) < < " Could not create S9 backup directory :-/ " ;
}
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
// Copy Identification files to backup folder
QFile : : copy ( path + idfile + ext_TGT , backup_path + idfile + ext_TGT ) ;
QFile : : copy ( path + idfile + ext_CRC , backup_path + idfile + ext_CRC ) ;
//copy STR files to backup folder
if ( strpath . endsWith ( ext_gz ) ) // Already compressed.
QFile : : copy ( strpath , backup_path + strfile + ext_EDF + ext_gz ) ;
else { // Compress STR file to backup folder
2012-01-06 16:07:54 +00:00
QString strf = backup_path + strfile + ext_EDF ;
if ( QFile : : exists ( strf ) )
QFile : : remove ( strf ) ;
2012-01-05 04:37:22 +00:00
compress_backups ?
2012-01-06 16:07:54 +00:00
compressFile ( strpath , strf )
2012-01-05 04:37:22 +00:00
:
2012-01-06 16:07:54 +00:00
QFile : : copy ( strpath , strf ) ;
2012-01-05 04:37:22 +00:00
}
2011-12-30 23:02:45 +00:00
2012-01-05 04:37:22 +00:00
QFile : : copy ( path + " STR.crc " , backup_path + " STR.crc " ) ;
}
2011-12-30 23:02:45 +00:00
///////////////////////////////////////////////////////////////////////////////////
// Process the actual STR.edf data
///////////////////////////////////////////////////////////////////////////////////
2011-09-17 12:39:00 +00:00
qint64 duration = stredf . GetNumDataRecords ( ) * stredf . GetDuration ( ) ;
int days = duration / 86400000L ;
2011-12-31 06:54:51 +00:00
//QDateTime dt1=QDateTime::fromTime_t(stredf.startdate/1000L);
//QDateTime dt2=QDateTime::fromTime_t(stredf.enddate/1000L);
//QDate dd1=dt1.date();
//QDate dd2=dt2.date();
2011-09-17 12:39:00 +00:00
2011-12-31 06:54:51 +00:00
// for (int s=0;s<stredf.GetNumSignals();s++) {
// EDFSignal & es=*stredf.edfsignals[s];
// long recs=es.nr*stredf.GetNumDataRecords();
// //qDebug() << "STREDF:" << es.label << recs;
// }
2011-09-17 12:39:00 +00:00
2011-12-30 23:02:45 +00:00
// Process STR.edf and find first and last time for each day
QVector < qint8 > dayused ;
dayused . resize ( days ) ;
QList < SessionID > strfirst ;
QList < SessionID > strlast ;
QList < int > strday ;
QList < bool > dayfoo ;
QHash < qint16 , QList < time_t > > daystarttimes ;
QHash < qint16 , QList < time_t > > dayendtimes ;
qint16 on , off ;
qint16 o1 [ 10 ] , o2 [ 10 ] ;
time_t st , et ;
time_t time = stredf . startdate / 1000L ; // == 12pm on first day
for ( int i = 0 ; i < days ; i + + ) {
EDFSignal * maskon = stredf . lookup [ " Mask On " ] ;
EDFSignal * maskoff = stredf . lookup [ " Mask Off " ] ;
int j = i * 10 ;
// Counts for on and off don't line up, and I'm not sure why
// The extra 'off' always seems to start with a 1 at the beginning
// A signal it's carried over from the day before perhaps? (noon boundary)
int ckon = 0 , ckoff = 0 ;
for ( int k = 0 ; k < 10 ; k + + ) {
on = maskon - > data [ j + k ] ;
off = maskoff - > data [ j + k ] ;
o1 [ k ] = on ;
o2 [ k ] = off ;
if ( on > = 0 ) ckon + + ;
if ( off > = 0 ) ckoff + + ;
}
2011-09-17 12:39:00 +00:00
2011-12-30 23:02:45 +00:00
// set to true if day starts with machine running
int offset = ckoff - ckon ;
dayfoo . push_back ( offset > 0 ) ;
2011-06-28 02:21:38 +00:00
2011-12-30 23:02:45 +00:00
st = 0 , et = 0 ;
time_t l , f ;
// Find the Min & Max times for this day
for ( int k = 0 ; k < ckon ; k + + ) {
on = o1 [ k ] ;
off = o2 [ k + offset ] ;
f = time + on * 60 ;
l = time + off * 60 ;
daystarttimes [ i ] . push_back ( f ) ;
dayendtimes [ i ] . push_back ( l ) ;
if ( ! st | | ( st > f ) ) st = f ;
if ( ! et | | ( et < l ) ) et = l ;
}
strfirst . push_back ( st ) ;
strlast . push_back ( et ) ;
strday . push_back ( i ) ;
dayused [ i ] = ckon ;
time + = 86400 ;
}
// reset time to first day
time = stredf . startdate / 1000 ;
///////////////////////////////////////////////////////////////////////////////////
// Open DATALOG file and build list of session files
///////////////////////////////////////////////////////////////////////////////////
2011-06-28 02:21:38 +00:00
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-06-29 16:19:57 +00:00
2012-01-05 04:37:22 +00:00
QString datestr ;
2011-06-29 14:19:38 +00:00
SessionID sessionid ;
QDateTime date ;
2011-06-30 04:55:20 +00:00
int size = flist . size ( ) ;
2011-12-30 23:02:45 +00:00
sessfiles . clear ( ) ;
2012-01-05 04:37:22 +00:00
// For each file in flist...
2011-06-30 04:55:20 +00:00
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-12-30 23:02:45 +00:00
// Forget about it if it can't be read.
if ( ! fi . isReadable ( ) )
continue ;
2012-01-05 04:37:22 +00:00
// Accept only .edf and .edf.gz files
if ( ! ( ( filename . right ( 4 ) . toLower ( ) = = " . " + ext_EDF )
| | ( filename . right ( 7 ) . toLower ( ) = = " . " + ext_EDF + ext_gz ) ) )
continue ;
2011-06-29 16:19:57 +00:00
2012-01-05 04:37:22 +00:00
// Extract the session date out of the filename
2011-06-29 14:19:38 +00:00
datestr = filename . section ( " _ " , 0 , 1 ) ;
2011-12-30 23:02:45 +00:00
2012-01-05 04:37:22 +00:00
// Take the filename's date, and
2011-06-29 14:19:38 +00:00
date = QDateTime : : fromString ( datestr , " yyyyMMdd_HHmmss " ) ;
2012-01-05 04:37:22 +00:00
// Skip file if dates invalid, the filename is clearly wrong..
2011-12-30 23:02:45 +00:00
if ( ! date . isValid ( ) )
2012-01-05 04:37:22 +00:00
continue ;
2011-12-30 23:02:45 +00:00
2012-01-05 04:37:22 +00:00
// convert this date to UNIX epoch to form the sessionID
2011-06-29 14:19:38 +00:00
sessionid = date . toTime_t ( ) ;
2011-12-26 18:26:06 +00:00
2011-12-30 23:02:45 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
2011-12-26 18:26:06 +00:00
// Resmed bugs up on the session filenames.. 1 or 2 seconds either way
2012-01-05 04:37:22 +00:00
// Moral of the story, when writing firmware and saving in batches, use the same datetimes,
// and provide firmware updates for free to your customers.
2011-12-30 23:02:45 +00:00
////////////////////////////////////////////////////////////////////////////////////////////
2011-06-29 21:23:00 +00:00
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
2012-01-05 04:37:22 +00:00
// Push current filename to ordered-by-sessionid list
sessfiles [ sessionid ] . push_back ( filename ) ;
2011-07-30 00:36:31 +00:00
2011-12-30 23:02:45 +00:00
// Update the progress bar
if ( qprogress ) qprogress - > setValue ( ( float ( i + 1 ) / float ( size ) * 10.0 ) ) ;
QApplication : : processEvents ( ) ;
2011-06-29 16:19:57 +00:00
}
2011-06-29 14:19:38 +00:00
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-12-30 23:02:45 +00:00
QHash < SessionID , int > sessday ;
/////////////////////////////////////////////////////////////////////////////
// Scan over file list and knock out of dayused list
/////////////////////////////////////////////////////////////////////////////
2012-01-05 04:37:22 +00:00
int dn ;
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-12-30 23:02:45 +00:00
// Earliest possible day number
int edn = ( ( sessionid - time ) / 86400 ) - 1 ;
if ( edn < 0 ) edn = 0 ;
// Find real day number from str.edf mask on/off data.
2012-01-05 04:37:22 +00:00
dn = - 1 ;
2011-12-30 23:02:45 +00:00
for ( int j = edn ; j < strfirst . size ( ) ; j + + ) {
2012-01-05 04:37:22 +00:00
st = strfirst . at ( j ) ;
2011-12-30 23:02:45 +00:00
if ( sessionid > = st ) {
2012-01-05 04:37:22 +00:00
et = strlast . at ( j ) ;
2011-12-30 23:02:45 +00:00
if ( sessionid < ( et + 300 ) ) {
dn = j ;
break ;
}
}
}
2012-01-05 04:37:22 +00:00
// If found, mark day off so STR.edf summary data isn't used instead of the real thing.
2011-12-30 23:02:45 +00:00
if ( dn > = 0 ) {
dayused [ dn ] = 0 ;
}
}
EDFSignal * sig ;
/////////////////////////////////////////////////////////////////////////////
// For all days not in session lists, (to get at days without data records)
/////////////////////////////////////////////////////////////////////////////
2012-01-05 04:37:22 +00:00
for ( dn = 0 ; dn < days ; dn + + ) {
// Skip days with loadable data.
if ( ! dayused [ dn ] )
continue ;
2011-12-30 23:02:45 +00:00
2012-01-05 04:37:22 +00:00
if ( ! daystarttimes . contains ( dn ) )
continue ;
2011-12-30 23:02:45 +00:00
2011-07-03 03:53:09 +00:00
sess = NULL ;
2012-01-05 04:37:22 +00:00
int scnt = daystarttimes [ dn ] . size ( ) ; // count of sessions for this day
// Create a new session for each mask on/off segment in a day
// But only populate the last one with summary data.
for ( int j = 0 ; j < scnt ; j + + ) {
2011-12-30 23:02:45 +00:00
st = daystarttimes [ dn ] . at ( j ) ;
2011-06-29 16:19:57 +00:00
2011-12-30 23:02:45 +00:00
// Skip if session already exists
if ( m - > SessionExists ( st ) )
2011-06-29 16:19:57 +00:00
continue ;
2011-12-30 23:02:45 +00:00
et = dayendtimes [ dn ] . at ( j ) ;
2012-01-05 04:37:22 +00:00
// Create the session
2011-12-30 23:02:45 +00:00
sess = new Session ( m , st ) ;
sess - > really_set_first ( qint64 ( st ) * 1000L ) ;
sess - > really_set_last ( qint64 ( et ) * 1000L ) ;
2011-12-31 06:54:51 +00:00
sess - > SetChanged ( true ) ;
m - > AddSession ( sess , profile ) ;
}
// Add the actual data to the last session
EventDataType tmp , dur ;
if ( sess ) {
2011-12-30 23:02:45 +00:00
/////////////////////////////////////////////////////////////////////
// CPAP Mode
/////////////////////////////////////////////////////////////////////
int mode ;
sig = stredf . lookupSignal ( CPAP_Mode ) ;
if ( sig ) {
mode = sig - > data [ dn ] ;
} else mode = 0 ;
/////////////////////////////////////////////////////////////////////
// EPR Settings
/////////////////////////////////////////////////////////////////////
sess - > settings [ CPAP_PresReliefType ] = PR_EPR ;
// Note: AutoSV machines don't have both fields
sig = stredf . lookupSignal ( RMS9_EPR ) ;
if ( sig ) {
2012-01-02 07:48:49 +00:00
sess - > settings [ CPAP_PresReliefMode ] = EventDataType ( sig - > data [ dn ] ) * sig - > gain ;
2011-12-30 23:02:45 +00:00
}
sig = stredf . lookupSignal ( RMS9_EPRSet ) ;
if ( sig ) {
2012-01-02 07:48:49 +00:00
sess - > settings [ CPAP_PresReliefSet ] = EventDataType ( sig - > data [ dn ] ) * sig - > gain ;
2011-12-30 23:02:45 +00:00
}
/////////////////////////////////////////////////////////////////////
// Set Min & Max pressures depending on CPAP mode
/////////////////////////////////////////////////////////////////////
if ( mode = = 0 ) {
sess - > settings [ CPAP_Mode ] = MODE_CPAP ;
sig = stredf . lookupSignal ( RMS9_SetPressure ) ; // ?? What's meant by Set Pressure?
if ( sig ) {
EventDataType pressure = sig - > data [ dn ] * sig - > gain ;
2011-12-31 11:36:19 +00:00
sess - > settings [ CPAP_Pressure ] = pressure ;
2011-06-29 19:06:49 +00:00
}
2011-12-30 23:02:45 +00:00
} else { // VPAP or Auto
if ( mode > 5 ) {
2011-12-31 06:54:51 +00:00
if ( mode > = 7 )
sess - > settings [ CPAP_Mode ] = MODE_ASV ;
else
sess - > settings [ CPAP_Mode ] = MODE_BIPAP ;
EventDataType tmp , epap = 0 , ipap = 0 ;
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " EPAP " ) ) ) {
2011-12-31 06:54:51 +00:00
epap = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_EPAP ] = epap ;
sess - > setMin ( CPAP_EPAP , epap ) ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " IPAP " ) ) ) {
2011-12-31 06:54:51 +00:00
ipap = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_IPAP ] = ipap ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " PS " ) ) ) {
2011-12-31 06:54:51 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PS ] = tmp ; // technically this is IPAP-EPAP
if ( ! ipap ) {
// not really possible. but anyway, just in case..
sess - > settings [ CPAP_IPAP ] = epap + tmp ;
}
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Min PS " ) ) ) {
2011-12-31 06:54:51 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PSMin ] = tmp ;
sess - > settings [ CPAP_IPAPLo ] = epap + tmp ;
sess - > setMin ( CPAP_IPAP , epap + tmp ) ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Max PS " ) ) ) {
2011-12-31 06:54:51 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PSMax ] = tmp ;
sess - > settings [ CPAP_IPAPHi ] = epap + tmp ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " RR " ) ) ) { // Is this a setting to force respiratory rate on S/T machines?
2011-12-31 06:54:51 +00:00
tmp = sig - > data [ dn ] ;
sess - > settings [ CPAP_RespRate ] = tmp * sig - > gain ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Easy-Breathe " ) ) ) {
2011-12-31 06:54:51 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PresReliefSet ] = tmp ;
sess - > settings [ CPAP_PresReliefType ] = ( int ) PR_EASYBREATHE ;
sess - > settings [ CPAP_PresReliefMode ] = ( int ) PM_FullTime ;
}
2011-12-30 23:02:45 +00:00
} else {
sess - > settings [ CPAP_Mode ] = MODE_APAP ;
2011-12-31 06:54:51 +00:00
sig = stredf . lookupSignal ( CPAP_PressureMin ) ;
if ( sig ) {
EventDataType pressure = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PressureMin ] = pressure ;
2011-12-31 11:36:19 +00:00
//sess->setMin(CPAP_Pressure,pressure);
2011-12-31 06:54:51 +00:00
}
sig = stredf . lookupSignal ( CPAP_PressureMax ) ;
if ( sig ) {
EventDataType pressure = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PressureMax ] = pressure ;
2011-12-31 11:36:19 +00:00
//sess->setMax(CPAP_Pressure,pressure);
2011-12-31 06:54:51 +00:00
}
2011-12-30 23:02:45 +00:00
}
2011-06-29 16:19:57 +00:00
}
2011-12-30 23:02:45 +00:00
EventDataType valmed = 0 , valmax = 0 , val95 = 0 ;
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Leak Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Leak Med " ) ) ) {
2011-12-30 23:02:45 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_Leak ] = sig - > gain * 60.0 ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_Leak ] [ valmed ] = 51 ;
2011-12-30 23:02:45 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Leak 95 " ) ) ) {
2011-12-30 23:02:45 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_Leak ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Leak Max " ) ) ) {
2011-12-30 23:02:45 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_Leak , valmax * sig - > gain * 60.0 ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_Leak ] [ valmax ] = 4 ;
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Minute Ventilation Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Min Vent Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_MinuteVent ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_MinuteVent ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Min Vent 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_MinuteVent ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Min Vent Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_MinuteVent , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_MinuteVent ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Respiratory Rate Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " RR Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_RespRate ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_RespRate ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " RR 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_RespRate ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " RR Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_RespRate , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_RespRate ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Tidal Volume Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Tid Vol Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
2012-01-01 15:47:21 +00:00
sess - > m_gain [ CPAP_TidalVolume ] = sig - > gain * 1000.0 ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_TidalVolume ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Tid Vol 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_TidalVolume ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Tid Vol Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
2012-01-01 15:47:21 +00:00
sess - > setMax ( CPAP_TidalVolume , valmax * sig - > gain * 1000.0 ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_TidalVolume ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Target Minute Ventilation Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Targ Vent Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_TgMV ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_TgMV ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Targ Vent 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_TgMV ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Targ Vent Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_TgMV , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_TgMV ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// I:E Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " I:E Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_IE ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_IE ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " I:E 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_IE ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " I:E Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_IE , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_IE ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Mask Pressure Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Mask Pres Med " ) ) ) {
2011-12-30 23:02:45 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_Pressure ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_Pressure ] [ valmed ] = 51 ;
2011-12-30 23:02:45 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Mask Pres 95 " ) ) ) {
2011-12-30 23:02:45 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_Pressure ] [ val95 ] = 45 ;
2011-06-29 16:19:57 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Mask Pres Max " ) ) ) {
2011-12-30 23:02:45 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_Pressure , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_Pressure ] [ valmax ] = 4 ;
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Inspiratory Pressure (IPAP) Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Insp Pres Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_IPAP ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_IPAP ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Insp Pres 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_IPAP ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Insp Pres Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_IPAP , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_IPAP ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Expiratory Pressure (EPAP) Summary
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Exp Pres Med " ) ) ) {
2011-12-31 06:54:51 +00:00
valmed = sig - > data [ dn ] ;
sess - > m_gain [ CPAP_EPAP ] = sig - > gain ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_EPAP ] [ valmed ] = 51 ;
2011-12-31 06:54:51 +00:00
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Exp Pres 95 " ) ) ) {
2011-12-31 06:54:51 +00:00
val95 = sig - > data [ dn ] ;
sess - > m_valuesummary [ CPAP_EPAP ] [ val95 ] = 45 ;
}
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Exp Pres Max " ) ) ) {
2011-12-31 06:54:51 +00:00
valmax = sig - > data [ dn ] ;
sess - > setMax ( CPAP_EPAP , valmax * sig - > gain ) ;
2012-01-01 14:14:18 +00:00
sess - > m_valuesummary [ CPAP_EPAP ] [ valmax ] = 4 ;
2011-12-31 06:54:51 +00:00
}
2011-12-30 23:02:45 +00:00
2012-01-05 04:37:22 +00:00
/////////////////////////////////////////////////////////////////////
// Duration and Event Indices
/////////////////////////////////////////////////////////////////////
2012-01-03 07:15:02 +00:00
if ( ( sig = stredf . lookupName ( " Mask Dur " ) ) ) {
2011-12-30 23:02:45 +00:00
dur = sig - > data [ dn ] * sig - > gain ;
2012-01-05 04:37:22 +00:00
dur / = 60.0f ; // convert to hours.
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
if ( ( sig = stredf . lookupName ( " OAI " ) ) ) { // Obstructive Apnea Index
2011-12-30 23:02:45 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > setCph ( CPAP_Obstructive , tmp ) ;
2012-01-05 04:37:22 +00:00
sess - > setCount ( CPAP_Obstructive , tmp * dur ) ; // Converting from indice to counts..
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
if ( ( sig = stredf . lookupName ( " HI " ) ) ) { // Hypopnea Index
2011-12-30 23:02:45 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > setCph ( CPAP_Hypopnea , tmp ) ;
2012-01-05 04:37:22 +00:00
sess - > setCount ( CPAP_Hypopnea , tmp * dur ) ;
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
if ( ( sig = stredf . lookupName ( " UAI " ) ) ) { // Unspecified Apnea Index
2011-12-30 23:02:45 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > setCph ( CPAP_Apnea , tmp ) ;
2012-01-05 04:37:22 +00:00
sess - > setCount ( CPAP_Apnea , tmp * dur ) ;
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
if ( ( sig = stredf . lookupName ( " CAI " ) ) ) { // "Central" Apnea Index
2011-12-30 23:02:45 +00:00
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > setCph ( CPAP_ClearAirway , tmp ) ;
2012-01-05 04:37:22 +00:00
sess - > setCount ( CPAP_ClearAirway , tmp * dur ) ;
2011-12-30 23:02:45 +00:00
}
2011-06-29 14:19:38 +00:00
}
2011-12-30 23:02:45 +00:00
}
2012-01-05 04:37:22 +00:00
bool gz ;
backup_path + = datalog + " / " ;
2011-12-30 23:02:45 +00:00
/////////////////////////////////////////////////////////////////////////////
// Scan through new file list and import sessions
/////////////////////////////////////////////////////////////////////////////
for ( QMap < SessionID , QVector < QString > > : : iterator si = sessfiles . begin ( ) ; si ! = sessfiles . end ( ) ; si + + ) {
sessionid = si . key ( ) ;
// Skip file if already imported
if ( m - > SessionExists ( sessionid ) )
continue ;
// Create the session
sess = new Session ( m , sessionid ) ;
2012-01-05 04:37:22 +00:00
QString filename , fullpath , backupfile , backfile , crcfile ;
2011-12-30 23:02:45 +00:00
// Process EDF File List
for ( int i = 0 ; i < si . value ( ) . size ( ) ; + + i ) {
2012-01-05 04:37:22 +00:00
filename = si . value ( ) [ i ] ;
gz = ( filename . right ( 3 ) . toLower ( ) = = ext_gz ) ;
fullpath = newpath + filename ;
2011-12-30 23:02:45 +00:00
// Copy the EDF file to the backup folder
2012-01-05 04:37:22 +00:00
if ( create_backups ) {
backupfile = backup_path + filename ;
2012-01-06 16:07:54 +00:00
bool dobackup = true ;
if ( ! gz & & QFile : : exists ( backupfile + " .gz " ) ) {
dobackup = false ;
} else if ( QFile : : exists ( backupfile ) ) {
if ( gz ) {
// don't bother, it's already there and compressed.
dobackup = false ;
} else {
// non compressed file is there..
if ( compress_backups ) {
// remove old edf file, as we are writing a compressed one
QFile : : remove ( backupfile ) ;
} else { // don't bother copying it.
dobackup = false ;
}
}
}
if ( dobackup ) {
if ( ! gz ) {
compress_backups ?
compressFile ( fullpath , backupfile )
:
QFile : : copy ( fullpath , backupfile ) ;
} else {
// already compressed, just copy it.
2012-01-05 04:37:22 +00:00
QFile : : copy ( fullpath , backupfile ) ;
2012-01-06 16:07:54 +00:00
}
}
if ( ! gz ) {
backfile = filename . replace ( " .edf " , " .crc " , Qt : : CaseInsensitive ) ;
} else {
backfile = filename . replace ( " .edf.gz " , " .crc " , Qt : : CaseInsensitive ) ;
}
2012-01-05 04:37:22 +00:00
backupfile = backup_path + backfile ;
crcfile = newpath + backfile ;
QFile : : copy ( crcfile , backupfile ) ;
}
2011-12-30 23:02:45 +00:00
EDFParser edf ( fullpath ) ;
// Parse the actual file
if ( ! edf . Parse ( ) )
continue ;
// Give a warning if doesn't match the machine serial number in Identification.tgt
if ( edf . serialnumber ! = serial ) {
qDebug ( ) < < " edf Serial number doesn't match Identification.tgt " ;
}
2012-01-05 04:37:22 +00:00
fn = filename . section ( " _ " , - 1 ) . section ( " . " , 0 , 0 ) . toLower ( ) ;
2011-12-30 23:02:45 +00:00
2012-01-05 04:37:22 +00:00
if ( fn = = " eve " ) LoadEVE ( sess , edf ) ;
else if ( fn = = " pld " ) LoadPLD ( sess , edf ) ;
else if ( fn = = " brp " ) LoadBRP ( sess , edf ) ;
else if ( fn = = " sad " ) LoadSAD ( sess , edf ) ;
2011-12-30 23:02:45 +00:00
}
if ( qprogress ) qprogress - > setValue ( 10.0 + ( float ( + + cnt ) / float ( size ) * 90.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 ) ;
2011-09-17 12:39:00 +00:00
qint64 dif = sess - > first ( ) - stredf . startdate ;
int dn = dif / 86400000L ;
if ( dn < days ) {
int mode ;
2011-11-25 12:13:35 +00:00
sig = stredf . lookupSignal ( CPAP_Mode ) ;
if ( sig ) {
mode = sig - > data [ dn ] ;
} else mode = 0 ;
2011-10-21 05:50:31 +00:00
2011-12-30 23:02:45 +00:00
sess - > settings [ CPAP_PresReliefType ] = PR_EPR ;
2011-10-21 05:50:31 +00:00
// AutoSV machines don't have both fields
2011-12-21 14:24:09 +00:00
sig = stredf . lookupSignal ( RMS9_EPR ) ;
2011-11-25 12:13:35 +00:00
if ( sig ) {
2012-01-02 07:48:49 +00:00
sess - > settings [ CPAP_PresReliefMode ] = EventDataType ( sig - > data [ dn ] ) * sig - > gain ;
2011-11-25 12:13:35 +00:00
}
2011-12-21 14:24:09 +00:00
sig = stredf . lookupSignal ( RMS9_EPRSet ) ;
2011-11-25 12:13:35 +00:00
if ( sig ) {
2012-01-02 07:48:49 +00:00
sess - > settings [ CPAP_PresReliefSet ] = EventDataType ( sig - > data [ dn ] ) * sig - > gain ;
2011-11-25 12:13:35 +00:00
}
2011-09-17 12:39:00 +00:00
2011-12-31 06:54:51 +00:00
2011-09-17 12:39:00 +00:00
if ( mode = = 0 ) {
2011-09-18 15:43:14 +00:00
sess - > settings [ CPAP_Mode ] = MODE_CPAP ;
2011-12-21 17:00:19 +00:00
sig = stredf . lookupSignal ( RMS9_SetPressure ) ; // ?? What's meant by Set Pressure?
2011-11-25 12:13:35 +00:00
if ( sig ) {
EventDataType pressure = sig - > data [ dn ] * sig - > gain ;
2011-12-31 06:54:51 +00:00
sess - > settings [ CPAP_Pressure ] = pressure ;
2011-11-25 12:13:35 +00:00
}
2011-12-31 06:54:51 +00:00
} else if ( mode > 5 ) {
if ( mode > = 7 )
sess - > settings [ CPAP_Mode ] = MODE_ASV ;
else
2011-09-17 12:39:00 +00:00
sess - > settings [ CPAP_Mode ] = MODE_BIPAP ;
2011-11-25 12:13:35 +00:00
2011-12-31 06:54:51 +00:00
EventDataType tmp , epap = 0 , ipap = 0 ;
if ( stredf . lookup . contains ( " EPAP " ) ) {
sig = stredf . lookup [ " EPAP " ] ;
epap = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_EPAP ] = epap ;
}
if ( stredf . lookup . contains ( " IPAP " ) ) {
sig = stredf . lookup [ " IPAP " ] ;
ipap = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_IPAP ] = ipap ;
2011-12-12 09:16:26 +00:00
}
2011-12-31 06:54:51 +00:00
if ( stredf . lookup . contains ( " PS " ) ) {
sig = stredf . lookup [ " PS " ] ;
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PS ] = tmp ; // technically this is IPAP-EPAP
if ( ! ipap ) {
// not really possible. but anyway, just in case..
sess - > settings [ CPAP_IPAP ] = epap + tmp ;
}
}
if ( stredf . lookup . contains ( " Min PS " ) ) {
sig = stredf . lookup [ " Min PS " ] ;
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PSMin ] = tmp ;
sess - > settings [ CPAP_IPAPLo ] = epap + tmp ;
sess - > setMin ( CPAP_IPAP , epap + tmp ) ;
}
if ( stredf . lookup . contains ( " Max PS " ) ) {
sig = stredf . lookup [ " Max PS " ] ;
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PSMax ] = tmp ;
sess - > settings [ CPAP_IPAPHi ] = epap + tmp ;
}
if ( stredf . lookup . contains ( " RR " ) ) { // Is this a setting to force respiratory rate on S/T machines?
sig = stredf . lookup [ " RR " ] ;
tmp = sig - > data [ dn ] ;
sess - > settings [ CPAP_RespRate ] = tmp * sig - > gain ;
}
if ( stredf . lookup . contains ( " Easy-Breathe " ) ) {
sig = stredf . lookup [ " Easy-Breathe " ] ;
tmp = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PresReliefSet ] = tmp ;
sess - > settings [ CPAP_PresReliefType ] = ( int ) PR_EASYBREATHE ;
sess - > settings [ CPAP_PresReliefMode ] = ( int ) PM_FullTime ;
}
} else {
sess - > settings [ CPAP_Mode ] = MODE_APAP ;
2011-12-21 14:24:09 +00:00
sig = stredf . lookupSignal ( CPAP_PressureMin ) ;
2011-12-12 09:16:26 +00:00
if ( sig ) {
EventDataType pressure = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PressureMin ] = pressure ;
2011-12-31 11:36:19 +00:00
//sess->setMin(CPAP_Pressure,pressure);
2011-12-12 09:16:26 +00:00
}
2011-12-21 14:24:09 +00:00
sig = stredf . lookupSignal ( CPAP_PressureMax ) ;
2011-12-12 09:16:26 +00:00
if ( sig ) {
EventDataType pressure = sig - > data [ dn ] * sig - > gain ;
sess - > settings [ CPAP_PressureMax ] = pressure ;
2011-12-31 11:36:19 +00:00
//sess->setMax(CPAP_Pressure,pressure);
2011-12-12 09:16:26 +00:00
}
2011-09-17 12:39:00 +00:00
2011-12-31 06:54:51 +00:00
}
2011-09-17 12:39:00 +00:00
}
2011-12-26 18:26:06 +00:00
// The following only happens when the STR.edf file is not up to date..
// This will only happen when the user fails to back up their SDcard properly.
// Basically takes a guess.
if ( ! sess - > settings . contains ( CPAP_Mode ) ) {
//The following is a lame assumption if 50th percentile == max, then it's CPAP
EventDataType p50 = sess - > percentile ( CPAP_Pressure , 0.50 ) ;
EventDataType max = sess - > Max ( CPAP_Pressure ) ;
if ( max = = p50 ) {
sess - > settings [ CPAP_Mode ] = MODE_CPAP ;
sess - > settings [ CPAP_PressureMin ] = p50 ;
} else {
// It's not cpap, so just take the highest setting for this machines history.
// This may fail if the missing str data is at the beginning of a fresh import.
CPAPMode mode = ( CPAPMode ) ( int ) PROFILE . calcSettingsMax ( CPAP_Mode , MT_CPAP , sess - > machine ( ) - > FirstDay ( ) , sess - > machine ( ) - > LastDay ( ) ) ;
if ( mode < MODE_APAP ) mode = MODE_APAP ;
sess - > settings [ CPAP_Mode ] = mode ;
// Assuming 10th percentile should cover for ramp/warmup
sess - > settings [ CPAP_PressureMin ] = sess - > percentile ( CPAP_Pressure , 0.10 ) ;
sess - > settings [ CPAP_PressureMax ] = sess - > Max ( CPAP_Pressure ) ;
}
}
//Rather than take a dodgy guess, EPR settings can take a hit, and this data can simply be missed..
// Add the session to the machine & profile objects
2011-12-30 23:02:45 +00:00
m - > AddSession ( sess , profile ) ;
2011-07-03 03:53:09 +00:00
}
2011-09-17 12:39:00 +00:00
}
2012-01-05 06:54:07 +00:00
# ifdef DEBUG_EFFICIENCY
{
qint64 totalbytes = 0 ;
qint64 totalns = 0 ;
qDebug ( ) < < " Time Delta Efficiency Information " ;
for ( QHash < ChannelID , qint64 > : : iterator it = channel_efficiency . begin ( ) ; it ! = channel_efficiency . end ( ) ; it + + ) {
ChannelID code = it . key ( ) ;
qint64 value = it . value ( ) ;
qint64 ns = channel_time [ code ] ;
totalbytes + = value ;
totalns + = ns ;
double secs = double ( ns ) / 1000000000.0 L ;
QString s = value < 0 ? " saved " : " cost " ;
qDebug ( ) < < " Time-Delta conversion for " + schema : : channel [ code ] . label ( ) + " " + s + " " + QString : : number ( qAbs ( value ) ) + " bytes and took " + QString : : number ( secs , ' f ' , 4 ) + " s " ;
}
qDebug ( ) < < " Total toTimeDelta function usage: " < < totalbytes < < " in " < < double ( totalns ) / 1000000000.0 < < " seconds " ;
}
# endif
2011-12-21 04:25:01 +00:00
2011-09-17 12:39:00 +00:00
if ( m ) {
m - > Save ( ) ;
2011-06-28 02:21:38 +00:00
}
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-12-31 06:54:51 +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
if ( t = = " obstructive apnea " ) {
if ( ! EL [ 0 ] ) {
2011-09-17 12:39:00 +00:00
if ( ! ( EL [ 0 ] = sess - > AddEventList ( CPAP_Obstructive , EVL_Event ) ) ) return false ;
2011-07-27 09:21:53 +00:00
}
EL [ 0 ] - > AddEvent ( tt , duration ) ;
} else if ( t = = " hypopnea " ) {
if ( ! EL [ 1 ] ) {
2011-09-17 12:39:00 +00:00
if ( ! ( EL [ 1 ] = sess - > AddEventList ( CPAP_Hypopnea , EVL_Event ) ) ) return false ;
2011-07-27 09:21:53 +00:00
}
2011-09-08 07:43:04 +00:00
EL [ 1 ] - > AddEvent ( tt , duration + 10 ) ; // Only Hyponea's Need the extra duration???
2011-07-27 09:21:53 +00:00
} else if ( t = = " apnea " ) {
if ( ! EL [ 2 ] ) {
2011-09-17 12:39:00 +00:00
if ( ! ( EL [ 2 ] = sess - > AddEventList ( CPAP_Apnea , EVL_Event ) ) ) return false ;
2011-07-27 09:21:53 +00:00
}
EL [ 2 ] - > AddEvent ( tt , duration ) ;
} else if ( t = = " central apnea " ) {
2011-12-31 06:54:51 +00:00
//code=CPAP_ClearAirway;
2011-07-27 09:21:53 +00:00
if ( ! EL [ 3 ] ) {
2011-09-17 12:39:00 +00:00
if ( ! ( EL [ 3 ] = sess - > AddEventList ( CPAP_ClearAirway , EVL_Event ) ) ) return false ;
2011-07-27 09:21:53 +00:00
}
EL [ 3 ] - > AddEvent ( tt , duration ) ;
2011-06-29 17:58:28 +00:00
} else {
if ( t ! = " recording starts " ) {
2011-08-09 23:44:36 +00:00
qDebug ( ) < < " Unobserved 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;
2012-01-06 16:07:54 +00:00
long recs = es . nr * edf . GetNumDataRecords ( ) ;
2011-07-31 20:24:43 +00:00
ChannelID code ;
2012-01-06 16:07:54 +00:00
if ( es . offset > 0 ) {
int i = 5 ;
}
if ( es . label = = " Flow " ) {
2011-08-07 11:37:56 +00:00
es . gain * = 60 ;
2011-08-09 23:44:36 +00:00
es . physical_dimension = " L/M " ;
2011-07-27 09:21:53 +00:00
code = CPAP_FlowRate ;
2012-01-06 16:07:54 +00:00
} else if ( es . label . startsWith ( " Mask Pres " ) ) {
2011-09-17 13:21:18 +00:00
code = CPAP_MaskPressureHi ;
2011-08-09 23:44:36 +00:00
} else if ( es . label . startsWith ( " Resp Event " ) ) {
2011-09-17 12:39:00 +00:00
code = CPAP_RespEvent ;
2011-07-10 17:49:02 +00:00
} else {
2011-08-09 23:44:36 +00:00
qDebug ( ) < < " Unobserved ResMed BRP 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 ) ;
2011-09-17 12:39:00 +00:00
EventList * a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
2011-08-09 23:44:36 +00:00
a - > setDimension ( es . physical_dimension ) ;
2011-07-27 09:21:53 +00:00
a - > AddWaveform ( edf . startdate , es . data , recs , duration ) ;
2011-12-17 14:38:15 +00:00
sess - > setMin ( code , a - > Min ( ) ) ;
sess - > setMax ( code , a - > Max ( ) ) ;
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-12-08 11:41:44 +00:00
EventList * ResmedLoader : : ToTimeDelta ( Session * sess , EDFParser & edf , EDFSignal & es , ChannelID code , long recs , qint64 duration , EventDataType min , EventDataType max , bool square )
2011-06-29 20:30:23 +00:00
{
2012-01-05 06:54:07 +00:00
# ifdef DEBUG_EFFICIENCY
QElapsedTimer time ;
time . start ( ) ;
# endif
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-09-17 12:39:00 +00:00
EventList * el = sess - > AddEventList ( code , EVL_Event , es . gain , es . offset , min , max ) ;
2011-09-13 08:12:07 +00:00
int startpos = 0 ;
2011-09-17 12:39:00 +00:00
if ( ( code = = CPAP_Pressure ) | | ( code = = CPAP_IPAP ) | | ( code = = CPAP_EPAP ) ) {
startpos = 20 ; // Shave the first 20 seconds of pressure data
2011-09-13 08:12:07 +00:00
tt + = rate * startpos ;
2011-09-17 12:39:00 +00:00
}
2011-09-13 08:12:07 +00:00
for ( int i = startpos ; 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-12-08 11:41:44 +00:00
if ( square ) el - > AddEvent ( tt , last ) ; // square waves look better on some charts.
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 ) ;
2012-01-05 06:54:07 +00:00
# ifdef DEBUG_EFFICIENCY
qint64 t = time . nsecsElapsed ( ) ;
int cnt = el - > count ( ) ;
int bytes = cnt * ( sizeof ( EventStoreType ) + sizeof ( quint32 ) ) ;
int wvbytes = recs * ( sizeof ( EventStoreType ) ) ;
QHash < ChannelID , qint64 > : : iterator it = channel_efficiency . find ( code ) ;
if ( it = = channel_efficiency . end ( ) ) {
channel_efficiency [ code ] = wvbytes - bytes ;
channel_time [ code ] = t ;
} else {
it . value ( ) + = wvbytes - bytes ;
channel_time [ code ] + = t ;
}
# endif
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;
2012-01-06 16:07:54 +00:00
long recs = es . nr * edf . GetNumDataRecords ( ) ;
2011-08-06 13:37:06 +00:00
ChannelID code ;
2012-01-06 16:07:54 +00:00
if ( es . label . startsWith ( " Puls " ) ) {
2011-09-17 12:39:00 +00:00
code = OXI_Pulse ;
2012-01-06 16:07:54 +00:00
} else if ( es . label = = " SpO2 " ) {
2011-09-17 12:39:00 +00:00
code = OXI_SPO2 ;
2011-08-06 13:37:06 +00:00
} else {
2011-08-09 23:44:36 +00:00
qDebug ( ) < < " Unobserved ResMed SAD Signal " < < edf . edfsignals [ s ] - > label ;
2011-08-06 13:37:06 +00:00
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 ) {
2011-12-17 14:38:15 +00:00
sess - > setMin ( code , a - > Min ( ) ) ;
sess - > setMax ( code , a - > Max ( ) ) ;
2011-08-06 13:37:06 +00:00
}
}
}
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-03 02:43:50 +00:00
qint64 duration = edf . GetNumDataRecords ( ) * edf . GetDuration ( ) ;
2011-09-17 12:39:00 +00:00
sess - > updateFirst ( edf . startdate ) ;
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 ] ;
2012-01-06 16:07:54 +00:00
if ( es . offset > 0 ) {
int i = 5 ;
}
2011-08-01 08:53:26 +00:00
recs = es . nr * edf . GetNumDataRecords ( ) ;
2011-09-17 12:39:00 +00:00
if ( recs < = 0 ) continue ;
2011-08-01 08:53:26 +00:00
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-08-01 08:53:26 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 ) ;
2011-11-25 12:13:35 +00:00
} else if ( es . label . startsWith ( " Therapy Pres " ) ) {
2011-09-17 12:39:00 +00:00
code = CPAP_Pressure ; //TherapyPressure;
2011-08-07 12:33:00 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 ) ;
2011-08-09 23:44:36 +00:00
} else if ( es . label = = " Insp Pressure " ) {
2011-08-07 12:33:00 +00:00
code = CPAP_IPAP ; //TherapyPressure;
sess - > settings [ CPAP_Mode ] = MODE_BIPAP ;
2011-08-01 08:53:26 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 ) ;
2011-11-25 12:42:41 +00:00
} else if ( ( es . label = = " MV " ) | | ( es . label = = " VM " ) ) {
2011-09-17 12:39:00 +00:00
code = CPAP_MinuteVent ;
2011-08-01 08:53:26 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 ) ;
2011-11-25 12:42:41 +00:00
} else if ( ( es . label = = " RR " ) | | ( es . label = = " AF " ) | | ( es . label = = " FR " ) ) {
2011-09-17 12:39:00 +00:00
code = CPAP_RespRate ;
a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
2011-07-27 09:21:53 +00:00
a - > AddWaveform ( edf . startdate , es . data , recs , duration ) ;
2011-11-25 12:42:41 +00:00
} else if ( ( es . label = = " Vt " ) | | ( es . label = = " VC " ) ) {
2011-06-30 04:55:20 +00:00
code = CPAP_TidalVolume ;
2011-07-27 09:21:53 +00:00
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-11-25 12:13:35 +00:00
} else if ( ( es . label = = " Leak " ) | | ( es . label . startsWith ( " Leck " ) ) ) {
2011-06-29 20:30:23 +00:00
code = CPAP_Leak ;
2011-08-07 08:52:12 +00:00
es . gain * = 60 ;
2011-08-09 23:44:36 +00:00
es . physical_dimension = " L/M " ;
2011-12-08 11:41:44 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 , true ) ;
2011-07-27 09:21:53 +00:00
} else if ( es . label = = " FFL Index " ) {
2011-09-17 12:39:00 +00:00
code = CPAP_FLG ;
2011-08-01 08:53:26 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 ) ;
2011-08-09 23:44:36 +00:00
} else if ( es . label . startsWith ( " Mask Pres " ) ) {
2011-09-17 12:39:00 +00:00
code = CPAP_MaskPressure ;
2011-08-01 08:53:26 +00:00
a = ToTimeDelta ( sess , edf , es , code , recs , duration , 0 , 0 ) ;
2011-08-09 23:44:36 +00:00
} else if ( es . label . startsWith ( " Exp Press " ) ) {
2011-11-14 09:26:58 +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-08-09 23:44:36 +00:00
} else if ( es . label . startsWith ( " I:E " ) ) {
2011-11-14 09:26:58 +00:00
code = CPAP_IE ; //I:E ratio?
2011-09-17 12:39:00 +00:00
a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
2011-08-09 23:44:36 +00:00
a - > AddWaveform ( edf . startdate , es . data , recs , duration ) ;
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
} else if ( es . label . startsWith ( " Ti " ) ) {
2011-11-14 09:26:58 +00:00
code = CPAP_Ti ;
2011-09-17 12:39:00 +00:00
a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
2011-08-09 23:44:36 +00:00
a - > AddWaveform ( edf . startdate , es . data , recs , duration ) ;
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
} else if ( es . label . startsWith ( " Te " ) ) {
2011-11-14 09:26:58 +00:00
code = CPAP_Te ;
a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
a - > AddWaveform ( edf . startdate , es . data , recs , duration ) ;
//a=ToTimeDelta(sess,edf,es, code,recs,duration,0,0);
} else if ( es . label . startsWith ( " TgMV " ) ) {
code = CPAP_TgMV ;
2011-09-17 12:39:00 +00:00
a = sess - > AddEventList ( code , EVL_Waveform , es . gain , es . offset , 0 , 0 , rate ) ;
2011-08-09 23:44:36 +00:00
a - > AddWaveform ( edf . startdate , es . data , recs , duration ) ;
//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-09-17 12:39:00 +00:00
code = RMS9_E01 ;
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-09-17 12:39:00 +00:00
code = RMS9_E02 ;
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-08-09 23:44:36 +00:00
qDebug ( ) < < " Unobserved ResMed PLD Signal " < < es . label ;
2011-08-01 08:53:26 +00:00
a = NULL ;
}
if ( a ) {
2011-12-17 14:38:15 +00:00
sess - > setMin ( code , a - > Min ( ) ) ;
sess - > setMax ( code , a - > Max ( ) ) ;
2011-08-09 23:44:36 +00:00
a - > setDimension ( es . physical_dimension ) ;
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
2011-12-12 09:16:26 +00:00
RMS9ModelMap [ 36001 ] = " S9 Escape " ;
RMS9ModelMap [ 36002 ] = " S9 Escape Auto " ;
RMS9ModelMap [ 36003 ] = " S9 Elite " ;
RMS9ModelMap [ 36004 ] = " S9 VPAP S " ;
RMS9ModelMap [ 36005 ] = " S9 AutoSet " ;
RMS9ModelMap [ 36006 ] = " S9 VPAP Auto " ;
RMS9ModelMap [ 36007 ] = " S9 VPAP Adapt " ;
RMS9ModelMap [ 36008 ] = " S9 VPAP ST " ;
2011-11-27 04:50:22 +00:00
/* S8 Series
2011-12-12 09:16:26 +00:00
RMS9ModelMap [ 33007 ] = " S8 Escape " ;
RMS9ModelMap [ 33039 ] = " S8 Elite II " ;
RMS9ModelMap [ 33051 ] = " S8 Escape II " ;
RMS9ModelMap [ 33064 ] = " S8 Escape II AutoSet " ;
RMS9ModelMap [ 33064 ] = " S8 Escape II AutoSet " ;
RMS9ModelMap [ 33129 ] = " S8 AutoSet II " ;
2011-11-27 04:50:22 +00:00
*/
2011-09-23 03:54:48 +00:00
resmed_codes [ CPAP_FlowRate ] . push_back ( " Flow " ) ;
resmed_codes [ CPAP_MaskPressureHi ] . push_back ( " Mask Pres " ) ;
resmed_codes [ CPAP_MaskPressureHi ] . push_back ( " Mask Pressure " ) ; // vpap
resmed_codes [ CPAP_RespEvent ] . push_back ( " Resp Event " ) ;
resmed_codes [ CPAP_MaskPressure ] . push_back ( " Mask Pres " ) ;
resmed_codes [ CPAP_MaskPressure ] . push_back ( " Mask Pressure " ) ; // vpap
resmed_codes [ CPAP_Pressure ] . push_back ( " Therapy Pres " ) ; // not on vpap
resmed_codes [ CPAP_IPAP ] . push_back ( " Insp Pressure " ) ; // on vpap
resmed_codes [ CPAP_EPAP ] . push_back ( " Exp Press " ) ;
resmed_codes [ CPAP_EPAP ] . push_back ( " Exp Pressure " ) ; // vpap
2011-11-25 12:13:35 +00:00
2011-09-23 03:54:48 +00:00
resmed_codes [ CPAP_Leak ] . push_back ( " Leak " ) ;
2011-11-25 12:13:35 +00:00
resmed_codes [ CPAP_Leak ] . push_back ( " Leck. " ) ;
2011-09-23 03:54:48 +00:00
resmed_codes [ CPAP_RespRate ] . push_back ( " RR " ) ;
2011-11-25 12:13:35 +00:00
resmed_codes [ CPAP_RespRate ] . push_back ( " AF " ) ;
2011-11-25 12:42:41 +00:00
resmed_codes [ CPAP_RespRate ] . push_back ( " FR " ) ;
2011-11-25 12:13:35 +00:00
2011-09-23 03:54:48 +00:00
resmed_codes [ CPAP_TidalVolume ] . push_back ( " Vt " ) ;
2011-11-25 12:42:41 +00:00
resmed_codes [ CPAP_TidalVolume ] . push_back ( " VC " ) ;
2011-09-23 03:54:48 +00:00
resmed_codes [ CPAP_MinuteVent ] . push_back ( " MV " ) ;
2011-11-25 12:42:41 +00:00
resmed_codes [ CPAP_MinuteVent ] . push_back ( " VM " ) ;
2011-09-23 03:54:48 +00:00
resmed_codes [ CPAP_IE ] . push_back ( " I:E " ) ; // vpap
resmed_codes [ CPAP_Snore ] . push_back ( " Snore Index " ) ;
resmed_codes [ CPAP_FLG ] . push_back ( " FFL Index " ) ;
resmed_codes [ CPAP_RespEvent ] . push_back ( " RE " ) ;
resmed_codes [ CPAP_Ti ] . push_back ( " Ti " ) ;
resmed_codes [ CPAP_Te ] . push_back ( " Te " ) ;
// Sad (oximetry)
resmed_codes [ OXI_Pulse ] . push_back ( " Pulse " ) ;
2011-11-25 12:13:35 +00:00
resmed_codes [ OXI_Pulse ] . push_back ( " Puls " ) ;
2011-09-23 03:54:48 +00:00
resmed_codes [ OXI_SPO2 ] . push_back ( " SpO2 " ) ;
// Event annotations
resmed_codes [ CPAP_Obstructive ] . push_back ( " Obstructive apnea " ) ;
resmed_codes [ CPAP_Hypopnea ] . push_back ( " Hypopnea " ) ;
resmed_codes [ CPAP_Apnea ] . push_back ( " Apnea " ) ;
resmed_codes [ CPAP_ClearAirway ] . push_back ( " Central apnea " ) ;
2011-11-25 12:13:35 +00:00
resmed_codes [ CPAP_Mode ] . push_back ( " Mode " ) ;
resmed_codes [ CPAP_Mode ] . push_back ( " Modus " ) ;
2011-12-21 17:00:19 +00:00
resmed_codes [ RMS9_SetPressure ] . push_back ( " Eingest. Druck " ) ;
resmed_codes [ RMS9_SetPressure ] . push_back ( " Set Pressure " ) ; // Prescription
resmed_codes [ RMS9_SetPressure ] . push_back ( " Pres. prescrite " ) ;
2011-12-21 14:24:09 +00:00
resmed_codes [ RMS9_EPR ] . push_back ( " EPR " ) ;
resmed_codes [ RMS9_EPRSet ] . push_back ( " EPR Level " ) ;
resmed_codes [ RMS9_EPRSet ] . push_back ( " EPR-Stufe " ) ;
resmed_codes [ RMS9_EPRSet ] . push_back ( " Niveau EPR " ) ;
2011-11-25 12:13:35 +00:00
resmed_codes [ CPAP_PressureMax ] . push_back ( " Max Pressure " ) ;
resmed_codes [ CPAP_PressureMax ] . push_back ( " Max. Druck " ) ;
2011-11-25 12:42:41 +00:00
resmed_codes [ CPAP_PressureMax ] . push_back ( " Pression max. " ) ;
2011-11-25 12:13:35 +00:00
resmed_codes [ CPAP_PressureMin ] . push_back ( " Min Pressure " ) ;
resmed_codes [ CPAP_PressureMin ] . push_back ( " Min. Druck " ) ;
2011-11-25 12:42:41 +00:00
resmed_codes [ CPAP_PressureMin ] . push_back ( " Pression min. " ) ;
2011-11-25 12:13:35 +00:00
2011-09-23 03:54:48 +00:00
// STR.edf
}
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 ;
}