diff --git a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp index 2f71405d..f013f9ee 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50_loader.cpp @@ -543,11 +543,13 @@ bool CMS50Loader::readSpoRFile(QString path) QFile file(path); if (!file.exists()) { qWarning() << "Can't find the oximeter file: " << path; + QMessageBox::warning(nullptr, STR_MessageBox_Error, "

"+tr("Could not find the oximeter file:")+"

"+path+"

"); return false; } if (!file.open(QFile::ReadOnly)) { qWarning() << "Can't open the oximeter file: " << path; + QMessageBox::warning(nullptr, STR_MessageBox_Error, "

"+tr("Could not open the oximeter file:")+"

"+path+"

"); return false; } @@ -592,8 +594,9 @@ bool CMS50Loader::readSpoRFile(QString path) if (m_startTime.date().year() < 2000) { m_startTime = m_startTime.addYears(100); } } else { m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0)); + cms50dplus = true; } - } else { // !spo2header + } else { // it is an spo2header quint32 samples = 0; // number of samples @@ -610,8 +613,10 @@ bool CMS50Loader::readSpoRFile(QString path) in >> year >> month >> day; in >> hour >> minute >> second; - if ( year == 0 ) // typically from a CMS50D+ - m_startTime = QDateTime(QDate::currentDate(), QTime(hour, minute, second)); + if ( year == 0 ) { // typically from a CMS50D+ + m_startTime = QDateTime(QDate::currentDate(), QTime(0, 0, 0)); + cms50dplus = true; + } else m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second)); diff --git a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp index 0872f8a2..570bd639 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp @@ -40,6 +40,7 @@ CMS50F37Loader::CMS50F37Loader() m_streaming = false; m_importing = false; started_reading = false; + cms50dplus = false; imp_callbacks = 0; @@ -166,18 +167,20 @@ unsigned char cms50_sequence[] = { 0xa7, 0xa2, 0xa0, 0xb0, 0xac, 0xb3, 0xad, 0xa const int TIMEOUT = 2000; -const quint8 COMMAND_CMS50_HELLO1 = 0xa7; -const quint8 COMMAND_CMS50_HELLO2 = 0xa2; +const quint8 COMMAND_GET_VERSION = 0xa0; // not entirely sure of this one +const quint8 COMMAND_CMS50_HELLO2 = 0xa2; // stop live data stream const quint8 COMMAND_GET_SESSION_COUNT = 0xA3; -const quint8 COMMAND_GET_SESSION_TIME = 0xA5; const quint8 COMMAND_GET_SESSION_DURATION = 0xA4; -const quint8 COMMAND_GET_USER_INFO = 0xAB; +const quint8 COMMAND_GET_SESSION_TIME = 0xA5; const quint8 COMMAND_GET_SESSION_DATA = 0xA6; -const quint8 COMMAND_GET_OXIMETER_DEVICEID = 0xAA; -const quint8 COMMAND_GET_OXIMETER_INFO = 0xb0; +const quint8 COMMAND_CMS50_HELLO1 = 0xa7; // stop stored data stream const quint8 COMMAND_GET_OXIMETER_MODEL = 0xA8; const quint8 COMMAND_GET_OXIMETER_VENDOR = 0xA9; +const quint8 COMMAND_GET_OXIMETER_DEVICEID = 0xAA; +const quint8 COMMAND_GET_USER_INFO = 0xAB; +const quint8 COMMAND_GET_USER_COUNT = 0xAD; // for future check and use const quint8 COMMAND_SESSION_ERASE = 0xAE; +const quint8 COMMAND_GET_OXIMETER_INFO = 0xb0; int cms50_seqlength = sizeof(cms50_sequence); @@ -195,7 +198,7 @@ QString CMS50F37Loader::getUser() QApplication::processEvents(); } while (user.isEmpty() && (time.elapsed() < TIMEOUT)); - qDebug() << "User = " << user; + qDebug() << "User 0 is " << user; return user; } @@ -244,7 +247,9 @@ QString CMS50F37Loader::getModel() QString CMS50F37Loader::getDeviceString() { - return QString("%1 %2").arg(getVendor()).arg(getModel()); + QString VendDev = QString("%1 %2").arg(getVendor()).arg(getModel()); + qDebug() << "USB Device String is " << VendDev; + return VendDev; } QString CMS50F37Loader::getDeviceID() @@ -263,6 +268,21 @@ QString CMS50F37Loader::getDeviceID() return devid; } +int CMS50F37Loader::getUserCount() // for future use, check, then add select user if > 1 +{ + userCount = -1; + sendCommand(COMMAND_GET_USER_COUNT); + + QTime time; + time.start(); + do { + QApplication::processEvents(); + } while ((userCount < 0) && (time.elapsed() < TIMEOUT)); + + qDebug() << "User count is " << userCount; + return userCount; +} + int CMS50F37Loader::getSessionCount() { session_count = -1; @@ -274,7 +294,7 @@ int CMS50F37Loader::getSessionCount() QApplication::processEvents(); } while ((session_count < 0) && (time.elapsed() < TIMEOUT)); - qDebug() << "Session count is " << session_count; + qDebug() << "Session count is " << session_count; return session_count; } @@ -322,12 +342,16 @@ QDateTime CMS50F37Loader::getDateTime(int session) QApplication::processEvents(); } while ((imp_date.isNull() || imp_time.isNull()) && (time.elapsed() < TIMEOUT)); - if (imp_date.isNull() || imp_time.isNull()) - datetime = QDateTime(); + if (imp_date.isNull() && imp_time.isNull()) + datetime = QDateTime(QDate::currentDate(), QTime(0,0,0)); + else if ( imp_date.isNull() ) + datetime = QDateTime(QDate::currentDate(), imp_time); + else if ( imp_time.isNull() ) + datetime = QDateTime(imp_date, QTime(0,0,0)); else datetime = QDateTime(imp_date, imp_time); - qDebug() << "Oximeter DateTime is " << datetime.toString("yyyy-MMM-dd HH:mm:ss"); + qDebug() << "Oximeter DateTime is " << datetime.toString("yyyy-MMM-dd HH:mm:ssap"); return datetime; } @@ -378,13 +402,15 @@ void CMS50F37Loader::processBytes(QByteArray bytes) } continue; } - for (int i = 1; i < len; i++) { - buffer[idx+i] = buffer[idx+i] ^ 0x80; + + // always copy the high bits into the following bytes + for (int i = 2, msb = buffer.at(idx+1); i < len; i++, msb>>= 1) { + buffer[idx+i] = (buffer[idx+i] & 0x7f) | (msb & 0x01 ? 0x80 : 0); } if (!started_reading) switch(res) { case 0x02: // Model name string (there are two in sequnce.. second might be the next chunk!) - data = buffer.at(idx+1); + data = buffer.at(idx+2); if (data == 0) { model = QString(buffer.mid(idx+3, 6)); modelsegments++; @@ -396,59 +422,55 @@ void CMS50F37Loader::processBytes(QByteArray bytes) } break; case 0x03: // Vendor string - data = buffer.at(idx+1); - if (data == 0) { - vendor = QString(buffer.mid(idx+2, 6)); - qDebug() << "pB: Vendor:" << vendor; - } + vendor = QString(buffer.mid(idx+2, 7)); + qDebug() << "pB: Vendor:" << vendor; break; case 0x04: // Device Identifiers - for (int i = 2, msb = buffer.at(idx+1); i < len; i++, msb>>= 1) { - buffer[idx+i] = (buffer[idx+i] & 0x7f) | (msb & 0x01 ? 0x80 : 0); - } - devid = QString(buffer.mid(idx+2, 7)); qDebug() << "pB: Device ID:" << devid; break; // COMMAND_GET_USER_INFO case 0x05: // 5,80,80,f5,f3,e5,f2,80,80 - // User + userIdx = buffer.at(idx+2); // for future use user = QString(buffer.mid(idx+3).trimmed()); qDebug() << "pB: 0x05:" << user; - break; + + // Command GET_VERSION case 0x6: // 6,80,80,87 - // data = buffer.at(idx+3) ^ 0x80; + protocolVersion = buffer.at(idx+3) | (buffer.at(idx+2)<<8); break; - // COMMAND_GET_SESSION_TIME + // COMMAND_GET_SESSION_TIME --- the date part case 0x07: // 7,80,80,80,94,8e,88,92 - for (int i = 2, msb = buffer.at(idx+1); i < len; i++, msb>>= 1) { - buffer[idx+i] = (buffer[idx+i] & 0x7f) | (msb & 0x01 ? 0x80 : 0); - } - year = QString().sprintf("%02i%02i",buffer.at(idx+4), buffer.at(idx+5)).toInt(); month = QString().sprintf("%02i", buffer.at(idx+6)).toInt(); day = QString().sprintf("%02i", buffer.at(idx+7)).toInt(); - imp_date = QDate(year,month,day); + if ( year == 0 ) { + imp_date = QDate::currentDate(); + cms50dplus = true; + } + else + imp_date = QDate(year,month,day); qDebug() << "pB: ymd " << year << month << day << " impDate " << imp_date; break; // COMMAND_GET_SESSION_DURATION case 0x08: // 8,80,80,80,a4,81,80,80 // 00, 00, 24, 01, 00, 00 - duration = ((buffer.at(idx+1) & 0x4) << 5); - duration |= buffer.at(idx+4); - duration |= (buffer.at(idx+5) | ((buffer.at(idx+1) & 0x8) << 4)) << 8; - duration |= (buffer.at(idx+6) | ((buffer.at(idx+1) & 0x10) << 3)) << 16; + duration = buffer.at(idx+4); + duration |= (buffer.at(idx+5) << 8); + duration |= (buffer.at(idx+6) << 16); + duration |= (buffer.at(idx+7) << 24); break; // COMMAND_GET_SESSION_COUNT case 0x0a: // a,80,80,81 + userIdx = buffer.at(idx+2); session_count = buffer.at(idx+3); break; @@ -460,12 +482,17 @@ void CMS50F37Loader::processBytes(QByteArray bytes) case 0xc: // a7 & a2 // responds with: c,80 //data = buffer.at(idx+1); break; + + // Query Perfusion Index available (0 is yes, 1 is no) case 0x0e: // e,80,81 - break; - case 0x10: // 10,80,81 //data = buffer.at(idx+2); break; + // Get User Count + case 0x10: // 10,80,81 + userCount = buffer.at(idx+2); + break; + // COMMAND_GET_OXIMETER_INFO case 0x11: // 11,80,81,81,80,80,80,80,80 device_info = buffer.at(idx+3); @@ -518,26 +545,15 @@ void CMS50F37Loader::processBytes(QByteArray bytes) if (res == 0x09) { // 9,80,e1,c4,ce,82 // cms50i data - quint8 * buf = &((quint8 *)buffer.data())[idx]; - - quint8 msb = buf[1]; - for (int i = 2; i < len-1; i++, msb >>= 1) { - buf[i] = (buf[i] & 0x7f) | ((msb & 0x01) ? 0x80 : 0); - } - - pi = buf[4] | buf[5] << 8; - pulse = buf[3]; - spo2 = buf[2] & 0x7f; + pi = buffer.at(idx+4) | (buffer.at(idx+5) << 8); + pulse = buffer.at(idx+3); + spo2 = buffer.at(idx+2); qDebug() << "pB: Pulse=" << pulse << "SPO2=" << spo2 << "PI=" << pi; oxirec->append(((spo2 == 0) || (pulse == 0)) ? OxiRecord(0,0,0) : OxiRecord(pulse, spo2, pi)); } else if (res == 0x0f) { // f,80,de,c2,de,c2,de,c2 cms50F data... - for (int i = 2, msb = buffer.at(idx+1); i < len; i++, msb>>= 1) { - buffer[idx+i] = (buffer[idx+i] & 0x7f) | (msb & 0x01 ? 0x80 : 0); - } - pulse = buffer.at(idx+3); spo2 = buffer.at(idx+2); qDebug() << "pB: Pulse=" << pulse << "SPO2=" << spo2; @@ -653,10 +669,10 @@ void CMS50F37Loader::eraseSession(int user, int session) QString out; for (int i=0; i < 9; ++i) out += QString().sprintf("%02X ",cmd[i]); - qDebug() << "Write:" << out; + qDebug() << "Erase Session: Write:" << out; if (serial.write((char *)cmd, 9) == -1) { - qDebug() << "Couldn't write data bytes to CMS50F"; + qDebug() << "Couldn't write Erase session bytes to CMS50F"; } int z = timectr; @@ -691,10 +707,10 @@ void CMS50F37Loader::setDeviceID(const QString & newid) QString out; for (int i=0; i < 9; ++i) out += QString().sprintf("%02X ",cmd[i]); - qDebug() << "Write:" << out; + qDebug() << "setDeviceID: Write:" << out; if (serial.write((char *)cmd, 9) == -1) { - qDebug() << "Couldn't write data bytes to CMS50F"; + qDebug() << "Couldn't write DeviceID data bytes to CMS50F"; } // Supposed to return 0x04 command, so reset devid.. @@ -721,7 +737,7 @@ void CMS50F37Loader::syncClock() timectr = 0; if (serial.write((char *)datecmd, 9) == -1) { - qDebug() << "Couldn't write data bytes to CMS50F"; + qDebug() << "Couldn't write date bytes to CMS50F"; } QTime time; @@ -743,7 +759,7 @@ void CMS50F37Loader::syncClock() timectr = 0; if (serial.write((char *)timecmd, 9) == -1) { - qDebug() << "Couldn't write data bytes to CMS50F"; + qDebug() << "Couldn't write time bytes to CMS50F"; } time.start(); @@ -854,11 +870,13 @@ bool CMS50F37Loader::readSpoRFile(const QString & path) QFile file(path); if (!file.exists()) { qWarning() << "Can't find the oximeter file: " << path; + QMessageBox::warning(nullptr, STR_MessageBox_Error, "

"+tr("Could not find the oximeter file:")+"

"+path+"

"); return false; } if (!file.open(QFile::ReadOnly)) { qWarning() << "Can't open the oximeter file: " << path; + QMessageBox::warning(nullptr, STR_MessageBox_Error, "

"+tr("Could not open the oximeter file:")+"

"+path+"

"); return false; } @@ -902,9 +920,10 @@ bool CMS50F37Loader::readSpoRFile(const QString & path) m_startTime = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss"); if (m_startTime.date().year() < 2000) { m_startTime = m_startTime.addYears(100); } } else { // this should probaly find the most recent SH data day - m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0)); + m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0)); // make it today at midnight + cms50dplus = true; } - } else { // !spo2header + } else { // it is an spo2header quint32 samples = 0; // number of samples @@ -921,7 +940,12 @@ bool CMS50F37Loader::readSpoRFile(const QString & path) in >> year >> month >> day; in >> hour >> minute >> second; - m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second)); + if ( year+month+day == 0 ) { + m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0)); + cms50dplus = true; + } + else + m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second)); // ignoring it for now pos += 0x1c + 200; diff --git a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.h b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.h index 9369e931..0d9abf7c 100644 --- a/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.h +++ b/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.h @@ -56,6 +56,7 @@ Q_OBJECT virtual QDateTime getDateTime(int session); virtual int getDuration(int session); virtual int getSessionCount(); + virtual int getUserCount(); virtual int getOximeterInfo(); virtual void eraseSession(int user, int session); @@ -119,6 +120,7 @@ protected: bool started_import; bool finished_import; bool started_reading; + bool cms50dplus; int cb_reset,imp_callbacks; @@ -136,6 +138,9 @@ protected: volatile int session_count; volatile int duration; + volatile int userCount; + volatile int userIdx; + volatile int protocolVersion; int device_info; QString model; QString vendor; diff --git a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp index 0631b88a..ed62393d 100644 --- a/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/md300w1_loader.cpp @@ -162,11 +162,13 @@ bool MD300W1Loader::readDATFile(const QString & path) if (!file.exists()) { qDebug() << "File does not exist: " << path; + QMessageBox::warning(nullptr, STR_MessageBox_Error, "

"+tr("Could not find the oximeter file:")+"

"+path+"

"); return false; } if (!file.open(QFile::ReadOnly)) { qDebug() << "Can't open file R/O: " << path; + QMessageBox::warning(nullptr, STR_MessageBox_Error, "

"+tr("Could not open the oximeter file:")+"

"+path+"

"); return false; }