diff --git a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp index 630d4617..6fbc5213 100644 --- a/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp +++ b/sleepyhead/SleepLib/loader_plugins/prs1_loader.cpp @@ -358,7 +358,7 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) QDir dir(path); 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); } - QList paths; + QStringList paths; for (int i = 0; i < flist.size(); i++) { QFileInfo fi = flist.at(i); filename = fi.fileName(); if ((filename[0].toLower() == 'p') && (isdigit(filename[1]))) { + // p0, p1, p2.. etc.. folders contain the session data paths.push_back(fi.canonicalFilePath()); } else if (filename.toLower() == "properties.txt") { ParseProperties(m, fi.canonicalFilePath()); @@ -393,13 +394,23 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) bool ok; int model = modelstr.toInt(&ok); - if (!ok || (model < 450)) { - QMessageBox::information(NULL, - QObject::tr("Non Data Capable Machine"), - 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 (ok) { + // Assumption is made here all PRS1 machines less than 450P are not data capable.. this could be wrong one day. + if ((model < 450)) { + QMessageBox::information(NULL, + QObject::tr("Non Data Capable Machine"), + 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; @@ -408,40 +419,43 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) int size = paths.size(); int cnt = 0; - new_sessions.clear(); + new_sessions.clear(); // this hash is used by OpenFile - for (QList::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); if (!dir.exists() || !dir.isReadable()) { continue; } flist = dir.entryInfoList(); + // Scan for individual session files for (int i = 0; i < flist.size(); i++) { QFileInfo fi = flist.at(i); QString ext_s = fi.fileName().section(".", -1); QString session_s = fi.fileName().section(".", 0, -2); ext = ext_s.toLong(&ok); - - if (!ok) { + if (!ok) { // not a numerical extension continue; } sid = session_s.toLong(&ok); - - if (!ok) { + if (!ok) { // not a numerical session ID continue; } 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)) { - 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 { - 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; size = new_sessions.size(); - for (QHash::iterator it = new_sessions.begin(); it != new_sessions.end(); - it++) { + // Scan through new sessions and parse event and waveform data + // NOTE: this function could be multithreaded. + for (auto it = new_sessions.begin(); it != new_sessions.end(); ++it) { sid = it.key(); if (sessfiles.contains(sid)) { @@ -458,8 +473,10 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) QString name = sessfiles[sid][j]; if (name.endsWith(".002")) { + // Parse event data OpenFile(m, name); } else if (name.endsWith(".005")) { + // Parse Flow Rate Waveform (and mask pressure if available) data OpenWaveforms(sid, name); } } @@ -467,6 +484,8 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) // 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 (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 cnt = 0; - //QVector KillList; - for (QHash::iterator it = new_sessions.begin(); it != new_sessions.end(); - it++) { + // Scan through parsed sessions, do a little cleanup and add to machine object + for (auto it = new_sessions.begin(); it != new_sessions.end(); ++it) { Session *sess = it.value(); if ((sess->length()) == 0) { @@ -489,7 +507,6 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) SessionID id = sess->session(); delete sess; new_sessions[id] = nullptr; - //KillList.push_back(id); continue; } @@ -540,9 +557,13 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) //sess->settings[CPAP_PressureMax]=sess->settings[CPAP_PressureMin]; } + // set session saved flag so machine::Save() knows it needs saving sess->SetChanged(true); + + // Add session to machine record m->AddSession(sess, profile); + // Update the progress bar if ((++cnt % 10) == 0) { //if (qprogress) qprogress->setValue(33.0+(float(cnt)/float(size)*33.0)); 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_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); } + // Return number of newly added sessions return cnt; } @@ -572,6 +597,11 @@ int PRS1Loader::OpenMachine(Machine *m, QString path, Profile *profile) //};// __attribute__((packed)); +// Family 0 = xPAP +// Family 5 = ASV + + + bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, unsigned char *data, quint16 size, int family, int familyVersion) { @@ -631,8 +661,8 @@ bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, offset = 2; } - session->settings[CPAP_RampTime] = (int)data[offset + - 0x06]; // Minutes. Convert to seconds/hours here? + // Minutes. Convert to seconds/hours here? + session->settings[CPAP_RampTime] = (int)data[offset + 0x06]; session->settings[CPAP_RampPressure] = (EventDataType)data[offset + 0x07] / 10.0; 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; - // up to this point appears to becorrect for 0x01 & 0x00 + // up to this point appears to be correct for 0x01 & 0x00 if (size < 59) { duration = data[offset + 0x12] | (data[offset + 0x13] << 8); duration *= 2; @@ -784,10 +814,10 @@ bool PRS1Loader::ParseSummary(Machine *mach, qint32 sequence, quint32 timestamp, session->setPhysMax(CPAP_PS, 25); session->setPhysMin(CPAP_PS, 0); - new_sessions[sequence] = session; return true; } + bool PRS1Loader::Parse002v5(qint32 sequence, quint32 timestamp, unsigned char *buffer, quint16 size) { @@ -1524,6 +1554,7 @@ bool PRS1Loader::OpenFile(Machine *mach, QString filename) family = header[4]; // == 5 familyVersion = header[5]; ext = header[6]; + sequence = (header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7]; 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); } } else if (ext == 5) { + // too messy to do here.. //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; vector whl; - // add the in reverse... + // add the signals to whl in reverse... for (int i = 0; i < num_signals; i++) { 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)); pos -= 3; } @@ -1684,7 +1716,7 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) lasttimestamp = start; duration = 0; int corrupt = 0; - char waveform[2][500000]; + char waveform[2][500000]; // eek. That's a lot of Stack. int wlength[2]; qint64 wdur[2]; @@ -1705,11 +1737,15 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) //QString MaskPressure="MaskPressure"; ChannelID wc[2] = {CPAP_FlowRate, CPAP_MaskPressure}; - do { - timestamp = m_buffer[pos + 0xb] | m_buffer[pos + 0xc] << 8 | m_buffer[pos + 0xd] << 16 | - m_buffer[pos + 0x0e] << 24; - unsigned char sum8 = 0; + unsigned char sum8; // additive checksum + 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]; } 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 duration = m_buffer[pos + 0xf] | m_buffer[pos + 0x10] << 8; // block duration in seconds - if (diff < 0) { qDebug() << "Padding waveform to keep sync" << block; @@ -1780,10 +1815,10 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) a->setMax(120); a->setMin(-120); } else if (wc[i] == CPAP_MaskPressure) { - // int v=ceil(a->Max()/5); - // a->setMax(v*5); - // v=floor(a->Min()/5); - // a->setMin(v*5); + //int v=ceil(a->Max()/5); + //a->setMax(v*5); + //v=floor(a->Min()/5); + //a->setMin(v*5); } session->updateLast(start + (qint64(wdur[i]) * 1000L)); @@ -1799,6 +1834,7 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) pos += hl + 1; //qDebug() << (duration*num_signals*whl[0].interleave) << duration; + if (num_signals == 1) { // no interleave.. this is much quicker. int bs = duration * whl[0].interleave; memcpy((char *) & (waveform[0])[wlength[0]], (char *)&m_buffer[pos], bs); @@ -1854,11 +1890,11 @@ bool PRS1Loader::OpenWaveforms(SessionID sid, QString filename) void InitModelMap() { - ModelMap[0x34] = "RemStar Pro with C-Flex+"; - ModelMap[0x35] = "RemStar Auto with A-Flex"; + ModelMap[0x34] = "RemStar Pro with C-Flex+"; // 450/460P + ModelMap[0x35] = "RemStar Auto with A-Flex"; // 550/560P ModelMap[0x36] = "RemStar BiPAP Pro 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"; } diff --git a/sleepyhead/SleepLib/machine.cpp b/sleepyhead/SleepLib/machine.cpp index 601608ab..ff5ecc1a 100644 --- a/sleepyhead/SleepLib/machine.cpp +++ b/sleepyhead/SleepLib/machine.cpp @@ -382,6 +382,8 @@ bool Machine::Save() if (!PROFILE.session->multithreading()) { for (int i = 0; i < savelistSize; i++) { + + // Update progress bar if ((i % 10) == 0) { qprogress->setValue(0 + (float(savelistCnt) / float(savelistSize) * 100.0)); QApplication::processEvents(); diff --git a/sleepyhead/mainwindow.cpp b/sleepyhead/mainwindow.cpp index c6d03621..5704c955 100644 --- a/sleepyhead/mainwindow.cpp +++ b/sleepyhead/mainwindow.cpp @@ -872,7 +872,7 @@ void MainWindow::on_action_About_triggered() // Credits section "

" +tr("Kudos & Credits") + "

" + tr("Bugfixes, Patches and Platform Help:") + " " + - 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.") + "

" "

" + tr("Translators:") + " " + tr("Arie Klerk (Dutch), Steffen Reitz (German).") +