2014-04-09 21:01:57 +00:00
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* vim : set ts = 8 sts = 4 et sw = 4 tw = 99 :
*
* SleepLib CMS50X Loader Implementation
*
* Copyright ( c ) 2011 Mark Watkins < jedimark @ users . sourceforge . net >
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of the Linux
* distribution for more details . */
2011-06-26 08:30:44 +00:00
//********************************************************************************************
/// IMPORTANT!!!
//********************************************************************************************
// Please INCREMENT the cms50_data_version in cms50_loader.h when making changes to this loader
// that change loader behaviour or modify channels.
//********************************************************************************************
2011-06-26 14:40:06 +00:00
# include <QProgressBar>
2011-07-30 00:36:31 +00:00
# include <QApplication>
2011-06-26 08:30:44 +00:00
# include <QDir>
# include <QString>
# 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>
2011-06-26 08:30:44 +00:00
using namespace std ;
# include "cms50_loader.h"
# include "SleepLib/machine.h"
# include "SleepLib/session.h"
2011-06-26 14:40:06 +00:00
extern QProgressBar * qprogress ;
2011-06-26 08:30:44 +00:00
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 ;
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-05-25 16:20:33 +00:00
Q_UNUSED ( path ) ;
return false ;
2014-05-25 07:07:08 +00:00
}
2014-05-25 16:20:33 +00:00
int CMS50Loader : : Open ( QString path , Profile * profile )
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
Q_UNUSED ( profile )
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-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-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 ) {
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 ) {
oxitime = QDateTime : : currentDateTime ( ) ;
setStatus ( LIVE ) ;
return 1 ;
}
QString ext = path . section ( " . " , 1 ) ;
if ( ( ext . compare ( " spo " , Qt : : CaseInsensitive ) = = 0 ) | | ( ext . compare ( " spor " , Qt : : CaseInsensitive ) = = 0 ) ) {
// 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 ;
quint8 msgcode = ( m_status = = IMPORTING ) ? 0xf0 : 0x80 ;
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 :
qDebug ( ) < < " Device mode not supported by " < < ClassName ( ) ;
}
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 ( )
{
if ( finished_import ) {
// CMS50E/F continue streaming after import, CMS50D+ stops dead
// there is a timer running at this stage that will kill the 50D
killTimers ( ) ;
closeDevice ( ) ;
m_importing = false ;
imp_callbacks = 0 ;
return 0 ;
}
int hour , minute ;
int available = buffer . size ( ) ;
int idx = 0 ;
while ( idx < available ) {
unsigned char c = ( unsigned char ) buffer . at ( idx ) ;
if ( ! started_import ) {
if ( c ! = 0xf2 ) { // If not started, continue scanning for a valie header.
idx + + ;
continue ;
} else {
received_bytes = 0 ;
hour = ( unsigned char ) buffer . at ( idx + 1 ) & 0x7f ;
minute = ( unsigned char ) buffer . at ( idx + 2 ) & 0x7f ;
qDebug ( ) < < QString ( " Receiving Oximeter transmission %1:%2 " ) . arg ( hour ) . arg ( minute ) ;
// set importing to true or whatever..
finished_import = false ;
started_import = true ;
started_reading = false ;
m_importing = true ;
m_itemCnt = 0 ;
m_itemTotal = 5000 ;
killTimers ( ) ;
qDebug ( ) < < " Getting ready for import " ;
oxirec . clear ( ) ;
oxirec . reserve ( 10000 ) ;
QDate oda = QDate : : currentDate ( ) ;
QTime oti = QTime ( hour , minute ) ; // Only CMS50E/F's have a realtime clock. CMS50D+ will set this to midnight
cms50dplus = ( hour = = 0 ) & & ( minute = = 0 ) ; // Either a CMS50D+ or it's really midnight, set a flag anyway for later to help choose the right sync time
// 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 ) ;
}
oxitime = QDateTime ( oda , oti ) ;
// Convert it to UTC
oxitime = oxitime . toTimeSpec ( Qt : : UTC ) ;
qDebug ( ) < < " Session start (according to CMS50) " < < oxitime < < hex < < buffer . at ( idx + 1 ) < < buffer . at ( idx + 2 ) < < " : " < < dec < < hour < < minute ;
cb_reset = 1 ;
// CMS50D+ needs an end timer because it just stops dead after uploading
resetTimer . singleShot ( 2000 , this , SLOT ( resetImportTimeout ( ) ) ) ;
}
idx + = 3 ;
} else { // have started import
if ( c = = 0xf2 ) { // Header is repeated 3 times, ignore the extras
hour = ( unsigned char ) buffer . at ( idx + 1 ) & 0x7f ;
minute = ( unsigned char ) buffer . at ( idx + 2 ) & 0x7f ;
// check..
idx + = 3 ;
continue ;
} else if ( ( c & 0xf0 ) = = 0xf0 ) { // Data trio
started_reading = true ; // Sometimes errornous crap is sent after data rec header
// Recording import
if ( ( idx + 2 ) > = available ) {
return idx ;
}
quint8 pulse = ( unsigned char ) buffer . at ( idx + 1 ) & 0xff ;
quint8 spo2 = ( unsigned char ) buffer . at ( idx + 2 ) & 0xff ;
oxirec . append ( OxiRecord ( pulse , spo2 ) ) ;
received_bytes + = 3 ;
// TODO: Store the data to the session
m_itemCnt + + ;
m_itemCnt = m_itemCnt % m_itemTotal ;
emit updateProgress ( m_itemCnt , m_itemTotal ) ;
idx + = 3 ;
} else if ( ! started_reading ) { // have not got a valid trio yet, skip...
idx + = 1 ;
} else {
// trio's are over.. finish up.
killTimers ( ) ;
closeDevice ( ) ;
started_import = false ;
started_reading = false ;
finished_import = true ;
m_importing = false ;
break ;
//Completed
}
}
}
if ( ! started_import ) {
imp_callbacks = 0 ;
} else {
imp_callbacks + + ;
}
return idx ;
}
int CMS50Loader : : doLiveMode ( )
{
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
oxirec . append ( OxiRecord ( pulse , spo2 ) ) ;
plethy . append ( pwave ) ;
int elapsed = m_time . elapsed ( ) ;
// update the graph plots
if ( elapsed > 1000 ) {
m_time . start ( ) ;
emit updateDisplay ( this ) ;
}
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();
static unsigned char b1 [ 3 ] = { 0xf6 , 0xf6 , 0xf6 } ;
if ( serial . write ( ( char * ) b1 , 3 ) = = - 1 ) {
qDebug ( ) < < " Couldn't write data reset bytes to CMS50 " ;
}
}
void CMS50Loader : : requestData ( ) // Switch CMS50D+ device to record transmission mode
{
static unsigned char b1 [ 2 ] = { 0xf5 , 0xf5 } ;
//qDebug() << "Sending request code to CMS50 device";
if ( serial . write ( ( char * ) b1 , 2 ) = = - 1 ) {
qDebug ( ) < < " Couldn't write data request bytes to CMS50 " ;
}
}
void CMS50Loader : : killTimers ( )
{
startTimer . stop ( ) ;
resetTimer . stop ( ) ;
}
void CMS50Loader : : startImportTimeout ( )
{
Q_ASSERT ( m_streaming = = true ) ;
if ( started_import ) {
return ;
}
Q_ASSERT ( finished_import = = false ) ;
//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
qDebug ( ) < < " Still receiving data in resetImportTimeout() " ;
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 ( ) ;
serial . flush ( ) ;
resetDevice ( ) ; // Send Reset to CMS50D+
//started_import = false;
finished_import = true ;
m_streaming = false ;
closeDevice ( ) ;
//emit transferComplete();
//doImportComplete();
return ;
}
qDebug ( ) < < " Should CMS50 resetImportTimeout reach here? " ;
// else what???
}
cb_reset = imp_callbacks ;
}
bool CMS50Loader : : readSpoRFile ( QString path )
{
QFile file ( path ) ;
if ( ! file . exists ( ) | | ! file . isReadable ( ) ) {
return false ;
}
file . open ( QFile : : ReadOnly ) ;
QByteArray data ;
2011-08-10 02:19:01 +00:00
2014-05-25 07:07:08 +00:00
data = file . readAll ( ) ;
long size = data . size ( ) ;
2011-06-26 08:30:44 +00:00
2014-05-25 07:07:08 +00:00
// position data stream starts at
int pos = ( ( unsigned char ) data . at ( 1 ) < < 8 ) | ( unsigned char ) data . at ( 0 ) ;
// 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 ) ;
2011-06-26 08:30:44 +00:00
}
2014-05-25 07:07:08 +00:00
dchr [ j ] = 0 ;
QString dstr ( dchr ) ;
oxitime = QDateTime : : fromString ( dstr , " MM/dd/yy HH:mm:ss " ) ;
if ( oxitime . date ( ) . year ( ) < 2000 ) { oxitime = oxitime . addYears ( 100 ) ; }
unsigned char o2 , pr ;
// Read all Pulse and SPO2 data
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 ;
}
// processing gets done later
return true ;
2011-06-26 08:30:44 +00:00
}
Machine * CMS50Loader : : CreateMachine ( Profile * profile )
{
2014-04-17 05:58:57 +00:00
if ( ! profile ) {
2014-04-23 13:19:56 +00:00
return nullptr ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
// NOTE: This only allows for one CMS50 machine per profile..
// Upgrading their oximeter will use this same record..
2014-04-17 05:58:57 +00:00
QList < Machine * > ml = profile - > GetMachines ( MT_OXIMETER ) ;
2011-06-26 08:30:44 +00:00
2014-04-17 05:58:57 +00:00
for ( QList < Machine * > : : iterator i = ml . begin ( ) ; i ! = ml . end ( ) ; i + + ) {
if ( ( * i ) - > GetClass ( ) = = cms50_class_name ) {
2011-06-26 08:30:44 +00:00
return ( * i ) ;
break ;
}
}
2011-07-01 10:10:44 +00:00
qDebug ( ) < < " Create CMS50 Machine Record " ;
2011-06-26 08:30:44 +00:00
2014-04-17 05:58:57 +00:00
Machine * m = new Oximeter ( profile , 0 ) ;
2011-06-26 08:30:44 +00:00
m - > SetClass ( cms50_class_name ) ;
2014-04-17 05:58:57 +00:00
m - > properties [ STR_PROP_Brand ] = " Contec " ;
m - > properties [ STR_PROP_Model ] = " CMS50X " ;
m - > properties [ STR_PROP_DataVersion ] = QString : : number ( cms50_data_version ) ;
2012-01-05 04:37:22 +00:00
2011-06-26 08:30:44 +00:00
profile - > AddMachine ( m ) ;
2014-04-17 05:58:57 +00:00
QString path = " { " + STR_GEN_DataFolder + " }/ " + m - > GetClass ( ) + " _ " + m - > hexid ( ) + " / " ;
m - > properties [ STR_PROP_Path ] = path ;
2011-06-26 08:30:44 +00:00
return m ;
}
2014-05-25 07:07:08 +00:00
void CMS50Loader : : process ( )
{
int size = oxirec . size ( ) ;
if ( size < 10 )
return ;
2014-05-25 16:20:33 +00:00
// EventList *PULSE=new EventList(EVL_Event);
// EventList *SPO2=new EventList(EVL_Event);
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
// quint64 ti = oxitime.toMSecsSinceEpoch();
2014-05-25 07:07:08 +00:00
2014-05-25 16:20:33 +00:00
// for (int i=0; i < size; ++i) {
// //PULSE->AddWaveform
// }
// qDebug() << "Processing" << oxirec.size() << "oximetry records";
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
}