2019-05-03 18:45:21 +00:00
/* SleepLib PRS1 Loader Header
2014-04-09 21:01:57 +00:00
*
2019-05-03 20:17:32 +00:00
* Copyright ( c ) 2019 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 20:48:38 +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
# ifndef PRS1LOADER_H
# define PRS1LOADER_H
//#include <map>
//using namespace std;
# include "SleepLib/machine.h" // Base class: MachineLoader
# include "SleepLib/machine_loader.h"
# include "SleepLib/profiles.h"
2019-05-05 01:50:45 +00:00
# ifdef UNITTEST_MODE
# define private public
# define protected public
# endif
2011-06-26 08:30:44 +00:00
//********************************************************************************************
/// IMPORTANT!!!
//********************************************************************************************
2018-05-05 15:48:32 +00:00
// Please INCREMENT the following value when making changes to this loaders implementation
// BEFORE making a release
2016-03-08 14:45:02 +00:00
const int prs1_data_version = 15 ;
2011-06-26 08:30:44 +00:00
//
//********************************************************************************************
2011-12-19 05:35:05 +00:00
/*! \class PRS1
\ brief PRS1 customized machine object ( via CPAP )
*/
2014-04-17 05:58:57 +00:00
class PRS1 : public CPAP
2011-06-26 08:30:44 +00:00
{
2014-04-17 05:58:57 +00:00
public :
2018-06-04 23:26:46 +00:00
PRS1 ( Profile * , MachineID id = 0 ) ;
2011-06-26 08:30:44 +00:00
virtual ~ PRS1 ( ) ;
} ;
2014-04-17 05:58:57 +00:00
const int max_load_buffer_size = 1024 * 1024 ;
2014-05-31 21:25:07 +00:00
const QString prs1_class_name = STR_MACH_PRS1 ;
/*! \struct PRS1Waveform
\ brief Used in PRS1 Waveform Parsing */
struct PRS1Waveform {
PRS1Waveform ( quint16 i , quint8 f ) {
interleave = i ;
sample_format = f ;
}
quint16 interleave ;
quint8 sample_format ;
} ;
2011-06-26 08:30:44 +00:00
2016-03-02 02:09:32 +00:00
/*! \class PRS1DataChunk
* \ brief Representing a chunk of event / summary / waveform data after the header is parsed . */
2014-05-31 21:25:07 +00:00
class PRS1DataChunk
{
friend class PRS1DataGroup ;
public :
PRS1DataChunk ( ) {
timestamp = 0 ;
ext = 255 ;
sessionid = 0 ;
htype = 0 ;
family = 0 ;
familyVersion = 0 ;
duration = 0 ;
}
~ PRS1DataChunk ( ) {
}
inline int size ( ) const { return m_data . size ( ) ; }
2019-05-14 22:57:04 +00:00
QByteArray m_header ;
2014-05-31 21:25:07 +00:00
QByteArray m_data ;
2016-01-19 04:26:28 +00:00
QByteArray m_headerblock ;
2014-05-31 21:25:07 +00:00
2019-05-14 20:20:32 +00:00
QString m_path ;
qint64 m_filepos ; // file offset
int m_index ; // nth chunk in file
2014-05-31 21:25:07 +00:00
SessionID sessionid ;
quint8 fileVersion ;
2019-05-14 22:57:04 +00:00
quint16 blockSize ;
2014-05-31 21:25:07 +00:00
quint8 ext ;
quint8 htype ;
quint8 family ;
quint8 familyVersion ;
quint32 timestamp ;
quint16 duration ;
QList < PRS1Waveform > waveformInfo ;
2018-05-05 15:48:32 +00:00
QMap < unsigned char , short > hblock ;
2019-05-15 00:47:00 +00:00
quint8 storedChecksum ; // header checksum stored in file, last byte of m_header
quint8 calcChecksum ; // header checksum as calculated when parsing
2014-05-31 21:25:07 +00:00
} ;
2014-08-04 15:40:56 +00:00
class PRS1Loader ;
2016-03-02 02:09:32 +00:00
/*! \class PRS1Import
* \ brief Contains the functions to parse a single session . . . multithreaded */
2014-08-04 15:40:56 +00:00
class PRS1Import : public ImportTask
2014-05-31 21:25:07 +00:00
{
public :
2014-08-04 15:40:56 +00:00
PRS1Import ( PRS1Loader * l , SessionID s , Machine * m ) : loader ( l ) , sessionid ( s ) , mach ( m ) {
summary = nullptr ;
compliance = nullptr ;
event = nullptr ;
2014-05-31 21:25:07 +00:00
session = nullptr ;
}
2014-08-04 15:40:56 +00:00
virtual ~ PRS1Import ( ) {
2014-05-31 21:25:07 +00:00
delete compliance ;
delete summary ;
delete event ;
2016-03-02 02:09:32 +00:00
for ( int i = 0 ; i < waveforms . size ( ) ; + + i ) { delete waveforms . at ( i ) ; }
2014-05-31 21:25:07 +00:00
}
2014-09-17 06:59:58 +00:00
2016-03-02 02:09:32 +00:00
//! \brief PRS1Import thread starts execution here.
2014-08-04 15:40:56 +00:00
virtual void run ( ) ;
PRS1DataChunk * compliance ;
PRS1DataChunk * summary ;
PRS1DataChunk * event ;
QList < PRS1DataChunk * > waveforms ;
2018-06-04 20:48:38 +00:00
QList < PRS1DataChunk * > oximetry ;
2014-08-04 15:40:56 +00:00
2018-05-05 15:48:32 +00:00
QMap < unsigned char , QByteArray > mainblock ;
QMap < unsigned char , QByteArray > hbdata ;
2014-09-29 14:41:31 +00:00
QString wavefile ;
QString oxifile ;
2014-05-31 21:25:07 +00:00
2016-03-02 02:09:32 +00:00
//! \brief As it says on the tin.. Parses .001 files for bricks.
2014-05-31 21:25:07 +00:00
bool ParseCompliance ( ) ;
2016-03-02 02:09:32 +00:00
//! \brief Figures out which Summary Parser to call, based on machine family/version and calls it.
2014-05-31 21:25:07 +00:00
bool ParseSummary ( ) ;
2016-03-02 02:09:32 +00:00
//! \brief Figures out which Event Parser to call, based on machine family/version and calls it.
2014-05-31 21:25:07 +00:00
bool ParseEvents ( ) ;
2016-03-02 02:09:32 +00:00
2019-05-14 01:20:11 +00:00
//! \brief Coalesce contiguous .005 or .006 waveform chunks from the file into larger chunks for import.
QList < PRS1DataChunk * > CoalesceWaveformChunks ( QList < PRS1DataChunk * > & allchunks ) ;
2016-03-02 02:09:32 +00:00
//! \brief Takes the parsed list of Flow/MaskPressure waveform chunks and adds them to the database
2014-05-31 21:25:07 +00:00
bool ParseWaveforms ( ) ;
2016-03-02 02:09:32 +00:00
//! \brief Takes the parsed list of oximeter waveform chunks and adds them to the database.
2014-09-29 14:41:31 +00:00
bool ParseOximetery ( ) ;
2014-05-31 21:25:07 +00:00
2016-03-02 02:09:32 +00:00
//! \brief Summary parser for 50 series Family 0 CPAP/APAP models
2014-08-03 13:00:13 +00:00
bool ParseSummaryF0 ( ) ;
2016-03-02 02:09:32 +00:00
//! \brief Summary parser for 60 series Family 0 CPAP/APAP models
2014-08-03 13:00:13 +00:00
bool ParseSummaryF0V4 ( ) ;
2016-03-02 02:09:32 +00:00
//! \brief Summary parser for 1060 series AVAPS models
2014-08-03 13:00:13 +00:00
bool ParseSummaryF3 ( ) ;
2016-03-11 00:00:01 +00:00
//! \brief Summary parser for 50 series Family 5-0 BiPAP/AutoSV models
2014-08-17 12:56:05 +00:00
bool ParseSummaryF5V0 ( ) ;
2016-03-11 00:00:01 +00:00
//! \brief Summary parser for 60 series Family 5-1 BiPAP/AutoSV models
2014-08-17 12:56:05 +00:00
bool ParseSummaryF5V1 ( ) ;
2016-03-11 00:00:01 +00:00
//! \brief Summary parser for 60 series Family 5-2 BiPAP/AutoSV models
bool ParseSummaryF5V2 ( ) ;
2019-05-03 18:45:21 +00:00
//! \brief Summary parser for 60 series Family 5-3 BiPAP/AutoSV models
2018-03-23 19:24:29 +00:00
bool ParseSummaryF5V3 ( ) ;
2016-03-11 00:00:01 +00:00
2016-03-02 02:09:32 +00:00
//! \brief Summary parser for DreamStation series CPAP/APAP models
2016-01-20 06:15:15 +00:00
bool ParseSummaryF0V6 ( ) ;
2014-08-03 13:00:13 +00:00
2014-05-31 21:25:07 +00:00
//! \brief Parse a single data chunk from a .002 file containing event data for a standard system one machine
bool ParseF0Events ( ) ;
2016-02-28 15:11:53 +00:00
//! \brief Parse a single data chunk from a .002 file containing event data for a AVAPS 1060P machine
bool ParseF3Events ( ) ;
2018-05-05 07:14:44 +00:00
//! \brief Parse a single data chunk from a .002 file containing event data for a AVAPS 1060P machine file version 3
bool ParseF3EventsV3 ( ) ;
2014-05-31 21:25:07 +00:00
//! \brief Parse a single data chunk from a .002 file containing event data for a family 5 ASV machine (which has a different format)
bool ParseF5Events ( ) ;
2018-03-23 19:24:29 +00:00
//! \brief Parse a single data chunk from a .002 file containing event data for a family 5 ASV file version 3 machine (which has a different format again)
bool ParseF5EventsFV3 ( ) ;
2014-05-31 21:25:07 +00:00
protected :
2014-08-04 15:40:56 +00:00
Session * session ;
2014-05-31 21:25:07 +00:00
PRS1Loader * loader ;
SessionID sessionid ;
Machine * mach ;
2014-08-17 12:56:05 +00:00
int summary_duration ;
2019-05-03 18:45:21 +00:00
//! \brief Parse all the chunks in a single machine session
bool ParseSession ( void ) ;
//! \brief Save parsed session data to the database
void SaveSessionToDatabase ( void ) ;
2014-05-31 21:25:07 +00:00
} ;
2011-06-26 08:30:44 +00:00
2011-12-19 05:35:05 +00:00
/*! \class PRS1Loader
\ brief Philips Respironics System One Loader Module
*/
2014-08-03 13:00:13 +00:00
class PRS1Loader : public CPAPLoader
2011-06-26 08:30:44 +00:00
{
2018-06-12 16:32:18 +00:00
Q_OBJECT
2014-04-17 05:58:57 +00:00
public :
2011-06-26 08:30:44 +00:00
PRS1Loader ( ) ;
virtual ~ PRS1Loader ( ) ;
2014-04-26 09:54:08 +00:00
2016-03-02 02:09:32 +00:00
//! \brief Examine path and return it back if it contains what looks to be a valid PRS1 SD card structure
2014-09-29 14:41:31 +00:00
QString checkDir ( const QString & path ) ;
2016-03-02 02:09:32 +00:00
//! \brief Peek into PROP.TXT or properties.txt at given path, and use it to fill MachineInfo structure
2018-04-27 04:29:03 +00:00
bool PeekProperties ( MachineInfo & info , const QString & path , Machine * mach = nullptr ) ;
2014-09-29 14:41:31 +00:00
2014-04-26 09:54:08 +00:00
//! \brief Detect if the given path contains a valid Folder structure
virtual bool Detect ( const QString & path ) ;
2016-03-02 02:09:32 +00:00
//! \brief Wrapper for PeekProperties that creates the MachineInfo structure.
2014-09-29 14:41:31 +00:00
virtual MachineInfo PeekInfo ( const QString & path ) ;
2011-12-19 05:35:05 +00:00
//! \brief Scans directory path for valid PRS1 signature
2018-04-27 04:29:03 +00:00
virtual int Open ( const QString & path ) ;
2011-12-19 05:35:05 +00:00
//! \brief Returns the database version of this loader
2011-07-15 13:30:41 +00:00
virtual int Version ( ) { return prs1_data_version ; }
2011-12-19 05:35:05 +00:00
2014-07-28 13:56:29 +00:00
//! \brief Return the loaderName, in this case "PRS1"
virtual const QString & loaderName ( ) { return prs1_class_name ; }
2011-12-19 05:35:05 +00:00
2016-03-02 02:09:32 +00:00
//! \brief Parse a PRS1 summary/event/waveform file and break into invidivual session or waveform chunks
2018-04-27 04:29:03 +00:00
QList < PRS1DataChunk * > ParseFile ( const QString & path ) ;
2014-08-04 15:40:56 +00:00
2019-05-14 22:57:04 +00:00
//! \brief Parse and return the next chunk from a PRS1 file
PRS1DataChunk * ParseChunk ( class QFile & f , int index = 0 ) ;
2011-12-19 05:35:05 +00:00
//! \brief Register this Module to the list of Loaders, so it knows to search for PRS1 data.
2011-06-26 08:30:44 +00:00
static void Register ( ) ;
2014-07-11 06:13:44 +00:00
2016-03-02 02:09:32 +00:00
//! \brief Generate a generic MachineInfo structure, with basic PRS1 info to be expanded upon.
2014-07-28 13:56:29 +00:00
virtual MachineInfo newInfo ( ) {
2014-08-03 13:00:13 +00:00
return MachineInfo ( MT_CPAP , 0 , prs1_class_name , QObject : : tr ( " Philips Respironics " ) , QString ( ) , QString ( ) , QString ( ) , QObject : : tr ( " System One " ) , QDateTime : : currentDateTime ( ) , prs1_data_version ) ;
2014-07-28 13:56:29 +00:00
}
2014-08-04 15:40:56 +00:00
2014-08-03 13:00:13 +00:00
virtual QString PresReliefLabel ( ) { return QObject : : tr ( " " ) ; }
2016-03-02 02:09:32 +00:00
//! \brief Returns the PRS1 specific code for Pressure Relief Mode
2014-08-03 13:00:13 +00:00
virtual ChannelID PresReliefMode ( ) { return PRS1_FlexMode ; }
2016-03-02 02:09:32 +00:00
//! \brief Returns the PRS1 specific code for Pressure Relief Setting
2014-08-03 13:00:13 +00:00
virtual ChannelID PresReliefLevel ( ) { return PRS1_FlexLevel ; }
2014-07-28 13:56:29 +00:00
2016-03-02 02:09:32 +00:00
//! \brief Returns the PRS1 specific code for Humidifier Connected
2014-08-04 16:12:49 +00:00
virtual ChannelID HumidifierConnected ( ) { return PRS1_HumidStatus ; }
2016-03-02 02:09:32 +00:00
//! \brief Returns the PRS1 specific code for Humidifier Level
2014-08-04 16:12:49 +00:00
virtual ChannelID HumidifierLevel ( ) { return PRS1_HumidLevel ; }
2016-03-02 02:09:32 +00:00
//! \brief Called at application init, to set up any custom PRS1 Channels
2014-08-06 14:06:44 +00:00
void initChannels ( ) ;
2014-08-04 16:12:49 +00:00
2014-08-04 15:40:56 +00:00
QHash < SessionID , PRS1Import * > sesstasks ;
2018-05-05 07:14:44 +00:00
QMap < unsigned char , QStringList > unknownCodes ;
2014-07-11 06:13:44 +00:00
2014-04-17 05:58:57 +00:00
protected :
2011-06-26 08:30:44 +00:00
QString last ;
2014-04-17 05:58:57 +00:00
QHash < QString , Machine * > PRS1List ;
2011-12-19 05:35:05 +00:00
//! \brief Opens the SD folder structure for this machine, scans for data files and imports any new sessions
2018-04-27 04:29:03 +00:00
int OpenMachine ( const QString & path ) ;
2011-12-19 05:35:05 +00:00
2019-05-03 20:59:26 +00:00
//! \brief Finds the P0,P1,... session paths and property pathname and returns the base (10 or 16) of the session filenames
int FindSessionDirsAndProperties ( const QString & path , QStringList & paths , QString & propertyfile ) ;
//! \brief Reads the model number from the property file, evaluates its capabilities, and returns a machine instance
Machine * CreateMachineFromProperties ( QString propertyfile ) ;
//! \brief Scans the given directories for session data and create an import task for each logical session.
2019-05-03 19:07:15 +00:00
void ScanFiles ( const QStringList & paths , int sessionid_base , Machine * m ) ;
2014-09-30 05:25:11 +00:00
// //! \brief Parses "properties.txt" file containing machine information
// bool ParseProperties(Machine *m, QString filename);
2011-12-19 05:35:05 +00:00
//! \brief Parse a .005 waveform file, extracting Flow Rate waveform (and Mask Pressure data if available)
2018-04-27 04:29:03 +00:00
bool OpenWaveforms ( SessionID sid , const QString & filename ) ;
2011-12-19 05:35:05 +00:00
//! \brief Parse a data chunk from the .000 (brick) and .001 (summary) files.
2014-04-17 05:58:57 +00:00
bool ParseSummary ( Machine * mach , qint32 sequence , quint32 timestamp , unsigned char * data ,
quint16 size , int family , int familyVersion ) ;
2011-12-19 05:35:05 +00:00
2011-12-10 12:14:48 +00:00
2011-12-19 05:35:05 +00:00
//! \brief Open a PRS1 data file, and break into data chunks, delivering them to the correct parser.
2018-04-27 04:29:03 +00:00
bool OpenFile ( Machine * mach , const QString & filename ) ;
2011-12-19 05:35:05 +00:00
2011-12-06 14:39:14 +00:00
QHash < SessionID , Session * > extra_session ;
2011-12-19 05:35:05 +00:00
//! \brief PRS1 Data files can store multiple sessions, so store them in this list for later processing.
2011-12-10 12:14:48 +00:00
QHash < SessionID , Session * > new_sessions ;
2014-05-31 21:25:07 +00:00
2011-12-10 12:14:48 +00:00
qint32 summary_duration ;
} ;
2011-06-26 08:30:44 +00:00
# endif // PRS1LOADER_H