diff --git a/Graphs/gGraphView.cpp b/Graphs/gGraphView.cpp index a1a2ef8a..12209ef7 100644 --- a/Graphs/gGraphView.cpp +++ b/Graphs/gGraphView.cpp @@ -730,7 +730,7 @@ gToolTip::gToolTip(gGraphView * graphview) gToolTip::~gToolTip() { - disconnect(timer,SLOT(timerDone())); + disconnect(timer,0,0,0); delete timer; } //void gToolTip::calcSize(QString text,int &w, int &h) @@ -3765,14 +3765,15 @@ bool gGraphView::isEmpty() void gGraphView::refreshTimeout() { - redraw(); + if (this->isVisible()) + redraw(); } void gGraphView::timedRedraw(int ms) { - if (timer->isActive()) - timer->stop(); - timer->setSingleShot(true); - timer->start(ms); + if (!timer->isActive()) { + timer->setSingleShot(true); + timer->start(ms); + } } void gGraphView::resetLayout() { diff --git a/SleepLib/calcs.cpp b/SleepLib/calcs.cpp index bf01aeb6..d2b4cc25 100644 --- a/SleepLib/calcs.cpp +++ b/SleepLib/calcs.cpp @@ -1550,10 +1550,15 @@ int calcSPO2Drop(Session *session) cnt++; } } - qSort(med); - int midx=float(med.size())*0.90; - if (midx>med.size()-1) midx=med.size()-1; - EventDataType baseline=med[midx]; + EventDataType baseline=0; + if (med.size()>0) { + qSort(med); + + int midx=float(med.size())*0.90; + if (midx>med.size()-1) midx=med.size()-1; + if (midx<0) midx=0; + baseline=med[midx]; + } session->settings[OXI_SPO2Drop]=baseline; //EventDataType baseline=round(tmp/EventDataType(cnt)); EventDataType current; diff --git a/oximetry.cpp b/oximetry.cpp index 3e3c7965..a3d60e3b 100644 --- a/oximetry.cpp +++ b/oximetry.cpp @@ -22,6 +22,9 @@ #include "Graphs/gYAxis.h" #include "Graphs/gLineOverlay.h" +#define SERIAL_DEBUG 1 + + extern QLabel * qstatus2; #include "mainwindow.h" extern MainWindow * mainwin; @@ -111,11 +114,14 @@ bool SerialOximeter::Open(QextSerialPort::QueryMode mode) void SerialOximeter::Close() { qDebug() << "Closing serial port" << m_portname; - if (!m_opened) return; + if (!m_opened) + return; - if (m_port) m_port->close(); + m_port->flush(); + disconnect(m_port, 0,0,0); // SIGNAL(readyRead()), this, SLOT(ReadyRead())); + if (m_port) + m_port->close(); //if (m_portmode==QextSerialPort::EventDriven) - disconnect(m_port, SIGNAL(readyRead()), this, SLOT(ReadyRead())); m_mode=SO_OFF; m_opened=false; } @@ -402,6 +408,7 @@ bool SerialOximeter::startLive() import_mode=false; m_mode=SO_LIVE; lastpr=lasto2=0; + buffer.clear(); if (!m_opened && !Open(QextSerialPort::EventDriven)) return false; createSession(); @@ -423,10 +430,23 @@ void SerialOximeter::stopLive() CMS50Serial::CMS50Serial(QObject * parent, QString portname="") : SerialOximeter(parent,"CMS50", portname, BAUD19200, FLOW_OFF, PAR_ODD, DATA_8, STOP_1) { + cms50timer=new QTimer(this); + cms50timer2=new QTimer(this); } CMS50Serial::~CMS50Serial() { + delete cms50timer2; + delete cms50timer; +} +void CMS50Serial::killTimers() +{ + if (cms50timer->isActive()) { + cms50timer->stop(); + } + if (cms50timer2->isActive()) { + cms50timer2->stop(); + } } void CMS50Serial::import_process() @@ -435,6 +455,9 @@ void CMS50Serial::import_process() qDebug() << "User pushing import too many times in a row?"; return; } + mainwin->getOximetry()->graphView()->setEmptyText("Processing..."); + mainwin->getOximetry()->graphView()->redraw(); + qDebug() << "CMS50 import complete. Processing" << data.size() << "bytes"; unsigned short a,pl,o2,lastpl=0,lasto2=0; int i=0; @@ -443,7 +466,7 @@ void CMS50Serial::import_process() EventList * pulse=(session->eventlist[OXI_Pulse][0]); EventList * spo2=(session->eventlist[OXI_SPO2][0]); - lasttime=f2time[0].toTime_t(); + lasttime=f2time.toTime_t(); session->SetSessionID(lasttime); lasttime*=1000; @@ -455,7 +478,7 @@ void CMS50Serial::import_process() qint64 lastpltime=0,lasto2time=0; bool first=true; while (i<(size-3)) { - a=data.at(i++); // low bits are supposedly the high bits of the heart rate + a=data.at(i++); // low bits are supposedly the high bits of the heart rate? not here pl=((data.at(i++) & 0x7f) | ((a & 1) << 7)) & 0xff; o2=data.at(i++) & 0x7f; if (pl!=0) { @@ -550,209 +573,243 @@ void CMS50Serial::import_process() void CMS50Serial::ReadyRead() { - QByteArray bytes; - int a = m_port->bytesAvailable(); - bytes.resize(a); - m_port->read(bytes.data(), bytes.size()); - m_callbacks++; - - if (m_mode==SO_WAIT) return; - - int i=0; - - // Was going out of sync previously.. To fix this unfortunately requires 4.7 - -#if QT_VERSION >= QT_VERSION_CHECK(4,7,0) - qint64 current=QDateTime::currentMSecsSinceEpoch(); //double(QDateTime::currentDateTime().toTime_t())*1000L; - //qint64 since=current-lasttime; - //if (since>25) - lasttime=current; -#endif - // else (don't bother - we can work some magic at the end of recording.) - + if (m_mode==SO_OFF) + return; static int lastbytesize=0; - int size=bytes.size(); - // Process all incoming serial data packets - unsigned short c; - unsigned short pl,o2; + QByteArray bytes; + int available = m_port->bytesAvailable(); + bytes.resize(available); + m_port->read(bytes.data(), bytes.size()); + + EventList * epulse=NULL; + EventList * espo2=NULL; + + if (m_mode==SO_WAIT) { + killTimers(); + // Close(); + return; + } + + m_callbacks++; if (!import_mode) { - QString data="Read: "; -#ifdef SERIAL_DEBUG - for (int i=0;iisActive()) { timer->stop(); } + // Set the Shutdown timer timer->setSingleShot(true); timer->setInterval(10000); timer->start(); } + qDebug() << "Oximeter switched off.. wait for timeout?" << hex << bytes.at(0); return; - //qDebug() << "Oximeter switched of.. wait for timeout?"; } else { + + // Cancel any shutdown timer if running if (timer->isActive()) { timer->stop(); } } } - lastbytesize=size; + lastbytesize=available; + + unsigned char c; + int i=0; + + while ((i < available) && (((c=(unsigned char)bytes.at(i)) & 0x80)!=0x80)) { + if (buffer.length()==0) { + // Lost frame, there is no previous data, so ignore.. + ++i; + continue; + } + if (((unsigned char)buffer.at(0)&0x80)==0x80) { + // If buffer is the start of a valid but short frame, add to it.. + buffer.append(c); + }// otherwise dump these bytes, as they are corrupt. + ++i; + } + + // Copy the rest to the buffer. + for (;i < available; ++i) { + buffer.append(bytes.at(i)); + } + + unsigned char pulse,spo2,pwave,pbeat; + + available=buffer.length(); + + bool pkt_short=false; + i=0; + short hour,minute; + bool updated=false; bool fixtime=false; - while (ieventlist[OXI_Pulse][0]); + espo2=(session->eventlist[OXI_SPO2][0]); + } - // why can't this stay in a waitf6 loop for 30 or so seconds? + while (i < available) { + c=buffer.at(i); + if ((c & 0xf0)==0xf0) { + if (c==0xf2) { // Is this a record block header? + if ((i+9) >= available) { + pkt_short=true; - if (waitf6) { //ack sequence from f6 command. - if ((unsigned char)bytes.at(i++)==0xf2) { - c=bytes.at(i); - if (c & 0x80) { - int h=(c & 0x1f); - int m=(bytes.at(i+1) % 60); - if (!((h==0) && (m==0))) { // CMS50E's have a realtime clock (apparently) - QDateTime d(PROFILE.LastDay(),QTime(h,m,0)); - f2time.push_back(d); - qDebug() << "Session start (according to CMS50)" << d << h << m; - } else { - // otherwise pick the first session of the last days data.. - Day *day=PROFILE.GetDay(PROFILE.LastDay(),MT_CPAP); - QDateTime d; - fixtime=true; - if (day) { - int ti=day->first()/1000L; + break; + } +#ifdef SERIAL_DEBUG + QString zdata="RecHeader:"; + for (int j=0;j<9;j++) zdata+=QString().sprintf("%02X",(unsigned char)buffer.at(i+j)); + qDebug() << zdata; +#endif - d=QDateTime::fromTime_t(ti); - qDebug() << "Guessing session starting from CPAP data" << d; - } else { - qDebug() << "Can't guess start time, defaulting to ending at 7:30am this morning" << d; - d=QDateTime::currentDateTime(); - d.setTime(QTime(7,30,0)); - //d.addDays(-1); - } - f2time.push_back(d); - } - i+=2; - cntf6++; - } else continue; + if (((unsigned char)buffer.at(i+3)!=c) || ((unsigned char)buffer.at(i+6)!=c)) { + buffer.clear(); + qDebug() << "Dodgy CMS50X recording header, destroying buffer"; + cms50timer->singleShot(1000,this,SLOT(startImportTimeout())); + break; + } + + qDebug() << "Got Record Header:" << hex << buffer.at(i+1) << buffer.at(i+2); + hour=(unsigned char)buffer.at(i+1) & 0x7f; + minute=(unsigned char)buffer.at(i+2) & 0x7f; + + if ((((unsigned char)buffer.at(i+4)&0x7f)!=hour) ||(((unsigned char)buffer.at(i+5)&0x7f)!=minute)) { + qDebug() << "Non matching time2 in CMS50X recording header"; + } + if ((((unsigned char)buffer.at(i+7)&0x7f)!=hour) ||(((unsigned char)buffer.at(i+8)&0x7f)!=minute)) { + qDebug() << "Non matching time3 in CMS50X recording header"; + } + + // set importing to true or whatever.. + + finished_import=false; + started_import=true; + started_reading=false; + + mainwin->getOximetry()->graphView()->setEmptyText("Please Wait, Importing..."); + mainwin->getOximetry()->graphView()->timedRedraw(50); + + if ((hour!=0) && (minute!=0)) { // CMS50E/F's have a realtime clock + f2time=QDateTime(QDate::currentDate(),QTime(hour,minute)); + f2time=f2time.toTimeSpec(Qt::UTC); + qDebug() << "Session start (according to CMS50)" << f2time << hex << buffer.at(i+1) << buffer.at(i+2) << ":" << dec << hour << minute ; } else { - if (cntf6>0) { - qDebug() << "Got Acknowledge Sequence" << cntf6; - i--; - if ((i+3)first()/1000L; - datasize=(((unsigned char)bytes.at(i) & 0x3f) << 14) | (((unsigned char)bytes.at(i+1)&0x7f) << 7) | ((unsigned char)bytes.at(i+2) & 0x7f); - received_bytes=0; - qDebug() << "Data Size=" << datasize << "??"; - if (fixtime) { - QDateTime time; - for (int i=0;igetOximetry()->graphView()->setEmptyText("Please Wait, Importing..."); - mainwin->getOximetry()->graphView()->redraw(); - - data.clear(); - for (z=i;z=datasize) - done_import=true; - waitf6=false; - break; + f2time=QDateTime::fromTime_t(ti); + qDebug() << "Guessing session starting from CPAP data" << f2time; } + + if (!f2time.isValid()) { + // CMS50D+ are rather stupid. + qDebug() << "Can't guess start time, defaulting to midnight" << f2time; + f2time=QDateTime::currentDateTime(); + f2time.setTime(QTime(0,0,0)); + //f2time.addDays(-1); + } + } + qDebug() << "Record Import:" << f2time; + i+=9; } else { - qDebug() << "Recieving Block" << size << "(" << received_bytes << "of" << datasize <<")"; - for (int z=i;z=datasize) || (((received_bytes/datasize)>0.7) && (size<250))) { - done_import=true; - } - break; - //read data blocks.. - } - } else { - static unsigned short hb=0; + started_reading=true; // Sometimes errornous crap is sent after data rec header - if (bytes[i]&0x80) { // 0x80 == sync bit - EventDataType d=bytes[i+1] & 0x7f; - hb=(bytes[i+2] & 0x40) << 1; + // Recording import + if ((i+2) >= available) { + pkt_short=true; + break; + } + + pulse=(unsigned char)buffer.at(i+1) & 0x7f; + spo2=(unsigned char)buffer.at(i+2) & 0x7f; + qDebug() << "Record: " << pulse << spo2; + + data.push_back(buffer.at(i)); + data.push_back(buffer.at(i+1)); + data.push_back(buffer.at(i+2)); + received_bytes+=3; - addPlethy(lasttime,d); - lasttime+=20; i+=3; - } else { - pl=(bytes[i] & 0x7f) | hb; - o2=bytes[i+1] & 0x7f; - addPulse(lasttime,pl); - addSpO2(lasttime,o2); - i+=2; } + + } else if ((c & 0x80)==0x80) { + if (started_reading) { // (Sometimes errornous bytes get transfered after header) + qDebug() << "Stopped importing due to live data"; + // We were importing, but now are done + + killTimers(); + + started_import=false; + started_reading=false; + finished_import=true; + m_mode=SO_WAIT; // Temporarily pause until complete shutdown. + + emit(importProcess()); + } + // Standard frame.. make sure theres the full 5 byte sequence + if ((i+4) > available) { + pkt_short=true; + break; + } + if (!import_mode) { + pwave=(unsigned char)buffer.at(i+1); + pbeat=(unsigned char)buffer.at(i+2); + pulse=((unsigned char)buffer.at(i+3) & 0x7f) | ((pbeat & 0x40) << 1); + spo2=(unsigned char)buffer.at(i+4) & 0x7f; + + addPlethy(lasttime,pwave); + addPulse(lasttime,pulse); + addSpO2(lasttime,spo2); + lasttime+=20; + updated=true; + } + +#if SERIAL_DEBUG + qDebug() << "Live: " << pulse << spo2 << pwave << pbeat; +#endif + // whatever depending on mode.. + i+=5; + } else break; + } + + if (i>0) { + // Trim any processed bytes from the buffer. + buffer=buffer.mid(i); + i=0; + // available should be < 5 or 3 here.. + } + available=buffer.length(); + + if (available>0) { + if ((buffer.at(0) & 0x80)!=0x80) { + buffer.clear(); } } - if (import_mode && waitf6 && (cntf6==0)) { - int i=imptime.elapsed(); + if (import_mode) { + // Stop any unnecessary timer callbacks + if (cms50timer->isActive()) cms50timer->stop(); + if (cms50timer2->isActive()) cms50timer2->stop(); - //mainwin->getOximetry()->graphView()->setEmptyText("fun"); - //mainwin->getOximetry()->graphView()->redraw(); - - if (i>1000) { - //mainwin->getOximetry()->graphView()->setEmptyText("fun"); - //mainwin->getOximetry()->graphView()->redraw(); - imptime.start(); - failcnt++; - QString a; - - if (failcnt>15) { // Give up after ~15 seconds - m_mode=SO_WAIT; - emit(importAborted()); - mainwin->getOximetry()->graphView()->setEmptyText("Import Failed"); - mainwin->getOximetry()->graphView()->redraw(); - return; - } else { - a="Waiting"; - for (int i=0;igetOximetry()->graphView()->setEmptyText(a); - mainwin->getOximetry()->graphView()->redraw(); - requestData(); // retransmit the data request code - } + m_callbacks=0; + if (!started_import) { // Haven't started import process yet, CMS50D+ needs a send command + if (!finished_import) + cms50timer->singleShot(250,this,SLOT(startImportTimeout())); + } else { + // CMS50D+ needs an end timer because it just stops dead after uploading + if (!finished_import) + cms50timer2->singleShot(250,this,SLOT(resetImportTimeout())); } } - if (!import_mode) + + if (updated) emit(dataChanged()); - else if (done_import) { - qDebug() << "End"; - resetDevice(); - m_mode=SO_WAIT; - emit(importProcess()); - - } } void CMS50Serial::resetDevice() { @@ -774,44 +831,105 @@ void CMS50Serial::requestData() bool CMS50Serial::startImport() { - m_mode=SO_WAIT; + buffer.clear(); + data.clear(); + m_mode=SO_IMPORT; + import_mode=true; + started_import=false; + started_reading=false; + finished_import=false; if (!m_opened && !Open(QextSerialPort::EventDriven)) return false; + connect(this,SIGNAL(importProcess()),this,SLOT(import_process())); + imptime.start(); m_callbacks=0; import_fails=0; + failcnt=0; - QTimer::singleShot(5000,this,SLOT(startImportTimeout())); - //make sure there is a data stream first.. + // doesn't need to happen until process.. createSession(); + //requestData(); return true; } void CMS50Serial::startImportTimeout() { - if (m_callbacks>0) { - connect(this,SIGNAL(importProcess()),this,SLOT(import_process())); - import_mode=true; - waitf6=true; - done_import=false; - cntf6=0; - failcnt=0; - m_mode=SO_IMPORT; - requestData(); - } else { - import_fails++; - if (import_fails<5) { - // resetDevice(); - QTimer::singleShot(250,this,SLOT(startImportTimeout())); + if ((m_mode==SO_WAIT)||(finished_import)) + return; + + // Make sure no other timer scheduled.. + if (cms50timer->isActive()) + cms50timer->stop(); + + if (!started_import) { + + // Wait until events really are jammed on the CMS50D+ before re-requesting data. + + if (m_callbacks>0) { + m_callbacks=0; + // Try again later.. device still erronously dumping crap. + + cms50timer->singleShot(1000,this,SLOT(startImportTimeout())); } else { - qDebug() << "No oximeter signal!!!!!!!!!!!!!!!!"; - emit(importAborted()); + int i=imptime.elapsed()/1000; + //imptime.start(); + + if (i>20) { // Give up after ~20 tries + m_mode=SO_WAIT; + cms50timer->stop(); + emit(importAborted()); + mainwin->getOximetry()->graphView()->setEmptyText("Import Failed"); + mainwin->getOximetry()->graphView()->timedRedraw(50); + return; + } else { + QString a="Waiting"; + for (int j=0;jgetOximetry()->graphView()->setEmptyText(a); + mainwin->getOximetry()->graphView()->timedRedraw(50); + + // Note: Newer CMS50 devices transmit from user input + requestData(); + + // Set the callback again + } + cms50timer->singleShot(1000,this,SLOT(startImportTimeout())); } } + +} +void CMS50Serial::resetImportTimeout() +{ + if (finished_import) return; + if (m_callbacks>0) { + // Still receiving data.. reset timer + m_callbacks=0; + qDebug() << "reseting timer in resetImportTimeout()"; + cms50timer2->singleShot(500,this,SLOT(resetImportTimeout())); + } else { + qDebug() << "no callbacks"; + // We were importing, but now are done + if (!finished_import && (started_import && started_reading)) { + qDebug() << "resetting CMS50 device"; + // Turn back on live streaming so the end of capture can be dealt with + cms50timer2->stop(); + + m_port->flush(); + resetDevice(); // Send Reset to CMS50D+ + //started_import=false; + finished_import=true; + m_mode=SO_WAIT; // Temporarily pause until complete shutdown. + + emit(importProcess()); + return; + } + qDebug() << "Should CMS50 resetImportTimeout reach here?"; + // else what??? + } } @@ -1139,7 +1257,7 @@ void Oximetry::data_changed() } GraphView->updateScale(); - GraphView->redraw(); + GraphView->timedRedraw(25); } @@ -1198,9 +1316,10 @@ void Oximetry::on_ImportButton_clicked() if (!oximeter->startImport()) { mainwin->Notify(tr("Oximeter Error\n\nThe device did not respond.. Make sure it's switched on.")); - disconnect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(import_complete(Session*))); - disconnect(oximeter,SIGNAL(importAborted()),this,SLOT(import_aborted())); - disconnect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(update_progress(float))); + import_finished(); +// disconnect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(import_complete(Session*))); +// disconnect(oximeter,SIGNAL(importAborted()),this,SLOT(import_aborted())); +// disconnect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(update_progress(float))); //qDebug() << "Error starting oximetry serial import process"; return; } @@ -1219,9 +1338,15 @@ void Oximetry::on_ImportButton_clicked() void Oximetry::import_finished() { + disconnect(oximeter,SIGNAL(importComplete(Session*)),this,SLOT(import_complete(Session*))); disconnect(oximeter,SIGNAL(importAborted()),this,SLOT(import_aborted())); disconnect(oximeter,SIGNAL(updateProgress(float)),this,SLOT(update_progress(float))); + oximeter->disconnect(oximeter,SIGNAL(importProcess()),0,0); + +// oximeter->killTimers(); +// oximeter->Close(); + // Hanging here.. :( ui->SerialPortsCombo->setEnabled(true); qstatus->setText(tr("Ready")); @@ -1236,13 +1361,12 @@ void Oximetry::import_finished() void Oximetry::import_aborted() { - oximeter->disconnect(oximeter,SIGNAL(importProcess()),0,0); day->getSessions().clear(); //QMessageBox::warning(mainwin,tr("Oximeter Error"),tr("Please make sure your oximeter is switched on, and able to transmit data.\n(You may need to enter the oximeters Settings screen for it to be able to transmit.)"),QMessageBox::Ok); mainwin->Notify(tr("Please make sure your oximeter is switched on, and in the right mode to transmit data."),tr("Oximeter Error!"),5000); //qDebug() << "Oximetry import failed"; - import_finished(); + import_finished(); } void Oximetry::import_complete(Session * session) { diff --git a/oximetry.h b/oximetry.h index 2f3690a3..db338d7a 100644 --- a/oximetry.h +++ b/oximetry.h @@ -135,6 +135,7 @@ public: virtual void addPulse(qint64 time, EventDataType pr); virtual void addSpO2(qint64 time, EventDataType o2); virtual void addPlethy(qint64 time, EventDataType pleth); + virtual void killTimers()=0; signals: void sessionCreated(Session *); @@ -171,8 +172,10 @@ protected slots: //! \brief Override this to start the Import Timeout virtual void startImportTimeout()=0; + virtual void resetImportTimeout()=0; protected: + //virtual void addEvents(EventDataType pr, EventDataType o2, EventDataType pleth=-1000000); //! \brief Pointer to current session object @@ -202,9 +205,12 @@ protected: int m_callbacks; bool done_import; + bool started_import,started_reading,finished_import; QTimer *timer; EventDataType lasto2,lastpr; + QByteArray buffer; + }; /*! \class CMS50Serial @@ -226,9 +232,14 @@ public: //! \brief Sends the 0xf5, 0xf5 data string to request devices serial recording virtual void requestData(); + //! \brief Kill any CMS50 specific timers (used internally) + virtual void killTimers(); + protected: //! \brief CMS50 Time-out detection virtual void startImportTimeout(); + virtual void resetImportTimeout(); + //! \brief Called on completion of data import, to convert bytearray into event data virtual void import_process(); @@ -240,13 +251,23 @@ protected: short failcnt; QByteArray data; - QVector f2time; + QByteArray buffer; + + QDateTime f2time; int datasize; int received_bytes; int import_fails; QTime imptime; + + EventDataType plmin,plmax; + EventDataType o2min,o2max; + int plcnt,o2cnt; + qint64 lastpltime,lasto2time; + short lastpl,lasto2; + bool first; + QTimer *cms50timer,*cms50timer2; }; namespace Ui {