OSCAR-code/sleepyhead/SleepLib/loader_plugins/cms50f37_loader.cpp

805 lines
20 KiB
C++
Raw Normal View History

2014-08-17 23:29:30 +00:00
/* 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. */
//********************************************************************************************
/// IMPORTANT!!!
//********************************************************************************************
// Please INCREMENT the cms50_data_version in cms50_loader.h when making changes to this loader
// that change loader behaviour or modify channels.
//********************************************************************************************
#include <QProgressBar>
#include <QApplication>
#include <QDir>
#include <QString>
#include <QDataStream>
#include <QDateTime>
#include <QFile>
#include <QDebug>
#include <QList>
#include <QMessageBox>
#include <QLabel>
#include <QVBoxLayout>
#include <QPushButton>
using namespace std;
#include "cms50f37_loader.h"
#include "SleepLib/machine.h"
#include "SleepLib/session.h"
extern QProgressBar *qprogress;
CMS50F37Loader::CMS50F37Loader()
{
m_type = MT_OXIMETER;
m_abort = false;
m_streaming = false;
m_importing = false;
imp_callbacks = 0;
m_vendorID = 0x10c4;
m_productID = 0xea60;
oxirec = nullptr;
startTimer.setParent(this);
resetTimer.setParent(this);
2014-08-19 02:44:48 +00:00
duration_divisor = 2;
2014-08-19 04:30:27 +00:00
model = QString();
vendor = QString();
2014-08-17 23:29:30 +00:00
}
CMS50F37Loader::~CMS50F37Loader()
{
}
2014-08-18 00:34:41 +00:00
bool CMS50F37Loader::openDevice()
{
if (port.isEmpty()) {
if (!scanDevice("",m_vendorID, m_productID))
return false;
}
serial.setPortName(port);
if (!serial.open(QSerialPort::ReadWrite))
return false;
// forward this stuff
// Set up serial port attributes
serial.setBaudRate(QSerialPort::Baud115200);
2014-08-18 00:58:12 +00:00
serial.setParity(QSerialPort::NoParity);
2014-08-18 00:34:41 +00:00
serial.setStopBits(QSerialPort::OneStop);
serial.setDataBits(QSerialPort::Data8);
serial.setFlowControl(QSerialPort::NoFlowControl);
m_streaming = true;
m_abort = false;
m_importing = false;
// connect relevant signals
connect(&serial,SIGNAL(readyRead()), this, SLOT(dataAvailable()));
2014-08-19 04:01:31 +00:00
started_import = false;
started_reading = false;
finished_import = false;
2014-08-19 02:44:48 +00:00
resetDevice();
2014-08-18 00:34:41 +00:00
return true;
}
2014-08-17 23:29:30 +00:00
bool CMS50F37Loader::Detect(const QString &path)
{
if (p_profile->oxi->oximeterType() == QString("Contec CMS50F v3.7+")) {
return true;
}
Q_UNUSED(path);
return false;
}
int CMS50F37Loader::Open(QString path)
{
// Only one active Oximeter module at a time, set in preferences
m_itemCnt = 0;
m_itemTotal = 0;
m_abort = false;
m_importing = false;
started_import = false;
started_reading = false;
finished_import = false;
setStatus(NEUTRAL);
imp_callbacks = 0;
cb_reset = 0;
m_time.start();
if (oxirec) {
trashRecords();
}
// Cheating using path for two serial oximetry modes
if (path.compare("import") == 0) {
2014-08-18 08:00:21 +00:00
serial.clear();
2014-08-17 23:29:30 +00:00
sequence = 0;
2014-08-18 08:00:21 +00:00
buffer.clear();
2014-08-19 02:44:48 +00:00
// nextCommand();
2014-08-18 00:50:17 +00:00
setStatus(IMPORTING);
2014-08-17 23:29:30 +00:00
return 1;
} else if (path.compare("live") == 0) {
return 0;
}
QString ext = path.section(".",1);
if ((ext.compare("spo2", Qt::CaseInsensitive)==0) || (ext.compare("spo", Qt::CaseInsensitive)==0) || (ext.compare("spor", Qt::CaseInsensitive)==0)) {
// try to read and process SpoR file..
return readSpoRFile(path) ? 1 : 0;
}
return 0;
}
2014-08-18 08:00:21 +00:00
unsigned char cms50_sequence[] = { 0xa7, 0xa2, 0xa0, 0xb0, 0xac, 0xb3, 0xad, 0xa3, 0xab, 0xa4, 0xa5, 0xaf, 0xa7, 0xa2, 0xa6 };
2014-08-19 02:44:48 +00:00
const int TIMEOUT = 2000;
const quint8 COMMAND_CMS50_HELLO1 = 0x27; // 0xa7
const quint8 COMMAND_CMS50_HELLO2 = 0x22; // 0xa2
const quint8 COMMAND_GET_SESSION_COUNT = 0x23; // 0xA3
const quint8 COMMAND_GET_SESSION_TIME = 0x25; // 0xA5
const quint8 COMMAND_GET_SESSION_DURATION = 0x24; // 0xA4
const quint8 COMMAND_GET_USER_INFO = 0x2B; // 0xAB
const quint8 COMMAND_GET_SESSION_DATA = 0x26; // 0xA6
const quint8 COMMAND_GET_OXIMETER_INFO = 0x30; // 0xb0
const quint8 COMMAND_GET_OXIMETER_MODEL = 0x28; // 0xA8
const quint8 COMMAND_GET_OXIMETER_VENDOR = 0x29; // 0xA9
2014-08-18 08:00:21 +00:00
2014-08-18 00:22:16 +00:00
int cms50_seqlength = sizeof(cms50_sequence);
2014-08-17 23:29:30 +00:00
2014-08-19 02:44:48 +00:00
QString CMS50F37Loader::getUser()
{
user = QString();
sendCommand(COMMAND_GET_USER_INFO);
QTime time;
time.start();
do {
QApplication::processEvents();
} while (user.isEmpty() && (time.elapsed() < TIMEOUT));
return user;
}
QString CMS50F37Loader::getVendor()
{
2014-08-19 04:19:16 +00:00
if (!vendor.isEmpty()) return vendor;
2014-08-19 02:44:48 +00:00
sendCommand(COMMAND_GET_OXIMETER_VENDOR);
QTime time;
time.start();
do {
QApplication::processEvents();
} while (vendor.isEmpty() && (time.elapsed() < TIMEOUT));
return vendor;
}
QString CMS50F37Loader::getModel()
{
2014-08-19 04:19:16 +00:00
if (!model.isEmpty()) return model;
2014-08-19 02:44:48 +00:00
sendCommand(COMMAND_GET_OXIMETER_MODEL);
QTime time;
time.start();
do {
QApplication::processEvents();
} while (model.isEmpty() && (time.elapsed() < TIMEOUT));
if (model.startsWith("CMS50I")) {
duration_divisor = 4;
} else {
duration_divisor = 2;
}
return model;
}
QString CMS50F37Loader::getDeviceString()
{
return QString("%1 %2").arg(getVendor()).arg(getModel());
}
int CMS50F37Loader::getUserIndex()
{
user_index = -1;
sendCommand(COMMAND_GET_USER_INFO);
QTime time;
time.start();
do {
QApplication::processEvents();
} while ((user_index < 0) && (time.elapsed() < TIMEOUT));
return user_index;
}
int CMS50F37Loader::getSessionCount()
{
session_count = -1;
sendCommand(COMMAND_GET_SESSION_COUNT);
QTime time;
time.start();
do {
QApplication::processEvents();
} while ((session_count < 0) && (time.elapsed() < TIMEOUT));
return session_count;
}
int CMS50F37Loader::getOximeterInfo()
{
device_info = -1;
sendCommand(COMMAND_GET_OXIMETER_INFO);
QTime time;
time.start();
do {
QApplication::processEvents();
} while ((device_info < 0) && (time.elapsed() < TIMEOUT));
return device_info;
}
int CMS50F37Loader::getDuration(int session)
{
getOximeterInfo();
duration = -1;
sendCommand(COMMAND_GET_SESSION_DURATION, session);
QTime time;
time.start();
do {
QApplication::processEvents();
} while ((duration < 0) && (time.elapsed() < TIMEOUT));
2014-08-19 04:10:38 +00:00
return duration / duration_divisor;
2014-08-19 02:44:48 +00:00
}
QDateTime CMS50F37Loader::getDateTime(int session)
{
imp_date = QDate();
imp_time = QTime();
sendCommand(COMMAND_GET_SESSION_TIME, session);
QTime time;
time.start();
do {
QApplication::processEvents();
} while ((imp_date.isNull() || imp_time.isNull()) && (time.elapsed() < TIMEOUT));
if (imp_date.isNull() || imp_time.isNull())
return QDateTime();
return QDateTime(imp_date, imp_time);
}
2014-08-17 23:29:30 +00:00
void CMS50F37Loader::processBytes(QByteArray bytes)
{
2014-08-18 08:00:21 +00:00
int data;
QString tmpstr;
2014-08-19 02:44:48 +00:00
int lengths[32] = { 0, 0, 9, 9, 0, 9, 4, 8, 8, 6, 4, 0, 2, 0, 3, 8, 3, 9, 8, 9, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };
2014-08-18 08:00:21 +00:00
buffer.append(bytes);
int size = buffer.size();
int idx = 0;
int len;
2014-08-18 15:12:34 +00:00
int year, month, day;
2014-08-19 14:25:03 +00:00
quint8 msb;
2014-08-19 02:44:48 +00:00
quint8 pulse;
2014-08-18 17:13:06 +00:00
2014-08-18 08:00:21 +00:00
do {
unsigned char res = bytes.at(idx);
len = lengths[res & 0x1f];
2014-08-18 00:22:16 +00:00
if (len > size)
2014-08-18 08:00:21 +00:00
break;
2014-08-18 00:22:16 +00:00
2014-08-18 08:00:21 +00:00
if (len == 0) {
// lost sync
idx++;
continue;
}
for (int i = 1; i < len; i++) {
buffer[idx+i] = buffer[idx+i] ^ 0x80;
2014-08-18 00:22:16 +00:00
}
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
switch(res) {
2014-08-19 02:44:48 +00:00
case 0x02:
data = buffer.at(idx+1);
if (data == 0) {
2014-08-19 05:02:16 +00:00
model = QString(buffer.mid(idx+3, 6));
2014-08-19 04:43:04 +00:00
qDebug() << "Model:" << model;
2014-08-19 02:44:48 +00:00
}
break;
case 0x03:
data = buffer.at(idx+1);
if (data == 0) {
2014-08-19 04:52:23 +00:00
vendor = QString(buffer.mid(idx+2, 6));
2014-08-19 04:43:04 +00:00
qDebug() << "Vendor:" << vendor;
2014-08-19 02:44:48 +00:00
}
break;
// COMMAND_GET_USER_INFO
2014-08-18 08:00:21 +00:00
case 0x05: // 5,80,80,f5,f3,e5,f2,80,80
2014-08-18 15:12:34 +00:00
// User
user = QString(buffer.mid(idx+3,4));
2014-08-19 02:44:48 +00:00
user_index = buffer.at(idx+8);
2014-08-18 15:12:34 +00:00
qDebug() << "0x05:" << user;
2014-08-18 08:00:21 +00:00
break;
case 0x6: // 6,80,80,87
data = buffer.at(idx+3) ^ 0x80;
break;
2014-08-19 02:44:48 +00:00
// COMMAND_GET_SESSION_TIME
2014-08-18 08:00:21 +00:00
case 0x07: // 7,80,80,80,94,8e,88,92
2014-08-18 15:12:34 +00:00
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);
qDebug() << imp_date;
2014-08-18 08:00:21 +00:00
break;
2014-08-19 02:44:48 +00:00
// COMMAND_GET_SESSION_DURATION
case 0x08: // 8,80,80,80,a4,81,80,80 // 00, 00, 24, 01, 00, 00
// duration
duration = ((buffer.at(idx+1) & 0x4) << 5);
2014-08-19 04:34:45 +00:00
duration |= buffer.at(idx+4);
duration |= buffer.at(idx+5) << 8;
2014-08-18 08:00:21 +00:00
break;
2014-08-19 02:44:48 +00:00
// COMMAND_GET_SESSION_COUNT
2014-08-18 08:00:21 +00:00
case 0x0a: // a,80,80,81
2014-08-19 02:44:48 +00:00
session_count = buffer.at(idx+3);
2014-08-18 08:00:21 +00:00
break;
2014-08-19 02:44:48 +00:00
// COMMAND_CMS50_HELLO1 && COMMAND_CMS50_HELLO2
2014-08-18 08:00:21 +00:00
case 0xc: // a7 & a2 // responds with: c,80
data = buffer.at(idx+1);
break;
case 0x0e: // e,80,81
break;
case 0x10: // 10,80,81
data = buffer.at(idx+2);
break;
2014-08-19 02:44:48 +00:00
// COMMAND_GET_OXIMETER_INFO
2014-08-18 08:00:21 +00:00
case 0x11: // 11,80,81,81,80,80,80,80,80
2014-08-19 02:44:48 +00:00
device_info = buffer.at(idx+3);
2014-08-18 08:00:21 +00:00
break;
2014-08-19 02:44:48 +00:00
// COMMAND_GET_SESSION_TIME
2014-08-18 08:00:21 +00:00
case 0x12: // 12,80,80,80,82,a6,92,80
tmpstr = QString().sprintf("%02i:%02i:%02i",buffer.at(idx+4), buffer.at(idx+5), buffer.at(idx+6));
2014-08-18 14:50:38 +00:00
imp_time = QTime::fromString(tmpstr, "HH:mm:ss");
2014-08-18 15:12:34 +00:00
qDebug() << imp_time;
2014-08-18 08:00:21 +00:00
break;
case 0x13: // 13,80,a0,a0,a0,a0,a0,a0,a0
break;
case 0x14: //14,80,80,80,80,80,80,80,80
break;
case 0x09: // cms50i data sequence
case 0x0f: // f,80,de,c2,de,c2,de,c2 cms50F data...
if (!started_import) {
started_import = true;
started_reading = true;
finished_import = false;
m_importing = true;
m_itemCnt=0;
m_itemTotal=5000;
have_perfindex = (res == 0x9);
2014-08-18 08:00:21 +00:00
oxirec = new QVector<OxiRecord>;
oxirec->reserve(30000);
2014-08-18 15:26:59 +00:00
oxisessions[m_startTime] = oxirec;
2014-08-18 08:00:21 +00:00
cb_reset = 1;
resetTimer.singleShot(2000,this,SLOT(resetImportTimeout()));
}
break;
default:
qDebug() << "unknown cms50 result?" << hex << (int)res;
break;
}
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
if (res == 0x09) {
2014-08-19 14:29:29 +00:00
msb = buffer.at(idx+1);
2014-08-18 08:00:21 +00:00
// 9,80,e1,c4,ce,82 // cms50i data
2014-08-19 14:46:50 +00:00
for (int i = 2, msb = buffer.at(idx+1); i < len; i++, msb>>= 1) {
2014-08-19 14:29:29 +00:00
buffer[idx+i] = (buffer[idx+i] & 0x7f) | (msb & 0x01 ? 0x80 : 0);
}
2014-08-19 14:58:33 +00:00
quint16 pi = quint8(buffer.at(idx + 4)) | (quint16(buffer.at(idx + 5)) << 8);
2014-08-19 06:07:28 +00:00
2014-08-19 14:29:29 +00:00
pulse = buffer.at(idx+3);
2014-08-19 06:07:28 +00:00
quint8 spo2 = buffer.at(idx+2);
2014-08-19 14:58:33 +00:00
oxirec->append(((spo2 == 0) || (pulse == 0) || (pi == 0xfff6)) ? OxiRecord(0,0,0) : OxiRecord(pulse, spo2, pi));
2014-08-18 08:00:21 +00:00
} else if (res == 0x0f) {
// f,80,de,c2,de,c2,de,c2 cms50F data...
2014-08-19 14:46:50 +00:00
for (int i = 2, msb = buffer.at(idx+1); i < len; i++, msb>>= 1) {
2014-08-19 14:29:29 +00:00
buffer[idx+i] = (buffer[idx+i] & 0x7f) | (msb & 0x01 ? 0x80 : 0);
2014-08-19 14:25:03 +00:00
}
// pulse = buffer.at(idx+3) | ((mask & 2) << 6);
oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(buffer.at(idx+3), buffer.at(idx+2)));
2014-08-18 17:13:06 +00:00
2014-08-19 14:25:03 +00:00
// pulse = buffer.at(idx+5) | ((mask & 8) << 4);
oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(buffer.at(idx+5), buffer.at(idx+4)));
2014-08-18 17:13:06 +00:00
2014-08-19 14:25:03 +00:00
// pulse = buffer.at(idx+7) | ((mask & 0x20) << 2);
2014-08-19 14:29:29 +00:00
oxirec->append((pulse == 0xff) ? OxiRecord(0,0) : OxiRecord(buffer.at(idx+7), buffer.at(idx+6)));
2014-08-18 08:00:21 +00:00
}
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
QStringList str;
for (int i=0; i < len; ++i) {
str.append(QString::number((unsigned char)buffer.at(idx + i),16));
2014-08-18 08:00:21 +00:00
}
if (!started_import) {
2014-08-19 02:47:26 +00:00
// startTimer.singleShot(2000, this, SLOT(requestData()));
qDebug() << "Read:" << len << size << str.join(",");
2014-08-18 08:00:21 +00:00
} else {
qDebug() << "Import:" << len << size << str.join(",");
2014-08-18 08:00:21 +00:00
}
idx += len;
} while (idx < size);
if (!started_import) {
imp_callbacks = 0;
2014-08-17 23:29:30 +00:00
} else {
2014-08-18 08:00:21 +00:00
imp_callbacks++;
2014-08-17 23:29:30 +00:00
}
2014-08-18 08:00:21 +00:00
buffer = buffer.mid(idx);
2014-08-17 23:29:30 +00:00
}
2014-08-18 08:00:21 +00:00
//int CMS50F37Loader::doLiveMode()
//{
// Q_ASSERT(oxirec != nullptr);
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
// int available = buffer.size();
// int idx = 0;
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
// QByteArray plethy;
// while (idx < available-5) {
// 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-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
// oxirec->append(OxiRecord(pulse, spo2));
// plethy.append(pwave);
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
// idx += 5;
// }
// emit updatePlethy(plethy);
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
// return idx;
//}
2014-08-17 23:29:30 +00:00
2014-08-19 04:01:31 +00:00
void CMS50F37Loader::sendCommand(quint8 c)
2014-08-17 23:29:30 +00:00
{
2014-08-19 04:01:31 +00:00
quint8 cmd[] = { 0x7d, 0x81, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80 };
2014-08-19 02:44:48 +00:00
cmd[2] |= (c & 0x7f);
2014-08-17 23:29:30 +00:00
QString out;
2014-08-18 00:50:17 +00:00
for (int i=0;i < 9;i++) out += QString().sprintf("%02X ",cmd[i]);
2014-08-17 23:29:30 +00:00
qDebug() << "Write:" << out;
2014-08-18 00:50:17 +00:00
if (serial.write((char *)cmd, 9) == -1) {
2014-08-17 23:29:30 +00:00
qDebug() << "Couldn't write data reset bytes to CMS50";
}
}
2014-08-19 04:01:31 +00:00
void CMS50F37Loader::sendCommand(quint8 c, quint8 c2)
2014-08-19 02:44:48 +00:00
{
2014-08-19 04:01:31 +00:00
quint8 cmd[] = { 0x7d, 0x81, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80, 0x80 };
2014-08-19 02:44:48 +00:00
cmd[2] |= (c & 0x7f);
cmd[4] |= (c2 & 0x7f);
QString out;
for (int i=0; i < 9; ++i) out += QString().sprintf("%02X ",cmd[i]);
qDebug() << "Write:" << out;
if (serial.write((char *)cmd, 9) == -1) {
qDebug() << "Couldn't write data reset bytes to CMS50";
}
}
2014-08-17 23:29:30 +00:00
void CMS50F37Loader::nextCommand()
{
2014-08-18 14:35:30 +00:00
if (++sequence < cms50_seqlength) {
// Send the next command packet in sequence
2014-08-17 23:29:30 +00:00
sendCommand(cms50_sequence[sequence]);
2014-08-18 14:35:30 +00:00
} else {
qDebug() << "Run out of startup tasks to do and import failed!";
}
2014-08-17 23:29:30 +00:00
}
2014-08-19 02:44:48 +00:00
void CMS50F37Loader::getSessionData(int session)
{
resetDevice();
for (int i=0;i<5;i++) {
QApplication::processEvents();
QThread::msleep(50);
}
selected_session = session;
requestData();
}
2014-08-17 23:29:30 +00:00
2014-08-19 02:44:48 +00:00
void CMS50F37Loader::resetDevice()
2014-08-17 23:29:30 +00:00
{
2014-08-19 02:44:48 +00:00
sendCommand(COMMAND_CMS50_HELLO1);
sendCommand(COMMAND_CMS50_HELLO2);
2014-08-17 23:29:30 +00:00
}
2014-08-19 02:44:48 +00:00
void CMS50F37Loader::requestData()
2014-08-17 23:29:30 +00:00
{
2014-08-19 02:44:48 +00:00
sendCommand(COMMAND_GET_SESSION_DATA, selected_session);
2014-08-17 23:29:30 +00:00
}
void CMS50F37Loader::killTimers()
{
2014-08-18 08:00:21 +00:00
if (resetTimer.isActive()) resetTimer.stop();
2014-08-17 23:29:30 +00:00
}
void CMS50F37Loader::startImportTimeout()
{
}
void CMS50F37Loader::resetImportTimeout()
{
if (finished_import) {
return;
}
if (imp_callbacks != cb_reset) {
// Still receiving data.. reset timer
qDebug() << "Still receiving data in resetImportTimeout()" << imp_callbacks << cb_reset;
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
2014-08-18 08:00:21 +00:00
2014-08-18 15:12:34 +00:00
killTimers();
2014-08-17 23:29:30 +00:00
2014-08-18 08:00:21 +00:00
finished_import = true;
m_streaming = false;
2014-08-18 15:12:34 +00:00
m_importing = false;
started_reading = false;
started_import = false;
2014-08-18 08:00:21 +00:00
emit importComplete(this);
2014-08-18 15:26:59 +00:00
m_status = NEUTRAL;
2014-08-18 15:12:34 +00:00
shutdownPorts();
2014-08-17 23:29:30 +00:00
return;
}
qDebug() << "Should CMS50 resetImportTimeout reach here?";
// else what???
}
cb_reset = imp_callbacks;
}
void CMS50F37Loader::shutdownPorts()
{
closeDevice();
}
bool CMS50F37Loader::readSpoRFile(QString path)
{
QFile file(path);
if (!file.exists()) {
return false;
}
if (!file.open(QFile::ReadOnly)) {
return false;
}
bool spo2header = false;
QString ext = path.section('.', -1);
if (ext.compare("spo2",Qt::CaseInsensitive) == 0) {
spo2header = true;
}
QByteArray data;
qint64 filesize = file.size();
data = file.readAll();
QDataStream in(data);
in.setByteOrder(QDataStream::LittleEndian);
quint16 pos;
in >> pos;
in.skipRawData(pos - 2);
//long size = data.size();
int bytes_per_record = 2;
if (!spo2header) {
// next is 0x0002
// followed by 16bit duration in seconds
// 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);
m_startTime = QDateTime::fromString(dstr, "MM/dd/yy HH:mm:ss");
if (m_startTime.date().year() < 2000) { m_startTime = m_startTime.addYears(100); }
} else {
m_startTime = QDateTime(QDate::currentDate(), QTime(0,0,0));
}
} else { // !spo2header
quint32 samples = 0; // number of samples
quint32 year, month, day;
quint32 hour, minute, second;
if (data.at(pos) != 1) {
qWarning() << ".spo2 file" << path << "might be a different";
}
// Unknown cruft header...
in.skipRawData(200);
in >> year >> month >> day;
in >> hour >> minute >> second;
m_startTime = QDateTime(QDate(year, month, day), QTime(hour, minute, second));
// ignoring it for now
pos += 0x1c + 200;
in >> samples;
int remainder = filesize - pos;
bytes_per_record = remainder / samples;
qDebug() << samples << "samples of" << bytes_per_record << "bytes each";
// CMS50I .spo2 data have 4 digits, a 16bit, followed by spo2 then pulse
}
oxirec = new QVector<OxiRecord>;
oxisessions[m_startTime] = oxirec;
unsigned char o2, pr;
quint16 un;
// Read all Pulse and SPO2 data
do {
if (bytes_per_record > 2) {
in >> un;
}
in >> o2;
in >> pr;
if ((o2 == 0x7f) && (pr == 0xff)) {
o2 = pr = 0;
un = 0;
}
if (spo2header) {
oxirec->append(OxiRecord(pr, o2));
} else {
oxirec->append(OxiRecord(o2, pr));
}
} 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;
// }
// processing gets done later
return true;
}
void CMS50F37Loader::process()
{
// Just clean up any extra crap before oximeterimport parses the oxirecords..
return;
// if (!oxirec)
// return;
// int size=oxirec->size();
// if (size<10)
// return;
}
static bool cms50f37_initialized = false;
void CMS50F37Loader::Register()
{
if (cms50f37_initialized) { return; }
qDebug() << "Registering CMS50F37Loader";
RegisterLoader(new CMS50F37Loader());
cms50f37_initialized = true;
}