2018-04-25 10:34:23 +00:00
/* SleepLib CMS50X Loader Implementation
2014-04-09 21:01:57 +00:00
*
2021-11-02 20:34:12 +00:00
* Copyright ( c ) 2019 - 2022 The OSCAR Team
2018-03-28 07:10:52 +00:00
* Copyright ( c ) 2011 - 2018 Mark Watkins < mark @ jedimark . net >
2014-04-09 21:01:57 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
2018-06-04 20:48:38 +00:00
* License . See the file COPYING in the main directory of the source code
* for more details . */
2011-06-26 08:30:44 +00:00
//********************************************************************************************
2021-03-22 22:43:14 +00:00
// Please only INCREMENT the cms50_data_version in cms50_loader.h when making changes
// that change loader behaviour or modify channels in a manner that fixes old data imports.
// Note that changing the data version will require a reimport of existing data for which OSCAR
// does not keep a backup - so it should be avoided if possible.
// i.e. there is no need to change the version when adding support for new devices
2011-06-26 08:30:44 +00:00
//********************************************************************************************
2011-07-30 00:36:31 +00:00
# include <QApplication>
2011-06-26 08:30:44 +00:00
# include <QDir>
# include <QString>
2014-06-30 10:41:50 +00:00
# include <QDataStream>
2011-06-26 08:30:44 +00:00
# include <QDateTime>
# include <QFile>
2011-07-01 10:10:44 +00:00
# include <QDebug>
2011-07-31 20:24:43 +00:00
# include <QList>
2014-05-25 07:07:08 +00:00
# include <QMessageBox>
# include <QLabel>
# include <QVBoxLayout>
# include <QPushButton>
2023-02-12 12:09:39 +00:00
// The qt5.15 obsolescence of hex requires this change.
// this solution to QT's obsolescence is only used in debug statements
# if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
# define QTHEX Qt::hex
# define QTDEC Qt::dec
# else
# define QTHEX hex
# define QTDEC dec
# endif
2011-06-26 08:30:44 +00:00
using namespace std ;
# include "cms50_loader.h"
# include "SleepLib/machine.h"
# include "SleepLib/session.h"
2014-05-25 16:20:33 +00:00
CMS50Loader : : CMS50Loader ( )
{
m_type = MT_OXIMETER ;
m_abort = false ;
m_streaming = false ;
m_importing = false ;
imp_callbacks = 0 ;
m_vendorID = 0x10c4 ;
m_productID = 0xea60 ;
2014-06-30 10:41:50 +00:00
cms50dplus = false ;
2014-05-25 16:20:33 +00:00
2014-05-28 09:35:21 +00:00
oxirec = nullptr ;
2014-05-25 16:20:33 +00:00
startTimer . setParent ( this ) ;
resetTimer . setParent ( this ) ;
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
}
2011-10-18 12:19:06 +00:00
2014-05-25 16:20:33 +00:00
CMS50Loader : : ~ CMS50Loader ( )
2014-05-25 07:07:08 +00:00
{
}
2014-05-25 16:20:33 +00:00
bool CMS50Loader : : Detect ( const QString & path )
2014-05-25 07:07:08 +00:00
{
2014-09-18 10:53:59 +00:00
if ( p_profile - > oxi - > oximeterType ( ) = = 1 ) {
2014-08-17 23:29:30 +00:00
return true ;
}
2014-05-25 16:20:33 +00:00
Q_UNUSED ( path ) ;
return false ;
2014-05-25 07:07:08 +00:00
}
2018-04-27 04:29:03 +00:00
int CMS50Loader : : Open ( const QString & path )
2014-05-25 07:07:08 +00:00
{
2014-05-25 16:20:33 +00:00
// Only one active Oximeter module at a time, set in preferences
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
m_itemCnt = 0 ;
m_itemTotal = 0 ;
2014-05-25 07:07:08 +00:00
m_abort = false ;
m_importing = false ;
2014-05-25 16:20:33 +00:00
started_import = false ;
started_reading = false ;
finished_import = false ;
2014-06-20 14:17:41 +00:00
setStatus ( NEUTRAL ) ;
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
imp_callbacks = 0 ;
cb_reset = 0 ;
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
m_time . start ( ) ;
2014-05-25 07:07:08 +00:00
2014-05-28 09:35:21 +00:00
if ( oxirec ) {
trashRecords ( ) ;
}
2014-05-25 16:20:33 +00:00
// Cheating using path for two serial oximetry modes
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
if ( path . compare ( " import " ) = = 0 ) {
2014-06-20 14:17:41 +00:00
for ( int i = 0 ; i < 5 ; + + i ) {
resetDevice ( ) ;
serial . flush ( ) ;
// QThread::msleep(50);
QApplication : : processEvents ( ) ;
}
serial . clear ( ) ;
2014-05-25 16:20:33 +00:00
setStatus ( IMPORTING ) ;
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
startTimer . stop ( ) ;
startImportTimeout ( ) ;
return 1 ;
} else if ( path . compare ( " live " ) = = 0 ) {
2014-06-20 14:17:41 +00:00
for ( int i = 0 ; i < 5 ; + + i ) {
resetDevice ( ) ;
serial . flush ( ) ;
QApplication : : processEvents ( ) ;
}
serial . clear ( ) ;
2014-05-28 09:35:21 +00:00
m_startTime = QDateTime : : currentDateTime ( ) ;
oxirec = new QVector < OxiRecord > ;
oxisessions [ m_startTime ] = oxirec ;
2014-05-25 16:20:33 +00:00
setStatus ( LIVE ) ;
return 1 ;
}
2017-11-12 16:40:02 +00:00
QString ext = path . section ( " . " , - 1 ) ; // find the last '.'
2014-06-30 10:41:50 +00:00
if ( ( ext . compare ( " spo2 " , Qt : : CaseInsensitive ) = = 0 ) | | ( ext . compare ( " spo " , Qt : : CaseInsensitive ) = = 0 ) | | ( ext . compare ( " spor " , Qt : : CaseInsensitive ) = = 0 ) ) {
2014-05-25 16:20:33 +00:00
// try to read and process SpoR file..
return readSpoRFile ( path ) ? 1 : 0 ;
2014-05-25 07:07:08 +00:00
}
2014-05-25 16:20:33 +00:00
return 0 ;
2014-05-25 07:07:08 +00:00
}
2014-05-25 16:20:33 +00:00
2014-05-25 07:07:08 +00:00
void CMS50Loader : : processBytes ( QByteArray bytes )
{
// Sync to start of message type we are interested in
quint8 c ;
2014-05-26 07:37:28 +00:00
quint8 msgcode = 0x80 ;
2014-05-25 07:07:08 +00:00
int idx = 0 ;
int bytesread = bytes . size ( ) ;
while ( ( idx < bytesread ) & & ( ( ( c = ( quint8 ) bytes . at ( idx ) ) & msgcode ) ! = msgcode ) ) {
if ( buffer . length ( ) > 0 ) {
// If buffer is the start of a valid but short frame, add to it..
buffer . append ( c ) ;
} // otherwise dump these bytes, as they are out of sequence.
+ + idx ;
}
// Copy the rest to the buffer.
buffer . append ( bytes . mid ( idx ) ) ;
int available = buffer . length ( ) ;
switch ( status ( ) ) {
case IMPORTING :
idx = doImportMode ( ) ;
break ;
case LIVE :
idx = doLiveMode ( ) ;
break ;
default :
2014-06-20 14:17:41 +00:00
;
2014-07-28 13:56:29 +00:00
// qDebug() << "Device mode not supported by" << loaderName();
2014-05-25 07:07:08 +00:00
}
if ( idx > = available ) {
buffer . clear ( ) ;
} else if ( idx > 0 ) {
// Trim any processed bytes from the buffer.
buffer = buffer . mid ( idx ) ;
}
if ( buffer . length ( ) > 0 ) {
// If what's left doesn't start with a marker bit, dump it
if ( ( ( unsigned char ) buffer . at ( 0 ) & 0x80 ) ! = 0x80 ) {
buffer . clear ( ) ;
}
}
}
int CMS50Loader : : doImportMode ( )
{
int available = buffer . size ( ) ;
2018-04-25 10:34:23 +00:00
2014-05-26 07:37:28 +00:00
int hour , minute ;
2014-05-25 07:07:08 +00:00
int idx = 0 ;
while ( idx < available ) {
unsigned char c = ( unsigned char ) buffer . at ( idx ) ;
if ( ! started_import ) {
2014-07-01 04:57:54 +00:00
// There are three [0xf2 0xXX 0xXX] trio's at start of recording
// Followed by [0xfX 0xXX 0xXX] trios containing spo2 and pulse till the end of recording
// Scan for first header trio starting byte.
if ( c ! = 0xf2 ) {
2014-05-25 07:07:08 +00:00
idx + + ;
continue ;
2014-06-20 14:17:41 +00:00
}
2014-07-01 04:57:54 +00:00
// sometimes a f2 starting trio can be corrupted by live data
// peek ahead and see where the f2 headers are
int f2cnt = 0 ;
int f2idx [ 3 ] = { - 1 } ;
for ( int i = 0 ; i < 30 ; + + i ) {
if ( ( idx + i ) > = available ) {
qDebug ( ) < < " Not enough bytes to read CMS50 headers " ;
break ;
}
c = ( unsigned char ) buffer . at ( idx + i ) ;
if ( c = = 0xf2 ) {
f2idx [ f2cnt + + ] = idx + i ;
if ( f2cnt > = 3 )
break ; // got all 3 headers
// Skip the following two bytes
i + = 2 ;
}
}
if ( f2cnt < 3 ) {
qDebug ( ) < < " Did not get all header Trio's " ;
}
f2cnt - - ;
// CHECK: Check there might be length data after the last header trio..
2014-06-20 14:17:41 +00:00
received_bytes = 0 ;
2014-07-01 04:57:54 +00:00
bool badheader = false ;
// Look for the best of three headers trios
int bestf2 = 0 ;
if ( ( f2cnt > = 1 ) & & ( ( f2idx [ 1 ] - f2idx [ 0 ] ) = = 3 ) ) {
bestf2 = f2idx [ 0 ] ;
} else if ( ( f2cnt > = 2 ) & & ( ( f2idx [ 2 ] - f2idx [ 1 ] ) = = 3 ) ) {
bestf2 = f2idx [ 1 ] ;
} else {
bestf2 = f2idx [ f2cnt ] ;
// ouch.. check if f0 starts afterwards
if ( ( ( unsigned char ) buffer . at ( bestf2 + 3 ) & 0xf0 ) ! = 0xf0 ) {
// crap.. bad time
badheader = true ;
}
}
if ( ! badheader ) {
hour = ( unsigned char ) buffer . at ( bestf2 + 1 ) & 0x7f ;
minute = ( unsigned char ) buffer . at ( bestf2 + 2 ) & 0x7f ;
} else {
hour = 0 ;
minute = 0 ;
}
2014-05-28 09:35:21 +00:00
2014-07-01 04:57:54 +00:00
// Either a CMS50D+, has a bad header, or it's really midnight, set a flag anyway for later to help choose the right sync time
cms50dplus = ( hour = = 0 ) & & ( minute = = 0 ) ;
2014-05-25 07:07:08 +00:00
2014-09-29 14:41:31 +00:00
MachineInfo info = newInfo ( ) ;
info . model = cms50dplus ? QObject : : tr ( " CMS50D+ " ) : QObject : : tr ( " CMS50E/F " ) ;
info . serial = QString ( ) ;
2018-04-22 12:06:48 +00:00
Machine * mach = p_profile - > CreateMachine ( info ) ;
2014-09-29 14:41:31 +00:00
2014-10-02 07:56:57 +00:00
Q_UNUSED ( mach ) ;
2014-06-20 14:17:41 +00:00
qDebug ( ) < < QString ( " Receiving Oximeter transmission %1:%2 " ) . arg ( hour ) . arg ( minute ) ;
// set importing to true or whatever..
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
finished_import = false ;
started_import = true ;
started_reading = false ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
m_importing = true ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
m_itemCnt = 0 ;
m_itemTotal = 5000 ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
killTimers ( ) ;
qDebug ( ) < < " Getting ready for import " ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
oxirec = new QVector < OxiRecord > ;
oxirec - > reserve ( 30000 ) ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
QDate oda = QDate : : currentDate ( ) ;
QTime oti = QTime ( hour , minute ) ; // Only CMS50E/F's have a realtime clock. CMS50D+ will set this to midnight
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
// If the oximeter record time is more than the current time, then assume it was from the day before
// Or should I use split time preference instead??? Foggy Confusements..
if ( oti > QTime : : currentTime ( ) ) {
oda = oda . addDays ( - 1 ) ;
}
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
m_startTime = QDateTime ( oda , oti ) ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
oxisessions [ m_startTime ] = oxirec ;
2023-02-12 12:09:39 +00:00
qDebug ( ) < < " Session start (according to CMS50) " < < m_startTime < < QTHEX < < buffer . at ( idx + 1 ) < < buffer . at ( idx + 2 ) < < " : " < < QTDEC < < hour < < minute ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
cb_reset = 1 ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
// CMS50D+ needs an end timer because it just stops dead after uploading
resetTimer . singleShot ( 2000 , this , SLOT ( resetImportTimeout ( ) ) ) ;
2014-05-25 07:07:08 +00:00
2014-06-20 14:17:41 +00:00
QStringList data ;
2014-05-25 07:07:08 +00:00
2014-07-01 04:57:54 +00:00
int len = f2idx [ f2cnt ] + 3 ;
for ( int i = idx ; i < len ; + + i ) {
data . append ( QString : : number ( ( unsigned char ) buffer . at ( i ) , 16 ) ) ;
2014-06-20 14:17:41 +00:00
}
2014-07-01 04:57:54 +00:00
qDebug ( ) < < " CMS50 Record Header bytes: " < < data . join ( " , " ) ;
idx = len ;
2014-05-25 07:07:08 +00:00
2014-07-01 04:57:54 +00:00
// peek ahead to see if there really is data length bytes..
2014-06-20 14:17:41 +00:00
data . clear ( ) ;
for ( int i = 0 ; i < 12 ; + + i ) {
if ( ( idx + i ) > available ) break ;
data . push_back ( QString : : number ( ( unsigned char ) buffer . at ( idx + i ) , 16 ) ) ;
}
qDebug ( ) < < " bytes directly following header trio's: " < < data . join ( " , " ) ;
2014-07-01 04:57:54 +00:00
2014-06-20 14:17:41 +00:00
} else { // have started import
if ( ( c & 0xf0 ) = = 0xf0 ) { // Data trio
2014-05-25 07:07:08 +00:00
started_reading = true ; // Sometimes errornous crap is sent after data rec header
// Recording import
if ( ( idx + 2 ) > = available ) {
return idx ;
}
2014-05-28 09:35:21 +00:00
quint8 pulse = ( unsigned char ) ( ( buffer . at ( idx + 1 ) & 0x7f ) | ( ( c & 1 ) < < 7 ) ) ;
2014-05-25 07:07:08 +00:00
quint8 spo2 = ( unsigned char ) buffer . at ( idx + 2 ) & 0xff ;
2014-05-28 09:35:21 +00:00
oxirec - > append ( OxiRecord ( pulse , spo2 ) ) ;
2014-05-25 07:07:08 +00:00
received_bytes + = 3 ;
// TODO: Store the data to the session
2014-09-04 14:59:54 +00:00
emit updateProgress ( 0 , 0 ) ;
2014-05-25 07:07:08 +00:00
idx + = 3 ;
} else if ( ! started_reading ) { // have not got a valid trio yet, skip...
idx + = 1 ;
} else {
2014-07-06 17:40:09 +00:00
// scan ahead for another 0xf0 in case it's corrupted..
bool resync = false ;
for ( int i = idx ; i < available ; + + i ) {
c = ( unsigned char ) buffer . at ( i ) ;
if ( ( c & 0xf0 ) = = 0xf0 ) {
idx = i ;
resync = true ;
break ;
}
}
if ( ! resync ) {
// Data transfer has completed
finished_import = true ;
killTimers ( ) ;
m_importing = false ;
m_status = NEUTRAL ;
emit importComplete ( this ) ;
resetTimer . singleShot ( 2000 , this , SLOT ( shutdownPorts ( ) ) ) ;
return available ;
}
2014-05-25 07:07:08 +00:00
}
}
}
if ( ! started_import ) {
imp_callbacks = 0 ;
} else {
imp_callbacks + + ;
}
return idx ;
}
int CMS50Loader : : doLiveMode ( )
{
2018-04-25 10:34:23 +00:00
if ( oxirec = = nullptr ) {
qWarning ( ) < < " CMS50Loader::doLiveMode() called when null oxirec object " ;
return 0 ;
}
2014-05-28 09:35:21 +00:00
2014-05-25 07:07:08 +00:00
int available = buffer . size ( ) ;
int idx = 0 ;
2014-05-25 16:20:33 +00:00
QByteArray plethy ;
while ( idx < available - 5 ) {
2014-05-25 07:07:08 +00:00
if ( ( ( unsigned char ) buffer . at ( idx ) & 0x80 ) ! = 0x80 ) {
idx + + ;
continue ;
}
int pwave = ( unsigned char ) buffer . at ( idx + 1 ) ;
int pbeat = ( unsigned char ) buffer . at ( idx + 2 ) ;
int pulse = ( ( unsigned char ) buffer . at ( idx + 3 ) & 0x7f ) | ( ( pbeat & 0x40 ) < < 1 ) ;
int spo2 = ( unsigned char ) buffer . at ( idx + 4 ) & 0x7f ;
2014-05-25 16:20:33 +00:00
2014-05-28 09:35:21 +00:00
oxirec - > append ( OxiRecord ( pulse , spo2 ) ) ;
2014-05-25 16:20:33 +00:00
plethy . append ( pwave ) ;
2014-05-25 07:07:08 +00:00
idx + = 5 ;
}
2014-05-25 16:20:33 +00:00
emit updatePlethy ( plethy ) ;
2014-05-25 07:07:08 +00:00
return idx ;
}
void CMS50Loader : : resetDevice ( ) // Switch CMS50D+ device to live streaming mode
{
//qDebug() << "Sending reset code to CMS50 device";
//m_port->flush();
2014-06-20 14:17:41 +00:00
2014-05-25 07:07:08 +00:00
static unsigned char b1 [ 3 ] = { 0xf6 , 0xf6 , 0xf6 } ;
2014-06-20 14:17:41 +00:00
if ( serial . write ( ( char * ) b1 , 3 ) = = - 1 ) {
2014-05-25 07:07:08 +00:00
qDebug ( ) < < " Couldn't write data reset bytes to CMS50 " ;
}
2014-06-20 14:17:41 +00:00
QApplication : : processEvents ( ) ;
2014-05-25 07:07:08 +00:00
}
void CMS50Loader : : requestData ( ) // Switch CMS50D+ device to record transmission mode
{
static unsigned char b1 [ 2 ] = { 0xf5 , 0xf5 } ;
//qDebug() << "Sending request code to CMS50 device";
2014-06-20 14:17:41 +00:00
if ( serial . write ( ( char * ) b1 , 2 ) = = - 1 ) {
2014-05-25 07:07:08 +00:00
qDebug ( ) < < " Couldn't write data request bytes to CMS50 " ;
}
2014-06-20 14:17:41 +00:00
QApplication : : processEvents ( ) ;
2014-05-25 07:07:08 +00:00
}
void CMS50Loader : : killTimers ( )
{
startTimer . stop ( ) ;
resetTimer . stop ( ) ;
}
void CMS50Loader : : startImportTimeout ( )
{
2014-05-26 03:48:22 +00:00
if ( ! m_streaming )
return ;
2014-05-25 07:07:08 +00:00
if ( started_import ) {
return ;
}
2018-04-25 10:34:23 +00:00
if ( finished_import ! = false ) {
qWarning ( ) < < " CMS50Loader::startImportTimeout() called when finished_import != false " ;
return ;
}
2014-05-25 07:07:08 +00:00
//qDebug() << "Starting oximeter import timeout";
// Wait until events really are jammed on the CMS50D+ before re-requesting data.
const int delay = 500 ;
if ( m_abort ) {
m_streaming = false ;
closeDevice ( ) ;
return ;
}
if ( imp_callbacks = = 0 ) { // Frozen, but still hasn't started?
m_itemCnt = m_time . elapsed ( ) ;
if ( m_itemCnt > START_TIMEOUT ) { // Give up after START_TIMEOUT
closeDevice ( ) ;
abort ( ) ;
QMessageBox : : warning ( nullptr , STR_MessageBox_Error , " <h2> " + tr ( " Could not get data transmission from oximeter. " ) + " <br/><br/> " + tr ( " Please ensure you select 'upload' from the oximeter devices menu. " ) + " </h2> " ) ;
return ;
} else {
// Note: Newer CMS50 devices transmit from user input, but there is no way of differentiating between models
requestData ( ) ;
}
emit updateProgress ( m_itemCnt , START_TIMEOUT ) ;
// Schedule another callback to make sure it's started
startTimer . singleShot ( delay , this , SLOT ( startImportTimeout ( ) ) ) ;
}
}
void CMS50Loader : : resetImportTimeout ( )
{
if ( finished_import ) {
return ;
}
if ( imp_callbacks ! = cb_reset ) {
// Still receiving data.. reset timer
2014-05-26 07:37:28 +00:00
qDebug ( ) < < " Still receiving data in resetImportTimeout() " < < imp_callbacks < < cb_reset ;
2014-05-25 07:07:08 +00:00
if ( resetTimer . isActive ( ) )
resetTimer . stop ( ) ;
if ( ! finished_import ) resetTimer . singleShot ( 2000 , this , SLOT ( resetImportTimeout ( ) ) ) ;
} else {
qDebug ( ) < < " Oximeter device stopped transmitting.. Transfer complete " ;
// We were importing, but now are done
if ( ! finished_import & & ( started_import & & started_reading ) ) {
qDebug ( ) < < " Switching CMS50 back to live mode and finalizing import " ;
// Turn back on live streaming so the end of capture can be dealt with
resetTimer . stop ( ) ;
2014-06-20 14:17:41 +00:00
resetDevice ( ) ; // Send Reset to CMS50D+
2014-05-25 07:07:08 +00:00
serial . flush ( ) ;
2014-06-20 14:17:41 +00:00
QThread : : msleep ( 200 ) ;
2014-05-25 07:07:08 +00:00
resetDevice ( ) ; // Send Reset to CMS50D+
2014-06-20 14:17:41 +00:00
serial . flush ( ) ;
serial . clear ( ) ;
2014-05-25 07:07:08 +00:00
//started_import = false;
2014-06-20 14:17:41 +00:00
// finished_import = true;
2014-05-26 07:37:28 +00:00
//m_streaming=false;
2014-05-25 07:07:08 +00:00
2014-05-26 07:37:28 +00:00
//closeDevice();
2014-05-25 07:07:08 +00:00
//emit transferComplete();
//doImportComplete();
return ;
}
qDebug ( ) < < " Should CMS50 resetImportTimeout reach here? " ;
// else what???
}
2014-06-20 14:17:41 +00:00
cb_reset = imp_callbacks ;
}
void CMS50Loader : : shutdownPorts ( )
{
closeDevice ( ) ;
2014-05-25 07:07:08 +00:00
}
bool CMS50Loader : : readSpoRFile ( QString path )
{
QFile file ( path ) ;
2014-05-28 09:35:21 +00:00
if ( ! file . exists ( ) ) {
2017-11-12 16:40:02 +00:00
qWarning ( ) < < " Can't find the oximeter file: " < < path ;
2019-01-31 15:47:52 +00:00
QMessageBox : : warning ( nullptr , STR_MessageBox_Error , " <h2> " + tr ( " Could not find the oximeter file: " ) + " <br/><br/> " + path + " </h2> " ) ;
2014-05-25 07:07:08 +00:00
return false ;
}
2014-05-28 09:35:21 +00:00
if ( ! file . open ( QFile : : ReadOnly ) ) {
2017-11-12 16:40:02 +00:00
qWarning ( ) < < " Can't open the oximeter file: " < < path ;
2019-01-31 15:47:52 +00:00
QMessageBox : : warning ( nullptr , STR_MessageBox_Error , " <h2> " + tr ( " Could not open the oximeter file: " ) + " <br/><br/> " + path + " </h2> " ) ;
2014-05-28 09:35:21 +00:00
return false ;
}
2014-05-25 07:07:08 +00:00
2014-06-30 10:41:50 +00:00
bool spo2header = false ;
QString ext = path . section ( ' . ' , - 1 ) ;
2017-11-12 16:40:02 +00:00
qDebug ( ) < < " Oximeter file extention is " < < ext ;
2014-06-30 10:41:50 +00:00
if ( ext . compare ( " spo2 " , Qt : : CaseInsensitive ) = = 0 ) {
spo2header = true ;
2017-11-12 16:40:02 +00:00
qDebug ( ) < < " Oximeter file looks like an SpO2 type " ;
2014-06-30 10:41:50 +00:00
}
2014-05-25 07:07:08 +00:00
QByteArray data ;
2011-08-10 02:19:01 +00:00
2014-07-30 17:14:28 +00:00
qint64 filesize = file . size ( ) ;
2014-05-25 07:07:08 +00:00
data = file . readAll ( ) ;
2014-06-30 10:41:50 +00:00
QDataStream in ( data ) ;
in . setByteOrder ( QDataStream : : LittleEndian ) ;
quint16 pos ;
in > > pos ;
2011-06-26 08:30:44 +00:00
2014-06-30 10:41:50 +00:00
in . skipRawData ( pos - 2 ) ;
2014-05-25 07:07:08 +00:00
2014-06-30 10:41:50 +00:00
//long size = data.size();
2014-07-30 17:14:28 +00:00
int bytes_per_record = 2 ;
2014-05-28 09:35:21 +00:00
2014-06-30 10:41:50 +00:00
if ( ! spo2header ) {
// next is 0x0002
// followed by 16bit duration in seconds
2011-06-26 08:30:44 +00:00
2014-06-30 10:41:50 +00:00
// Read date and time (it's a 16bit charset)
char dchr [ 20 ] ;
int j = 0 ;
for ( int i = 0 ; i < 18 * 2 ; i + = 2 ) {
dchr [ j + + ] = data . at ( 8 + i ) ;
}
dchr [ j ] = 0 ;
if ( dchr [ 0 ] ) {
QString dstr ( dchr ) ;
2021-10-09 15:47:45 +00:00
// Ensure date is correct first to ensure DST is handled correctly
QDate date = QDate : : fromString ( dstr . left ( 8 ) , " MM/dd/yy " ) ;
QTime time = QTime : : fromString ( dstr . right ( 8 ) , " HH:mm:ss " ) ;
if ( date . year ( ) < 2000 ) {
date = date . addYears ( 100 ) ;
2019-05-09 17:05:53 +00:00
}
2021-10-09 15:47:45 +00:00
m_startTime = QDateTime ( date , time ) ;
2014-06-30 10:41:50 +00:00
} else {
m_startTime = QDateTime ( QDate : : currentDate ( ) , QTime ( 0 , 0 , 0 ) ) ;
2019-01-31 15:47:52 +00:00
cms50dplus = true ;
2014-06-30 10:41:50 +00:00
}
2019-01-31 15:47:52 +00:00
} else { // it is an spo2header
2014-06-30 10:41:50 +00:00
quint32 samples = 0 ; // number of samples
quint32 year , month , day ;
quint32 hour , minute , second ;
if ( data . at ( pos ) ! = 1 ) {
2017-11-12 16:40:02 +00:00
qWarning ( ) < < " oximeter file " < < path < < " might be odd format " ;
2014-06-30 10:41:50 +00:00
}
2014-07-30 17:14:28 +00:00
// Unknown cruft header...
2014-06-30 10:41:50 +00:00
in . skipRawData ( 200 ) ;
in > > year > > month > > day ;
in > > hour > > minute > > second ;
2017-09-21 14:41:35 +00:00
2019-01-31 15:47:52 +00:00
if ( year = = 0 ) { // typically from a CMS50D+
2019-05-09 17:05:53 +00:00
m_startTime = QDateTime ( QDate : : currentDate ( ) , QTime ( hour , minute , second ) ) ;
2019-01-31 15:47:52 +00:00
cms50dplus = true ;
}
2017-09-21 14:41:35 +00:00
else
m_startTime = QDateTime ( QDate ( year , month , day ) , QTime ( hour , minute , second ) ) ;
2014-06-30 10:41:50 +00:00
// ignoring it for now
pos + = 0x1c + 200 ;
in > > samples ;
2014-07-30 17:14:28 +00:00
int remainder = filesize - pos ;
bytes_per_record = remainder / samples ;
qDebug ( ) < < samples < < " samples of " < < bytes_per_record < < " bytes each " ;
2017-09-21 14:41:35 +00:00
// CMS50I .spo2 data have 4 bytes: a 16bit, followed by spo2 then pulse
2014-07-30 17:14:28 +00:00
2014-06-30 10:41:50 +00:00
}
2014-05-25 07:07:08 +00:00
2014-05-28 09:35:21 +00:00
oxirec = new QVector < OxiRecord > ;
oxisessions [ m_startTime ] = oxirec ;
2014-05-25 07:07:08 +00:00
unsigned char o2 , pr ;
2014-07-30 17:14:28 +00:00
quint16 un ;
2014-05-25 07:07:08 +00:00
// Read all Pulse and SPO2 data
2014-06-30 10:41:50 +00:00
do {
2014-07-30 17:14:28 +00:00
if ( bytes_per_record > 2 ) {
in > > un ;
}
2014-06-30 10:41:50 +00:00
in > > o2 ;
in > > pr ;
2014-07-30 17:14:28 +00:00
if ( ( o2 = = 0x7f ) & & ( pr = = 0xff ) ) {
o2 = pr = 0 ;
un = 0 ;
}
if ( spo2header ) {
oxirec - > append ( OxiRecord ( pr , o2 ) ) ;
} else {
oxirec - > append ( OxiRecord ( o2 , pr ) ) ;
}
2014-06-30 10:41:50 +00:00
} while ( ! in . atEnd ( ) ) ;
// for (int i = pos; i < size - 2;) {
// o2 = (unsigned char)(data.at(i + 1));
// pr = (unsigned char)(data.at(i + 0));
// oxirec->append(OxiRecord(pr, o2));
// i += 2;
// }
2014-05-25 07:07:08 +00:00
// processing gets done later
return true ;
2011-06-26 08:30:44 +00:00
}
2014-05-25 07:07:08 +00:00
void CMS50Loader : : process ( )
{
2014-05-28 09:35:21 +00:00
// Just clean up any extra crap before oximeterimport parses the oxirecords..
return ;
// if (!oxirec)
// return;
// int size=oxirec->size();
// if (size<10)
// return;
2014-05-25 07:07:08 +00:00
}
2011-06-26 08:30:44 +00:00
2014-04-17 05:58:57 +00:00
static bool cms50_initialized = false ;
2011-06-26 08:30:44 +00:00
void CMS50Loader : : Register ( )
{
2014-04-17 05:58:57 +00:00
if ( cms50_initialized ) { return ; }
2011-07-01 10:10:44 +00:00
qDebug ( ) < < " Registering CMS50Loader " ;
2011-06-26 08:30:44 +00:00
RegisterLoader ( new CMS50Loader ( ) ) ;
2014-04-17 05:58:57 +00:00
cms50_initialized = true ;
2011-06-26 08:30:44 +00:00
}