Minor PRS1 cleaning

This commit is contained in:
Mark Watkins 2014-05-02 14:34:34 +10:00
parent 027a81d915
commit c9a5b712f3
3 changed files with 82 additions and 44 deletions

View File

@ -358,7 +358,7 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
QDir dir(path); QDir dir(path);
if (!dir.exists() || (!dir.isReadable())) { if (!dir.exists() || (!dir.isReadable())) {
return false; return 0;
} }
@ -370,13 +370,14 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
if (qprogress) { qprogress->setValue(0); } if (qprogress) { qprogress->setValue(0); }
QList<QString> paths; QStringList paths;
for (int i = 0; i < flist.size(); i++) { for (int i = 0; i < flist.size(); i++) {
QFileInfo fi = flist.at(i); QFileInfo fi = flist.at(i);
filename = fi.fileName(); filename = fi.fileName();
if ((filename[0].toLower() == 'p') && (isdigit(filename[1]))) { if ((filename[0].toLower() == 'p') && (isdigit(filename[1]))) {
// p0, p1, p2.. etc.. folders contain the session data
paths.push_back(fi.canonicalFilePath()); paths.push_back(fi.canonicalFilePath());
} else if (filename.toLower() == "properties.txt") { } else if (filename.toLower() == "properties.txt") {
ParseProperties(m, fi.canonicalFilePath()); ParseProperties(m, fi.canonicalFilePath());
@ -393,13 +394,23 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
bool ok; bool ok;
int model = modelstr.toInt(&ok); int model = modelstr.toInt(&ok);
if (!ok || (model < 450)) { if (ok) {
QMessageBox::information(NULL, // Assumption is made here all PRS1 machines less than 450P are not data capable.. this could be wrong one day.
QObject::tr("Non Data Capable Machine"), if ((model < 450)) {
QString(QObject::tr("Your Philips Respironics CPAP machine (Model %1) is unfortunately not a data capable model.")+"\n\n"+ QMessageBox::information(NULL,
QObject::tr("I'm sorry to report that SleepyHead can only track hours of use for this machine.")). QObject::tr("Non Data Capable Machine"),
arg(m->properties["ModelNumber"]),QMessageBox::Ok); QString(QObject::tr("Your Philips Respironics CPAP machine (Model %1) is unfortunately not a data capable model.")+"\n\n"+
QObject::tr("I'm sorry to report that SleepyHead can only track hours of use for this machine.")).
arg(m->properties["ModelNumber"]),QMessageBox::Ok);
}
if ((model % 100) >= 60) {
// 60 series machine warning??
}
} else {
// model number didn't parse.. Meh...
} }
SessionID sid; SessionID sid;
@ -408,40 +419,43 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
int size = paths.size(); int size = paths.size();
int cnt = 0; int cnt = 0;
new_sessions.clear(); new_sessions.clear(); // this hash is used by OpenFile
for (QList<QString>::iterator p = paths.begin(); p != paths.end(); p++) { // for each p0/p1/p2/etc... folder
for (auto p = paths.begin(); p != paths.end(); p++) {
dir.setPath(*p); dir.setPath(*p);
if (!dir.exists() || !dir.isReadable()) { continue; } if (!dir.exists() || !dir.isReadable()) { continue; }
flist = dir.entryInfoList(); flist = dir.entryInfoList();
// Scan for individual session files
for (int i = 0; i < flist.size(); i++) { for (int i = 0; i < flist.size(); i++) {
QFileInfo fi = flist.at(i); QFileInfo fi = flist.at(i);
QString ext_s = fi.fileName().section(".", -1); QString ext_s = fi.fileName().section(".", -1);
QString session_s = fi.fileName().section(".", 0, -2); QString session_s = fi.fileName().section(".", 0, -2);
ext = ext_s.toLong(&ok); ext = ext_s.toLong(&ok);
if (!ok) { // not a numerical extension
if (!ok) {
continue; continue;
} }
sid = session_s.toLong(&ok); sid = session_s.toLong(&ok);
if (!ok) { // not a numerical session ID
if (!ok) {
continue; continue;
} }
if (m->SessionExists(sid)) { if (m->SessionExists(sid)) {
continue; // could skip this and error check data by reloading summary. // Skip already imported session
continue;
} }
if ((ext == 1) || (ext == 0)) { if ((ext == 1) || (ext == 0)) {
OpenFile(m, fi.canonicalFilePath()); // Open just the summary files first round // Open just the summary files first round and create a session record if succesful
OpenFile(m, fi.canonicalFilePath());
} else { } else {
sessfiles[sid].push_back(fi.canonicalFilePath()); // and keep the rest of the names // and keep the rest of the names for next stage
sessfiles[sid].push_back(fi.canonicalFilePath());
} }
} }
} }
@ -449,8 +463,9 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
cnt = 0; cnt = 0;
size = new_sessions.size(); size = new_sessions.size();
for (QHash<SessionID, Session *>::iterator it = new_sessions.begin(); it != new_sessions.end(); // Scan through new sessions and parse event and waveform data
it++) { // NOTE: this function could be multithreaded.
for (auto it = new_sessions.begin(); it != new_sessions.end(); ++it) {
sid = it.key(); sid = it.key();
if (sessfiles.contains(sid)) { if (sessfiles.contains(sid)) {
@ -458,8 +473,10 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
QString name = sessfiles[sid][j]; QString name = sessfiles[sid][j];
if (name.endsWith(".002")) { if (name.endsWith(".002")) {
// Parse event data
OpenFile(m, name); OpenFile(m, name);
} else if (name.endsWith(".005")) { } else if (name.endsWith(".005")) {
// Parse Flow Rate Waveform (and mask pressure if available) data
OpenWaveforms(sid, name); OpenWaveforms(sid, name);
} }
} }
@ -467,6 +484,8 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
// sessions are fully loaded here.. // sessions are fully loaded here..
// Update the progress bar.. this could be done better by an emit if MachineLoader was a QObject and an
// event loop was running.
if ((++cnt % 10) == 0) { if ((++cnt % 10) == 0) {
if (qprogress) { qprogress->setValue(0.0 + (float(cnt) / float(size) * 100.0)); } if (qprogress) { qprogress->setValue(0.0 + (float(cnt) / float(size) * 100.0)); }
@ -477,9 +496,8 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
// strictly can do this in the above loop, but this is cautionary // strictly can do this in the above loop, but this is cautionary
cnt = 0; cnt = 0;
//QVector<SessionID> KillList; // Scan through parsed sessions, do a little cleanup and add to machine object
for (QHash<SessionID, Session *>::iterator it = new_sessions.begin(); it != new_sessions.end(); for (auto it = new_sessions.begin(); it != new_sessions.end(); ++it) {
it++) {
Session *sess = it.value(); Session *sess = it.value();
if ((sess->length()) == 0) { if ((sess->length()) == 0) {
@ -489,7 +507,6 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
SessionID id = sess->session(); SessionID id = sess->session();
delete sess; delete sess;
new_sessions[id] = nullptr; new_sessions[id] = nullptr;
//KillList.push_back(id);
continue; continue;
} }
@ -540,9 +557,13 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
//sess->settings[CPAP_PressureMax]=sess->settings[CPAP_PressureMin]; //sess->settings[CPAP_PressureMax]=sess->settings[CPAP_PressureMin];
} }
// set session saved flag so machine::Save() knows it needs saving
sess->SetChanged(true); sess->SetChanged(true);
// Add session to machine record
m->AddSession(sess, profile); m->AddSession(sess, profile);
// Update the progress bar
if ((++cnt % 10) == 0) { if ((++cnt % 10) == 0) {
//if (qprogress) qprogress->setValue(33.0+(float(cnt)/float(size)*33.0)); //if (qprogress) qprogress->setValue(33.0+(float(cnt)/float(size)*33.0));
QApplication::processEvents(); QApplication::processEvents();
@ -552,10 +573,14 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
m->properties[STR_PROP_DataVersion] = QString::number(prs1_data_version); m->properties[STR_PROP_DataVersion] = QString::number(prs1_data_version);
m->properties[STR_PROP_LastImported] = QDateTime::currentDateTime().toString(Qt::ISODate); m->properties[STR_PROP_LastImported] = QDateTime::currentDateTime().toString(Qt::ISODate);
m->Save(); // Save any new sessions to disk in our format
// Save any new sessions to disk in our format, updating summaries as it goes..
m->Save();
// Set progress bar to completed.
if (qprogress) { qprogress->setValue(100); } if (qprogress) { qprogress->setValue(100); }
// Return number of newly added sessions
return cnt; return cnt;
} }
@ -572,6 +597,11 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile)
//};// __attribute__((packed)); //};// __attribute__((packed));
// Family 0 = xPAP
// Family 5 = ASV
bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp,
unsigned char *data, quint16 size, int family, int familyVersion) unsigned char *data, quint16 size, int family, int familyVersion)
{ {
@ -631,8 +661,8 @@ bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp,
offset = 2; offset = 2;
} }
session->settings[CPAP_RampTime] = (int)data[offset + // Minutes. Convert to seconds/hours here?
0x06]; // Minutes. Convert to seconds/hours here? session->settings[CPAP_RampTime] = (int)data[offset + 0x06];
session->settings[CPAP_RampPressure] = (EventDataType)data[offset + 0x07] / 10.0; session->settings[CPAP_RampPressure] = (EventDataType)data[offset + 0x07] / 10.0;
if (max > 0) { // Ignoring bipap until we see some more data during import if (max > 0) { // Ignoring bipap until we see some more data during import
@ -693,7 +723,7 @@ bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp,
unsigned duration; unsigned duration;
// up to this point appears to becorrect for 0x01 & 0x00 // up to this point appears to be correct for 0x01 & 0x00
if (size < 59) { if (size < 59) {
duration = data[offset + 0x12] | (data[offset + 0x13] << 8); duration = data[offset + 0x12] | (data[offset + 0x13] << 8);
duration *= 2; duration *= 2;
@ -784,10 +814,10 @@ bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp,
session->setPhysMax(CPAP_PS, 25); session->setPhysMax(CPAP_PS, 25);
session->setPhysMin(CPAP_PS, 0); session->setPhysMin(CPAP_PS, 0);
new_sessions[sequence] = session; new_sessions[sequence] = session;
return true; return true;
} }
bool PRS1Loader::Parse002v5(qint32 sequence, quint32 timestamp, unsigned char *buffer, bool PRS1Loader::Parse002v5(qint32 sequence, quint32 timestamp, unsigned char *buffer,
quint16 size) quint16 size)
{ {
@ -1524,6 +1554,7 @@ bool PRS1Loader::OpenFile(Machine *mach, QString filename)
family = header[4]; // == 5 family = header[4]; // == 5
familyVersion = header[5]; familyVersion = header[5];
ext = header[6]; ext = header[6];
sequence = (header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7]; sequence = (header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7];
timestamp = (header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11]; timestamp = (header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11];
@ -1611,6 +1642,7 @@ bool PRS1Loader::OpenFile(Machine *mach, QString filename)
Parse002(sequence, timestamp, data, datasize, family, familyVersion); Parse002(sequence, timestamp, data, datasize, family, familyVersion);
} }
} else if (ext == 5) { } else if (ext == 5) {
// too messy to do here..
//ParseWaveform(mach,sequence,timestamp,data,datasize,duration,num_signals,interleave,sample_format); //ParseWaveform(mach,sequence,timestamp,data,datasize,duration,num_signals,interleave,sample_format);
} }
} }
@ -1666,10 +1698,10 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
pos = 0x14 + (num_signals - 1) * 3; pos = 0x14 + (num_signals - 1) * 3;
vector<WaveHeaderList> whl; vector<WaveHeaderList> whl;
// add the in reverse... // add the signals to whl in reverse...
for (int i = 0; i < num_signals; i++) { for (int i = 0; i < num_signals; i++) {
quint16 interleave = m_buffer[pos] | m_buffer[pos + 1] << 8; quint16 interleave = m_buffer[pos] | m_buffer[pos + 1] << 8;
quint8 sample_format = m_buffer[pos + 2]; quint8 sample_format = m_buffer[pos + 2];
whl.push_back(WaveHeaderList(interleave, sample_format)); whl.push_back(WaveHeaderList(interleave, sample_format));
pos -= 3; pos -= 3;
} }
@ -1684,7 +1716,7 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
lasttimestamp = start; lasttimestamp = start;
duration = 0; duration = 0;
int corrupt = 0; int corrupt = 0;
char waveform[2][500000]; char waveform[2][500000]; // eek. That's a lot of Stack.
int wlength[2]; int wlength[2];
qint64 wdur[2]; qint64 wdur[2];
@ -1705,11 +1737,15 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
//QString MaskPressure="MaskPressure"; //QString MaskPressure="MaskPressure";
ChannelID wc[2] = {CPAP_FlowRate, CPAP_MaskPressure}; ChannelID wc[2] = {CPAP_FlowRate, CPAP_MaskPressure};
do { unsigned char sum8; // additive checksum
timestamp = m_buffer[pos + 0xb] | m_buffer[pos + 0xc] << 8 | m_buffer[pos + 0xd] << 16 |
m_buffer[pos + 0x0e] << 24;
unsigned char sum8 = 0;
do {
timestamp = m_buffer[pos + 0xb] | m_buffer[pos + 0xc] << 8 |
m_buffer[pos + 0xd] << 16 | m_buffer[pos + 0x0e] << 24;
// Calculate header checksum
sum8 = 0;
for (int i = 0; i < hl; i++) { sum8 += m_buffer[pos + i]; } for (int i = 0; i < hl; i++) { sum8 += m_buffer[pos + i]; }
if (m_buffer[pos + hl] != sum8) { if (m_buffer[pos + hl] != sum8) {
@ -1738,7 +1774,6 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
length = m_buffer[pos + 0x1] | m_buffer[pos + 0x2] << 8; // block length in bytes length = m_buffer[pos + 0x1] | m_buffer[pos + 0x2] << 8; // block length in bytes
duration = m_buffer[pos + 0xf] | m_buffer[pos + 0x10] << 8; // block duration in seconds duration = m_buffer[pos + 0xf] | m_buffer[pos + 0x10] << 8; // block duration in seconds
if (diff < 0) { if (diff < 0) {
qDebug() << "Padding waveform to keep sync" << block; qDebug() << "Padding waveform to keep sync" << block;
@ -1780,10 +1815,10 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
a->setMax(120); a->setMax(120);
a->setMin(-120); a->setMin(-120);
} else if (wc[i] == CPAP_MaskPressure) { } else if (wc[i] == CPAP_MaskPressure) {
// int v=ceil(a->Max()/5); //int v=ceil(a->Max()/5);
// a->setMax(v*5); //a->setMax(v*5);
// v=floor(a->Min()/5); //v=floor(a->Min()/5);
// a->setMin(v*5); //a->setMin(v*5);
} }
session->updateLast(start + (qint64(wdur[i]) * 1000L)); session->updateLast(start + (qint64(wdur[i]) * 1000L));
@ -1799,6 +1834,7 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
pos += hl + 1; pos += hl + 1;
//qDebug() << (duration*num_signals*whl[0].interleave) << duration; //qDebug() << (duration*num_signals*whl[0].interleave) << duration;
if (num_signals == 1) { // no interleave.. this is much quicker. if (num_signals == 1) { // no interleave.. this is much quicker.
int bs = duration * whl[0].interleave; int bs = duration * whl[0].interleave;
memcpy((char *) & (waveform[0])[wlength[0]], (char *)&m_buffer[pos], bs); memcpy((char *) & (waveform[0])[wlength[0]], (char *)&m_buffer[pos], bs);
@ -1854,11 +1890,11 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename)
void InitModelMap() void InitModelMap()
{ {
ModelMap[0x34] = "RemStar Pro with C-Flex+"; ModelMap[0x34] = "RemStar Pro with C-Flex+"; // 450/460P
ModelMap[0x35] = "RemStar Auto with A-Flex"; ModelMap[0x35] = "RemStar Auto with A-Flex"; // 550/560P
ModelMap[0x36] = "RemStar BiPAP Pro with Bi-Flex"; ModelMap[0x36] = "RemStar BiPAP Pro with Bi-Flex";
ModelMap[0x37] = "RemStar BiPAP Auto with Bi-Flex"; ModelMap[0x37] = "RemStar BiPAP Auto with Bi-Flex";
ModelMap[0x38] = "RemStar Plus :("; ModelMap[0x38] = "RemStar Plus :("; // 150/250P/260P
ModelMap[0x41] = "BiPAP autoSV Advanced"; ModelMap[0x41] = "BiPAP autoSV Advanced";
} }

View File

@ -382,6 +382,8 @@ bool Machine::Save()
if (!PROFILE.session->multithreading()) { if (!PROFILE.session->multithreading()) {
for (int i = 0; i < savelistSize; i++) { for (int i = 0; i < savelistSize; i++) {
// Update progress bar
if ((i % 10) == 0) { if ((i % 10) == 0) {
qprogress->setValue(0 + (float(savelistCnt) / float(savelistSize) * 100.0)); qprogress->setValue(0 + (float(savelistCnt) / float(savelistSize) * 100.0));
QApplication::processEvents(); QApplication::processEvents();

View File

@ -872,7 +872,7 @@ void MainWindow::on_action_About_triggered()
// Credits section // Credits section
"<hr/><p><b><font size='+1'>" +tr("Kudos & Credits") + "</font></b></p><b>" + "<hr/><p><b><font size='+1'>" +tr("Kudos & Credits") + "</font></b></p><b>" +
tr("Bugfixes, Patches and Platform Help:") + "</b> " + tr("Bugfixes, Patches and Platform Help:") + "</b> " +
tr("James Marshall, Rich Freeman, John Masters, Patricia Shanahan, Alec Clews, manders99, and Sean Stangl.") tr("James Marshall, Rich Freeman, John Masters, Keary Griffin, Patricia Shanahan, Alec Clews, manders99, and Sean Stangl.")
+ "</p>" + "</p>"
"<p><b>" + tr("Translators:") + "</b> " + tr("Arie Klerk (Dutch), Steffen Reitz (German).") + "<p><b>" + tr("Translators:") + "</b> " + tr("Arie Klerk (Dutch), Steffen Reitz (German).") +