2019-05-03 18:45:21 +00:00
/* SleepLib PRS1 Loader Implementation
2014-04-09 21:01:57 +00:00
*
2023-09-19 00:27:37 +00:00
* Copyright ( c ) 2019 - 2023 The OSCAR Team
2018-03-28 07:10:52 +00:00
* Copyright ( c ) 2011 - 2018 Mark Watkins < mark @ jedimark . net >
2014-04-09 21:01:57 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
2018-06-04 23:26:46 +00:00
* License . See the file COPYING in the main directory of the source code
* for more details . */
2011-06-26 08:30:44 +00:00
2023-02-18 13:58:47 +00:00
# define TEST_MACROS_ENABLEDoff
# include <test_macros.h>
2011-07-30 00:36:31 +00:00
# include <QApplication>
2011-06-26 08:30:44 +00:00
# include <QString>
# include <QDateTime>
# include <QDir>
# include <QFile>
2021-11-03 17:13:44 +00:00
# include <QBuffer>
2014-05-31 21:25:07 +00:00
# include <QDataStream>
2011-06-26 08:30:44 +00:00
# include <QMessageBox>
2011-07-01 10:10:44 +00:00
# include <QDebug>
2011-07-27 09:21:53 +00:00
# include <cmath>
2018-05-07 18:42:23 +00:00
2011-09-17 12:39:00 +00:00
# include "SleepLib/schema.h"
2021-09-01 19:31:06 +00:00
# include "SleepLib/importcontext.h"
2011-06-26 08:30:44 +00:00
# include "prs1_loader.h"
2021-05-31 17:18:39 +00:00
# include "prs1_parser.h"
2011-06-26 08:30:44 +00:00
# include "SleepLib/session.h"
2011-12-10 12:14:48 +00:00
# include "SleepLib/calcs.h"
2021-12-03 21:47:06 +00:00
# include "SleepLib/crypto.h"
2021-05-23 16:43:31 +00:00
# include "rawdata.h"
2011-06-26 08:30:44 +00:00
2011-07-10 14:23:07 +00:00
2018-05-05 07:14:44 +00:00
// Disable this to cut excess debug messages
2018-05-05 15:48:32 +00:00
# define DEBUG_SUMMARY
2018-05-05 07:14:44 +00:00
2014-06-02 11:22:45 +00:00
//const int PRS1_MAGIC_NUMBER = 2;
2014-04-17 04:56:04 +00:00
//const int PRS1_SUMMARY_FILE=1;
//const int PRS1_EVENT_FILE=2;
//const int PRS1_WAVEFORM_FILE=5;
2011-07-10 14:23:07 +00:00
2011-06-26 08:30:44 +00:00
//********************************************************************************************
/// IMPORTANT!!!
//********************************************************************************************
// Please INCREMENT the prs1_data_version in prs1_loader.h when making changes to this loader
// that change loader behaviour or modify channels.
//********************************************************************************************
2021-06-01 00:24:09 +00:00
QString ts ( qint64 msecs )
2019-09-30 14:23:28 +00:00
{
// TODO: make this UTC so that tests don't vary by where they're run
return QDateTime : : fromMSecsSinceEpoch ( msecs ) . toString ( Qt : : ISODate ) ;
}
2020-03-24 21:15:29 +00:00
ChannelID PRS1_Mode = 0 ;
2020-01-12 00:14:01 +00:00
ChannelID PRS1_TimedBreath = 0 , PRS1_HumidMode = 0 , PRS1_TubeTemp = 0 ;
2020-03-23 16:59:06 +00:00
ChannelID PRS1_FlexLock = 0 , PRS1_TubeLock = 0 , PRS1_RampType = 0 ;
2020-03-24 01:09:55 +00:00
ChannelID PRS1_BackupBreathMode = 0 , PRS1_BackupBreathRate = 0 , PRS1_BackupBreathTi = 0 ;
ChannelID PRS1_AutoTrial = 0 , PRS1_EZStart = 0 , PRS1_RiseTime = 0 , PRS1_RiseTimeLock = 0 ;
2020-04-22 20:52:21 +00:00
ChannelID PRS1_PeakFlow = 0 ;
2020-03-26 13:01:28 +00:00
ChannelID PRS1_VariableBreathing = 0 ; // TODO: UNCONFIRMED, but seems to match sample data
2014-08-03 13:00:13 +00:00
2020-03-24 21:15:29 +00:00
QString PRS1Loader : : PresReliefLabel ( ) { return QObject : : tr ( " " ) ; }
ChannelID PRS1Loader : : PresReliefMode ( ) { return PRS1_FlexMode ; }
ChannelID PRS1Loader : : PresReliefLevel ( ) { return PRS1_FlexLevel ; }
ChannelID PRS1Loader : : CPAPModeChannel ( ) { return PRS1_Mode ; }
ChannelID PRS1Loader : : HumidifierConnected ( ) { return PRS1_HumidStatus ; }
ChannelID PRS1Loader : : HumidifierLevel ( ) { return PRS1_HumidLevel ; }
2019-05-27 14:05:16 +00:00
struct PRS1TestedModel
{
QString model ;
int family ;
int familyVersion ;
2019-06-20 04:09:28 +00:00
const char * name ;
2019-05-27 14:05:16 +00:00
} ;
2011-06-26 08:30:44 +00:00
2019-05-27 14:05:16 +00:00
static const PRS1TestedModel s_PRS1TestedModels [ ] = {
2019-06-20 03:23:15 +00:00
// This first set says "(Philips Respironics)" intead of "(System One)" on official reports.
2019-06-20 04:09:28 +00:00
{ " 251P " , 0 , 2 , " REMstar Plus (System One) " } , // (brick)
2020-09-12 20:54:29 +00:00
{ " 450P " , 0 , 2 , " REMstar Pro (System One) " } ,
2019-06-20 04:09:28 +00:00
{ " 450P " , 0 , 3 , " REMstar Pro (System One) " } ,
2020-03-09 17:13:33 +00:00
{ " 451P " , 0 , 2 , " REMstar Pro (System One) " } ,
2019-06-20 04:09:28 +00:00
{ " 451P " , 0 , 3 , " REMstar Pro (System One) " } ,
2020-08-17 01:19:11 +00:00
{ " 452P " , 0 , 3 , " REMstar Pro (System One) " } ,
2019-06-20 04:09:28 +00:00
{ " 550P " , 0 , 2 , " REMstar Auto (System One) " } ,
{ " 550P " , 0 , 3 , " REMstar Auto (System One) " } ,
{ " 551P " , 0 , 2 , " REMstar Auto (System One) " } ,
2022-05-02 20:52:19 +00:00
{ " 552P " , 0 , 3 , " REMstar Auto (System One) " } ,
2020-08-17 01:19:11 +00:00
{ " 650P " , 0 , 2 , " BiPAP Pro (System One) " } ,
2019-06-20 04:09:28 +00:00
{ " 750P " , 0 , 2 , " BiPAP Auto (System One) " } ,
2020-08-05 00:59:51 +00:00
{ " 261CA " , 0 , 4 , " REMstar Plus (System One 60 Series) " } , // (brick)
2021-11-29 15:07:11 +00:00
{ " 261P " , 0 , 4 , " REMstar Plus (System One 60 Series) " } , // (brick)
2019-06-20 04:09:28 +00:00
{ " 460P " , 0 , 4 , " REMstar Pro (System One 60 Series) " } ,
2021-11-11 20:40:20 +00:00
{ " 460PBT " , 0 , 4 , " REMstar Pro (System One 60 Series) " } , // evidently built-in bluetooth
2019-06-20 04:09:28 +00:00
{ " 461P " , 0 , 4 , " REMstar Pro (System One 60 Series) " } ,
2020-08-03 18:18:25 +00:00
{ " 462P " , 0 , 4 , " REMstar Pro (System One 60 Series) " } ,
2020-05-12 18:08:24 +00:00
{ " 461CA " , 0 , 4 , " REMstar Pro (System One 60 Series) " } ,
2019-06-20 04:09:28 +00:00
{ " 560P " , 0 , 4 , " REMstar Auto (System One 60 Series) " } ,
{ " 560PBT " , 0 , 4 , " REMstar Auto (System One 60 Series) " } ,
{ " 561P " , 0 , 4 , " REMstar Auto (System One 60 Series) " } ,
2019-08-06 01:46:05 +00:00
{ " 562P " , 0 , 4 , " REMstar Auto (System One 60 Series) " } ,
2019-06-20 04:09:28 +00:00
{ " 660P " , 0 , 4 , " BiPAP Pro (System One 60 Series) " } ,
{ " 760P " , 0 , 4 , " BiPAP Auto (System One 60 Series) " } ,
2022-02-20 19:37:05 +00:00
{ " 761P " , 0 , 4 , " BiPAP Auto (System One 60 Series) " } ,
2019-05-27 14:05:16 +00:00
2020-09-13 00:12:31 +00:00
{ " 501V " , 0 , 5 , " Dorma 500 Auto (System One 60 Series) " } , // (brick)
2019-06-20 04:09:28 +00:00
{ " 200X110 " , 0 , 6 , " DreamStation CPAP " } , // (brick)
{ " 400G110 " , 0 , 6 , " DreamStation Go " } ,
{ " 400X110 " , 0 , 6 , " DreamStation CPAP Pro " } ,
2020-08-03 18:18:25 +00:00
{ " 400X120 " , 0 , 6 , " DreamStation CPAP Pro " } ,
2020-05-29 16:38:39 +00:00
{ " 400X130 " , 0 , 6 , " DreamStation CPAP Pro " } ,
2019-06-20 04:09:28 +00:00
{ " 400X150 " , 0 , 6 , " DreamStation CPAP Pro " } ,
2021-12-10 18:43:03 +00:00
{ " 401X150 " , 0 , 6 , " DreamStation CPAP Pro with Auto-Trial " } ,
2019-06-20 04:09:28 +00:00
{ " 500X110 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2020-04-04 21:26:50 +00:00
{ " 500X120 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2020-01-28 18:04:36 +00:00
{ " 500X130 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2020-12-29 22:38:00 +00:00
{ " 500X140 " , 0 , 6 , " DreamStation Auto CPAP with A-Flex " } ,
2019-06-20 04:09:28 +00:00
{ " 500X150 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2020-04-10 20:48:28 +00:00
{ " 500X180 " , 0 , 6 , " DreamStation Auto CPAP " } ,
2019-12-04 00:04:09 +00:00
{ " 501X120 " , 0 , 6 , " DreamStation Auto CPAP with P-Flex " } ,
2019-09-07 19:02:16 +00:00
{ " 500G110 " , 0 , 6 , " DreamStation Go Auto " } ,
2020-12-29 22:38:00 +00:00
{ " 500G120 " , 0 , 6 , " DreamStation Go Auto " } ,
2023-09-19 00:27:37 +00:00
{ " 500G150 " , 0 , 6 , " DreamStation Go Auto " } ,
2019-06-20 04:09:28 +00:00
{ " 502G150 " , 0 , 6 , " DreamStation Go Auto " } ,
{ " 600X110 " , 0 , 6 , " DreamStation BiPAP Pro " } ,
2020-04-21 00:24:58 +00:00
{ " 600X150 " , 0 , 6 , " DreamStation BiPAP Pro " } ,
2019-06-20 04:09:28 +00:00
{ " 700X110 " , 0 , 6 , " DreamStation Auto BiPAP " } ,
2020-05-12 18:08:24 +00:00
{ " 700X120 " , 0 , 6 , " DreamStation Auto BiPAP " } ,
2021-05-13 14:15:57 +00:00
{ " 700X130 " , 0 , 6 , " DreamStation Auto BiPAP " } ,
2020-04-21 00:24:58 +00:00
{ " 700X150 " , 0 , 6 , " DreamStation Auto BiPAP " } ,
2019-05-27 14:05:16 +00:00
2022-04-24 21:29:19 +00:00
{ " 410X150C " , 0 , 6 , " DreamStation 2 CPAP " } ,
2022-05-16 18:42:27 +00:00
{ " 420X150C " , 0 , 6 , " DreamStation 2 Advanced CPAP " } , // from FDA filing
2021-11-04 20:40:44 +00:00
{ " 520X110C " , 0 , 6 , " DreamStation 2 Auto CPAP Advanced " } , // based on bottom label, boot screen says "Advanced Auto CPAP"
2022-05-02 00:00:30 +00:00
{ " 520X130C " , 0 , 6 , " DreamStation 2 Auto CPAP Advanced " } , // from user report
2022-01-26 16:11:24 +00:00
{ " 520X150C " , 0 , 6 , " DreamStation 2 Auto CPAP Advanced " } , // from user report
2022-02-17 00:04:17 +00:00
{ " 521X120C " , 0 , 6 , " DreamStation 2 Auto CPAP Advanced with P-Flex " } , // inferred from 501X120 and presence of "P-Flex" on bottom label
2022-05-09 14:40:49 +00:00
{ " 521X140C " , 0 , 6 , " DreamStation 2 avec GSM + Humidificateur " } , // from brochure
2021-11-03 20:05:26 +00:00
2019-06-20 04:09:28 +00:00
{ " 950P " , 5 , 0 , " BiPAP AutoSV Advanced System One " } ,
2019-11-20 17:33:45 +00:00
{ " 951P " , 5 , 0 , " BiPAP AutoSV Advanced System One " } ,
2019-06-20 04:09:28 +00:00
{ " 960P " , 5 , 1 , " BiPAP autoSV Advanced (System One 60 Series) " } ,
{ " 961P " , 5 , 1 , " BiPAP autoSV Advanced (System One 60 Series) " } ,
{ " 960T " , 5 , 2 , " BiPAP autoSV Advanced 30 (System One 60 Series) " } , // omits "(System One 60 Series)" on official reports
2022-04-24 21:02:25 +00:00
{ " 961TCA " , 5 , 2 , " BiPAP autoSV Advanced 30 (System One 60 Series) " } ,
2019-06-20 04:09:28 +00:00
{ " 900X110 " , 5 , 3 , " DreamStation BiPAP autoSV " } ,
{ " 900X120 " , 5 , 3 , " DreamStation BiPAP autoSV " } ,
2020-06-26 16:53:52 +00:00
{ " 900X150 " , 5 , 3 , " DreamStation BiPAP autoSV " } ,
2019-05-27 14:05:16 +00:00
2020-04-22 00:49:15 +00:00
{ " 1061401 " , 3 , 0 , " BiPAP S/T (C Series) " } ,
2019-06-20 04:09:28 +00:00
{ " 1061T " , 3 , 3 , " BiPAP S/T 30 (System One 60 Series) " } ,
{ " 1160P " , 3 , 3 , " BiPAP AVAPS 30 (System One 60 Series) " } ,
{ " 1030X110 " , 3 , 6 , " DreamStation BiPAP S/T 30 " } ,
2019-11-13 16:39:02 +00:00
{ " 1030X150 " , 3 , 6 , " DreamStation BiPAP S/T 30 with AAM " } ,
2019-06-20 04:09:28 +00:00
{ " 1130X110 " , 3 , 6 , " DreamStation BiPAP AVAPS 30 " } ,
2019-10-04 16:20:18 +00:00
{ " 1131X150 " , 3 , 6 , " DreamStation BiPAP AVAPS 30 AE " } ,
2021-03-04 22:34:05 +00:00
{ " 1130X200 " , 3 , 6 , " DreamStation BiPAP AVAPS 30 " } ,
2019-05-27 14:05:16 +00:00
2019-06-20 02:19:16 +00:00
{ " " , 0 , 0 , " " } ,
2011-12-19 05:35:05 +00:00
} ;
2019-05-27 14:05:16 +00:00
PRS1ModelInfo s_PRS1ModelInfo ;
PRS1ModelInfo : : PRS1ModelInfo ( )
{
for ( int i = 0 ; ! s_PRS1TestedModels [ i ] . model . isEmpty ( ) ; i + + ) {
const PRS1TestedModel & model = s_PRS1TestedModels [ i ] ;
m_testedModels [ model . family ] [ model . familyVersion ] . append ( model . model ) ;
2019-06-20 02:19:16 +00:00
m_modelNames [ model . model ] = model . name ;
2019-05-27 14:05:16 +00:00
}
2019-06-20 02:19:16 +00:00
2021-11-29 15:07:11 +00:00
m_bricks = { " 251P " , " 261CA " , " 261P " , " 200X110 " , " 501V " } ;
2019-05-27 14:05:16 +00:00
}
bool PRS1ModelInfo : : IsSupported ( int family , int familyVersion ) const
{
if ( m_testedModels . value ( family ) . contains ( familyVersion ) ) {
return true ;
}
return false ;
}
bool PRS1ModelInfo : : IsTested ( const QString & model , int family , int familyVersion ) const
{
if ( m_testedModels . value ( family ) . value ( familyVersion ) . contains ( model ) ) {
return true ;
}
2019-06-03 02:05:10 +00:00
// Some 500X150 C0/C1 folders have contained this bogus model number in their PROP.TXT file,
// with the same serial number seen in the main PROP.TXT file that shows the real model number.
if ( model = = " 100X100 " ) {
2019-06-05 03:19:35 +00:00
# ifndef UNITTEST_MODE
2019-06-03 02:05:10 +00:00
qDebug ( ) < < " Ignoring 100X100 for untested alert " ;
2019-06-05 03:19:35 +00:00
# endif
2019-06-03 02:05:10 +00:00
return true ;
}
2019-05-27 14:05:16 +00:00
return false ;
} ;
2021-05-23 19:32:56 +00:00
static bool getVersionFromProps ( const QHash < QString , QString > & props , int & family , int & familyVersion )
2019-05-27 14:05:16 +00:00
{
bool ok ;
2021-05-23 19:32:56 +00:00
family = props [ " Family " ] . toInt ( & ok , 10 ) ;
2019-05-27 14:05:16 +00:00
if ( ok ) {
2021-05-23 19:32:56 +00:00
familyVersion = props [ " FamilyVersion " ] . toInt ( & ok , 10 ) ;
}
return ok ;
}
bool PRS1ModelInfo : : IsSupported ( const QHash < QString , QString > & props ) const
{
int family , familyVersion ;
bool ok = getVersionFromProps ( props , family , familyVersion ) ;
if ( ok ) {
ok = IsSupported ( family , familyVersion ) ;
2019-05-27 14:05:16 +00:00
}
return ok ;
}
bool PRS1ModelInfo : : IsTested ( const QHash < QString , QString > & props ) const
{
2021-05-23 19:32:56 +00:00
int family , familyVersion ;
bool ok = getVersionFromProps ( props , family , familyVersion ) ;
2019-05-27 14:05:16 +00:00
if ( ok ) {
2021-05-23 19:32:56 +00:00
ok = IsTested ( props [ " ModelNumber " ] , family , familyVersion ) ;
2019-05-27 14:05:16 +00:00
}
return ok ;
} ;
2019-06-20 02:19:16 +00:00
bool PRS1ModelInfo : : IsBrick ( const QString & model ) const
{
2021-05-23 19:02:01 +00:00
bool is_brick = false ;
2019-06-20 02:19:16 +00:00
if ( m_modelNames . contains ( model ) ) {
is_brick = m_bricks . contains ( model ) ;
2021-05-23 19:02:01 +00:00
} else if ( model . length ( ) > 0 ) {
2019-06-20 02:19:16 +00:00
// If we haven't seen it before, assume any 2xx is a brick.
is_brick = ( model . at ( 0 ) = = QChar ( ' 2 ' ) ) ;
}
return is_brick ;
} ;
2019-06-20 04:09:28 +00:00
const char * PRS1ModelInfo : : Name ( const QString & model ) const
2019-06-20 02:19:16 +00:00
{
2019-06-20 04:09:28 +00:00
const char * name ;
2019-06-20 02:19:16 +00:00
if ( m_modelNames . contains ( model ) ) {
name = m_modelNames [ model ] ;
} else {
2019-06-20 04:09:28 +00:00
name = " Unknown Model " ;
2019-06-20 02:19:16 +00:00
}
return name ;
} ;
2011-12-19 05:35:05 +00:00
2021-05-23 16:43:31 +00:00
//********************************************************************************************
// Decoder for DreamStation 2 files, which encrypt the actual data after a header with the key.
// The public read/seek/pos/etc. functions are all in terms of the decoded stream.
class PRDS2File : public RawDataFile
{
public :
2021-12-06 21:14:11 +00:00
PRDS2File ( class QFile & file , QHash < QByteArray , QByteArray > & keycache ) ;
2021-05-23 16:43:31 +00:00
virtual ~ PRDS2File ( ) { } ;
2021-11-03 17:13:44 +00:00
bool isValid ( ) const ;
2021-11-29 15:25:52 +00:00
QString guid ( ) const ;
2021-05-23 16:43:31 +00:00
private :
2021-11-03 17:13:44 +00:00
bool parseDS2Header ( ) ;
2021-05-23 16:43:31 +00:00
int read16 ( ) ;
QByteArray readBytes ( ) ;
2021-11-03 17:13:44 +00:00
bool initializeKey ( ) ;
bool decryptData ( ) ;
QByteArray m_iv ;
2021-12-03 14:57:29 +00:00
QByteArray m_salt ;
QByteArray m_export_key ;
QByteArray m_export_key_tag ;
2021-11-03 17:13:44 +00:00
QByteArray m_payload_key ;
QByteArray m_payload_tag ;
QBuffer m_payload ;
bool m_valid ;
2021-05-23 16:43:31 +00:00
protected :
virtual qint64 readData ( char * data , qint64 maxSize ) ;
virtual bool seek ( qint64 pos ) ;
virtual qint64 pos ( ) const ;
virtual qint64 size ( ) const ;
QByteArray m_guid ;
static const int m_header_size = 0xCA ;
} ;
2021-12-06 21:14:11 +00:00
PRDS2File : : PRDS2File ( class QFile & file , QHash < QByteArray , QByteArray > & keycache )
2021-05-23 16:43:31 +00:00
: RawDataFile ( file )
{
2021-11-03 17:13:44 +00:00
bool valid = parseDS2Header ( ) ;
if ( valid ) {
2021-12-06 21:14:11 +00:00
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 ;
}
}
2021-11-03 17:13:44 +00:00
if ( valid ) {
valid = decryptData ( ) ;
}
}
m_valid = valid ;
if ( m_valid ) {
seek ( 0 ) ; // initialize internal position
}
}
bool PRDS2File : : isValid ( ) const {
return m_valid ;
2021-05-23 16:43:31 +00:00
}
2021-11-29 15:25:52 +00:00
QString PRDS2File : : guid ( ) const {
QString guid ( m_guid ) ;
return guid ;
}
2021-05-23 16:43:31 +00:00
bool PRDS2File : : seek ( qint64 pos )
{
2021-11-03 17:13:44 +00:00
if ( ! m_valid ) {
qWarning ( ) < < " seeking in unsupported DS2 file " ;
return false ;
}
2021-05-23 16:43:31 +00:00
QIODevice : : seek ( pos ) ;
2021-11-03 17:13:44 +00:00
return m_payload . seek ( pos ) ;
2021-05-23 16:43:31 +00:00
}
qint64 PRDS2File : : pos ( ) const
{
2021-11-03 17:13:44 +00:00
if ( ! m_valid ) {
qWarning ( ) < < " querying pos in unsupported DS2 file " ;
return 0 ;
}
return m_payload . pos ( ) ;
2021-05-23 16:43:31 +00:00
}
qint64 PRDS2File : : size ( ) const
{
2021-11-03 17:13:44 +00:00
return m_payload . size ( ) ;
2021-05-23 16:43:31 +00:00
}
qint64 PRDS2File : : readData ( char * data , qint64 maxSize )
{
2021-11-03 17:13:44 +00:00
if ( ! m_valid ) {
qWarning ( ) < < " reading from unsupported DS2 file " ;
2021-05-25 19:57:09 +00:00
return - 1 ;
}
2021-11-03 17:13:44 +00:00
return m_payload . read ( data , maxSize ) ;
}
bool PRDS2File : : decryptData ( )
{
bool valid = false ;
2021-12-03 21:47:06 +00:00
QByteArray ciphertext = m_device . read ( m_device . size ( ) - m_device . pos ( ) ) ;
QByteArray plaintext ;
CryptoResult error = decrypt_aes256_gcm ( m_payload_key , m_iv , ciphertext , m_payload_tag , plaintext ) ;
if ( error ) {
if ( error = = InvalidTag ) {
2022-05-02 19:32:28 +00:00
static const QByteArray s_zero_tag ( 16 , 0 ) ;
if ( m_payload_tag = = s_zero_tag ) {
// This has been observed where the tag is zero and the data appears truncated.
// Decrypt and ignore the tag. Rely on the decrypted payload's CRC for integrity.
qWarning ( ) < < name ( ) < < " DS2 payload has zero tag, recovering data " ;
error = encrypt_aes256_gcm ( m_payload_key , m_iv , ciphertext , plaintext , m_payload_tag ) ;
if ( error ) {
qWarning ( ) < < " *** DS2 unexpected exception decrypting " < < name ( ) ;
}
} else {
qWarning ( ) < < name ( ) < < " DS2 payload doesn't match tag, skipping " ;
}
2021-12-03 21:47:06 +00:00
} else {
qWarning ( ) < < " *** DS2 unexpected exception decrypting " < < name ( ) ;
2021-05-25 19:57:09 +00:00
}
2022-05-02 19:32:28 +00:00
}
if ( ! error ) {
2021-12-03 21:47:06 +00:00
m_payload . setData ( plaintext ) ;
m_payload . open ( QIODevice : : ReadOnly ) ;
valid = true ;
2021-11-03 17:13:44 +00:00
}
return valid ;
2021-05-23 16:43:31 +00:00
}
2021-12-03 14:57:29 +00:00
static const int KEY_SIZE = 256 / 8 ; // AES-256
static const uint8_t OSCAR_KEY [ KEY_SIZE + 1 ] = " Patient access to their own data " ;
static const uint8_t COMMON_KEY [ KEY_SIZE ] = { 0x75 , 0xB3 , 0xA2 , 0x12 , 0x4A , 0x65 , 0xAF , 0x97 , 0x54 , 0xD8 , 0xC1 , 0xF3 , 0xE5 , 0x2E , 0xB6 , 0xF0 , 0x23 , 0x20 , 0x57 , 0x69 , 0x7E , 0x38 , 0x0E , 0xC9 , 0x4A , 0xDC , 0x46 , 0x45 , 0xB6 , 0x92 , 0x5A , 0x98 } ;
2021-12-03 21:47:06 +00:00
static const QByteArray s_oscar_key ( ( const char * ) OSCAR_KEY , KEY_SIZE ) ;
static const QByteArray s_common_key ( ( const char * ) COMMON_KEY , KEY_SIZE ) ;
2021-11-03 17:13:44 +00:00
bool PRDS2File : : initializeKey ( )
2021-05-25 19:57:09 +00:00
{
2021-11-03 17:13:44 +00:00
bool valid = false ;
2021-12-03 21:47:06 +00:00
QByteArray common_key ;
CryptoResult error = decrypt_aes256 ( s_oscar_key , s_common_key , common_key ) ;
if ( error ) {
qWarning ( ) < < " *** DS2 unexpected exception deriving common key " ;
return false ;
}
QByteArray salted_key ( KEY_SIZE , 0 ) ;
error = pbkdf2_sha256 ( common_key , m_salt , 10000 , salted_key ) ;
if ( error ) {
qWarning ( ) < < " *** DS2 unexpected exception deriving salted key for " < < name ( ) ;
return false ;
}
error = decrypt_aes256_gcm ( salted_key , m_iv , m_export_key , m_export_key_tag , m_payload_key ) ;
if ( error ) {
if ( error = = InvalidTag ) {
2021-12-03 14:57:29 +00:00
qWarning ( ) < < " DS2 validation of payload key failed for " < < name ( ) ;
2021-12-03 21:47:06 +00:00
} else {
qWarning ( ) < < " *** DS2 unexpected exception deriving key for " < < name ( ) ;
2021-12-03 14:57:29 +00:00
}
2021-12-03 21:47:06 +00:00
} else {
valid = true ;
2021-05-25 19:57:09 +00:00
}
2021-11-03 17:13:44 +00:00
return valid ;
2021-05-25 19:57:09 +00:00
}
2021-11-03 17:13:44 +00:00
bool PRDS2File : : parseDS2Header ( )
2021-05-23 16:43:31 +00:00
{
2021-12-09 17:23:08 +00:00
if ( m_device . size ( ) = = 0 ) {
qWarning ( ) < < name ( ) < < " is empty, skipping " ;
return false ;
}
2021-05-23 16:43:31 +00:00
int a = read16 ( ) ;
int b = read16 ( ) ;
int c = read16 ( ) ;
if ( a ! = 0x0D | | b ! = 1 | | c ! = 1 ) {
qWarning ( ) < < " DS2 unexpected first bytes = " < < a < < b < < c ;
2021-11-03 17:13:44 +00:00
return false ;
2021-05-23 16:43:31 +00:00
}
m_guid = readBytes ( ) ;
if ( m_guid . size ( ) ! = 36 ) {
qWarning ( ) < < " DS2 guid unexpected length " < < m_guid . size ( ) ;
} else {
2021-11-03 20:38:20 +00:00
//qDebug() << "DS2 guid {" << m_guid << "}";
2021-05-23 16:43:31 +00:00
}
2021-11-03 17:13:44 +00:00
m_iv = readBytes ( ) ; // 96-bit IV
2021-12-03 14:57:29 +00:00
m_salt = readBytes ( ) ; // 128-bit salt used to decrypt export key
if ( m_iv . size ( ) ! = 12 | | m_salt . size ( ) ! = 16 ) {
qWarning ( ) < < " DS2 IV,salt sizes = " < < m_iv . size ( ) < < m_salt . size ( ) ;
2021-05-23 16:43:31 +00:00
} else {
2021-12-03 14:57:29 +00:00
//qDebug() << "DS2 IV,salt =" << m_iv.toHex() << m_salt.toHex();
2021-05-23 16:43:31 +00:00
}
int f = read16 ( ) ;
int g = read16 ( ) ;
if ( f ! = 0 | | g ! = 1 ) {
qWarning ( ) < < " DS2 unexpected middle bytes = " < < f < < g ;
}
2022-02-27 16:50:10 +00:00
QByteArray import_key = readBytes ( ) ; // payload key encrypted with device-specific key
2021-12-03 14:57:29 +00:00
QByteArray import_key_tag = readBytes ( ) ; // tag of import key
if ( import_key . size ( ) ! = 32 | | import_key_tag . size ( ) ! = 16 ) {
qWarning ( ) < < " DS2 import_key sizes = " < < import_key . size ( ) < < import_key_tag . size ( ) ;
2021-05-23 16:43:31 +00:00
} else {
2021-12-03 14:57:29 +00:00
//qDebug() << "DS2 import_key,tag =" << import_key.toHex() << import_key_tag.toHex();
2021-05-23 16:43:31 +00:00
}
2021-12-03 14:57:29 +00:00
m_export_key = readBytes ( ) ; // payload key encrypted with salted common key
m_export_key_tag = readBytes ( ) ; // tag of export key
if ( m_export_key . size ( ) ! = 32 | | m_export_key_tag . size ( ) ! = 16 ) {
qWarning ( ) < < " DS2 export_key sizes = " < < m_export_key . size ( ) < < m_export_key_tag . size ( ) ;
2021-05-23 16:43:31 +00:00
} else {
2021-12-03 14:57:29 +00:00
//qDebug() << "DS2 export_key,tag =" << m_export_key.toHex() << m_export_key_tag.toHex();
2021-05-23 16:43:31 +00:00
}
2021-11-03 17:13:44 +00:00
m_payload_tag = readBytes ( ) ;
if ( m_payload_tag . size ( ) ! = 16 ) {
qWarning ( ) < < " DS2 payload tag size = " < < m_payload_tag . size ( ) ;
2021-05-23 16:43:31 +00:00
} else {
2021-11-03 20:38:20 +00:00
//qDebug() << "DS2 payload tag =" << m_payload_tag.toHex();
2021-05-23 16:43:31 +00:00
}
2021-05-25 19:57:09 +00:00
if ( m_device . pos ( ) ! = m_header_size ) {
2021-05-23 16:43:31 +00:00
qWarning ( ) < < " DS2 header size != " < < m_header_size ;
}
2021-11-03 17:13:44 +00:00
return true ;
2021-05-23 16:43:31 +00:00
}
int PRDS2File : : read16 ( )
{
unsigned char data [ 2 ] ;
int result ;
2021-05-25 19:57:09 +00:00
result = m_device . read ( ( char * ) data , sizeof ( data ) ) ; // access the underlying data for the header
2021-05-23 16:43:31 +00:00
if ( result = = sizeof ( data ) ) {
result = data [ 0 ] | ( data [ 1 ] < < 8 ) ;
} else {
result = 0 ;
}
return result ;
}
QByteArray PRDS2File : : readBytes ( )
{
int length = read16 ( ) ;
2021-05-25 19:57:09 +00:00
QByteArray result = m_device . read ( length ) ; // access the underlying data for the header
2021-05-23 16:43:31 +00:00
if ( result . size ( ) < length ) {
result . clear ( ) ;
}
return result ;
}
//********************************************************************************************
2019-06-20 04:09:28 +00:00
QMap < const char * , const char * > s_PRS1Series = {
{ " System One 60 Series " , " :/icons/prs1_60s.png " } , // needs to come before following substring
{ " System One " , " :/icons/prs1.png " } ,
2020-04-22 19:28:36 +00:00
{ " C Series " , " :/icons/prs1vent.png " } ,
2021-11-03 20:05:26 +00:00
{ " DreamStation 2 " , " :/icons/prds2.png " } , // needs to come before following substring
2019-06-20 04:09:28 +00:00
{ " DreamStation " , " :/icons/dreamstation.png " } ,
} ;
2011-12-19 05:35:05 +00:00
2011-06-26 08:30:44 +00:00
PRS1Loader : : PRS1Loader ( )
{
2019-05-05 01:49:50 +00:00
# ifndef UNITTEST_MODE // no QPixmap without a QGuiApplication
2019-06-20 04:09:28 +00:00
for ( auto & series : s_PRS1Series . keys ( ) ) {
QString path = s_PRS1Series [ series ] ;
m_pixmap_paths [ series ] = path ;
m_pixmaps [ series ] = QPixmap ( path ) ;
}
2019-05-05 01:49:50 +00:00
# endif
2014-09-17 17:20:01 +00:00
2014-05-25 07:07:08 +00:00
m_type = MT_CPAP ;
2011-06-26 08:30:44 +00:00
}
PRS1Loader : : ~ PRS1Loader ( )
{
}
2012-01-05 04:37:22 +00:00
2011-06-26 08:30:44 +00:00
bool isdigit ( QChar c )
{
2014-04-17 05:58:57 +00:00
if ( ( c > = ' 0 ' ) & & ( c < = ' 9 ' ) ) { return true ; }
2011-06-26 08:30:44 +00:00
return false ;
}
2013-09-14 23:32:14 +00:00
2014-04-28 03:27:33 +00:00
// Tests path to see if it has (what looks like) a valid PRS1 folder structure
2020-03-09 16:47:54 +00:00
// This is used both to detect newly inserted media and to decide which loader
// to use after the user selects a folder.
2020-03-09 17:03:28 +00:00
//
// TODO: Ideally there should be a way to handle the two scenarios slightly
// differently. In the latter case, it should clean up the selection and
// return the canonical path if it detects one, allowing us to remove the
// notification about selecting the root of the card. That kind of cleanup
// wouldn't be appropriate when scanning devices.
2020-03-09 16:47:54 +00:00
bool PRS1Loader : : Detect ( const QString & selectedPath )
2014-09-29 14:41:31 +00:00
{
2020-03-09 16:47:54 +00:00
QString path = selectedPath ;
if ( GetPSeriesPath ( path ) . isEmpty ( ) ) {
// Try up one level in case the user selected the P-Series folder within the SD card.
path = QFileInfo ( path ) . canonicalPath ( ) ;
}
2020-03-09 15:17:59 +00:00
QStringList machines = FindMachinesOnCard ( path ) ;
return ! machines . isEmpty ( ) ;
2014-09-29 14:41:31 +00:00
}
2020-03-09 14:48:10 +00:00
QString PRS1Loader : : GetPSeriesPath ( const QString & path )
{
QString outpath = " " ;
QDir root ( path ) ;
QStringList dirs = root . entryList ( QDir : : NoDotAndDotDot | QDir : : Dirs | QDir : : Hidden | QDir : : NoSymLinks ) ;
for ( auto & dir : dirs ) {
// We've seen P-Series, P-SERIES, and p-series, so we need to search for the directory
// in a way that won't break on a case-sensitive filesystem.
if ( dir . toUpper ( ) = = " P-SERIES " ) {
outpath = path + QDir : : separator ( ) + dir ;
break ;
}
}
return outpath ;
}
2020-03-09 14:28:34 +00:00
QStringList PRS1Loader : : FindMachinesOnCard ( const QString & cardPath )
{
QStringList machinePaths ;
2020-03-09 14:48:10 +00:00
QString pseriesPath = this - > GetPSeriesPath ( cardPath ) ;
QDir pseries ( pseriesPath ) ;
2020-03-09 14:28:34 +00:00
// If it contains a P-Series folder, it's a PRS1 SD card
2020-03-09 14:48:10 +00:00
if ( ! pseriesPath . isEmpty ( ) & & pseries . exists ( ) ) {
2020-03-09 14:28:34 +00:00
pseries . setFilter ( QDir : : NoDotAndDotDot | QDir : : Dirs | QDir : : Files | QDir : : Hidden | QDir : : NoSymLinks ) ;
pseries . setSorting ( QDir : : Name ) ;
QFileInfoList plist = pseries . entryInfoList ( ) ;
2022-02-27 16:50:10 +00:00
// Look for device directories (containing a PROP.TXT or properties.txt)
2020-03-09 14:28:34 +00:00
QFileInfoList propertyfiles ;
for ( auto & pfi : plist ) {
if ( pfi . isDir ( ) ) {
QString machinePath = pfi . canonicalFilePath ( ) ;
QDir machineDir ( machinePath ) ;
QFileInfoList mlist = machineDir . entryInfoList ( ) ;
for ( auto & mfi : mlist ) {
if ( QDir : : match ( " PROP*.TXT " , mfi . fileName ( ) ) ) {
2022-02-27 16:50:10 +00:00
// Found a properties file, this is a device folder
2020-03-09 14:28:34 +00:00
propertyfiles . append ( mfi ) ;
}
2021-05-13 18:20:54 +00:00
if ( QDir : : match ( " PROP.BIN " , mfi . fileName ( ) ) ) {
2022-02-27 16:50:10 +00:00
// Found a DreamStation 2 properties file, this is a device folder
2021-05-13 18:20:54 +00:00
propertyfiles . append ( mfi ) ;
}
2020-03-09 14:28:34 +00:00
}
}
}
2022-02-27 16:50:10 +00:00
// Sort devices from oldest to newest.
2020-03-09 14:28:34 +00:00
std : : sort ( propertyfiles . begin ( ) , propertyfiles . end ( ) ,
[ ] ( const QFileInfo & a , const QFileInfo & b )
{
return a . lastModified ( ) < b . lastModified ( ) ;
} ) ;
for ( auto & propertyfile : propertyfiles ) {
machinePaths . append ( propertyfile . canonicalPath ( ) ) ;
}
}
return machinePaths ;
}
2018-04-27 04:29:03 +00:00
void parseModel ( MachineInfo & info , const QString & modelnum )
2014-09-29 14:41:31 +00:00
{
info . modelnumber = modelnum ;
2014-09-30 16:13:05 +00:00
2019-06-20 04:09:28 +00:00
const char * name = s_PRS1ModelInfo . Name ( modelnum ) ;
const char * series = nullptr ;
for ( auto & s : s_PRS1Series . keys ( ) ) {
if ( QString ( name ) . contains ( s ) ) {
series = s ;
break ;
}
}
if ( series = = nullptr ) {
2019-08-14 01:09:55 +00:00
if ( modelnum ! = " 100X100 " ) { // Bogus model number seen in empty C0/Clear0 directories.
qWarning ( ) < < " unknown series for " < < name < < modelnum ;
}
2019-06-20 04:09:28 +00:00
series = " unknown " ;
2014-09-29 14:41:31 +00:00
}
2019-08-18 21:03:52 +00:00
info . model = QObject : : tr ( name ) ;
info . series = series ;
2014-09-29 14:41:31 +00:00
}
2019-05-27 14:05:16 +00:00
bool PRS1Loader : : PeekProperties ( const QString & filename , QHash < QString , QString > & props )
2014-09-29 14:41:31 +00:00
{
2019-05-27 14:05:16 +00:00
const static QMap < QString , QString > s_longFieldNames = {
// CF?
{ " SN " , " SerialNumber " } ,
{ " MN " , " ModelNumber " } ,
{ " PT " , " ProductType " } ,
{ " DF " , " DataFormat " } ,
{ " DFV " , " DataFormatVersion " } ,
{ " F " , " Family " } ,
{ " FV " , " FamilyVersion " } ,
{ " SV " , " SoftwareVersion " } ,
{ " FD " , " FirstDate " } ,
{ " LD " , " LastDate " } ,
// SID?
// SK?
2021-11-04 20:40:44 +00:00
// TS?
// DC?
2019-05-27 14:05:16 +00:00
{ " BK " , " BasicKey " } ,
{ " DK " , " DetailsKey " } ,
{ " EK " , " ErrorKey " } ,
{ " FN " , " PatientFolderNum " } , // most recent Pn directory
{ " PFN " , " PatientFileNum " } , // number of files in the most recent Pn directory
{ " EFN " , " EquipFileNum " } , // number of .004 files in the E directory
{ " DFN " , " DFileNum " } , // number of .003 files in the D directory
{ " VC " , " ValidCheck " } ,
} ;
2014-09-29 14:41:31 +00:00
QFile f ( filename ) ;
if ( ! f . open ( QFile : : ReadOnly ) ) {
2014-04-28 03:27:33 +00:00
return false ;
}
2021-05-23 16:43:31 +00:00
RawDataFile * src ;
if ( QFileInfo ( f ) . suffix ( ) . toUpper ( ) = = " BIN " ) {
// If it's a DS2 file, insert the DS2 wrapper to decode the chunk stream.
2021-12-06 21:14:11 +00:00
PRDS2File * ds2 = new PRDS2File ( f , m_keycache ) ;
2021-11-03 20:38:20 +00:00
if ( ! ds2 - > isValid ( ) ) {
2021-12-03 14:57:29 +00:00
//qWarning() << filename << "unable to decrypt";
2021-11-03 20:38:20 +00:00
delete ds2 ;
return false ;
}
src = ds2 ;
2021-11-29 15:25:52 +00:00
props [ " GUID " ] = ds2 - > guid ( ) ;
2021-05-23 16:43:31 +00:00
} else {
// Otherwise just use the file as input.
src = new RawDataFile ( f ) ;
}
2021-05-23 19:02:01 +00:00
{
QTextStream in ( src ) ; // Scope this here so that it's torn down before we delete src below.
2014-09-29 14:41:31 +00:00
do {
QString line = in . readLine ( ) ;
QStringList pair = line . split ( " = " ) ;
2021-05-23 19:02:01 +00:00
if ( pair . size ( ) ! = 2 ) {
qWarning ( ) < < src - > name ( ) < < " malformed line: " < < line ;
2021-05-25 19:57:09 +00:00
QHashIterator < QString , QString > i ( props ) ;
while ( i . hasNext ( ) ) {
i . next ( ) ;
qDebug ( ) < < i . key ( ) < < " : " < < i . value ( ) ;
}
2021-05-23 19:02:01 +00:00
break ;
}
2014-09-29 14:41:31 +00:00
2019-05-27 14:05:16 +00:00
if ( s_longFieldNames . contains ( pair [ 0 ] ) ) {
pair [ 0 ] = s_longFieldNames [ pair [ 0 ] ] ;
}
if ( pair [ 0 ] = = " Family " ) {
if ( pair [ 1 ] = = " xPAP " ) {
pair [ 1 ] = " 0 " ;
2020-04-22 00:49:15 +00:00
} else if ( pair [ 1 ] = = " Ventilator " ) {
pair [ 1 ] = " 3 " ;
2019-05-27 14:05:16 +00:00
}
}
props [ pair [ 0 ] ] = pair [ 1 ] ;
} while ( ! in . atEnd ( ) ) ;
2021-05-23 19:02:01 +00:00
}
2021-05-23 16:43:31 +00:00
delete src ;
2021-05-23 19:02:01 +00:00
return props . size ( ) > 0 ;
2019-05-27 14:05:16 +00:00
}
2021-10-26 19:23:08 +00:00
bool PRS1Loader : : PeekProperties ( MachineInfo & info , const QString & filename )
2019-05-27 14:05:16 +00:00
{
QHash < QString , QString > props ;
if ( ! PeekProperties ( filename , props ) ) {
return false ;
}
QString modelnum ;
for ( auto & key : props . keys ( ) ) {
2014-09-30 05:25:11 +00:00
bool skip = false ;
2019-05-27 14:05:16 +00:00
if ( key = = " ModelNumber " ) {
modelnum = props [ key ] ;
2014-09-30 05:25:11 +00:00
skip = true ;
2014-09-29 14:41:31 +00:00
}
2019-05-27 14:05:16 +00:00
if ( key = = " SerialNumber " ) {
info . serial = props [ key ] ;
2014-09-30 05:25:11 +00:00
skip = true ;
}
2019-05-27 14:05:16 +00:00
if ( key = = " ProductType " ) {
2019-08-19 16:23:45 +00:00
bool ok ;
props [ key ] . toInt ( & ok , 16 ) ;
2019-05-27 14:05:16 +00:00
if ( ! ok ) qWarning ( ) < < " ProductType " < < props [ key ] ;
2016-01-22 00:07:10 +00:00
skip = true ;
}
2021-10-26 19:23:08 +00:00
if ( skip ) continue ;
2014-09-30 05:25:11 +00:00
2021-10-26 19:23:08 +00:00
info . properties [ key ] = props [ key ] ;
2019-05-27 14:05:16 +00:00
} ;
2016-01-22 00:07:10 +00:00
if ( ! modelnum . isEmpty ( ) ) {
parseModel ( info , modelnum ) ;
2019-06-20 02:19:16 +00:00
} else {
qWarning ( ) < < " missing model number " < < filename ;
2016-01-22 00:07:10 +00:00
}
2014-04-28 03:27:33 +00:00
return true ;
}
2014-09-29 14:41:31 +00:00
MachineInfo PRS1Loader : : PeekInfo ( const QString & path )
{
2020-03-09 15:17:59 +00:00
QStringList machines = FindMachinesOnCard ( path ) ;
if ( machines . isEmpty ( ) ) {
2014-09-29 14:41:31 +00:00
return MachineInfo ( ) ;
2020-03-09 15:17:59 +00:00
}
2014-09-29 14:41:31 +00:00
2022-02-27 16:50:10 +00:00
// Present information about the newest device on the card.
2020-03-09 15:17:59 +00:00
QString newpath = machines . last ( ) ;
2014-09-29 14:41:31 +00:00
MachineInfo info = newInfo ( ) ;
2016-03-01 11:51:14 +00:00
if ( ! PeekProperties ( info , newpath + " /properties.txt " ) ) {
2020-03-09 17:03:28 +00:00
if ( ! PeekProperties ( info , newpath + " /PROP.TXT " ) ) {
2021-05-13 18:20:54 +00:00
// Detect (unsupported) DreamStation 2
QString filepath ( newpath + " /PROP.BIN " ) ;
2021-05-23 16:43:31 +00:00
if ( ! PeekProperties ( info , filepath ) ) {
2021-05-13 18:20:54 +00:00
qWarning ( ) < < " No properties file found in " < < newpath ;
}
2020-03-09 17:03:28 +00:00
}
2016-03-01 11:51:14 +00:00
}
2014-09-29 14:41:31 +00:00
return info ;
}
2020-03-09 16:47:54 +00:00
int PRS1Loader : : Open ( const QString & selectedPath )
2011-06-26 08:30:44 +00:00
{
2020-03-09 16:47:54 +00:00
QString path = selectedPath ;
if ( GetPSeriesPath ( path ) . isEmpty ( ) ) {
// Try up one level in case the user selected the P-Series folder within the SD card.
path = QFileInfo ( path ) . canonicalPath ( ) ;
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
2020-03-09 16:47:54 +00:00
QStringList machines = FindMachinesOnCard ( path ) ;
2022-02-27 16:50:10 +00:00
// Return an error if no devices were found.
2020-03-09 16:47:54 +00:00
if ( machines . isEmpty ( ) ) {
2022-02-27 16:50:10 +00:00
qDebug ( ) < < " No PRS1 devices found at " < < path ;
2014-07-29 14:38:59 +00:00
return - 1 ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
2022-02-27 16:50:10 +00:00
// Import each device, from oldest to newest.
// TODO: Loaders should return the set of devices during detection, so that Open() will
2021-09-01 19:31:06 +00:00
// open a unique device, instead of surprising the user.
2014-07-29 14:38:59 +00:00
int c = 0 ;
2021-11-03 20:06:54 +00:00
bool failures = false ;
2020-03-09 16:47:54 +00:00
for ( auto & machinePath : machines ) {
2021-09-02 14:27:50 +00:00
if ( m_ctx = = nullptr ) {
qWarning ( ) < < " PRS1Loader::Open() called without a valid m_ctx object present " ;
2021-09-01 19:31:06 +00:00
return 0 ;
}
2021-10-26 16:24:31 +00:00
int imported = OpenMachine ( machinePath ) ;
2021-11-03 20:06:54 +00:00
if ( imported > = 0 ) { // don't let errors < 0 suppress subsequent successes
2021-10-26 16:24:31 +00:00
c + = imported ;
2021-11-03 20:06:54 +00:00
} else {
failures = true ;
2021-10-26 16:24:31 +00:00
}
2021-09-02 14:27:50 +00:00
m_ctx - > FlushUnexpectedMessages ( ) ;
2011-06-26 08:30:44 +00:00
}
2021-11-03 20:06:54 +00:00
if ( c = = 0 & & failures ) {
// report an error when there were failures and no successess
c = - 1 ;
}
2014-07-29 14:38:59 +00:00
return c ;
2011-06-26 08:30:44 +00:00
}
2011-07-15 13:30:41 +00:00
2011-06-26 08:30:44 +00:00
2018-04-27 04:29:03 +00:00
int PRS1Loader : : OpenMachine ( const QString & path )
2011-06-26 08:30:44 +00:00
{
2021-09-01 19:31:06 +00:00
Q_ASSERT ( m_ctx ) ;
2011-06-26 08:30:44 +00:00
2011-07-01 10:10:44 +00:00
qDebug ( ) < < " Opening PRS1 " < < path ;
2011-06-26 08:30:44 +00:00
QDir dir ( path ) ;
2014-04-17 05:58:57 +00:00
if ( ! dir . exists ( ) | | ( ! dir . isReadable ( ) ) ) {
2014-05-02 04:34:34 +00:00
return 0 ;
2014-04-17 05:58:57 +00:00
}
2018-05-07 01:57:58 +00:00
m_abort = false ;
2011-06-26 08:30:44 +00:00
2018-05-07 00:37:22 +00:00
emit updateMessage ( QObject : : tr ( " Getting Ready... " ) ) ;
QCoreApplication : : processEvents ( ) ;
2019-05-03 20:59:26 +00:00
emit setProgressValue ( 0 ) ;
QStringList paths ;
QString propertyfile ;
int sessionid_base ;
sessionid_base = FindSessionDirsAndProperties ( path , paths , propertyfile ) ;
2021-10-26 19:23:08 +00:00
bool supported = CreateMachineFromProperties ( propertyfile ) ;
if ( ! supported ) {
2021-09-02 14:27:50 +00:00
// Device is unsupported.
2019-05-03 20:59:26 +00:00
return - 1 ;
}
2020-03-09 16:47:54 +00:00
emit updateMessage ( QObject : : tr ( " Backing Up Files... " ) ) ;
QCoreApplication : : processEvents ( ) ;
2021-09-03 16:45:00 +00:00
QString backupPath = context ( ) - > GetBackupPath ( ) + path . section ( " / " , - 2 ) ;
2019-05-03 20:59:26 +00:00
if ( QDir : : cleanPath ( path ) . compare ( QDir : : cleanPath ( backupPath ) ) ! = 0 ) {
copyPath ( path , backupPath ) ;
}
emit updateMessage ( QObject : : tr ( " Scanning Files... " ) ) ;
QCoreApplication : : processEvents ( ) ;
// Walk through the files and create an import task for each logical session.
2021-09-03 16:45:00 +00:00
ScanFiles ( paths , sessionid_base ) ;
2019-05-03 20:59:26 +00:00
int tasks = countTasks ( ) ;
emit updateMessage ( QObject : : tr ( " Importing Sessions... " ) ) ;
QCoreApplication : : processEvents ( ) ;
runTasks ( AppSetting - > multithreading ( ) ) ;
2021-09-02 14:27:50 +00:00
return tasks ;
2019-05-03 20:59:26 +00:00
}
int PRS1Loader : : FindSessionDirsAndProperties ( const QString & path , QStringList & paths , QString & propertyfile )
{
QDir dir ( path ) ;
2011-06-26 08:30:44 +00:00
dir . setFilter ( QDir : : NoDotAndDotDot | QDir : : Dirs | QDir : : Files | QDir : : Hidden | QDir : : NoSymLinks ) ;
dir . setSorting ( QDir : : Name ) ;
2014-04-17 05:58:57 +00:00
QFileInfoList flist = dir . entryInfoList ( ) ;
2011-06-26 08:30:44 +00:00
QString filename ;
2014-04-17 05:58:57 +00:00
2014-09-29 14:41:31 +00:00
int sessionid_base = 10 ;
2014-09-30 05:25:11 +00:00
2014-04-17 05:58:57 +00:00
for ( int i = 0 ; i < flist . size ( ) ; i + + ) {
QFileInfo fi = flist . at ( i ) ;
filename = fi . fileName ( ) ;
2014-09-29 14:41:31 +00:00
if ( fi . isDir ( ) ) {
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 ( ) = = " e " ) {
// Error files..
// Reminder: I have been given some info about these. should check it over.
}
} else if ( filename . compare ( " properties.txt " , Qt : : CaseInsensitive ) = = 0 ) {
2014-09-30 05:25:11 +00:00
propertyfile = fi . canonicalFilePath ( ) ;
2014-09-29 14:41:31 +00:00
} else if ( filename . compare ( " PROP.TXT " , Qt : : CaseInsensitive ) = = 0 ) {
sessionid_base = 16 ;
2014-09-30 05:25:11 +00:00
propertyfile = fi . canonicalFilePath ( ) ;
2021-05-13 18:20:54 +00:00
} else if ( filename . compare ( " PROP.BIN " , Qt : : CaseInsensitive ) = = 0 ) {
sessionid_base = 16 ;
propertyfile = fi . canonicalFilePath ( ) ;
2011-06-26 08:30:44 +00:00
}
}
2019-05-03 20:59:26 +00:00
return sessionid_base ;
}
2011-06-26 08:30:44 +00:00
2019-05-03 20:59:26 +00:00
2021-10-26 19:23:08 +00:00
bool PRS1Loader : : CreateMachineFromProperties ( QString propertyfile )
2019-05-03 20:59:26 +00:00
{
2021-12-06 21:14:11 +00:00
m_keycache . clear ( ) ;
2021-09-01 21:00:19 +00:00
MachineInfo info = newInfo ( ) ;
2021-05-23 19:32:56 +00:00
QHash < QString , QString > props ;
if ( ! PeekProperties ( propertyfile , props ) | | ! s_PRS1ModelInfo . IsSupported ( props ) ) {
if ( props . contains ( " ModelNumber " ) ) {
int family , familyVersion ;
getVersionFromProps ( props , family , familyVersion ) ;
QString model_number = props [ " ModelNumber " ] ;
qWarning ( ) . noquote ( ) < < " Model " < < model_number < < QString ( " (F%1V%2) " ) . arg ( family ) . arg ( familyVersion ) < < " unsupported. " ;
2021-09-01 21:00:19 +00:00
info . modelnumber = QObject : : tr ( " model %1 " ) . arg ( model_number ) ;
2021-05-23 19:32:56 +00:00
} else {
qWarning ( ) < < " Unable to identify model or series! " ;
2021-09-01 21:00:19 +00:00
info . modelnumber = QObject : : tr ( " unknown model " ) ;
2021-05-23 19:32:56 +00:00
}
2021-09-01 21:00:19 +00:00
emit deviceIsUnsupported ( info ) ;
2021-10-26 19:23:08 +00:00
return false ;
2021-05-13 18:20:54 +00:00
}
2019-05-03 20:59:26 +00:00
// Have a peek first to get the model number.
2014-09-30 05:25:11 +00:00
PeekProperties ( info , propertyfile ) ;
2014-05-01 04:59:35 +00:00
2021-09-01 21:00:19 +00:00
if ( s_PRS1ModelInfo . IsBrick ( info . modelnumber ) ) {
emit deviceReportsUsageOnly ( info ) ;
2014-09-30 05:25:11 +00:00
}
2022-02-27 16:50:10 +00:00
// Which is needed to get the right device record..
2021-10-26 19:23:08 +00:00
m_ctx - > CreateMachineFromInfo ( info ) ;
2014-09-30 05:25:11 +00:00
2019-12-30 04:09:46 +00:00
if ( ! s_PRS1ModelInfo . IsTested ( props ) ) {
2019-05-30 20:32:57 +00:00
qDebug ( ) < < info . modelnumber < < " untested " ;
2021-09-01 21:00:19 +00:00
emit deviceIsUntested ( info ) ;
2019-05-30 20:32:57 +00:00
}
2019-05-27 14:05:16 +00:00
2021-10-26 19:23:08 +00:00
return true ;
2019-05-03 19:07:15 +00:00
}
2019-10-10 14:34:36 +00:00
static QString relativePath ( const QString & inpath )
{
2023-02-18 13:58:47 +00:00
# if QT_VERSION >= QT_VERSION_CHECK(5,14,0)
QStringList pathlist = QDir : : toNativeSeparators ( inpath ) . split ( QDir : : separator ( ) , Qt : : SkipEmptyParts ) ;
# else
QStringList pathlist = QDir : : toNativeSeparators ( inpath ) . split ( QDir : : separator ( ) , QString : : SkipEmptyParts ) ;
# endif
2019-10-10 14:34:36 +00:00
QString relative = pathlist . mid ( pathlist . size ( ) - 3 ) . join ( QDir : : separator ( ) ) ;
return relative ;
}
static bool chunksIdentical ( const PRS1DataChunk * a , const PRS1DataChunk * b )
{
2019-10-25 21:27:35 +00:00
return ( a - > hash ( ) = = b - > hash ( ) ) ;
2019-10-10 14:34:36 +00:00
}
static QString chunkComparison ( const PRS1DataChunk * a , const PRS1DataChunk * b )
{
return QString ( " Session %1 in %2 @ %3 %4 %5 @ %6, skipping " )
. arg ( a - > sessionid )
. arg ( relativePath ( a - > m_path ) ) . arg ( a - > m_filepos )
. arg ( chunksIdentical ( a , b ) ? " is identical to " : " differs from " )
. arg ( relativePath ( b - > m_path ) ) . arg ( b - > m_filepos ) ;
}
2021-09-03 16:45:00 +00:00
void PRS1Loader : : ScanFiles ( const QStringList & paths , int sessionid_base )
2019-05-03 19:07:15 +00:00
{
2021-09-02 14:09:11 +00:00
Q_ASSERT ( m_ctx ) ;
2011-12-10 12:14:48 +00:00
SessionID sid ;
2011-06-26 08:30:44 +00:00
long ext ;
2014-07-28 13:56:29 +00:00
2019-05-03 19:07:15 +00:00
QDir dir ;
dir . setFilter ( QDir : : NoDotAndDotDot | QDir : : Dirs | QDir : : Files | QDir : : Hidden | QDir : : NoSymLinks ) ;
dir . setSorting ( QDir : : Name ) ;
2014-04-17 05:58:57 +00:00
int size = paths . size ( ) ;
2011-12-10 12:14:48 +00:00
2014-08-04 15:40:56 +00:00
sesstasks . clear ( ) ;
2014-05-02 04:34:34 +00:00
new_sessions . clear ( ) ; // this hash is used by OpenFile
2014-04-17 05:58:57 +00:00
2014-07-11 06:13:44 +00:00
2014-08-04 15:40:56 +00:00
PRS1Import * task = nullptr ;
2014-07-11 06:13:44 +00:00
// Note, I have observed p0/p1/etc folders containing duplicates session files (in Robin Sanders data.)
2018-05-07 01:30:42 +00:00
QDateTime datetime ;
2021-09-01 19:31:06 +00:00
qint64 ignoreBefore = m_ctx - > IgnoreSessionsOlderThan ( ) . toMSecsSinceEpoch ( ) / 1000 ;
bool ignoreOldSessions = m_ctx - > ShouldIgnoreOldSessions ( ) ;
2020-01-19 01:16:31 +00:00
QSet < SessionID > skipped ;
2018-05-07 01:30:42 +00:00
2014-05-02 04:34:34 +00:00
// for each p0/p1/p2/etc... folder
2014-05-06 18:03:13 +00:00
for ( int p = 0 ; p < size ; + + p ) {
dir . setPath ( paths . at ( p ) ) ;
2011-06-26 08:30:44 +00:00
2019-05-13 16:11:04 +00:00
if ( ! dir . exists ( ) | | ! dir . isReadable ( ) ) {
qWarning ( ) < < dir . canonicalPath ( ) < < " can't read directory " ;
continue ;
}
2014-04-17 05:58:57 +00:00
2019-05-03 19:07:15 +00:00
QFileInfoList flist = dir . entryInfoList ( ) ;
2011-06-26 08:30:44 +00:00
2014-05-02 04:34:34 +00:00
// Scan for individual session files
2014-04-17 05:58:57 +00:00
for ( int i = 0 ; i < flist . size ( ) ; i + + ) {
2020-03-09 15:17:59 +00:00
# ifndef UNITTEST_MODE
QCoreApplication : : processEvents ( ) ;
# endif
2019-05-13 16:11:04 +00:00
if ( isAborted ( ) ) {
qDebug ( ) < < " received abort signal " ;
break ;
}
2014-04-17 05:58:57 +00:00
QFileInfo fi = flist . at ( i ) ;
2019-05-13 16:11:04 +00:00
QString path = fi . canonicalFilePath ( ) ;
2019-05-03 19:07:15 +00:00
bool ok ;
2014-04-17 05:58:57 +00:00
2019-05-31 20:58:58 +00:00
if ( fi . fileName ( ) = = " .DS_Store " ) {
continue ;
}
2014-08-04 15:40:56 +00:00
QString ext_s = fi . fileName ( ) . section ( " . " , - 1 ) ;
2021-11-03 20:38:20 +00:00
if ( ext_s . toUpper ( ) . startsWith ( " B " ) ) { // .B01, .B02, etc.
ext_s = ext_s . mid ( 1 ) ;
}
2014-08-04 15:40:56 +00:00
ext = ext_s . toInt ( & ok ) ;
if ( ! ok ) {
// not a numerical extension
2019-09-20 19:38:14 +00:00
qInfo ( ) < < path < < " unexpected filename " ;
2012-01-14 05:59:01 +00:00
continue ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
2014-08-04 15:40:56 +00:00
QString session_s = fi . fileName ( ) . section ( " . " , 0 , - 2 ) ;
2014-09-29 14:41:31 +00:00
sid = session_s . toInt ( & ok , sessionid_base ) ;
2014-08-04 15:40:56 +00:00
if ( ! ok ) {
// not a numerical session ID
2019-09-20 19:38:14 +00:00
qInfo ( ) < < path < < " unexpected filename " ;
2012-01-14 05:59:01 +00:00
continue ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
2019-06-01 01:47:28 +00:00
// TODO: BUG: This isn't right, since files can have multiple session
// chunks, which might not correspond to the filename. But before we can
// fix this we need to come up with a reasonably fast way to filter previously
// imported files without re-reading all of them.
2021-09-03 16:45:00 +00:00
if ( context ( ) - > SessionExists ( sid ) ) {
2014-05-02 04:34:34 +00:00
// Skip already imported session
2021-10-28 18:01:33 +00:00
// TODO: Consider reinstating this debug statement if/when we scan only new/changed files.
//qDebug() << path << "session already exists, skipping" << sid;
2014-05-02 04:34:34 +00:00
continue ;
2014-04-17 05:58:57 +00:00
}
2011-06-26 08:30:44 +00:00
2014-09-29 14:41:31 +00:00
if ( ( ext = = 5 ) | | ( ext = = 6 ) ) {
2020-01-19 01:16:31 +00:00
if ( skipped . contains ( sid ) ) {
// We don't know the timestamp until the file is parsed, which we only do for
// waveform data at import (after scanning) since it's so large. If we relied
// solely on the chunks' timestamps at that point, we'd get half of an otherwise
// skipped session (the half after midnight).
//
// So we skip the entire file here based on the session's other data.
continue ;
}
2014-08-04 15:40:56 +00:00
// Waveform files aren't grouped... so we just want to add the filename for later
QHash < SessionID , PRS1Import * > : : iterator it = sesstasks . find ( sid ) ;
if ( it ! = sesstasks . end ( ) ) {
task = it . value ( ) ;
} else {
2014-09-30 05:25:11 +00:00
// Should probably check if session already imported has this data missing..
2014-08-04 15:40:56 +00:00
// Create the group if we see it first..
2021-09-03 16:45:00 +00:00
task = new PRS1Import ( this , sid , sessionid_base ) ;
2014-08-04 15:40:56 +00:00
sesstasks [ sid ] = task ;
queTask ( task ) ;
}
2012-01-14 06:22:24 +00:00
2014-09-29 14:41:31 +00:00
if ( ext = = 5 ) {
2019-10-26 01:17:41 +00:00
// Occasionally waveforms in a session can be split into multiple files.
//
2022-02-27 16:50:10 +00:00
// This seems to happen when the device begins writing the waveform file
2019-10-26 01:17:41 +00:00
// before realizing that it will hit its 500-file-per-directory limit
// for the remaining session files, at which point it appears to write
// the rest of the waveform data along with the summary and event files
// in the next directory.
//
// All samples exhibiting this behavior are DreamStations.
task - > m_wavefiles . append ( fi . canonicalFilePath ( ) ) ;
2014-09-29 14:41:31 +00:00
} else if ( ext = = 6 ) {
2020-04-29 14:22:12 +00:00
// Oximetry data can also be split into multiple files, see waveform
// comment above.
task - > m_oxifiles . append ( fi . canonicalFilePath ( ) ) ;
2014-09-29 14:41:31 +00:00
}
2014-07-11 06:13:44 +00:00
2014-08-04 15:40:56 +00:00
continue ;
}
2014-05-31 21:25:07 +00:00
2014-08-04 15:40:56 +00:00
// Parse the data chunks and read the files..
2018-04-27 04:29:03 +00:00
QList < PRS1DataChunk * > Chunks = ParseFile ( fi . canonicalFilePath ( ) ) ;
2014-05-31 21:25:07 +00:00
2014-08-04 15:40:56 +00:00
for ( int i = 0 ; i < Chunks . size ( ) ; + + i ) {
2019-05-13 16:11:04 +00:00
if ( isAborted ( ) ) {
qDebug ( ) < < " received abort signal 2 " ;
break ;
}
2018-05-07 01:57:58 +00:00
2014-08-04 15:40:56 +00:00
PRS1DataChunk * chunk = Chunks . at ( i ) ;
2014-09-29 14:41:31 +00:00
2014-09-30 05:25:11 +00:00
SessionID chunk_sid = chunk - > sessionid ;
2019-06-04 02:01:02 +00:00
if ( i = = 0 & & chunk_sid ! = sid ) { // log session ID mismatches
2022-02-27 16:50:10 +00:00
// This appears to be benign, probably when a card is out of the device one night and
2020-01-28 18:04:36 +00:00
// then inserted in the morning. It writes out all of the still-in-memory summaries and
// events up through the last night (and no waveform data).
//
// This differs from the first time a card is inserted, because in that case the filename
// *is* equal to the first session contained within it, and then filenames for the
// remaining sessions contained in that file are skipped.
//
// Because the card was present and previous sessions were written with their filenames,
// the first available filename isn't the first session contained in the file.
//qDebug() << fi.canonicalFilePath() << "first session is" << chunk_sid << "instead of" << sid;
2019-05-05 19:50:38 +00:00
}
2021-09-03 16:45:00 +00:00
if ( context ( ) - > SessionExists ( chunk_sid ) ) {
2019-10-09 17:35:02 +00:00
qDebug ( ) < < path < < " session already imported, skipping " < < sid < < chunk_sid ;
2014-09-30 05:25:11 +00:00
delete chunk ;
continue ;
}
2020-01-19 01:16:31 +00:00
if ( ignoreOldSessions & & chunk - > timestamp < ignoreBefore ) {
2020-01-21 18:49:02 +00:00
qDebug ( ) . noquote ( ) < < relativePath ( path ) < < " skipping session " < < chunk_sid < < " : "
< < QDateTime : : fromMSecsSinceEpoch ( chunk - > timestamp * 1000 ) . toString ( ) < < " older than "
< < QDateTime : : fromMSecsSinceEpoch ( ignoreBefore * 1000 ) . toString ( ) ;
2020-01-19 01:16:31 +00:00
skipped + = chunk_sid ;
delete chunk ;
continue ;
}
2014-08-04 15:40:56 +00:00
task = nullptr ;
2014-09-30 05:25:11 +00:00
QHash < SessionID , PRS1Import * > : : iterator it = sesstasks . find ( chunk_sid ) ;
2014-08-04 15:40:56 +00:00
if ( it ! = sesstasks . end ( ) ) {
task = it . value ( ) ;
} else {
2021-09-03 16:45:00 +00:00
task = new PRS1Import ( this , chunk_sid , sessionid_base ) ;
2014-09-30 05:25:11 +00:00
sesstasks [ chunk_sid ] = task ;
2014-08-04 15:40:56 +00:00
// save a loop an que this now
queTask ( task ) ;
}
switch ( ext ) {
case 0 :
2019-05-13 16:11:04 +00:00
if ( task - > compliance ) {
2022-02-16 21:45:26 +00:00
if ( chunksIdentical ( chunk , task - > compliance ) ) {
2019-10-10 14:34:36 +00:00
// Never seen identical compliance chunks, so keep logging this for now.
2022-02-16 21:45:26 +00:00
qDebug ( ) < < chunkComparison ( chunk , task - > compliance ) ;
2019-10-10 14:34:36 +00:00
} else {
2022-02-16 21:45:26 +00:00
qWarning ( ) < < chunkComparison ( chunk , task - > compliance ) ;
2019-10-10 14:34:36 +00:00
}
2019-05-15 21:41:37 +00:00
delete chunk ;
2019-05-13 16:11:04 +00:00
continue ; // (skipping to avoid duplicates)
}
2014-08-04 15:40:56 +00:00
task - > compliance = chunk ;
break ;
case 1 :
2019-05-13 16:11:04 +00:00
if ( task - > summary ) {
2019-10-10 14:34:36 +00:00
if ( chunksIdentical ( chunk , task - > summary ) ) {
// This seems to be benign. It happens most often when a single file contains
// a bunch of chunks and subsequent files each contain a single chunk that was
// already covered by the first file. It also sometimes happens with entirely
// duplicate files between e.g. a P1 and P0 directory.
//
// It's common enough that we don't emit a message about it by default.
//qDebug() << chunkComparison(chunk, task->summary);
} else {
// Warn about any non-identical duplicate session IDs.
2019-12-02 22:30:28 +00:00
//
// This seems to happen with F5V1 slice 8, which is the only slice in a session,
// and which doesn't update the session ID, so the following slice 7 session
// (which can be hours later) has the same session ID. Neither affects import.
2019-10-10 14:34:36 +00:00
qWarning ( ) < < chunkComparison ( chunk , task - > summary ) ;
}
2019-05-15 21:41:37 +00:00
delete chunk ;
2019-05-13 16:11:04 +00:00
continue ;
}
2014-08-04 15:40:56 +00:00
task - > summary = chunk ;
break ;
case 2 :
2019-10-24 20:25:36 +00:00
if ( task - > m_event_chunks . count ( ) > 0 ) {
PRS1DataChunk * previous ;
2020-04-22 00:49:15 +00:00
if ( chunk - > family = = 3 & & chunk - > familyVersion < = 3 ) {
// F3V0 and F3V3 events are formatted as waveforms, with one chunk per mask-on slice,
2019-10-24 20:25:36 +00:00
// and thus multiple chunks per session.
previous = task - > m_event_chunks [ chunk - > timestamp ] ;
if ( previous ! = nullptr ) {
// Skip any chunks with identical timestamps.
qWarning ( ) < < chunkComparison ( chunk , previous ) ;
delete chunk ;
continue ;
}
// fall through to add the new chunk
2019-10-10 14:34:36 +00:00
} else {
2019-10-24 20:25:36 +00:00
// Nothing else should have multiple event chunks per session.
previous = task - > m_event_chunks . first ( ) ;
if ( chunksIdentical ( chunk , previous ) ) {
// See comment above regarding identical summary chunks.
//qDebug() << chunkComparison(chunk, previous);
} else {
qWarning ( ) < < chunkComparison ( chunk , previous ) ;
2019-10-10 14:34:36 +00:00
}
2019-10-24 20:25:36 +00:00
delete chunk ;
continue ;
2019-10-10 14:34:36 +00:00
}
2019-05-13 16:11:04 +00:00
}
2019-10-24 20:25:36 +00:00
task - > m_event_chunks [ chunk - > timestamp ] = chunk ;
2014-08-04 15:40:56 +00:00
break ;
default :
2019-05-13 16:11:04 +00:00
qWarning ( ) < < path < < " unexpected file " ;
2014-08-04 15:40:56 +00:00
break ;
}
2011-06-26 08:30:44 +00:00
}
2014-04-17 05:58:57 +00:00
}
2019-05-13 16:11:04 +00:00
if ( isAborted ( ) ) {
qDebug ( ) < < " received abort signal 3 " ;
break ;
}
2012-12-09 14:01:23 +00:00
}
2011-06-26 08:30:44 +00:00
}
2014-05-02 04:34:34 +00:00
2019-05-28 01:02:28 +00:00
2019-10-20 21:46:18 +00:00
// The set of PRS1 "on-demand" channels that only get created on import if the session
// contains events of that type. Any channels not on this list always get created if
// they're reported/supported by the parser.
static const QVector < PRS1ParsedEventType > PRS1OnDemandChannels =
{
2020-03-26 13:01:28 +00:00
PRS1TimedBreathEvent : : TYPE ,
2019-10-20 21:46:18 +00:00
PRS1PressurePulseEvent : : TYPE ,
// Pressure initialized on-demand for F0 due to the possibility of bilevel vs. single pressure.
PRS1PressureSetEvent : : TYPE ,
PRS1IPAPSetEvent : : TYPE ,
PRS1EPAPSetEvent : : TYPE ,
2020-01-28 16:26:31 +00:00
// Pressure average initialized on-demand for F0 due to the different semantics of bilevel vs. single pressure.
PRS1PressureAverageEvent : : TYPE ,
PRS1FlexPressureAverageEvent : : TYPE ,
2019-10-20 21:46:18 +00:00
} ;
2019-11-19 17:29:45 +00:00
// The set of "non-slice" channels are independent of mask-on slices, i.e. they
// are continuously reported and charted regardless of whether the mask is on.
static const QSet < PRS1ParsedEventType > PRS1NonSliceChannels =
{
PRS1PressureSetEvent : : TYPE ,
PRS1IPAPSetEvent : : TYPE ,
PRS1EPAPSetEvent : : TYPE ,
PRS1SnoresAtPressureEvent : : TYPE ,
} ;
2019-10-20 21:46:18 +00:00
// The channel ID (referenced by pointer because their values aren't initialized
// prior to runtime) to which a given PRS1 event should be added. Events with
// no channel IDs are silently dropped, and events with more than one channel ID
// must be handled specially.
static const QHash < PRS1ParsedEventType , QVector < ChannelID * > > PRS1ImportChannelMap =
{
{ PRS1ClearAirwayEvent : : TYPE , { & CPAP_ClearAirway } } ,
{ PRS1ObstructiveApneaEvent : : TYPE , { & CPAP_Obstructive } } ,
{ PRS1HypopneaEvent : : TYPE , { & CPAP_Hypopnea } } ,
{ PRS1FlowLimitationEvent : : TYPE , { & CPAP_FlowLimit } } ,
{ PRS1SnoreEvent : : TYPE , { & CPAP_Snore , & CPAP_VSnore2 } } , // VSnore2 is calculated from snore count, used to flag nonzero intervals on overview
{ PRS1VibratorySnoreEvent : : TYPE , { & CPAP_VSnore } } ,
{ PRS1RERAEvent : : TYPE , { & CPAP_RERA } } ,
{ PRS1PeriodicBreathingEvent : : TYPE , { & CPAP_PB } } ,
{ PRS1LargeLeakEvent : : TYPE , { & CPAP_LargeLeak } } ,
2021-08-24 22:48:31 +00:00
{ PRS1TotalLeakEvent : : TYPE , { & CPAP_LeakTotal } } ,
2019-10-20 21:46:18 +00:00
{ PRS1LeakEvent : : TYPE , { & CPAP_Leak } } ,
{ PRS1RespiratoryRateEvent : : TYPE , { & CPAP_RespRate } } ,
{ PRS1TidalVolumeEvent : : TYPE , { & CPAP_TidalVolume } } ,
{ PRS1MinuteVentilationEvent : : TYPE , { & CPAP_MinuteVent } } ,
{ PRS1PatientTriggeredBreathsEvent : : TYPE , { & CPAP_PTB } } ,
{ PRS1TimedBreathEvent : : TYPE , { & PRS1_TimedBreath } } ,
2020-04-22 20:52:21 +00:00
{ PRS1FlowRateEvent : : TYPE , { & PRS1_PeakFlow } } , // Only reported by F3V0 and F3V3 // TODO: should this stat be calculated from flow waveforms on other models?
2019-10-20 21:46:18 +00:00
2019-10-29 15:14:57 +00:00
{ PRS1PressureSetEvent : : TYPE , { & CPAP_PressureSet } } ,
{ PRS1IPAPSetEvent : : TYPE , { & CPAP_IPAPSet , & CPAP_PS } } , // PS is calculated from IPAPset and EPAPset when both are supported (F0) TODO: Should this be a separate channel since it's not a 2-minute average?
{ PRS1EPAPSetEvent : : TYPE , { & CPAP_EPAPSet } } , // EPAPset is supported on F5 without any corresponding IPAPset, so it shouldn't always create a PS channel
2020-01-28 16:26:31 +00:00
{ PRS1PressureAverageEvent : : TYPE , { & CPAP_Pressure } } , // This is the time-weighted average pressure in bilevel modes.
{ PRS1FlexPressureAverageEvent : : TYPE , { & CPAP_EPAP } } , // This is effectively EPAP due to Flex reduced pressure in single-pressure modes.
2019-10-20 21:46:18 +00:00
{ PRS1IPAPAverageEvent : : TYPE , { & CPAP_IPAP } } ,
2019-10-29 15:14:57 +00:00
{ PRS1EPAPAverageEvent : : TYPE , { & CPAP_EPAP , & CPAP_PS } } , // PS is calculated from IPAP and EPAP averages (F3 and F5)
2019-10-20 21:46:18 +00:00
{ PRS1IPAPLowEvent : : TYPE , { & CPAP_IPAPLo } } ,
{ PRS1IPAPHighEvent : : TYPE , { & CPAP_IPAPHi } } ,
2019-10-21 01:09:34 +00:00
{ PRS1Test1Event : : TYPE , { & CPAP_Test1 } } , // ??? F3V6
{ PRS1Test2Event : : TYPE , { & CPAP_Test2 } } , // ??? F3V6
2019-10-20 21:46:18 +00:00
{ PRS1PressurePulseEvent : : TYPE , { & CPAP_PressurePulse } } ,
{ PRS1ApneaAlarmEvent : : TYPE , { /* Not imported */ } } ,
2019-10-22 16:30:42 +00:00
{ PRS1SnoresAtPressureEvent : : TYPE , { /* Not imported */ } } ,
2019-10-20 21:46:18 +00:00
{ PRS1AutoPressureSetEvent : : TYPE , { /* Not imported */ } } ,
2020-03-26 13:01:28 +00:00
{ PRS1VariableBreathingEvent : : TYPE , { & PRS1_VariableBreathing } } , // UNCONFIRMED
2019-11-12 22:35:13 +00:00
{ PRS1HypopneaCount : : TYPE , { & CPAP_Hypopnea } } , // F3V3 only, generates individual events on import
{ PRS1ObstructiveApneaCount : : TYPE , { & CPAP_Obstructive } } , // F3V3 only, generates individual events on import
{ PRS1ClearAirwayCount : : TYPE , { & CPAP_ClearAirway } } , // F3V3 only, generates individual events on import
2019-10-20 21:46:18 +00:00
} ;
2019-05-29 16:11:53 +00:00
//********************************************************************************************
2019-05-28 22:25:08 +00:00
2021-05-31 18:53:23 +00:00
PRS1Import : : ~ PRS1Import ( )
{
delete compliance ;
delete summary ;
for ( auto & e : m_event_chunks . values ( ) ) {
delete e ;
}
for ( int i = 0 ; i < waveforms . size ( ) ; + + i ) {
delete waveforms . at ( i ) ;
}
for ( auto & c : oximetry ) {
delete c ;
}
}
2019-11-19 17:29:45 +00:00
void PRS1Import : : CreateEventChannels ( const PRS1DataChunk * chunk )
2019-10-20 21:46:18 +00:00
{
2019-11-19 17:29:45 +00:00
const QVector < PRS1ParsedEventType > & supported = GetSupportedEvents ( chunk ) ;
2022-02-27 16:50:10 +00:00
// Generate the list of channels created by non-slice events for this device.
// We can't just use the full list of non-slice events, since on some devices
2019-11-19 17:29:45 +00:00
// PS is generated by slice events (EPAP/IPAP average).
2023-02-18 13:58:47 +00:00
// Duplicates need to be removed. QSet does the removal.
# if QT_VERSION < QT_VERSION_CHECK(5,14,0)
// convert QVvector to QList then to QSet
QSet < PRS1ParsedEventType > supportedNonSliceEvents = QSet < PRS1ParsedEventType > : : fromList ( QList < PRS1ParsedEventType > : : fromVector ( supported ) ) ;
# else
// release 5.14 supports the direct conversion.
QSet < PRS1ParsedEventType > supportedNonSliceEvents ( supported . begin ( ) , supported . end ( ) ) ;
# endif
2019-11-19 17:29:45 +00:00
supportedNonSliceEvents . intersect ( PRS1NonSliceChannels ) ;
QSet < ChannelID > supportedNonSliceChannels ;
for ( auto & e : supportedNonSliceEvents ) {
for ( auto & pChannelID : PRS1ImportChannelMap [ e ] ) {
supportedNonSliceChannels + = * pChannelID ;
}
}
// Clear channels to prepare for a new slice, except for channels created by
// non-slice events.
for ( auto & c : m_importChannels . keys ( ) ) {
if ( supportedNonSliceChannels . contains ( c ) = = false ) {
m_importChannels . remove ( c ) ;
}
}
2023-02-18 13:58:47 +00:00
2019-10-20 21:46:18 +00:00
// Create all supported channels (except for on-demand ones that only get created if an event appears)
for ( auto & e : supported ) {
if ( ! PRS1OnDemandChannels . contains ( e ) ) {
for ( auto & pChannelID : PRS1ImportChannelMap [ e ] ) {
2019-11-09 20:09:02 +00:00
GetImportChannel ( * pChannelID ) ;
2019-10-20 21:46:18 +00:00
}
}
}
}
EventList * PRS1Import : : GetImportChannel ( ChannelID channel )
{
2020-03-08 20:27:18 +00:00
if ( ! channel ) {
qCritical ( ) < < this - > sessionid < < " channel in import table has not been added to schema! " ;
}
2019-10-20 21:46:18 +00:00
EventList * C = m_importChannels [ channel ] ;
if ( C = = nullptr ) {
C = session - > AddEventList ( channel , EVL_Event ) ;
2019-11-09 20:09:02 +00:00
Q_ASSERT ( C ) ; // Once upon a time AddEventList could return nullptr, but not any more.
m_importChannels [ channel ] = C ;
2019-10-20 21:46:18 +00:00
}
return C ;
}
2019-11-09 20:09:02 +00:00
void PRS1Import : : AddEvent ( ChannelID channel , qint64 t , float value , float gain )
2019-10-20 21:46:18 +00:00
{
EventList * C = GetImportChannel ( channel ) ;
2019-11-09 20:09:02 +00:00
Q_ASSERT ( C ) ;
2019-10-20 21:46:18 +00:00
if ( C - > count ( ) = = 0 ) {
// Initialize the gain (here, since required channels are created with default gain).
C - > setGain ( gain ) ;
} else {
// Any change in gain is a programming error.
if ( gain ! = C - > gain ( ) ) {
qWarning ( ) < < " gain mismatch for channel " < < channel < < " at " < < ts ( t ) ;
}
}
2023-02-18 13:58:47 +00:00
2019-10-20 21:46:18 +00:00
// Add the event
C - > AddEvent ( t , value , gain ) ;
}
2019-11-19 17:29:45 +00:00
bool PRS1Import : : UpdateCurrentSlice ( PRS1DataChunk * chunk , qint64 t )
2019-11-13 14:27:47 +00:00
{
2019-11-19 17:29:45 +00:00
bool updated = false ;
2023-02-18 13:58:47 +00:00
2019-11-19 17:29:45 +00:00
if ( ! m_currentSliceInitialized ) {
m_currentSliceInitialized = true ;
m_currentSlice = m_slices . constBegin ( ) ;
2020-01-07 02:45:52 +00:00
m_lastIntervalEvents . clear ( ) ; // there was no previous slice, so there are no pending end-of-slice events
m_lastIntervalEnd = 0 ;
2019-11-19 17:29:45 +00:00
updated = true ;
}
// Update the slice iterator to point to the mask-on slice encompassing time t.
while ( ( * m_currentSlice ) . status ! = MaskOn | | t > ( * m_currentSlice ) . end ) {
m_currentSlice + + ;
updated = true ;
if ( m_currentSlice = = m_slices . constEnd ( ) ) {
qWarning ( ) < < sessionid < < " Events after last mask-on slice? " ;
m_currentSlice - - ;
break ;
}
}
2023-02-18 13:58:47 +00:00
2020-01-07 02:45:52 +00:00
if ( updated ) {
// Write out any pending end-of-slice events.
FinishSlice ( ) ;
}
2023-02-18 13:58:47 +00:00
2019-11-19 17:29:45 +00:00
if ( updated & & ( * m_currentSlice ) . status = = MaskOn ) {
2020-01-07 02:45:52 +00:00
// Set the interval start times based on the new slice's start time.
m_statIntervalStart = 0 ;
StartNewInterval ( ( * m_currentSlice ) . start ) ;
2019-11-13 14:27:47 +00:00
2019-11-19 17:29:45 +00:00
// Create a new eventlist for this new slice, to allow for a gap in the data between slices.
CreateEventChannels ( chunk ) ;
}
2023-02-18 13:58:47 +00:00
2019-11-19 17:29:45 +00:00
return updated ;
2019-11-13 14:27:47 +00:00
}
2020-01-07 02:45:52 +00:00
void PRS1Import : : FinishSlice ( )
{
qint64 t = m_lastIntervalEnd ; // end of the slice (at least of its interval data)
// If the most recently recorded interval stats aren't at the end of the slice,
// import additional events marking the end of the data.
if ( t ! = m_prevIntervalStart ) {
// Make sure to use the same pressure used to import the original events,
// otherwise calculated channels (such as PS or LEAK) will be wrong.
EventDataType orig_pressure = m_currentPressure ;
m_currentPressure = m_intervalPressure ;
// Import duplicates of each event with the end-of-slice timestamp.
for ( auto & e : m_lastIntervalEvents ) {
ImportEvent ( t , e ) ;
}
// Restore the current pressure.
m_currentPressure = orig_pressure ;
}
m_lastIntervalEvents . clear ( ) ;
}
void PRS1Import : : StartNewInterval ( qint64 t )
{
if ( t = = m_prevIntervalStart ) {
qWarning ( ) < < sessionid < < " Multiple zero-length intervals at end of slice? " ;
}
m_prevIntervalStart = m_statIntervalStart ;
m_statIntervalStart = t ;
}
2019-11-17 01:07:52 +00:00
bool PRS1Import : : IsIntervalEvent ( PRS1ParsedEvent * e )
{
bool intervalEvent = false ;
// Statistical timestamps are reported at the end of a (generally) 2-minute
// interval, rather than the start time that OSCAR expects for its imported
// events. (When a session or slice ends, there will be a shorter interval,
// the previous statistics to the end of the session/slice.)
switch ( e - > m_type ) {
case PRS1PressureAverageEvent : : TYPE :
2020-01-28 16:26:31 +00:00
case PRS1FlexPressureAverageEvent : : TYPE :
2019-11-17 01:07:52 +00:00
case PRS1IPAPAverageEvent : : TYPE :
case PRS1IPAPLowEvent : : TYPE :
case PRS1IPAPHighEvent : : TYPE :
case PRS1EPAPAverageEvent : : TYPE :
case PRS1TotalLeakEvent : : TYPE :
case PRS1LeakEvent : : TYPE :
case PRS1RespiratoryRateEvent : : TYPE :
case PRS1PatientTriggeredBreathsEvent : : TYPE :
case PRS1MinuteVentilationEvent : : TYPE :
case PRS1TidalVolumeEvent : : TYPE :
case PRS1FlowRateEvent : : TYPE :
case PRS1Test1Event : : TYPE :
case PRS1Test2Event : : TYPE :
case PRS1SnoreEvent : : TYPE :
case PRS1HypopneaCount : : TYPE :
case PRS1ClearAirwayCount : : TYPE :
case PRS1ObstructiveApneaCount : : TYPE :
intervalEvent = true ;
break ;
default :
break ;
}
2023-02-18 13:58:47 +00:00
2019-11-17 01:07:52 +00:00
return intervalEvent ;
}
2019-10-24 20:25:36 +00:00
bool PRS1Import : : ImportEventChunk ( PRS1DataChunk * event )
2018-05-05 07:14:44 +00:00
{
2019-11-07 19:19:58 +00:00
m_currentPressure = 0 ;
2019-10-11 00:13:39 +00:00
2019-10-29 15:14:57 +00:00
const QVector < PRS1ParsedEventType > & supported = GetSupportedEvents ( event ) ;
// Calculate PS from IPAP/EPAP set events only when both are supported. This includes F0, but excludes
// F5, which only reports EPAP set events, but both IPAP/EPAP average, from which PS will be calculated.
2019-11-07 19:19:58 +00:00
m_calcPSfromSet = supported . contains ( PRS1IPAPSetEvent : : TYPE ) & & supported . contains ( PRS1EPAPSetEvent : : TYPE ) ;
2019-10-29 15:14:57 +00:00
2019-07-26 01:44:36 +00:00
qint64 t = qint64 ( event - > timestamp ) * 1000L ;
2019-11-13 16:25:59 +00:00
if ( session - > first ( ) = = 0 ) {
qWarning ( ) < < sessionid < < " Start time not set by summary? " ;
} else if ( t < session - > first ( ) ) {
qWarning ( ) < < sessionid < < " Events start before summary? " ;
}
2019-10-23 20:26:25 +00:00
2019-05-24 23:41:42 +00:00
bool ok ;
2019-10-09 17:35:02 +00:00
ok = event - > ParseEvents ( ) ;
2019-05-24 23:41:42 +00:00
2019-11-19 17:29:45 +00:00
// Set up the (possibly initial) slice based on the chunk's starting timestamp.
UpdateCurrentSlice ( event , t ) ;
2019-11-13 14:27:47 +00:00
2019-05-24 23:41:42 +00:00
for ( int i = 0 ; i < event - > m_parsedData . count ( ) ; i + + ) {
PRS1ParsedEvent * e = event - > m_parsedData . at ( i ) ;
t = qint64 ( event - > timestamp + e - > m_start ) * 1000L ;
2019-11-17 01:07:52 +00:00
2019-11-19 17:29:45 +00:00
// Skip unknown events with no timestamp
if ( e - > m_type = = PRS1UnknownDataEvent : : TYPE ) {
continue ;
}
2020-03-26 13:01:28 +00:00
// Skip zero-length PB or LL or VB events
if ( ( e - > m_type = = PRS1PeriodicBreathingEvent : : TYPE | | e - > m_type = = PRS1LargeLeakEvent : : TYPE | | e - > m_type = = PRS1VariableBreathingEvent : : TYPE ) & &
2019-11-19 17:29:45 +00:00
( e - > m_duration = = 0 ) ) {
// LL occasionally appear about a minute before a new mask-on slice
// begins, when the previous mask-on slice ended with a large leak.
// This probably indicates the end of LL and beginning
// of breath detection, but we don't get any real data until mask-on.
//
2020-03-26 13:01:28 +00:00
// It has also happened once in a similar scenario for PB and VB, even when
2019-11-19 17:29:45 +00:00
// the two mask-on slices are in different sessions!
continue ;
}
2020-01-06 16:59:15 +00:00
if ( e - > m_type = = PRS1IntervalBoundaryEvent : : TYPE ) {
2020-01-07 02:45:52 +00:00
StartNewInterval ( t ) ;
continue ; // these internal pseudo-events don't get imported
2020-01-06 16:59:15 +00:00
}
2019-11-17 01:07:52 +00:00
bool intervalEvent = IsIntervalEvent ( e ) ;
2019-11-19 19:18:13 +00:00
qint64 interval_end_t = 0 ;
2019-11-15 00:24:45 +00:00
if ( intervalEvent ) {
2020-01-07 02:45:52 +00:00
// Deal with statistics that are reported at the end of an interval, but which need to be imported
// at the start of the interval.
2020-04-22 00:49:15 +00:00
if ( event - > family = = 3 & & event - > familyVersion < = 3 ) {
// In F3V0 and F3V3, each slice has its own chunk, so the initial call to UpdateCurrentSlice()
2020-01-07 02:45:52 +00:00
// for this chunk is all that's needed.
//
// We can't just call it again here for simplicity, since the timestamps of F3V3 stat events
// can go past the end of the slice.
} else {
2022-02-27 16:50:10 +00:00
// For all other devices, the event's time stamp will be within bounds of its slice, so
2020-01-07 02:45:52 +00:00
// we can use it to find the current slice.
UpdateCurrentSlice ( event , t ) ;
2019-11-15 00:24:45 +00:00
}
2019-11-19 19:18:13 +00:00
// Clamp this interval's end time to the end of the slice.
interval_end_t = min ( t , ( * m_currentSlice ) . end ) ;
2019-11-15 00:24:45 +00:00
// Set this event's timestamp as the start of the interval, since that what OSCAR assumes.
t = m_statIntervalStart ;
// TODO: ideally we would also set the duration of the event, but OSCAR doesn't have any notion of that yet.
2019-11-19 17:29:45 +00:00
} else {
// Advance the slice if needed for the regular event's timestamp.
if ( ! PRS1NonSliceChannels . contains ( e - > m_type ) ) {
UpdateCurrentSlice ( event , t ) ;
}
}
// Sanity check: warn if a (non-slice) event is earlier than the current mask-on slice
if ( t < ( * m_currentSlice ) . start & & ( * m_currentSlice ) . status = = MaskOn ) {
if ( ! PRS1NonSliceChannels . contains ( e - > m_type ) ) {
2020-03-26 13:01:28 +00:00
// LL and VB at the beginning of a mask-on session sometimes start 1 second early,
2019-11-19 17:29:45 +00:00
// so suppress that warning.
2020-03-26 13:01:28 +00:00
if ( ( * m_currentSlice ) . start - t > 1000 | | ( e - > m_type ! = PRS1LargeLeakEvent : : TYPE & & e - > m_type ! = PRS1VariableBreathingEvent : : TYPE ) ) {
2019-11-19 17:29:45 +00:00
qWarning ( ) < < sessionid < < " Event " < < e - > m_type < < " before mask-on slice: " < < ts ( t ) ;
}
}
2019-11-15 00:24:45 +00:00
}
2019-10-29 20:25:04 +00:00
2020-01-07 02:45:52 +00:00
// Import the event.
2019-11-12 22:35:13 +00:00
switch ( e - > m_type ) {
// F3V3 doesn't have individual HY/CA/OA events, only counts every 2 minutes, where
// nonzero counts show up as overview flags. Currently OSCAR doesn't have a way to
// chart those numeric statistics, so we generate events based on the count.
//
// TODO: This (and VS2) would probably be better handled as numeric charts only,
// along with enhancing overview flags to be drawn when channels have nonzero values,
// instead of the fictitious "events" that are currently generated.
case PRS1HypopneaCount : : TYPE :
case PRS1ClearAirwayCount : : TYPE :
case PRS1ObstructiveApneaCount : : TYPE :
// Make sure PRS1ClearAirwayEvent/etc. isn't supported before generating events from counts.
CHECK_VALUE ( supported . contains ( PRS1HypopneaEvent : : TYPE ) , false ) ;
CHECK_VALUE ( supported . contains ( PRS1ClearAirwayEvent : : TYPE ) , false ) ;
CHECK_VALUE ( supported . contains ( PRS1ObstructiveApneaEvent : : TYPE ) , false ) ;
// Divide each count into events evenly spaced over the interval.
// NOTE: This is slightly fictional, but there's no waveform data for F3V3, so it won't
// incorrectly associate specific events with specific flow or pressure events.
if ( e - > m_value > 0 ) {
2019-11-19 19:18:13 +00:00
qint64 blockduration = interval_end_t - m_statIntervalStart ;
2019-11-12 22:35:13 +00:00
qint64 div = blockduration / e - > m_value ;
qint64 tt = t ;
PRS1ParsedDurationEvent ee ( e - > m_type , t , 0 ) ;
for ( int i = 0 ; i < e - > m_value ; + + i ) {
ImportEvent ( tt , & ee ) ;
tt + = div ;
}
}
// TODO: Consider whether to have a numeric channel for HY/CA/OA that gets charted like VS does,
// in which case we can fall through.
break ;
default :
ImportEvent ( t , e ) ;
2020-01-07 02:45:52 +00:00
// Cache the most recently imported interval events so that we can import duplicate end-of-slice events if needed.
// We can't write them here because we don't yet know if they're the last in the slice.
if ( intervalEvent ) {
// Reset the list of pending events when we encounter a stat event in a new interval.
//
// This logic has grown sufficiently complex that it may eventually be worth encapsulating
// each batch of parsed interval events into a composite interval event when parsing,
// rather than requiring timestamp-based gymnastics to infer that structure on import.
if ( m_lastIntervalEnd ! = interval_end_t ) {
m_lastIntervalEvents . clear ( ) ;
m_lastIntervalEnd = interval_end_t ;
m_intervalPressure = m_currentPressure ;
}
// The events need to be in order so that any dynamically calculated channels (such as PS) are properly computed.
m_lastIntervalEvents . append ( e ) ;
2019-11-19 19:18:13 +00:00
}
2019-11-12 22:35:13 +00:00
break ;
}
2019-11-07 19:19:58 +00:00
}
2020-01-07 02:45:52 +00:00
// Write out any pending end-of-slice events.
FinishSlice ( ) ;
2019-11-07 19:19:58 +00:00
if ( ! ok ) {
return false ;
}
2020-04-22 00:49:15 +00:00
// TODO: This needs to be special-cased for F3V0 and F3V3 due to their weird interval-based event format
2019-11-07 19:19:58 +00:00
// until there's a way for its parser to correctly set the timestamps for truncated
// intervals in sessions that don't end on a 2-minute boundary.
2020-04-22 00:49:15 +00:00
if ( ! ( event - > family = = 3 & & event - > familyVersion < = 3 ) ) {
2019-11-07 19:19:58 +00:00
// If the last event has a non-zero duration, t will not reflect the full duration of the chunk, so update it.
t = qint64 ( event - > timestamp + event - > duration ) * 1000L ;
2019-11-13 16:25:59 +00:00
if ( session - > last ( ) = = 0 ) {
qWarning ( ) < < sessionid < < " End time not set by summary? " ;
} else if ( t > session - > last ( ) ) {
2020-06-04 14:01:26 +00:00
// This has only been seen in two instances:
// 1. Once with corrupted data, in which the summary and event files each contained
// multiple conflicting sessions (all brief) with the same ID.
// 2. On one 500G110, multiple PRS1PressureSetEvents appear after the end of the session,
// across roughtly two dozen sessions. These seem to be discarded on official reports,
// see ImportEvent() below.
2019-11-13 16:25:59 +00:00
qWarning ( ) < < sessionid < < " Events continue after summary? " ;
}
// Events can end before the session if the mask was off before the equipment turned off.
2019-11-07 19:19:58 +00:00
}
return true ;
}
void PRS1Import : : ImportEvent ( qint64 t , PRS1ParsedEvent * e )
{
qint64 duration ;
2019-11-19 21:00:08 +00:00
// TODO: Filter out duplicate/overlapping PB and RE events.
//
2022-02-27 16:50:10 +00:00
// These actually get reported by the devices, but they cause "unordered time" warnings
2019-11-19 21:00:08 +00:00
// and they throw off the session statistics. Even official reports show the wrong stats,
// for example counting each of 3 duplicate PBs towards the total time in PB.
//
// It's not clear whether filtering can reasonably be done here or whether it needs
// to be done in ImportEventChunk.
2019-11-07 19:19:58 +00:00
2019-11-19 21:00:08 +00:00
const QVector < ChannelID * > & channels = PRS1ImportChannelMap [ e - > m_type ] ;
2021-08-24 22:48:31 +00:00
ChannelID channel = NoChannel , PS , VS2 ;
2019-11-19 21:00:08 +00:00
if ( channels . count ( ) > 0 ) {
channel = * channels . at ( 0 ) ;
}
switch ( e - > m_type ) {
case PRS1PressureSetEvent : : TYPE : // currentPressure is used to calculate unintentional leak, not just PS
2020-06-04 14:01:26 +00:00
// TODO: These have sometimes been observed with t > session->last() on a 500G110.
// Official reports seem to discard such events, OSCAR currently doesn't.
// Test this more thoroughly before changing behavior here.
// fall through
2019-11-19 21:00:08 +00:00
case PRS1IPAPSetEvent : : TYPE :
case PRS1IPAPAverageEvent : : TYPE :
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ;
m_currentPressure = e - > m_value ;
break ;
case PRS1EPAPSetEvent : : TYPE :
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ;
if ( m_calcPSfromSet ) {
PS = * ( PRS1ImportChannelMap [ PRS1IPAPSetEvent : : TYPE ] . at ( 1 ) ) ;
2019-11-07 19:19:58 +00:00
AddEvent ( PS , t , m_currentPressure - e - > m_value , e - > m_gain ) ; // Pressure Support
2019-11-19 21:00:08 +00:00
}
break ;
case PRS1EPAPAverageEvent : : TYPE :
PS = * channels . at ( 1 ) ;
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ;
AddEvent ( PS , t , m_currentPressure - e - > m_value , e - > m_gain ) ; // Pressure Support
break ;
2019-10-21 01:22:49 +00:00
2019-11-19 21:00:08 +00:00
case PRS1TimedBreathEvent : : TYPE :
// The duration appears to correspond to the length of the timed breath in seconds when multiplied by 0.1 (100ms)!
// TODO: consider changing parsers to use milliseconds for time, since it turns out there's at least one way
// they can express durations less than 1 second.
// TODO: consider allowing OSCAR to record millisecond durations so that the display will say "2.1" instead of "21" or "2".
duration = e - > m_duration * 100L ; // for now do this here rather than in parser, since parser events don't use milliseconds
AddEvent ( * channels . at ( 0 ) , t - duration , e - > m_duration * 0.1F , 0.1F ) ; // TODO: a gain of 0.1 should render this unnecessary, but gain doesn't seem to work currently
break ;
2019-10-21 01:22:49 +00:00
2019-11-19 21:00:08 +00:00
case PRS1ObstructiveApneaEvent : : TYPE :
case PRS1ClearAirwayEvent : : TYPE :
case PRS1HypopneaEvent : : TYPE :
case PRS1FlowLimitationEvent : : TYPE :
AddEvent ( channel , t , e - > m_duration , e - > m_gain ) ;
break ;
2019-10-21 01:22:49 +00:00
2019-11-19 21:00:08 +00:00
case PRS1PeriodicBreathingEvent : : TYPE :
case PRS1LargeLeakEvent : : TYPE :
2020-03-26 13:01:28 +00:00
case PRS1VariableBreathingEvent : : TYPE :
2019-11-19 21:00:08 +00:00
// TODO: The graphs silently treat the timestamp of a span as an end time rather than start (see gFlagsLine::paint).
// Decide whether to preserve that behavior or change it universally and update either this code or comment.
duration = e - > m_duration * 1000L ;
AddEvent ( channel , t + duration , e - > m_duration , e - > m_gain ) ;
break ;
2019-10-21 01:22:49 +00:00
2019-11-19 21:00:08 +00:00
case PRS1SnoreEvent : : TYPE : // snore count that shows up in flags but not waveform
// TODO: The numeric snore graph is the right way to present this information,
// but it needs to be shifted left 2 minutes, since it's not a starting value
// but a past statistic.
2020-05-07 21:36:31 +00:00
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ; // Snore count, continuous data
2019-11-19 21:00:08 +00:00
if ( e - > m_value > 0 ) {
// TODO: currently these get drawn on our waveforms, but they probably shouldn't,
// since they don't have a precise timestamp. They should continue to be drawn
// on the flags overview. See the comment in ImportEventChunk regarding flags
// for numeric channels.
2020-05-07 21:36:31 +00:00
//
// We need to pass the count along so that the VS2 index will tabulate correctly.
2019-11-19 21:00:08 +00:00
VS2 = * channels . at ( 1 ) ;
2020-05-07 21:36:31 +00:00
AddEvent ( VS2 , t , e - > m_value , 1 ) ;
2019-11-19 21:00:08 +00:00
}
break ;
case PRS1VibratorySnoreEvent : : TYPE : // real VS marker on waveform
// TODO: These don't need to be drawn separately on the flag overview, since
// they're presumably included in the overall snore count statistic. They should
// continue to be drawn on the waveform, due to their precise timestamp.
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ;
break ;
2019-10-21 01:22:49 +00:00
2019-11-19 21:00:08 +00:00
default :
if ( channels . count ( ) = = 1 ) {
// For most events, simply pass the value through to the mapped channel.
AddEvent ( channel , t , e - > m_value , e - > m_gain ) ;
} else if ( channels . count ( ) > 1 ) {
// Anything mapped to more than one channel must have a case statement above.
qWarning ( ) < < " Missing import handler for PRS1 event type " < < ( int ) e - > m_type ;
2019-05-24 23:41:42 +00:00
break ;
2019-11-19 21:00:08 +00:00
} else {
// Not imported, no channels mapped to this event
// These will show up in chunk YAML and any user alerts will be driven by the parser.
}
break ;
}
2019-05-24 23:41:42 +00:00
}
2018-05-05 07:14:44 +00:00
2019-08-29 01:30:25 +00:00
CPAPMode PRS1Import : : importMode ( int prs1mode )
{
2020-03-24 21:15:29 +00:00
CPAPMode mode = MODE_UNKNOWN ;
2019-08-29 01:30:25 +00:00
switch ( prs1mode ) {
2020-03-24 21:15:29 +00:00
case PRS1_MODE_CPAPCHECK : mode = MODE_CPAP ; break ;
2019-08-29 01:30:25 +00:00
case PRS1_MODE_CPAP : mode = MODE_CPAP ; break ;
case PRS1_MODE_AUTOCPAP : mode = MODE_APAP ; break ;
2020-03-24 21:15:29 +00:00
case PRS1_MODE_AUTOTRIAL : mode = MODE_APAP ; break ;
2019-08-29 01:30:25 +00:00
case PRS1_MODE_BILEVEL : mode = MODE_BILEVEL_FIXED ; break ;
case PRS1_MODE_AUTOBILEVEL : mode = MODE_BILEVEL_AUTO_VARIABLE_PS ; break ;
case PRS1_MODE_ASV : mode = MODE_ASV_VARIABLE_EPAP ; break ;
2020-03-25 01:33:50 +00:00
case PRS1_MODE_S : mode = MODE_BILEVEL_FIXED ; break ;
case PRS1_MODE_ST : mode = MODE_BILEVEL_FIXED ; break ;
case PRS1_MODE_PC : mode = MODE_BILEVEL_FIXED ; break ;
case PRS1_MODE_ST_AVAPS : mode = MODE_AVAPS ; break ;
case PRS1_MODE_PC_AVAPS : mode = MODE_AVAPS ; break ;
2020-03-24 21:15:29 +00:00
default :
UNEXPECTED_VALUE ( prs1mode , " known PRS1 mode " ) ;
break ;
2019-08-29 01:30:25 +00:00
}
return mode ;
}
2019-05-31 20:58:58 +00:00
bool PRS1Import : : ImportCompliance ( )
2014-05-31 21:25:07 +00:00
{
2019-05-26 18:17:58 +00:00
bool ok ;
ok = compliance - > ParseCompliance ( ) ;
qint64 start = qint64 ( compliance - > timestamp ) * 1000L ;
for ( int i = 0 ; i < compliance - > m_parsedData . count ( ) ; i + + ) {
PRS1ParsedEvent * e = compliance - > m_parsedData . at ( i ) ;
2019-05-28 01:02:28 +00:00
if ( e - > m_type = = PRS1ParsedSliceEvent : : TYPE ) {
2019-11-09 20:09:02 +00:00
AddSlice ( start , e ) ;
2019-05-26 18:17:58 +00:00
continue ;
2019-05-28 01:02:28 +00:00
} else if ( e - > m_type ! = PRS1ParsedSettingEvent : : TYPE ) {
2019-05-26 18:17:58 +00:00
qWarning ( ) < < " Compliance had non-setting event: " < < ( int ) e - > m_type ;
continue ;
}
PRS1ParsedSettingEvent * s = ( PRS1ParsedSettingEvent * ) e ;
switch ( s - > m_setting ) {
case PRS1_SETTING_CPAP_MODE :
2020-03-24 21:15:29 +00:00
session - > settings [ PRS1_Mode ] = ( PRS1Mode ) e - > m_value ;
2019-08-29 01:30:25 +00:00
session - > settings [ CPAP_Mode ] = importMode ( e - > m_value ) ;
2019-05-26 18:17:58 +00:00
break ;
case PRS1_SETTING_PRESSURE :
session - > settings [ CPAP_Pressure ] = e - > value ( ) ;
break ;
2020-09-13 18:00:59 +00:00
case PRS1_SETTING_PRESSURE_MIN :
session - > settings [ CPAP_PressureMin ] = e - > value ( ) ;
break ;
case PRS1_SETTING_PRESSURE_MAX :
session - > settings [ CPAP_PressureMax ] = e - > value ( ) ;
break ;
2019-05-26 18:17:58 +00:00
case PRS1_SETTING_FLEX_MODE :
session - > settings [ PRS1_FlexMode ] = e - > m_value ;
break ;
case PRS1_SETTING_FLEX_LEVEL :
session - > settings [ PRS1_FlexLevel ] = e - > m_value ;
break ;
2020-03-23 16:59:06 +00:00
case PRS1_SETTING_FLEX_LOCK :
session - > settings [ PRS1_FlexLock ] = ( bool ) e - > m_value ;
break ;
2019-05-26 18:17:58 +00:00
case PRS1_SETTING_RAMP_TIME :
session - > settings [ CPAP_RampTime ] = e - > m_value ;
break ;
case PRS1_SETTING_RAMP_PRESSURE :
session - > settings [ CPAP_RampPressure ] = e - > value ( ) ;
break ;
2020-03-23 16:59:06 +00:00
case PRS1_SETTING_RAMP_TYPE :
session - > settings [ PRS1_RampType ] = e - > m_value ;
break ;
2019-05-26 18:17:58 +00:00
case PRS1_SETTING_HUMID_STATUS :
session - > settings [ PRS1_HumidStatus ] = ( bool ) e - > m_value ;
break ;
2020-01-12 00:14:01 +00:00
case PRS1_SETTING_HUMID_MODE :
session - > settings [ PRS1_HumidMode ] = e - > m_value ;
break ;
case PRS1_SETTING_HEATED_TUBE_TEMP :
session - > settings [ PRS1_TubeTemp ] = e - > m_value ;
break ;
2019-05-26 18:17:58 +00:00
case PRS1_SETTING_HUMID_LEVEL :
session - > settings [ PRS1_HumidLevel ] = e - > m_value ;
break ;
2019-09-20 19:38:14 +00:00
case PRS1_SETTING_MASK_RESIST_LOCK :
2020-03-23 17:07:08 +00:00
session - > settings [ PRS1_MaskResistLock ] = ( bool ) e - > m_value ;
2019-09-20 19:38:14 +00:00
break ;
2019-09-25 01:07:10 +00:00
case PRS1_SETTING_MASK_RESIST_SETTING :
2020-08-05 00:59:51 +00:00
session - > settings [ PRS1_MaskResistSet ] = e - > m_value ;
2019-09-25 01:07:10 +00:00
break ;
2019-09-20 19:38:14 +00:00
case PRS1_SETTING_HOSE_DIAMETER :
2020-01-28 19:07:58 +00:00
session - > settings [ PRS1_HoseDiam ] = e - > m_value ;
2019-09-20 19:38:14 +00:00
break ;
2020-03-23 16:59:06 +00:00
case PRS1_SETTING_TUBING_LOCK :
session - > settings [ PRS1_TubeLock ] = ( bool ) e - > m_value ;
break ;
2019-09-20 19:38:14 +00:00
case PRS1_SETTING_AUTO_ON :
session - > settings [ PRS1_AutoOn ] = ( bool ) e - > m_value ;
break ;
case PRS1_SETTING_AUTO_OFF :
session - > settings [ PRS1_AutoOff ] = ( bool ) e - > m_value ;
break ;
case PRS1_SETTING_MASK_ALERT :
session - > settings [ PRS1_MaskAlert ] = ( bool ) e - > m_value ;
break ;
case PRS1_SETTING_SHOW_AHI :
2020-03-23 16:59:06 +00:00
session - > settings [ PRS1_ShowAHI ] = ( bool ) e - > m_value ;
2019-09-20 19:38:14 +00:00
break ;
2019-05-26 18:17:58 +00:00
default :
qWarning ( ) < < " Unknown PRS1 setting type " < < ( int ) s - > m_setting ;
break ;
}
}
2019-06-05 15:12:08 +00:00
if ( ! ok ) {
return false ;
}
if ( compliance - > duration = = 0 ) {
// This does occasionally happen and merely indicates a brief session with no useful data.
2019-10-24 16:19:13 +00:00
// This requires the use of really_set_last below, which otherwise rejects 0 length.
qDebug ( ) < < compliance - > sessionid < < " compliance duration == 0 " ;
2019-05-26 18:17:58 +00:00
}
session - > setSummaryOnly ( true ) ;
session - > set_first ( start ) ;
2019-10-24 16:19:13 +00:00
session - > really_set_last ( qint64 ( compliance - > timestamp + compliance - > duration ) * 1000L ) ;
2019-05-26 18:17:58 +00:00
return true ;
}
2019-11-09 20:09:02 +00:00
void PRS1Import : : AddSlice ( qint64 start , PRS1ParsedEvent * e )
2019-11-07 22:35:09 +00:00
{
2019-11-09 20:09:02 +00:00
// Cache all slices and incrementally calculate their durations.
2019-11-07 22:35:09 +00:00
PRS1ParsedSliceEvent * s = ( PRS1ParsedSliceEvent * ) e ;
qint64 tt = start + qint64 ( s - > m_start ) * 1000L ;
2019-11-09 20:09:02 +00:00
if ( ! m_slices . isEmpty ( ) ) {
SessionSlice & prevSlice = m_slices . last ( ) ;
2019-11-07 22:35:09 +00:00
prevSlice . end = tt ;
}
2019-11-09 20:09:02 +00:00
m_slices . append ( SessionSlice ( tt , tt , ( SliceStatus ) s - > m_value ) ) ;
2019-11-07 22:35:09 +00:00
}
2019-05-27 14:25:57 +00:00
bool PRS1Import : : ImportSummary ( )
2014-05-31 21:25:07 +00:00
{
2019-06-05 15:12:08 +00:00
if ( ! summary ) {
qWarning ( ) < < " ImportSummary() called with no summary? " ;
return false ;
}
2018-05-05 15:48:32 +00:00
2019-06-05 00:26:43 +00:00
qint64 start = qint64 ( summary - > timestamp ) * 1000L ;
session - > set_first ( start ) ;
2018-05-05 15:48:32 +00:00
2020-03-26 01:14:25 +00:00
// TODO: The below max pressures aren't right for the 30 cmH2O models.
2019-05-27 14:25:57 +00:00
session - > setPhysMax ( CPAP_LeakTotal , 120 ) ;
session - > setPhysMin ( CPAP_LeakTotal , 0 ) ;
session - > setPhysMax ( CPAP_Pressure , 25 ) ;
session - > setPhysMin ( CPAP_Pressure , 4 ) ;
session - > setPhysMax ( CPAP_IPAP , 25 ) ;
session - > setPhysMin ( CPAP_IPAP , 4 ) ;
session - > setPhysMax ( CPAP_EPAP , 25 ) ;
session - > setPhysMin ( CPAP_EPAP , 4 ) ;
session - > setPhysMax ( CPAP_PS , 25 ) ;
session - > setPhysMin ( CPAP_PS , 0 ) ;
2019-05-27 14:28:14 +00:00
bool ok ;
2019-05-27 15:14:55 +00:00
ok = summary - > ParseSummary ( ) ;
2019-05-27 14:28:14 +00:00
2020-03-26 01:14:25 +00:00
PRS1Mode nativemode = PRS1_MODE_UNKNOWN ;
2019-08-29 01:30:25 +00:00
CPAPMode cpapmode = MODE_UNKNOWN ;
2020-12-30 01:23:37 +00:00
bool humidifierConnected = false ;
2019-05-27 14:28:14 +00:00
for ( int i = 0 ; i < summary - > m_parsedData . count ( ) ; i + + ) {
PRS1ParsedEvent * e = summary - > m_parsedData . at ( i ) ;
2019-06-05 00:26:43 +00:00
if ( e - > m_type = = PRS1ParsedSliceEvent : : TYPE ) {
2019-11-09 20:09:02 +00:00
AddSlice ( start , e ) ;
2019-06-05 00:26:43 +00:00
continue ;
} else if ( e - > m_type ! = PRS1ParsedSettingEvent : : TYPE ) {
2019-05-27 14:28:14 +00:00
qWarning ( ) < < " Summary had non-setting event: " < < ( int ) e - > m_type ;
continue ;
}
PRS1ParsedSettingEvent * s = ( PRS1ParsedSettingEvent * ) e ;
switch ( s - > m_setting ) {
case PRS1_SETTING_CPAP_MODE :
2020-03-26 01:14:25 +00:00
nativemode = ( PRS1Mode ) e - > m_value ;
2019-08-29 01:30:25 +00:00
cpapmode = importMode ( e - > m_value ) ;
2019-05-27 14:28:14 +00:00
break ;
case PRS1_SETTING_PRESSURE :
session - > settings [ CPAP_Pressure ] = e - > value ( ) ;
break ;
case PRS1_SETTING_PRESSURE_MIN :
session - > settings [ CPAP_PressureMin ] = e - > value ( ) ;
break ;
case PRS1_SETTING_PRESSURE_MAX :
session - > settings [ CPAP_PressureMax ] = e - > value ( ) ;
break ;
case PRS1_SETTING_EPAP :
session - > settings [ CPAP_EPAP ] = e - > value ( ) ;
break ;
case PRS1_SETTING_IPAP :
session - > settings [ CPAP_IPAP ] = e - > value ( ) ;
break ;
case PRS1_SETTING_PS :
session - > settings [ CPAP_PS ] = e - > value ( ) ;
break ;
case PRS1_SETTING_EPAP_MIN :
session - > settings [ CPAP_EPAPLo ] = e - > value ( ) ;
break ;
case PRS1_SETTING_EPAP_MAX :
session - > settings [ CPAP_EPAPHi ] = e - > value ( ) ;
break ;
case PRS1_SETTING_IPAP_MIN :
session - > settings [ CPAP_IPAPLo ] = e - > value ( ) ;
break ;
case PRS1_SETTING_IPAP_MAX :
session - > settings [ CPAP_IPAPHi ] = e - > value ( ) ;
break ;
case PRS1_SETTING_PS_MIN :
session - > settings [ CPAP_PSMin ] = e - > value ( ) ;
break ;
case PRS1_SETTING_PS_MAX :
session - > settings [ CPAP_PSMax ] = e - > value ( ) ;
break ;
case PRS1_SETTING_FLEX_MODE :
session - > settings [ PRS1_FlexMode ] = e - > m_value ;
break ;
case PRS1_SETTING_FLEX_LEVEL :
session - > settings [ PRS1_FlexLevel ] = e - > m_value ;
break ;
2020-03-23 16:59:06 +00:00
case PRS1_SETTING_FLEX_LOCK :
session - > settings [ PRS1_FlexLock ] = ( bool ) e - > m_value ;
break ;
2019-05-27 14:28:14 +00:00
case PRS1_SETTING_RAMP_TIME :
session - > settings [ CPAP_RampTime ] = e - > m_value ;
break ;
case PRS1_SETTING_RAMP_PRESSURE :
session - > settings [ CPAP_RampPressure ] = e - > value ( ) ;
break ;
2020-03-23 16:59:06 +00:00
case PRS1_SETTING_RAMP_TYPE :
session - > settings [ PRS1_RampType ] = e - > m_value ;
break ;
2019-05-27 14:28:14 +00:00
case PRS1_SETTING_HUMID_STATUS :
2020-12-30 01:23:37 +00:00
humidifierConnected = ( bool ) e - > m_value ;
session - > settings [ PRS1_HumidStatus ] = humidifierConnected ;
2019-05-27 14:28:14 +00:00
break ;
2020-01-12 00:14:01 +00:00
case PRS1_SETTING_HUMID_MODE :
session - > settings [ PRS1_HumidMode ] = e - > m_value ;
break ;
case PRS1_SETTING_HEATED_TUBE_TEMP :
session - > settings [ PRS1_TubeTemp ] = e - > m_value ;
2019-05-27 15:05:34 +00:00
break ;
2019-05-27 14:28:14 +00:00
case PRS1_SETTING_HUMID_LEVEL :
session - > settings [ PRS1_HumidLevel ] = e - > m_value ;
break ;
2020-12-30 01:23:37 +00:00
case PRS1_SETTING_HUMID_TARGET_TIME :
// Only import this setting if there's a humidifier connected.
// (This setting appears in the data even when it's disconnected.)
// TODO: Consider moving this logic into the parser for target time.
if ( humidifierConnected ) {
if ( e - > m_value > 1 ) {
// use scaled numeric value
session - > settings [ PRS1_HumidTargetTime ] = e - > value ( ) ;
} else {
// use unscaled 0 or 1 for Off or Auto respectively
session - > settings [ PRS1_HumidTargetTime ] = e - > m_value ;
}
}
break ;
2019-09-20 04:15:40 +00:00
case PRS1_SETTING_MASK_RESIST_LOCK :
2020-03-23 17:07:08 +00:00
session - > settings [ PRS1_MaskResistLock ] = ( bool ) e - > m_value ;
2019-05-27 14:28:14 +00:00
break ;
2019-09-20 04:15:40 +00:00
case PRS1_SETTING_MASK_RESIST_SETTING :
2020-03-23 17:07:08 +00:00
session - > settings [ PRS1_MaskResistSet ] = e - > m_value ;
2019-05-27 14:28:14 +00:00
break ;
case PRS1_SETTING_HOSE_DIAMETER :
2020-01-28 19:07:58 +00:00
session - > settings [ PRS1_HoseDiam ] = e - > m_value ;
2019-05-27 14:28:14 +00:00
break ;
2020-03-23 16:59:06 +00:00
case PRS1_SETTING_TUBING_LOCK :
session - > settings [ PRS1_TubeLock ] = ( bool ) e - > m_value ;
break ;
2019-05-27 14:28:14 +00:00
case PRS1_SETTING_AUTO_ON :
session - > settings [ PRS1_AutoOn ] = ( bool ) e - > m_value ;
break ;
case PRS1_SETTING_AUTO_OFF :
session - > settings [ PRS1_AutoOff ] = ( bool ) e - > m_value ;
break ;
case PRS1_SETTING_MASK_ALERT :
session - > settings [ PRS1_MaskAlert ] = ( bool ) e - > m_value ;
break ;
case PRS1_SETTING_SHOW_AHI :
2020-03-23 16:59:06 +00:00
session - > settings [ PRS1_ShowAHI ] = ( bool ) e - > m_value ;
2019-05-27 14:28:14 +00:00
break ;
2019-09-23 17:43:28 +00:00
case PRS1_SETTING_BACKUP_BREATH_MODE :
2020-03-24 01:09:55 +00:00
session - > settings [ PRS1_BackupBreathMode ] = e - > m_value ;
break ;
2019-09-23 17:43:28 +00:00
case PRS1_SETTING_BACKUP_BREATH_RATE :
2020-03-24 01:09:55 +00:00
session - > settings [ PRS1_BackupBreathRate ] = e - > m_value ;
break ;
2019-09-23 17:43:28 +00:00
case PRS1_SETTING_BACKUP_TIMED_INSPIRATION :
2020-03-24 21:15:29 +00:00
session - > settings [ PRS1_BackupBreathTi ] = e - > value ( ) ;
2020-03-24 01:09:55 +00:00
break ;
2019-09-20 16:59:14 +00:00
case PRS1_SETTING_TIDAL_VOLUME :
2020-03-24 01:09:55 +00:00
session - > settings [ CPAP_TidalVolume ] = e - > m_value ;
break ;
2020-03-26 01:14:25 +00:00
case PRS1_SETTING_AUTO_TRIAL : // new to F0V6
2020-03-24 01:09:55 +00:00
session - > settings [ PRS1_AutoTrial ] = e - > m_value ;
2020-03-26 01:14:25 +00:00
nativemode = PRS1_MODE_AUTOTRIAL ; // Note: F0V6 reports show the underlying CPAP mode rather than Auto-Trial.
cpapmode = importMode ( nativemode ) ;
2020-03-24 01:09:55 +00:00
break ;
2019-09-20 16:59:14 +00:00
case PRS1_SETTING_EZ_START :
2020-03-24 01:09:55 +00:00
session - > settings [ PRS1_EZStart ] = ( bool ) e - > m_value ;
break ;
2019-09-23 16:39:20 +00:00
case PRS1_SETTING_RISE_TIME :
2020-03-24 01:09:55 +00:00
session - > settings [ PRS1_RiseTime ] = e - > m_value ;
break ;
2020-03-23 00:00:09 +00:00
case PRS1_SETTING_RISE_TIME_LOCK :
2020-03-24 01:09:55 +00:00
session - > settings [ PRS1_RiseTimeLock ] = ( bool ) e - > m_value ;
break ;
2019-09-20 16:59:14 +00:00
case PRS1_SETTING_APNEA_ALARM :
case PRS1_SETTING_DISCONNECT_ALARM :
case PRS1_SETTING_LOW_MV_ALARM :
case PRS1_SETTING_LOW_TV_ALARM :
2020-03-24 01:09:55 +00:00
// TODO: define and add new channels for alarms once we have more samples and can reliably parse them.
2019-09-20 16:59:14 +00:00
break ;
2019-05-27 14:28:14 +00:00
default :
qWarning ( ) < < " Unknown PRS1 setting type " < < ( int ) s - > m_setting ;
break ;
}
}
if ( ! ok ) {
return false ;
}
2019-06-05 21:08:45 +00:00
2020-03-26 01:14:25 +00:00
if ( summary - > m_parsedData . count ( ) > 0 ) {
if ( nativemode = = PRS1_MODE_UNKNOWN ) UNEXPECTED_VALUE ( nativemode , " known mode " ) ;
if ( cpapmode = = MODE_UNKNOWN ) UNEXPECTED_VALUE ( cpapmode , " known mode " ) ;
session - > settings [ PRS1_Mode ] = nativemode ;
session - > settings [ CPAP_Mode ] = cpapmode ;
}
2019-06-05 15:12:08 +00:00
if ( summary - > duration = = 0 ) {
// This does occasionally happen and merely indicates a brief session with no useful data.
2019-11-13 16:25:59 +00:00
// This requires the use of really_set_last below, which otherwise rejects 0 length.
//qDebug() << summary->sessionid << "session duration == 0";
2019-06-05 15:12:08 +00:00
}
2019-11-13 16:25:59 +00:00
session - > really_set_last ( qint64 ( summary - > timestamp + summary - > duration ) * 1000L ) ;
2019-06-05 21:08:45 +00:00
2019-05-27 14:28:14 +00:00
return true ;
}
2019-10-24 20:25:36 +00:00
bool PRS1Import : : ImportEvents ( )
2014-05-31 21:25:07 +00:00
{
2019-11-09 20:09:02 +00:00
bool ok = true ;
2019-10-24 20:25:36 +00:00
2019-11-13 14:27:47 +00:00
for ( auto & event : m_event_chunks . values ( ) ) {
bool chunk_ok = this - > ImportEventChunk ( event ) ;
if ( ! chunk_ok & & m_event_chunks . count ( ) > 1 ) {
// Specify which chunk had problems if there's more than one. ParseSession will warn about the overall result.
qWarning ( ) < < event - > sessionid < < QString ( " Error parsing events in %1 @ %2, continuing " )
. arg ( relativePath ( event - > m_path ) )
. arg ( event - > m_filepos ) ;
2011-12-10 12:14:48 +00:00
}
2019-11-13 14:27:47 +00:00
ok & = chunk_ok ;
2014-05-31 21:25:07 +00:00
}
2019-10-24 20:25:36 +00:00
if ( ok ) {
2019-11-19 17:29:45 +00:00
// Sanity check: warn if channels' eventlists don't line up with the final mask-on slices.
// First make a list of the mask-on slices that will be imported (nonzero duration)
QVector < SessionSlice > maskOn ;
for ( auto & slice : m_slices ) {
2020-09-13 23:00:16 +00:00
if ( slice . status = = MaskOn ) {
if ( slice . end > slice . start ) {
maskOn . append ( slice ) ;
} else {
qWarning ( ) < < this - > sessionid < < " Dropping empty mask-on slice: " < < ts ( slice . start ) ;
}
2019-11-19 17:29:45 +00:00
}
}
// Then go through each required channel and make sure each eventlist is within
// the bounds of the corresponding slice, warn if not.
if ( maskOn . count ( ) > 0 & & m_event_chunks . count ( ) > 0 ) {
2020-04-22 17:25:59 +00:00
QVector < SessionSlice > maskOnWithEvents = maskOn ;
2020-04-22 00:49:15 +00:00
if ( m_event_chunks . first ( ) - > family = = 3 & & m_event_chunks . first ( ) - > familyVersion < = 3 ) {
2020-04-22 17:25:59 +00:00
// F3V0 and F3V3 sometimes omit (empty) event chunks if the mask-on slice is shorter than 2 minutes.
// Specifically, 1061401 and 1061T always do, but 1160P usually doesn't. Sometimes 1160P will omit
// just the first event chunk if the first mask-on slice is shorter than 2 minutes.
int empty = maskOn . count ( ) - m_event_chunks . count ( ) ;
if ( empty > 0 ) {
// If there are fewer event chunks than mask-on slices, filter the list to have just the
// mask-on slices that we expect to have events.
int skipped = 0 ;
maskOnWithEvents . clear ( ) ;
for ( auto & slice : maskOn ) {
if ( skipped < empty & & slice . end - slice . start < 120 * 1000L ) {
skipped + + ;
continue ;
}
maskOnWithEvents . append ( slice ) ;
}
2019-11-19 19:18:13 +00:00
}
}
2020-04-22 17:25:59 +00:00
if ( maskOnWithEvents . count ( ) < m_event_chunks . count ( ) ) {
qWarning ( ) < < sessionid < < " has more event chunks than mask-on slices! " ;
}
2019-11-19 17:29:45 +00:00
const QVector < PRS1ParsedEventType > & supported = GetSupportedEvents ( m_event_chunks . first ( ) ) ;
for ( auto & e : supported ) {
if ( ! PRS1OnDemandChannels . contains ( e ) & & ! PRS1NonSliceChannels . contains ( e ) ) {
for ( auto & pChannelID : PRS1ImportChannelMap [ e ] ) {
auto & eventlists = session - > eventlist [ * pChannelID ] ;
2020-04-22 17:25:59 +00:00
if ( eventlists . count ( ) ! = maskOnWithEvents . count ( ) ) {
qWarning ( ) < < sessionid < < " has " < < maskOnWithEvents . count ( ) < < " mask-on slices, channel "
2019-11-19 17:29:45 +00:00
< < * pChannelID < < " has " < < eventlists . count ( ) < < " eventlists " ;
continue ;
}
for ( int i = 0 ; i < eventlists . count ( ) ; i + + ) {
if ( eventlists [ i ] - > count ( ) = = 0 ) continue ; // no first/last timestamp
2020-04-22 17:25:59 +00:00
auto & list = eventlists [ i ] ;
auto & slice = maskOnWithEvents [ i ] ;
if ( list - > first ( ) < slice . start | | list - > first ( ) > slice . end | |
list - > last ( ) < slice . start | | list - > last ( ) > slice . end ) {
2019-11-19 19:18:13 +00:00
qWarning ( ) < < sessionid < < " channel " < < * pChannelID < < " has events outside of mask-on slice " < < i ;
2019-11-19 17:29:45 +00:00
}
}
}
}
}
}
2020-04-22 17:25:59 +00:00
// The above is just sanity-checking the results of our import process, that discontinuous
// data is fully contained within mask-on slices.
2019-11-19 17:29:45 +00:00
2019-10-24 20:25:36 +00:00
session - > m_cnt . clear ( ) ;
session - > m_cph . clear ( ) ;
session - > m_valuesummary [ CPAP_Pressure ] . clear ( ) ;
session - > m_valuesummary . erase ( session - > m_valuesummary . find ( CPAP_Pressure ) ) ;
}
2019-10-24 15:07:47 +00:00
return ok ;
2014-05-31 21:25:07 +00:00
}
2014-04-17 05:58:57 +00:00
2019-05-14 01:20:11 +00:00
QList < PRS1DataChunk * > PRS1Import : : CoalesceWaveformChunks ( QList < PRS1DataChunk * > & allchunks )
{
QList < PRS1DataChunk * > coalesced ;
PRS1DataChunk * chunk = nullptr , * lastchunk = nullptr ;
2019-05-14 20:20:32 +00:00
int num ;
2019-05-14 01:20:11 +00:00
for ( int i = 0 ; i < allchunks . size ( ) ; + + i ) {
chunk = allchunks . at ( i ) ;
2019-12-01 21:42:54 +00:00
// Log mismatched waveform session IDs
QFileInfo fi ( chunk - > m_path ) ;
bool numeric ;
QString session_s = fi . fileName ( ) . section ( " . " , 0 , - 2 ) ;
2023-10-30 16:47:22 +00:00
quint32 sid = session_s . toInt ( & numeric , m_sessionid_base ) ;
2019-12-01 21:42:54 +00:00
if ( ! numeric | | sid ! = chunk - > sessionid ) {
2019-12-02 22:30:28 +00:00
qWarning ( ) < < chunk - > m_path < < " @ " < < chunk - > m_filepos < < " session ID mismatch: " < < chunk - > sessionid ;
2019-12-01 21:42:54 +00:00
}
2019-05-14 01:20:11 +00:00
if ( lastchunk ! = nullptr ) {
2019-12-02 22:30:28 +00:00
// A handful of 960P waveform files have been observed to have multiple sessions.
//
// This breaks the current approach of deferring waveform parsing until the (multithreaded)
// import, since each session is in a separate import task and could be in a separate
// thread, or already imported by the time it is discovered that this file contains
// more than one session.
//
// For now, we just dump the chunks that don't belong to the session currently
// being imported in this thread, since this happens so rarely.
//
// TODO: Rework the import process to handle waveform data after compliance/summary/
// events (since we're no longer inferring session information from it) and add it to the
// newly imported sessions.
2019-05-14 01:20:11 +00:00
if ( lastchunk - > sessionid ! = chunk - > sessionid ) {
2019-12-02 22:30:28 +00:00
qWarning ( ) < < chunk - > m_path < < " @ " < < chunk - > m_filepos
< < " session ID " < < lastchunk - > sessionid < < " -> " < < chunk - > sessionid
< < " , skipping " < < allchunks . size ( ) - i < < " remaining chunks " ;
2019-05-14 01:20:11 +00:00
// Free any remaining chunks
for ( int j = i ; j < allchunks . size ( ) ; + + j ) {
chunk = allchunks . at ( j ) ;
delete chunk ;
}
break ;
}
2019-05-14 20:20:32 +00:00
// Check whether the data format is the same between the two chunks
bool same_format = ( lastchunk - > waveformInfo . size ( ) = = chunk - > waveformInfo . size ( ) ) ;
if ( same_format ) {
num = chunk - > waveformInfo . size ( ) ;
for ( int n = 0 ; n < num ; n + + ) {
const PRS1Waveform & a = lastchunk - > waveformInfo . at ( n ) ;
const PRS1Waveform & b = chunk - > waveformInfo . at ( n ) ;
if ( a . interleave ! = b . interleave ) {
// We've never seen this before
qWarning ( ) < < chunk - > m_path < < " format change? " < < a . interleave < < b . interleave ;
same_format = false ;
break ;
}
}
} else {
// We've never seen this before
qWarning ( ) < < chunk - > m_path < < " channels change? " < < lastchunk - > waveformInfo . size ( ) < < chunk - > waveformInfo . size ( ) ;
}
2019-05-14 01:20:11 +00:00
qint64 diff = ( chunk - > timestamp - lastchunk - > timestamp ) - lastchunk - > duration ;
2019-05-14 20:20:32 +00:00
if ( same_format & & diff = = 0 ) {
// Same format and in sync, so append waveform data to previous chunk
2019-05-14 01:20:11 +00:00
lastchunk - > m_data . append ( chunk - > m_data ) ;
lastchunk - > duration + = chunk - > duration ;
delete chunk ;
continue ;
}
// else start a new chunk to resync
}
2019-05-14 20:20:32 +00:00
// Report any formats we haven't seen before
num = chunk - > waveformInfo . size ( ) ;
if ( num > 2 ) {
qDebug ( ) < < chunk - > m_path < < num < < " channels " ;
}
for ( int n = 0 ; n < num ; n + + ) {
int interleave = chunk - > waveformInfo . at ( n ) . interleave ;
2019-12-02 22:30:28 +00:00
switch ( chunk - > ext ) {
case 5 : // flow data, 5 samples per second
if ( interleave ! = 5 ) {
qDebug ( ) < < chunk - > m_path < < " interleave? " < < interleave ;
}
break ;
case 6 : // oximetry, 1 sample per second
if ( interleave ! = 1 ) {
qDebug ( ) < < chunk - > m_path < < " interleave? " < < interleave ;
}
break ;
default :
qWarning ( ) < < chunk - > m_path < < " unknown waveform? " < < chunk - > ext ;
break ;
2019-05-14 20:20:32 +00:00
}
}
2019-05-14 01:20:11 +00:00
coalesced . append ( chunk ) ;
lastchunk = chunk ;
}
2020-01-19 01:16:31 +00:00
// In theory there could be broken sessions that have waveform data but no summary or events.
// Those waveforms won't be skipped by the scanner, so we have to check for them here.
//
// This won't be perfect, since any coalesced chunks starting after midnight of the threshhold
// date will also be imported, but those should be relatively few, and tolerable imprecision.
QList < PRS1DataChunk * > coalescedAndFiltered ;
2021-09-01 19:31:06 +00:00
qint64 ignoreBefore = loader - > context ( ) - > IgnoreSessionsOlderThan ( ) . toMSecsSinceEpoch ( ) / 1000 ;
bool ignoreOldSessions = loader - > context ( ) - > ShouldIgnoreOldSessions ( ) ;
2020-01-19 01:16:31 +00:00
for ( auto & chunk : coalesced ) {
if ( ignoreOldSessions & & chunk - > timestamp < ignoreBefore ) {
2020-01-21 18:49:02 +00:00
qWarning ( ) . noquote ( ) < < relativePath ( chunk - > m_path ) < < " skipping session " < < chunk - > sessionid < < " : "
< < QDateTime : : fromMSecsSinceEpoch ( chunk - > timestamp * 1000 ) . toString ( ) < < " older than "
< < QDateTime : : fromMSecsSinceEpoch ( ignoreBefore * 1000 ) . toString ( ) ;
2020-01-29 15:43:58 +00:00
delete chunk ;
2020-01-19 01:16:31 +00:00
continue ;
}
coalescedAndFiltered . append ( chunk ) ;
}
return coalescedAndFiltered ;
2019-05-14 01:20:11 +00:00
}
2020-04-29 14:40:23 +00:00
void PRS1Import : : ImportOximetry ( )
2014-09-29 14:41:31 +00:00
{
2018-06-04 20:48:38 +00:00
int size = oximetry . size ( ) ;
2014-09-29 14:41:31 +00:00
for ( int i = 0 ; i < size ; + + i ) {
2018-06-04 20:48:38 +00:00
PRS1DataChunk * oxi = oximetry . at ( i ) ;
2014-09-29 14:41:31 +00:00
int num = oxi - > waveformInfo . size ( ) ;
2019-12-02 22:30:28 +00:00
CHECK_VALUE ( num , 2 ) ;
2014-09-29 14:41:31 +00:00
int size = oxi - > m_data . size ( ) ;
if ( size = = 0 ) {
2019-05-13 16:11:04 +00:00
qDebug ( ) < < oxi - > sessionid < < oxi - > timestamp < < " empty? " ;
2014-09-29 14:41:31 +00:00
continue ;
}
quint64 ti = quint64 ( oxi - > timestamp ) * 1000L ;
qint64 dur = qint64 ( oxi - > duration ) * 1000L ;
if ( num > 1 ) {
2020-01-28 21:04:34 +00:00
CHECK_VALUE ( oxi - > waveformInfo . at ( 0 ) . interleave , 1 ) ;
CHECK_VALUE ( oxi - > waveformInfo . at ( 1 ) . interleave , 1 ) ;
2014-09-29 14:41:31 +00:00
// Process interleaved samples
QVector < QByteArray > data ;
data . resize ( num ) ;
int pos = 0 ;
do {
for ( int n = 0 ; n < num ; n + + ) {
int interleave = oxi - > waveformInfo . at ( n ) . interleave ;
data [ n ] . append ( oxi - > m_data . mid ( pos , interleave ) ) ;
pos + = interleave ;
}
} while ( pos < size ) ;
2020-01-28 21:04:34 +00:00
CHECK_VALUE ( data [ 0 ] . size ( ) , data [ 1 ] . size ( ) ) ;
2014-09-29 14:41:31 +00:00
2020-01-28 21:04:34 +00:00
ImportOximetryChannel ( OXI_Pulse , data [ 0 ] , ti , dur ) ;
2014-09-29 14:41:31 +00:00
2020-01-28 21:04:34 +00:00
ImportOximetryChannel ( OXI_SPO2 , data [ 1 ] , ti , dur ) ;
}
}
}
void PRS1Import : : ImportOximetryChannel ( ChannelID channel , QByteArray & data , quint64 ti , qint64 dur )
{
if ( data . size ( ) = = 0 )
return ;
2020-01-28 22:36:02 +00:00
unsigned char * raw = ( unsigned char * ) data . data ( ) ;
qint64 step = dur / data . size ( ) ;
CHECK_VALUE ( dur % data . size ( ) , 0 ) ;
bool pending_samples = false ;
quint64 start_ti ;
int start_i ;
2020-04-29 14:40:23 +00:00
// Split eventlist on invalid values (254-255)
2020-01-28 21:04:34 +00:00
for ( int i = 0 ; i < data . size ( ) ; i + + ) {
2020-01-28 22:36:02 +00:00
unsigned char value = raw [ i ] ;
2020-04-29 14:40:23 +00:00
bool valid = ( value < 254 ) ;
2020-01-28 22:36:02 +00:00
if ( valid ) {
if ( pending_samples = = false ) {
pending_samples = true ;
start_i = i ;
start_ti = ti ;
}
if ( channel = = OXI_Pulse ) {
2020-05-13 17:40:54 +00:00
// Values up through 253 are confirmed to be reported as valid on official reports.
2020-01-28 22:36:02 +00:00
} else {
if ( value > 100 ) UNEXPECTED_VALUE ( value , " <= 100% " ) ;
}
2020-01-28 21:04:34 +00:00
} else {
2020-01-28 22:36:02 +00:00
if ( pending_samples ) {
// Create the pending event list
EventList * el = session - > AddEventList ( channel , EVL_Waveform , 1.0 , 0.0 , 0.0 , 0.0 , step ) ;
el - > AddWaveform ( start_ti , & raw [ start_i ] , i - start_i , ti - start_ti ) ;
pending_samples = false ;
}
2014-09-29 14:41:31 +00:00
}
2020-01-28 22:36:02 +00:00
ti + = step ;
2014-09-29 14:41:31 +00:00
}
2020-01-28 21:04:34 +00:00
2020-01-28 22:36:02 +00:00
if ( pending_samples ) {
// Create the pending event list
EventList * el = session - > AddEventList ( channel , EVL_Waveform , 1.0 , 0.0 , 0.0 , 0.0 , step ) ;
el - > AddWaveform ( start_ti , & raw [ start_i ] , data . size ( ) - start_i , ti - start_ti ) ;
pending_samples = false ;
}
2014-09-29 14:41:31 +00:00
}
2019-06-05 12:34:36 +00:00
2020-04-29 14:40:23 +00:00
void PRS1Import : : ImportWaveforms ( )
2014-05-31 21:25:07 +00:00
{
int size = waveforms . size ( ) ;
2016-02-27 05:39:01 +00:00
quint64 s1 , s2 ;
2011-12-10 12:14:48 +00:00
2019-06-05 12:34:36 +00:00
int discontinuities = 0 ;
2016-03-08 07:22:46 +00:00
qint64 lastti = 0 ;
2016-03-06 03:15:54 +00:00
2014-05-31 21:25:07 +00:00
for ( int i = 0 ; i < size ; + + i ) {
PRS1DataChunk * waveform = waveforms . at ( i ) ;
int num = waveform - > waveformInfo . size ( ) ;
2014-04-17 05:58:57 +00:00
2014-05-31 21:25:07 +00:00
int size = waveform - > m_data . size ( ) ;
if ( size = = 0 ) {
2019-05-13 16:11:04 +00:00
qDebug ( ) < < waveform - > sessionid < < waveform - > timestamp < < " empty? " ;
2011-12-10 12:14:48 +00:00
continue ;
}
2014-05-31 21:25:07 +00:00
quint64 ti = quint64 ( waveform - > timestamp ) * 1000L ;
2016-02-27 05:39:01 +00:00
quint64 dur = qint64 ( waveform - > duration ) * 1000L ;
2014-05-31 21:25:07 +00:00
2019-06-05 12:34:36 +00:00
qint64 diff = ti - lastti ;
if ( ( lastti ! = 0 ) & & ( diff = = 1000 | | diff = = - 1000 ) ) {
2019-11-19 20:05:08 +00:00
// TODO: Handle discontinuities properly.
// Option 1: preserve the discontinuity and make it apparent:
// - In the case of a 1-sec overlap, truncate the previous waveform by 1s (+1 sample).
// - Then start a new eventlist for the new section.
// > The down side of this approach is gaps in the data.
// Option 2: slide the waveform data a fraction of a second to avoid the discontinuity
// - In the case of a single discontinuity, simply adjust the timestamps of each section by 0.5s so they meet.
// - In the case of multiple discontinuities, fitting them is more complicated
// > The down side of this approach is that events won't line up exactly the same as official reports.
//
2022-02-27 16:50:10 +00:00
// Evidently the devices' internal clock drifts slightly, and in some sessions that
2019-06-05 12:34:36 +00:00
// means two adjacent (5-minute) waveform chunks have have a +/- 1 second difference in
2022-02-27 16:50:10 +00:00
// their notion of the correct time, since the devices only record time at 1-second
2019-06-05 12:34:36 +00:00
// resolution. Presumably the real drift is fractional, but there's no way to tell from
// the data.
//
// Encore apparently drops the second chunk entirely if it overlaps with the first
// (even by 1 second), and inserts a 1-second gap in the data if it's 1 second later than
// the first ended.
//
// At worst in the former case it seems preferable to drop the overlap and then one
// additional second to mark the discontinuity. But depending how often these drifts
2019-11-19 20:05:08 +00:00
// occur, it may be possible to adjust all the data so that it's continuous. "Overlapping"
// data is not identical, so it seems like these discontinuities are simply an artifact
// of timestamping at 1-second intervals right around the 1-second boundary.
2019-11-19 17:29:45 +00:00
//qDebug() << waveform->sessionid << "waveform discontinuity:" << (diff / 1000L) << "s @" << ts(waveform->timestamp * 1000L);
2019-06-05 12:34:36 +00:00
discontinuities + + ;
2019-05-13 16:11:04 +00:00
}
2016-03-06 03:15:54 +00:00
2014-05-31 21:25:07 +00:00
if ( num > 1 ) {
2019-06-16 00:56:55 +00:00
float pressure_gain = 0.1F ; // standard pressure gain
2020-04-22 21:00:07 +00:00
if ( ( waveform - > family = = 5 & & ( waveform - > familyVersion = = 2 | | waveform - > familyVersion = = 3 ) ) | |
2019-07-25 02:42:00 +00:00
( waveform - > family = = 3 & & waveform - > familyVersion = = 6 ) ) {
2020-04-22 21:00:07 +00:00
// F5V2, F5V3, and F3V6 use a gain of 0.125 rather than 0.1 to allow for a maximum value of 30 cmH2O
2022-02-27 16:50:10 +00:00
pressure_gain = 0.125F ; // TODO: this should be parameterized somewhere better, once we have a clear idea of which devices use this
2019-06-16 00:56:55 +00:00
}
2014-05-31 21:25:07 +00:00
// Process interleaved samples
QVector < QByteArray > data ;
data . resize ( num ) ;
int pos = 0 ;
do {
for ( int n = 0 ; n < num ; n + + ) {
int interleave = waveform - > waveformInfo . at ( n ) . interleave ;
data [ n ] . append ( waveform - > m_data . mid ( pos , interleave ) ) ;
pos + = interleave ;
}
} while ( pos < size ) ;
2014-04-17 05:58:57 +00:00
2016-02-27 05:39:01 +00:00
s1 = data [ 0 ] . size ( ) ;
s2 = data [ 1 ] . size ( ) ;
if ( s1 > 0 ) {
2018-06-07 00:09:06 +00:00
EventList * flow = session - > AddEventList ( CPAP_FlowRate , EVL_Waveform , 1.0f , 0.0f , 0.0f , 0.0f , double ( dur ) / double ( s1 ) ) ;
2014-07-21 13:47:17 +00:00
flow - > AddWaveform ( ti , ( char * ) data [ 0 ] . data ( ) , data [ 0 ] . size ( ) , dur ) ;
2014-05-31 21:25:07 +00:00
}
2014-04-17 05:58:57 +00:00
2016-02-27 05:39:01 +00:00
if ( s2 > 0 ) {
2021-12-09 17:23:08 +00:00
// NOTE: The 900X (F5V3) firmware V1.0.1 clamps the values at 127 (15.875 cmH2O)
// due to incorrectly treating this value as a signed integer. This bug is fixed
// in firmware V1.0.6.
2019-06-16 00:56:55 +00:00
EventList * pres = session - > AddEventList ( CPAP_MaskPressureHi , EVL_Waveform , pressure_gain , 0.0f , 0.0f , 0.0f , double ( dur ) / double ( s2 ) ) ;
2014-07-21 13:47:17 +00:00
pres - > AddWaveform ( ti , ( unsigned char * ) data [ 1 ] . data ( ) , data [ 1 ] . size ( ) , dur ) ;
2011-12-10 12:14:48 +00:00
}
2014-05-31 21:25:07 +00:00
} else {
// Non interleaved, so can process it much faster
2018-06-07 00:09:06 +00:00
EventList * flow = session - > AddEventList ( CPAP_FlowRate , EVL_Waveform , 1.0f , 0.0f , 0.0f , 0.0f , double ( dur ) / double ( waveform - > m_data . size ( ) ) ) ;
2014-07-21 13:47:17 +00:00
flow - > AddWaveform ( ti , ( char * ) waveform - > m_data . data ( ) , waveform - > m_data . size ( ) , dur ) ;
2011-12-10 12:14:48 +00:00
}
2016-03-06 03:15:54 +00:00
lastti = dur + ti ;
2011-12-10 12:14:48 +00:00
}
2019-06-05 12:34:36 +00:00
if ( discontinuities > 1 ) {
2019-06-11 00:57:05 +00:00
qWarning ( ) < < session - > session ( ) < < " multiple discontinuities! " < < discontinuities ;
2019-06-05 12:34:36 +00:00
}
2011-12-10 12:14:48 +00:00
}
2014-05-31 21:25:07 +00:00
void PRS1Import : : run ( )
2011-07-21 11:02:23 +00:00
{
2019-05-03 18:45:21 +00:00
if ( ParseSession ( ) ) {
2021-09-03 19:42:10 +00:00
loader - > context ( ) - > AddSession ( session ) ;
2019-05-03 18:45:21 +00:00
}
}
bool PRS1Import : : ParseSession ( void )
{
2019-06-05 14:24:32 +00:00
bool ok = false ;
2019-05-03 18:45:21 +00:00
bool save = false ;
2021-09-03 16:45:00 +00:00
session = loader - > context ( ) - > CreateSession ( sessionid ) ;
2014-04-17 05:58:57 +00:00
2019-06-05 14:24:32 +00:00
do {
if ( compliance ! = nullptr ) {
ok = ImportCompliance ( ) ;
if ( ! ok ) {
2019-11-13 16:25:59 +00:00
// We don't see any parse errors with our test data, so warn if there's ever an error encountered.
2019-10-24 15:07:47 +00:00
qWarning ( ) < < sessionid < < " Error parsing compliance, skipping session " ;
2019-06-05 14:24:32 +00:00
break ;
}
}
if ( summary ! = nullptr ) {
if ( compliance ! = nullptr ) {
qWarning ( ) < < sessionid < < " Has both compliance and summary?! " ;
// Never seen this, but try the summary anyway.
}
ok = ImportSummary ( ) ;
if ( ! ok ) {
2019-11-13 16:25:59 +00:00
// We don't see any parse errors with our test data, so warn if there's ever an error encountered.
2019-10-24 15:07:47 +00:00
qWarning ( ) < < sessionid < < " Error parsing summary, skipping session " ;
2019-06-05 14:24:32 +00:00
break ;
}
}
if ( compliance = = nullptr & & summary = = nullptr ) {
2019-11-13 16:25:59 +00:00
// With one exception, the only time we've seen missing .000 or .001 data has been with a corrupted card,
// or occasionally with partial cards where the .002 is the first file in the Pn directory
// and we're missing the preceding directory. Since the lack of compliance or summary means we
// don't know the therapy settings or if the mask was ever off, we just skip this very rare case.
2019-06-05 14:24:32 +00:00
qWarning ( ) < < sessionid < < " No compliance or summary, skipping session " ;
break ;
}
2019-11-09 20:09:02 +00:00
// Import the slices into the session
for ( auto & slice : m_slices ) {
// Filter out 0-length slices, since they cause problems for Day::total_time().
if ( slice . end > slice . start ) {
// Filter out everything except mask on/off, since gSessionTimesChart::paint assumes those are the only options.
if ( slice . status = = MaskOn ) {
session - > m_slices . append ( slice ) ;
} else if ( slice . status = = MaskOff ) {
2019-11-19 20:05:08 +00:00
// Mark this slice as BND
AddEvent ( PRS1_BND , slice . end , ( slice . end - slice . start ) / 1000L , 1.0 ) ;
2019-11-09 20:09:02 +00:00
session - > m_slices . append ( slice ) ;
}
}
}
2019-11-13 16:39:02 +00:00
2019-11-14 01:44:35 +00:00
// If are no mask-on slices, then there's not any meaningful event or waveform data for the session.
// If there's no no event or waveform data, mark this session as a summary.
2020-04-29 14:22:12 +00:00
if ( session - > m_slices . count ( ) = = 0 | | ( m_event_chunks . count ( ) = = 0 & & m_wavefiles . isEmpty ( ) & & m_oxifiles . isEmpty ( ) ) ) {
2019-11-14 01:44:35 +00:00
session - > setSummaryOnly ( true ) ;
save = true ;
break ; // and skip the occasional fragmentary event or waveform data
}
2019-11-13 16:25:59 +00:00
// TODO: There should be a way to distinguish between no-data-to-import vs. parsing errors
// (once we figure out what's benign and what isn't).
2019-10-24 20:25:36 +00:00
if ( m_event_chunks . count ( ) > 0 ) {
ok = ImportEvents ( ) ;
2019-06-05 14:24:32 +00:00
if ( ! ok ) {
qWarning ( ) < < sessionid < < " Error parsing events, proceeding anyway? " ;
}
2014-05-31 21:25:07 +00:00
}
2018-05-05 07:14:44 +00:00
2019-10-26 01:17:41 +00:00
if ( ! m_wavefiles . isEmpty ( ) ) {
// Parse .005 Waveform files
2020-04-29 14:22:12 +00:00
waveforms = ReadWaveformData ( m_wavefiles , " Waveform " ) ;
2020-04-29 14:40:23 +00:00
// Extract and import raw data into channels.
ImportWaveforms ( ) ;
2018-05-05 07:14:44 +00:00
}
2014-08-04 15:40:56 +00:00
2020-04-29 14:22:12 +00:00
if ( ! m_oxifiles . isEmpty ( ) ) {
// Parse .006 Waveform files
oximetry = ReadWaveformData ( m_oxifiles , " Oximetry " ) ;
2020-04-29 14:40:23 +00:00
// Extract and import raw data into channels.
ImportOximetry ( ) ;
2019-06-05 14:24:32 +00:00
}
2014-09-29 14:41:31 +00:00
2019-11-13 16:25:59 +00:00
save = true ;
2019-06-05 14:24:32 +00:00
} while ( false ) ;
2019-05-03 18:45:21 +00:00
return save ;
}
2016-03-02 07:07:38 +00:00
2014-04-17 05:58:57 +00:00
2020-04-29 14:22:12 +00:00
QList < PRS1DataChunk * > PRS1Import : : ReadWaveformData ( QList < QString > & files , const char * label )
2019-10-26 01:17:41 +00:00
{
QMap < qint64 , PRS1DataChunk * > waveform_chunks ;
2020-04-29 14:22:12 +00:00
QList < PRS1DataChunk * > result ;
2019-10-26 01:17:41 +00:00
2020-04-29 14:22:12 +00:00
if ( files . count ( ) > 1 ) {
qDebug ( ) < < session - > session ( ) < < label < < " data split across multiple files " ;
2019-10-26 01:17:41 +00:00
}
2020-04-29 14:22:12 +00:00
for ( auto & f : files ) {
// Parse a single .005 or .006 waveform file
2019-10-26 01:17:41 +00:00
QList < PRS1DataChunk * > file_chunks = loader - > ParseFile ( f ) ;
for ( auto & chunk : file_chunks ) {
PRS1DataChunk * previous = waveform_chunks [ chunk - > timestamp ] ;
if ( previous ! = nullptr ) {
// Skip any chunks with identical timestamps. Never yet seen, so warn.
qWarning ( ) < < chunkComparison ( chunk , previous ) ;
delete chunk ;
continue ;
}
waveform_chunks [ chunk - > timestamp ] = chunk ;
}
}
// Get the list of pointers sorted by timestamp.
2020-04-29 14:22:12 +00:00
result = waveform_chunks . values ( ) ;
2019-10-26 01:17:41 +00:00
// Coalesce contiguous waveform chunks into larger chunks.
2020-04-29 14:22:12 +00:00
result = CoalesceWaveformChunks ( result ) ;
2019-10-26 01:17:41 +00:00
2020-04-29 14:22:12 +00:00
return result ;
2019-10-26 01:17:41 +00:00
}
2018-04-27 04:29:03 +00:00
QList < PRS1DataChunk * > PRS1Loader : : ParseFile ( const QString & path )
2018-03-23 19:24:29 +00:00
{
QList < PRS1DataChunk * > CHUNKS ;
2019-05-13 16:11:04 +00:00
if ( path . isEmpty ( ) ) {
// ParseSession passes empty filepaths for waveforms if none exist.
//qWarning() << path << "ParseFile given empty path";
2018-03-23 19:24:29 +00:00
return CHUNKS ;
2019-05-13 16:11:04 +00:00
}
2018-03-23 19:24:29 +00:00
QFile f ( path ) ;
if ( ! f . exists ( ) ) {
2019-05-13 16:11:04 +00:00
qWarning ( ) < < path < < " missing " ;
2018-03-23 19:24:29 +00:00
return CHUNKS ;
}
if ( ! f . open ( QIODevice : : ReadOnly ) ) {
2019-05-13 16:11:04 +00:00
qWarning ( ) < < path < < " can't open " ;
2018-03-23 19:24:29 +00:00
return CHUNKS ;
}
2021-05-23 16:43:31 +00:00
RawDataFile * src ;
2021-11-03 20:38:20 +00:00
if ( QFileInfo ( f ) . suffix ( ) . toUpper ( ) . startsWith ( " B " ) ) { // .B01, .B02, etc.
2021-05-23 16:43:31 +00:00
// If it's a DS2 file, insert the DS2 wrapper to decode the chunk stream.
2021-12-06 21:14:11 +00:00
PRDS2File * ds2 = new PRDS2File ( f , m_keycache ) ;
2021-11-03 20:38:20 +00:00
if ( ! ds2 - > isValid ( ) ) {
2021-12-03 14:57:29 +00:00
//qWarning() << path << "unable to decrypt";
2021-11-03 20:38:20 +00:00
delete ds2 ;
return CHUNKS ;
}
src = ds2 ;
2021-05-23 16:43:31 +00:00
} else {
// Otherwise just use the file as input.
src = new RawDataFile ( f ) ;
}
2018-03-23 19:24:29 +00:00
PRS1DataChunk * chunk = nullptr , * lastchunk = nullptr ;
2019-05-14 22:57:04 +00:00
int cnt = 0 ;
do {
2021-05-23 16:43:31 +00:00
chunk = PRS1DataChunk : : ParseNext ( * src , this ) ;
2019-05-14 22:57:04 +00:00
if ( chunk = = nullptr ) {
break ;
}
2019-05-15 02:49:41 +00:00
chunk - > SetIndex ( cnt ) ; // for logging/debugging purposes
2019-05-14 22:57:04 +00:00
if ( lastchunk ! = nullptr ) {
if ( ( lastchunk - > fileVersion ! = chunk - > fileVersion )
| | ( lastchunk - > ext ! = chunk - > ext )
| | ( lastchunk - > family ! = chunk - > family )
| | ( lastchunk - > familyVersion ! = chunk - > familyVersion )
| | ( lastchunk - > htype ! = chunk - > htype ) ) {
2020-01-28 14:34:02 +00:00
QString message = " *** unexpected change in header data " ;
qWarning ( ) < < path < < message ;
2021-09-02 14:09:11 +00:00
m_ctx - > LogUnexpectedMessage ( message ) ;
2020-01-28 14:34:02 +00:00
// There used to be error-recovery code here, written before we checked CRCs.
// If we ever encounter data with a valid CRC that triggers the above warnings,
// we can then revisit how to handle it.
2019-05-14 22:57:04 +00:00
}
}
CHUNKS . append ( chunk ) ;
lastchunk = chunk ;
cnt + + ;
2021-05-23 16:43:31 +00:00
} while ( ! src - > atEnd ( ) ) ;
2019-05-14 22:57:04 +00:00
2021-05-23 16:43:31 +00:00
delete src ;
2019-05-14 22:57:04 +00:00
return CHUNKS ;
}
2014-04-17 05:58:57 +00:00
bool initialized = false ;
2014-05-17 05:04:40 +00:00
using namespace schema ;
Channel PRS1Channels ;
2014-08-06 14:06:44 +00:00
void PRS1Loader : : initChannels ( )
2011-06-26 08:30:44 +00:00
{
2014-08-06 14:06:44 +00:00
Channel * chan = nullptr ;
2014-05-17 05:04:40 +00:00
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , new Channel ( CPAP_PressurePulse = 0x1009 , MINOR_FLAG , MT_CPAP , SESSION ,
2014-08-03 13:00:13 +00:00
" PressurePulse " ,
QObject : : tr ( " Pressure Pulse " ) ,
2014-05-17 05:04:40 +00:00
QObject : : tr ( " A pulse of pressure 'pinged' to detect a closed airway. " ) ,
2014-08-03 13:00:13 +00:00
QObject : : tr ( " PP " ) ,
STR_UNIT_EventsPerHour , DEFAULT , QColor ( " dark red " ) ) ) ;
2020-03-24 21:15:29 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_Mode = 0xe120 , SETTING , MT_CPAP , SESSION ,
" PRS1Mode " , QObject : : tr ( " Mode " ) ,
QObject : : tr ( " PAP Mode " ) ,
QObject : : tr ( " Mode " ) ,
" " , LOOKUP , Qt : : green ) ) ;
2020-03-25 01:04:21 +00:00
chan - > addOption ( PRS1_MODE_CPAPCHECK , QObject : : tr ( " CPAP-Check " ) ) ;
chan - > addOption ( PRS1_MODE_CPAP , QObject : : tr ( " CPAP " ) ) ;
chan - > addOption ( PRS1_MODE_AUTOCPAP , QObject : : tr ( " AutoCPAP " ) ) ;
chan - > addOption ( PRS1_MODE_AUTOTRIAL , QObject : : tr ( " Auto-Trial " ) ) ;
chan - > addOption ( PRS1_MODE_BILEVEL , QObject : : tr ( " Bi-Level " ) ) ;
chan - > addOption ( PRS1_MODE_AUTOBILEVEL , QObject : : tr ( " AutoBiLevel " ) ) ;
chan - > addOption ( PRS1_MODE_ASV , QObject : : tr ( " ASV " ) ) ;
chan - > addOption ( PRS1_MODE_S , QObject : : tr ( " S " ) ) ;
chan - > addOption ( PRS1_MODE_ST , QObject : : tr ( " S/T " ) ) ;
chan - > addOption ( PRS1_MODE_PC , QObject : : tr ( " PC " ) ) ;
chan - > addOption ( PRS1_MODE_ST_AVAPS , QObject : : tr ( " S/T - AVAPS " ) ) ;
chan - > addOption ( PRS1_MODE_PC_AVAPS , QObject : : tr ( " PC - AVAPS " ) ) ;
2020-03-24 21:15:29 +00:00
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_FlexMode = 0xe105 , SETTING , MT_CPAP , SESSION ,
2014-08-03 13:00:13 +00:00
" PRS1FlexMode " , QObject : : tr ( " Flex Mode " ) ,
QObject : : tr ( " PRS1 pressure relief mode. " ) ,
QObject : : tr ( " Flex Mode " ) ,
2014-08-04 19:57:48 +00:00
" " , LOOKUP , Qt : : green ) ) ;
2014-08-03 13:00:13 +00:00
chan - > addOption ( FLEX_None , STR_TR_None ) ;
chan - > addOption ( FLEX_CFlex , QObject : : tr ( " C-Flex " ) ) ;
chan - > addOption ( FLEX_CFlexPlus , QObject : : tr ( " C-Flex+ " ) ) ;
chan - > addOption ( FLEX_AFlex , QObject : : tr ( " A-Flex " ) ) ;
2019-12-04 00:04:09 +00:00
chan - > addOption ( FLEX_PFlex , QObject : : tr ( " P-Flex " ) ) ;
2014-08-03 13:00:13 +00:00
chan - > addOption ( FLEX_RiseTime , QObject : : tr ( " Rise Time " ) ) ;
chan - > addOption ( FLEX_BiFlex , QObject : : tr ( " Bi-Flex " ) ) ;
2020-03-25 01:04:21 +00:00
//chan->addOption(FLEX_AVAPS, QObject::tr("AVAPS")); // Converted into AVAPS PRS1_Mode with FLEX_RiseTime
2020-09-13 18:00:59 +00:00
chan - > addOption ( FLEX_Flex , QObject : : tr ( " Flex " ) ) ;
2014-08-03 13:00:13 +00:00
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_FlexLevel = 0xe106 , SETTING , MT_CPAP , SESSION ,
2014-08-03 13:00:13 +00:00
" PRS1FlexSet " ,
QObject : : tr ( " Flex Level " ) ,
QObject : : tr ( " PRS1 pressure relief setting. " ) ,
QObject : : tr ( " Flex Level " ) ,
2014-08-04 19:57:48 +00:00
" " , LOOKUP , Qt : : blue ) ) ;
2014-08-03 13:00:13 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2014-05-17 05:04:40 +00:00
2020-03-23 16:59:06 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_FlexLock = 0xe111 , SETTING , MT_CPAP , SESSION ,
" PRS1FlexLock " ,
QObject : : tr ( " Flex Lock " ) ,
QObject : : tr ( " Whether Flex settings are available to you. " ) ,
QObject : : tr ( " Flex Lock " ) ,
" " , LOOKUP , Qt : : black ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2020-03-24 01:09:55 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_RiseTime = 0xe119 , SETTING , MT_CPAP , SESSION ,
" PRS1RiseTime " ,
QObject : : tr ( " Rise Time " ) ,
QObject : : tr ( " Amount of time it takes to transition from EPAP to IPAP, the higher the number the slower the transition " ) ,
QObject : : tr ( " Rise Time " ) ,
2020-03-26 01:14:25 +00:00
" " , LOOKUP , Qt : : blue ) ) ;
2020-03-24 01:09:55 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_RiseTimeLock = 0xe11a , SETTING , MT_CPAP , SESSION ,
" PRS1RiseTimeLock " ,
QObject : : tr ( " Rise Time Lock " ) ,
QObject : : tr ( " Whether Rise Time settings are available to you. " ) ,
QObject : : tr ( " Rise Lock " ) ,
" " , LOOKUP , Qt : : black ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_HumidStatus = 0xe101 , SETTING , MT_CPAP , SESSION ,
2014-08-04 16:12:49 +00:00
" PRS1HumidStat " ,
QObject : : tr ( " Humidifier Status " ) ,
QObject : : tr ( " PRS1 humidifier connected? " ) ,
2020-01-13 01:13:21 +00:00
QObject : : tr ( " Humidifier " ) ,
2014-08-04 19:57:48 +00:00
" " , LOOKUP , Qt : : green ) ) ;
chan - > addOption ( 0 , QObject : : tr ( " Disconnected " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " Connected " ) ) ;
2014-08-04 16:12:49 +00:00
2020-01-12 00:14:01 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_HumidMode = 0xe110 , SETTING , MT_CPAP , SESSION ,
" PRS1HumidMode " ,
QObject : : tr ( " Humidification Mode " ) ,
QObject : : tr ( " PRS1 Humidification Mode " ) ,
QObject : : tr ( " Humid. Mode " ) ,
2014-08-11 01:44:25 +00:00
" " , LOOKUP , Qt : : green ) ) ;
2020-04-10 20:48:28 +00:00
chan - > addOption ( HUMID_Fixed , QObject : : tr ( " Fixed (Classic) " ) ) ;
chan - > addOption ( HUMID_Adaptive , QObject : : tr ( " Adaptive (System One) " ) ) ;
chan - > addOption ( HUMID_HeatedTube , QObject : : tr ( " Heated Tube " ) ) ;
chan - > addOption ( HUMID_Passover , QObject : : tr ( " Passover " ) ) ;
chan - > addOption ( HUMID_Error , QObject : : tr ( " Error " ) ) ;
2020-01-12 00:14:01 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_TubeTemp = 0xe10f , SETTING , MT_CPAP , SESSION ,
" PRS1TubeTemp " ,
QObject : : tr ( " Tube Temperature " ) ,
QObject : : tr ( " PRS1 Heated Tube Temperature " ) ,
2020-01-13 01:13:21 +00:00
QObject : : tr ( " Tube Temp. " ) ,
2020-01-12 00:14:01 +00:00
" " , LOOKUP , Qt : : red ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
2014-08-11 01:44:25 +00:00
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_HumidLevel = 0xe102 , SETTING , MT_CPAP , SESSION ,
2014-08-04 16:12:49 +00:00
" PRS1HumidLevel " ,
2020-01-12 00:14:01 +00:00
QObject : : tr ( " Humidifier " ) , // label varies in reports, "Humidifier Setting" in 50-series, "Humidity Level" in 60-series, "Humidifier" in DreamStation
QObject : : tr ( " PRS1 Humidifier Setting " ) ,
2020-12-30 01:23:37 +00:00
QObject : : tr ( " Humid. Level " ) ,
2020-01-12 00:14:01 +00:00
" " , LOOKUP , Qt : : blue ) ) ;
2014-08-04 19:57:48 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2020-12-30 01:23:37 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_HumidTargetTime = 0xe11b , SETTING , MT_CPAP , SESSION ,
" PRS1HumidTargetTime " ,
QObject : : tr ( " Target Time " ) ,
QObject : : tr ( " PRS1 Humidifier Target Time " ) ,
QObject : : tr ( " Hum. Tgt Time " ) ,
STR_UNIT_Hours , DEFAULT , Qt : : green ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , QObject : : tr ( " Auto " ) ) ;
2020-03-23 17:07:08 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_MaskResistSet = 0xe104 , SETTING , MT_CPAP , SESSION ,
" MaskResistSet " ,
QObject : : tr ( " Mask Resistance Setting " ) ,
QObject : : tr ( " Mask Resistance Setting " ) ,
QObject : : tr ( " Mask Resist. " ) ,
2014-08-04 19:57:48 +00:00
" " , LOOKUP , Qt : : green ) ) ;
2014-08-04 16:12:49 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_HoseDiam = 0xe107 , SETTING , MT_CPAP , SESSION ,
2014-08-04 19:57:48 +00:00
" PRS1HoseDiam " ,
QObject : : tr ( " Hose Diameter " ) ,
QObject : : tr ( " Diameter of primary CPAP hose " ) ,
2020-03-24 21:15:29 +00:00
QObject : : tr ( " Hose Diam. " ) ,
2014-08-04 19:57:48 +00:00
" " , LOOKUP , Qt : : green ) ) ;
2020-01-28 19:07:58 +00:00
chan - > addOption ( 22 , QObject : : tr ( " 22mm " ) ) ;
chan - > addOption ( 15 , QObject : : tr ( " 15mm " ) ) ;
chan - > addOption ( 12 , QObject : : tr ( " 12mm " ) ) ;
2014-08-04 19:57:48 +00:00
2020-03-23 16:59:06 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_TubeLock = 0xe112 , SETTING , MT_CPAP , SESSION ,
" PRS1TubeLock " ,
QObject : : tr ( " Tubing Type Lock " ) ,
QObject : : tr ( " Whether tubing type settings are available to you. " ) ,
QObject : : tr ( " Tube Lock " ) ,
" " , LOOKUP , Qt : : black ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2020-03-23 17:07:08 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_MaskResistLock = 0xe108 , SETTING , MT_CPAP , SESSION ,
" MaskResistLock " ,
QObject : : tr ( " Mask Resistance Lock " ) ,
QObject : : tr ( " Whether mask resistance settings are available to you. " ) ,
2020-03-24 21:15:29 +00:00
QObject : : tr ( " Mask Res. Lock " ) ,
2020-03-23 16:59:06 +00:00
" " , LOOKUP , Qt : : black ) ) ;
2014-08-04 19:57:48 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_AutoOn = 0xe109 , SETTING , MT_CPAP , SESSION ,
2014-08-04 19:57:48 +00:00
" PRS1AutoOn " ,
QObject : : tr ( " Auto On " ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " A few breaths automatically starts device " ) ,
2014-08-04 19:57:48 +00:00
QObject : : tr ( " Auto On " ) ,
" " , LOOKUP , Qt : : green ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_AutoOff = 0xe10a , SETTING , MT_CPAP , SESSION ,
2014-08-04 19:57:48 +00:00
" PRS1AutoOff " ,
QObject : : tr ( " Auto Off " ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " Device automatically switches off " ) ,
2014-08-04 19:57:48 +00:00
QObject : : tr ( " Auto Off " ) ,
" " , LOOKUP , Qt : : green ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_MaskAlert = 0xe10b , SETTING , MT_CPAP , SESSION ,
2014-08-04 19:57:48 +00:00
" PRS1MaskAlert " ,
QObject : : tr ( " Mask Alert " ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " Whether or not device allows Mask checking. " ) ,
2014-08-04 19:57:48 +00:00
QObject : : tr ( " Mask Alert " ) ,
" " , LOOKUP , Qt : : green ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2020-03-23 16:59:06 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_ShowAHI = 0xe10c , SETTING , MT_CPAP , SESSION ,
2014-08-04 19:57:48 +00:00
" PRS1ShowAHI " ,
QObject : : tr ( " Show AHI " ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " Whether or not device shows AHI via built-in display. " ) ,
2014-08-04 19:57:48 +00:00
QObject : : tr ( " Show AHI " ) ,
" " , LOOKUP , Qt : : green ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2020-03-23 16:59:06 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_RampType = 0xe113 , SETTING , MT_CPAP , SESSION ,
" PRS1RampType " ,
QObject : : tr ( " Ramp Type " ) ,
QObject : : tr ( " Type of ramp curve to use. " ) ,
QObject : : tr ( " Ramp Type " ) ,
" " , LOOKUP , Qt : : black ) ) ;
chan - > addOption ( 0 , QObject : : tr ( " Linear " ) ) ;
chan - > addOption ( 1 , QObject : : tr ( " SmartRamp " ) ) ;
2021-11-05 13:30:44 +00:00
chan - > addOption ( 2 , QObject : : tr ( " Ramp+ " ) ) ;
2020-03-23 16:59:06 +00:00
2020-03-24 01:09:55 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_BackupBreathMode = 0xe114 , SETTING , MT_CPAP , SESSION ,
" PRS1BackupBreathMode " ,
QObject : : tr ( " Backup Breath Mode " ) ,
QObject : : tr ( " The kind of backup breath rate in use: none (off), automatic, or fixed " ) ,
QObject : : tr ( " Breath Rate " ) ,
" " , LOOKUP , Qt : : black ) ) ;
2020-03-26 01:14:25 +00:00
chan - > addOption ( 0 , STR_TR_Off ) ;
2020-03-24 01:09:55 +00:00
chan - > addOption ( 1 , QObject : : tr ( " Auto " ) ) ;
chan - > addOption ( 2 , QObject : : tr ( " Fixed " ) ) ;
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_BackupBreathRate = 0xe115 , SETTING , MT_CPAP , SESSION ,
" PRS1BackupBreathRate " ,
QObject : : tr ( " Fixed Backup Breath BPM " ) ,
QObject : : tr ( " Minimum breaths per minute (BPM) below which a timed breath will be initiated " ) ,
QObject : : tr ( " Breath BPM " ) ,
2020-03-26 01:14:25 +00:00
STR_UNIT_BreathsPerMinute , LOOKUP , Qt : : black ) ) ;
2020-03-24 01:09:55 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_BackupBreathTi = 0xe116 , SETTING , MT_CPAP , SESSION ,
" PRS1BackupBreathTi " ,
QObject : : tr ( " Timed Inspiration " ) ,
2020-03-24 21:15:29 +00:00
QObject : : tr ( " The time that a timed breath will provide IPAP before transitioning to EPAP " ) ,
2020-03-24 01:09:55 +00:00
QObject : : tr ( " Timed Insp. " ) ,
2020-03-24 21:15:29 +00:00
STR_UNIT_Seconds , DEFAULT , Qt : : blue ) ) ;
2020-03-24 01:09:55 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_AutoTrial = 0xe117 , SETTING , MT_CPAP , SESSION ,
" PRS1AutoTrial " ,
2020-03-26 01:14:25 +00:00
QObject : : tr ( " Auto-Trial Duration " ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " The number of days in the Auto-CPAP trial period, after which the device will revert to CPAP " ) ,
2020-03-26 01:14:25 +00:00
QObject : : tr ( " Auto-Trial Dur. " ) ,
2020-03-24 01:09:55 +00:00
" " , LOOKUP , Qt : : black ) ) ;
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_EZStart = 0xe118 , SETTING , MT_CPAP , SESSION ,
" PRS1EZStart " ,
QObject : : tr ( " EZ-Start " ) ,
QObject : : tr ( " Whether or not EZ-Start is enabled " ) ,
QObject : : tr ( " EZ-Start " ) ,
" " , LOOKUP , Qt : : black ) ) ;
chan - > addOption ( 0 , STR_TR_Off ) ;
chan - > addOption ( 1 , STR_TR_On ) ;
2020-03-26 13:01:28 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_VariableBreathing = 0x1156 , SPAN , MT_CPAP , SESSION ,
" PRS1_VariableBreathing " ,
QObject : : tr ( " Variable Breathing " ) ,
QObject : : tr ( " UNCONFIRMED: Possibly variable breathing, which are periods of high deviation from the peak inspiratory flow trend " ) ,
" VB " ,
2019-12-28 04:08:59 +00:00
STR_UNIT_Seconds ,
2019-10-29 18:06:57 +00:00
DEFAULT , QColor ( " #ffe8f0 " ) ) ) ;
2020-03-26 13:01:28 +00:00
chan - > setEnabled ( false ) ; // disable by default
2016-03-06 03:15:54 +00:00
channel . add ( GRP_CPAP , new Channel ( PRS1_BND = 0x1159 , SPAN , MT_CPAP , SESSION ,
" PRS1_BND " ,
2016-04-26 06:11:39 +00:00
QObject : : tr ( " Breathing Not Detected " ) ,
2022-02-27 16:50:10 +00:00
QObject : : tr ( " A period during a session where the device could not detect flow. " ) ,
2016-04-26 06:11:39 +00:00
QObject : : tr ( " BND " ) ,
2016-03-06 03:15:54 +00:00
STR_UNIT_Unknown ,
DEFAULT , QColor ( " light purple " ) ) ) ;
2014-08-23 06:21:50 +00:00
channel . add ( GRP_CPAP , new Channel ( PRS1_TimedBreath = 0x1180 , MINOR_FLAG , MT_CPAP , SESSION ,
2014-08-06 07:08:34 +00:00
" PRS1TimedBreath " ,
2014-08-06 14:06:44 +00:00
QObject : : tr ( " Timed Breath " ) ,
QObject : : tr ( " Machine Initiated Breath " ) ,
QObject : : tr ( " TB " ) ,
2019-06-17 21:33:39 +00:00
STR_UNIT_Seconds ,
2014-08-06 07:08:34 +00:00
DEFAULT , QColor ( " black " ) ) ) ;
2020-04-22 20:52:21 +00:00
channel . add ( GRP_CPAP , chan = new Channel ( PRS1_PeakFlow = 0x115a , WAVEFORM , MT_CPAP , SESSION ,
" PRS1PeakFlow " ,
QObject : : tr ( " Peak Flow " ) ,
QObject : : tr ( " Peak flow during a 2-minute interval " ) ,
QObject : : tr ( " Peak Flow " ) ,
STR_UNIT_LPM ,
DEFAULT , QColor ( " red " ) ) ) ;
chan - > setShowInOverview ( true ) ;
2014-08-06 14:06:44 +00:00
}
2014-08-06 07:08:34 +00:00
2014-08-06 14:06:44 +00:00
void PRS1Loader : : Register ( )
{
if ( initialized ) { return ; }
2014-08-06 07:08:34 +00:00
2014-08-06 14:06:44 +00:00
qDebug ( ) < < " Registering PRS1Loader " ;
RegisterLoader ( new PRS1Loader ( ) ) ;
initialized = true ;
2011-06-26 08:30:44 +00:00
}