Added support for importing .spo2 files from newer CMS50F oximeters, plus a little SleepyHead history

This commit is contained in:
Mark Watkins 2014-06-30 20:41:50 +10:00
parent 1735d9b51e
commit fe184a1f4a
5 changed files with 1864 additions and 23 deletions

9
history/README Normal file
View File

@ -0,0 +1,9 @@
This folder contains the python/GTK script I (jedimark) wrote that eventually turned into SleepyHead
I can't honestly tell you if this is the latest version, because the computer I originally wrote this on died.
I put it here for project history, reference (and humour at how sad it is.)
It requires matplotlib and pygtk to run.. Probably pytz too, I honestly can't remember if it needed any other libs.
I've mostly forgotten python since then.

1783
history/cpap.py Normal file

File diff suppressed because it is too large Load Diff

View File

@ -20,6 +20,7 @@
#include <QApplication>
#include <QDir>
#include <QString>
#include <QDataStream>
#include <QDateTime>
#include <QFile>
#include <QDebug>
@ -47,6 +48,7 @@ CMS50Loader::CMS50Loader()
m_vendorID = 0x10c4;
m_productID = 0xea60;
cms50dplus = false;
oxirec = nullptr;
@ -124,7 +126,7 @@ int CMS50Loader::Open(QString path, Profile *profile)
return 1;
}
QString ext = path.section(".",1);
if ((ext.compare("spo", Qt::CaseInsensitive)==0) || (ext.compare("spor", Qt::CaseInsensitive)==0)) {
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;
}
@ -471,18 +473,30 @@ bool CMS50Loader::readSpoRFile(QString path)
return false;
}
bool spo2header = false;
QString ext = path.section('.', -1);
if (ext.compare("spo2",Qt::CaseInsensitive) == 0) {
spo2header = true;
}
QByteArray data;
data = file.readAll();
long size = data.size();
QDataStream in(data);
in.setByteOrder(QDataStream::LittleEndian);
quint16 pos;
in >> pos;
// position data stream starts at
int pos = ((unsigned char)data.at(1) << 8) | (unsigned char)data.at(0);
in.skipRawData(pos - 2);
//long size = data.size();
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) {
@ -490,9 +504,37 @@ bool CMS50Loader::readSpoRFile(QString path)
}
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...
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;
}
oxirec = new QVector<OxiRecord>;
oxisessions[m_startTime] = oxirec;
@ -500,12 +542,19 @@ bool CMS50Loader::readSpoRFile(QString path)
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));
do {
in >> o2;
in >> pr;
oxirec->append(OxiRecord(pr, o2));
i += 2;
}
} 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;

View File

@ -177,7 +177,7 @@ void init()
QObject::tr("VS"), STR_UNIT_EventsPerHour, DEFAULT, QColor("red")));
schema::channel.add(GRP_CPAP, new Channel(CPAP_VSnore2 = 0x1008, FLAG, SESSION, "VSnore2",
QObject::tr("Vibratory Snore"),
QObject::tr("Vibratory Snore (VS2) "),
QObject::tr("A vibratory snore as detcted by a System One machine"),
QObject::tr("VS2"), STR_UNIT_EventsPerHour, DEFAULT, QColor("red")));

View File

@ -271,7 +271,7 @@ void OximeterImport::on_fileImportButton_clicked()
#endif
QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.dat)"));
QString filename = QFileDialog::getOpenFileName(nullptr , tr("Select a valid oximetry data file"), documentsFolder, tr("Oximetry Files (*.spo *.spor *.spo2 *.dat)"));
if (filename.isEmpty())
return;