From eac13160b99a21af92e66967975e298411836c16 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Mon, 6 Dec 2021 16:14:11 -0500 Subject: [PATCH] Cache DS2 keys during import for a 3x speedup. Crypto overhead now makes import take only twice as long, instead of of 22x before optimization. --- oscar/SleepLib/loader_plugins/prs1_loader.cpp | 21 ++++++++++++++----- oscar/SleepLib/loader_plugins/prs1_loader.h | 3 +++ 2 files changed, 19 insertions(+), 5 deletions(-) diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.cpp b/oscar/SleepLib/loader_plugins/prs1_loader.cpp index 17f2c5fd..d305c57b 100644 --- a/oscar/SleepLib/loader_plugins/prs1_loader.cpp +++ b/oscar/SleepLib/loader_plugins/prs1_loader.cpp @@ -254,7 +254,7 @@ const char* PRS1ModelInfo::Name(const QString & model) const class PRDS2File : public RawDataFile { public: - PRDS2File(class QFile & file); + PRDS2File(class QFile & file, QHash & keycache); virtual ~PRDS2File() {}; bool isValid() const; QString guid() const; @@ -282,12 +282,21 @@ class PRDS2File : public RawDataFile static const int m_header_size = 0xCA; }; -PRDS2File::PRDS2File(class QFile & file) +PRDS2File::PRDS2File(class QFile & file, QHash & keycache) : RawDataFile(file) { bool valid = parseDS2Header(); if (valid) { - valid = initializeKey(); + QByteArray key = m_iv + m_salt + m_export_key + m_export_key_tag; + m_payload_key = keycache[key]; + if (m_payload_key.isEmpty()) { + // Derive the key (slow). + valid = initializeKey(); + if (valid) { + // Cache the result for the next file. + keycache[key] = m_payload_key; + } + } if (valid) { valid = decryptData(); } @@ -668,7 +677,7 @@ bool PRS1Loader::PeekProperties(const QString & filename, QHash RawDataFile* src; if (QFileInfo(f).suffix().toUpper() == "BIN") { // If it's a DS2 file, insert the DS2 wrapper to decode the chunk stream. - PRDS2File* ds2 = new PRDS2File(f); + PRDS2File* ds2 = new PRDS2File(f, m_keycache); if (!ds2->isValid()) { //qWarning() << filename << "unable to decrypt"; delete ds2; @@ -914,6 +923,8 @@ int PRS1Loader::FindSessionDirsAndProperties(const QString & path, QStringList & bool PRS1Loader::CreateMachineFromProperties(QString propertyfile) { + m_keycache.clear(); + MachineInfo info = newInfo(); QHash props; if (!PeekProperties(propertyfile, props) || !s_PRS1ModelInfo.IsSupported(props)) { @@ -2720,7 +2731,7 @@ QList PRS1Loader::ParseFile(const QString & path) RawDataFile* src; if (QFileInfo(f).suffix().toUpper().startsWith("B")) { // .B01, .B02, etc. // If it's a DS2 file, insert the DS2 wrapper to decode the chunk stream. - PRDS2File* ds2 = new PRDS2File(f); + PRDS2File* ds2 = new PRDS2File(f, m_keycache); if (!ds2->isValid()) { //qWarning() << path << "unable to decrypt"; delete ds2; diff --git a/oscar/SleepLib/loader_plugins/prs1_loader.h b/oscar/SleepLib/loader_plugins/prs1_loader.h index c8d91aaf..4395c61d 100644 --- a/oscar/SleepLib/loader_plugins/prs1_loader.h +++ b/oscar/SleepLib/loader_plugins/prs1_loader.h @@ -228,6 +228,9 @@ class PRS1Loader : public CPAPLoader //! \brief PRS1 Data files can store multiple sessions, so store them in this list for later processing. QHash new_sessions; + + //! \brief DS2 key derivation is very slow, but keys are reused in multiple files, so we cache the derived keys. + QHash m_keycache; };